refactor: use services in routes plz
This commit is contained in:
@@ -40,6 +40,8 @@ export class SkillsService {
|
||||
|
||||
if (!categoriesMap.has(categoryId)) {
|
||||
categoriesMap.set(categoryId, {
|
||||
id: categoryId,
|
||||
name: row.category_name,
|
||||
category: row.category_name,
|
||||
icon: row.category_icon,
|
||||
skills: [],
|
||||
@@ -250,4 +252,241 @@ export class SkillsService {
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get all skills with usage count for admin
|
||||
*/
|
||||
static async getAllSkillsWithUsage(): Promise<
|
||||
Array<{
|
||||
id: string;
|
||||
name: string;
|
||||
description: string;
|
||||
icon: string;
|
||||
categoryId: string;
|
||||
category: string;
|
||||
usageCount: number;
|
||||
}>
|
||||
> {
|
||||
const pool = getPool();
|
||||
const query = `
|
||||
SELECT
|
||||
s.id,
|
||||
s.name,
|
||||
s.description,
|
||||
s.icon,
|
||||
sc.id as category_id,
|
||||
sc.name as category_name,
|
||||
COUNT(DISTINCT se.id) as usage_count
|
||||
FROM skills s
|
||||
LEFT JOIN skill_categories sc ON s.category_id = sc.id
|
||||
LEFT JOIN skill_evaluations se ON s.id = se.skill_id AND se.is_selected = true
|
||||
GROUP BY s.id, s.name, s.description, s.icon, sc.id, sc.name
|
||||
ORDER BY s.name
|
||||
`;
|
||||
|
||||
try {
|
||||
const result = await pool.query(query);
|
||||
|
||||
return result.rows.map((row) => ({
|
||||
id: row.id,
|
||||
name: row.name,
|
||||
description: row.description || "",
|
||||
icon: row.icon || "",
|
||||
categoryId: row.category_id,
|
||||
category: row.category_name,
|
||||
usageCount: parseInt(row.usage_count) || 0,
|
||||
}));
|
||||
} catch (error) {
|
||||
console.error("Error fetching skills with usage:", error);
|
||||
throw new Error("Failed to fetch skills with usage");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new skill for admin
|
||||
*/
|
||||
static async createSkillForAdmin(data: {
|
||||
name: string;
|
||||
categoryId: string;
|
||||
description?: string;
|
||||
icon?: string;
|
||||
}): Promise<{
|
||||
id: string;
|
||||
name: string;
|
||||
description: string;
|
||||
icon: string;
|
||||
categoryId: string;
|
||||
category: string;
|
||||
usageCount: number;
|
||||
}> {
|
||||
const pool = getPool();
|
||||
const client = await pool.connect();
|
||||
|
||||
try {
|
||||
await client.query("BEGIN");
|
||||
|
||||
// Vérifier si la skill existe déjà
|
||||
const existingSkill = await client.query(
|
||||
"SELECT id FROM skills WHERE LOWER(name) = LOWER($1)",
|
||||
[data.name]
|
||||
);
|
||||
|
||||
if (existingSkill.rows.length > 0) {
|
||||
throw new Error("Une skill avec ce nom existe déjà");
|
||||
}
|
||||
|
||||
// Créer la nouvelle skill
|
||||
const result = await client.query(
|
||||
`INSERT INTO skills (id, name, category_id, description, icon)
|
||||
VALUES (gen_random_uuid(), $1, $2, $3, $4)
|
||||
RETURNING id, name, description, icon, category_id`,
|
||||
[data.name, data.categoryId, data.description || "", data.icon || ""]
|
||||
);
|
||||
|
||||
const newSkill = result.rows[0];
|
||||
|
||||
// Récupérer le nom de la catégorie
|
||||
const categoryResult = await client.query(
|
||||
"SELECT name FROM skill_categories WHERE id = $1",
|
||||
[newSkill.category_id]
|
||||
);
|
||||
|
||||
await client.query("COMMIT");
|
||||
|
||||
return {
|
||||
id: newSkill.id,
|
||||
name: newSkill.name,
|
||||
description: newSkill.description,
|
||||
icon: newSkill.icon,
|
||||
categoryId: newSkill.category_id,
|
||||
category: categoryResult.rows[0]?.name || "Inconnue",
|
||||
usageCount: 0,
|
||||
};
|
||||
} catch (error) {
|
||||
await client.query("ROLLBACK");
|
||||
throw error;
|
||||
} finally {
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Update a skill for admin
|
||||
*/
|
||||
static async updateSkillForAdmin(data: {
|
||||
id: string;
|
||||
name: string;
|
||||
categoryId: string;
|
||||
description?: string;
|
||||
icon?: string;
|
||||
}): Promise<{
|
||||
id: string;
|
||||
name: string;
|
||||
description: string;
|
||||
icon: string;
|
||||
categoryId: string;
|
||||
category: string;
|
||||
}> {
|
||||
const pool = getPool();
|
||||
const client = await pool.connect();
|
||||
|
||||
try {
|
||||
await client.query("BEGIN");
|
||||
|
||||
// Vérifier si la skill existe
|
||||
const existingSkill = await client.query(
|
||||
"SELECT id FROM skills WHERE id = $1",
|
||||
[data.id]
|
||||
);
|
||||
|
||||
if (existingSkill.rows.length === 0) {
|
||||
throw new Error("Skill non trouvée");
|
||||
}
|
||||
|
||||
// Vérifier si le nom existe déjà (sauf pour cette skill)
|
||||
const duplicateName = await client.query(
|
||||
"SELECT id FROM skills WHERE LOWER(name) = LOWER($1) AND id != $2",
|
||||
[data.name, data.id]
|
||||
);
|
||||
|
||||
if (duplicateName.rows.length > 0) {
|
||||
throw new Error("Une skill avec ce nom existe déjà");
|
||||
}
|
||||
|
||||
// Mettre à jour la skill
|
||||
await client.query(
|
||||
`UPDATE skills
|
||||
SET name = $1, category_id = $2, description = $3, icon = $4
|
||||
WHERE id = $5`,
|
||||
[
|
||||
data.name,
|
||||
data.categoryId,
|
||||
data.description || "",
|
||||
data.icon || "",
|
||||
data.id,
|
||||
]
|
||||
);
|
||||
|
||||
// Récupérer la skill mise à jour
|
||||
const result = await client.query(
|
||||
`SELECT s.id, s.name, s.description, s.icon, s.category_id, sc.name as category_name
|
||||
FROM skills s
|
||||
LEFT JOIN skill_categories sc ON s.category_id = sc.id
|
||||
WHERE s.id = $1`,
|
||||
[data.id]
|
||||
);
|
||||
|
||||
await client.query("COMMIT");
|
||||
|
||||
const skill = result.rows[0];
|
||||
return {
|
||||
id: skill.id,
|
||||
name: skill.name,
|
||||
description: skill.description,
|
||||
icon: skill.icon,
|
||||
categoryId: skill.category_id,
|
||||
category: skill.category_name,
|
||||
};
|
||||
} catch (error) {
|
||||
await client.query("ROLLBACK");
|
||||
throw error;
|
||||
} finally {
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete a skill for admin
|
||||
*/
|
||||
static async deleteSkillForAdmin(id: string): Promise<void> {
|
||||
const pool = getPool();
|
||||
const client = await pool.connect();
|
||||
|
||||
try {
|
||||
await client.query("BEGIN");
|
||||
|
||||
// Vérifier si la skill est utilisée
|
||||
const usageCheck = await client.query(
|
||||
`SELECT COUNT(*) as count
|
||||
FROM skill_evaluations se
|
||||
WHERE se.skill_id = $1 AND se.is_selected = true`,
|
||||
[id]
|
||||
);
|
||||
|
||||
const usageCount = parseInt(usageCheck.rows[0].count);
|
||||
if (usageCount > 0) {
|
||||
throw new Error("Impossible de supprimer une skill qui est utilisée");
|
||||
}
|
||||
|
||||
// Supprimer la skill
|
||||
await client.query("DELETE FROM skills WHERE id = $1", [id]);
|
||||
|
||||
await client.query("COMMIT");
|
||||
} catch (error) {
|
||||
await client.query("ROLLBACK");
|
||||
throw error;
|
||||
} finally {
|
||||
client.release();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user