SOGOMS v1.0.3 - Admin UI, Cron, Config reload

Phase 13 : sogoms-cron
- Jobs planifiés avec schedule cron standard
- Types: query_email, http, service
- Actions: list, trigger, status

Phase 16 : Réorganisation config/apps/{app}/
- Tous les fichiers d'une app dans un seul dossier
- Migration prokov vers nouvelle structure

Phase 17 : sogoms-admin
- Interface web d'administration (Go templates + htmx)
- Auth sessions cookies signées HMAC-SHA256
- Rôles super_admin / app_admin avec permissions

Phase 19 : Création d'app via Admin UI
- Formulaire création app avec config DB/auth
- Bouton "Scanner la base" : introspection + schema.yaml
- Rechargement automatique sogoway via SIGHUP

Infrastructure :
- sogoctl : socket de contrôle /run/sogoctl.sock
- sogoway : reload config sur SIGHUP sans restart

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2025-12-19 20:30:56 +01:00
parent a4694a10d1
commit 65da4efdad
76 changed files with 5305 additions and 80 deletions

View File

@@ -0,0 +1,95 @@
# Scénario: Créer un projet
name: projects_create
version: "1.0"
description: Crée un nouveau projet
input:
required:
- name
optional:
- description
- parent_id
- position
- tags
defaults:
position: 0
validation:
name:
type: string
min_length: 1
max_length: 100
description:
type: string
max_length: 65535
parent_id:
type: int
position:
type: int
steps:
- id: check_parent
service: db
action: query_one
condition: "{{input.parent_id != null}}"
params:
query: "SELECT id FROM projects WHERE id = ? AND user_id = ?"
args: ["{{input.parent_id}}", "{{auth.user_id}}"]
on_error: abort
error_message: "Projet parent non trouvé"
error_status: 422
- id: insert_project
service: db
action: insert
params:
table: projects
data:
user_id: "{{auth.user_id}}"
parent_id: "{{input.parent_id}}"
name: "{{input.name}}"
description: "{{input.description}}"
position: "{{input.position}}"
- id: sync_tags
service: db
action: exec
condition: "{{input.tags != null && input.tags | length > 0}}"
foreach: "{{input.tags}}"
foreach_as: tag_id
params:
query: |
INSERT INTO project_tags (project_id, tag_id)
SELECT ?, id FROM tags WHERE id = ? AND user_id = ?
args: ["{{steps.insert_project.insert_id}}", "{{tag_id}}", "{{auth.user_id}}"]
- id: get_project
service: db
action: query_one
params:
query: "SELECT * FROM projects WHERE id = ?"
args: ["{{steps.insert_project.insert_id}}"]
- id: get_tags
service: db
action: query
params:
query: |
SELECT t.id, t.name, t.color
FROM tags t
JOIN project_tags pt ON t.id = pt.tag_id
WHERE pt.project_id = ?
args: ["{{steps.insert_project.insert_id}}"]
output:
status: 201
body:
success: true
message: "Projet créé"
data:
id: "{{steps.get_project.result.id}}"
name: "{{steps.get_project.result.name}}"
description: "{{steps.get_project.result.description}}"
parent_id: "{{steps.get_project.result.parent_id}}"
position: "{{steps.get_project.result.position}}"
created_at: "{{steps.get_project.result.created_at}}"
tags: "{{steps.get_tags.result}}"

View File

@@ -0,0 +1,36 @@
# Scénario: Supprimer un projet
name: projects_delete
version: "1.0"
description: Supprime un projet (cascade sur sous-projets et tâches)
input:
required:
- id
validation:
id:
type: int
steps:
- id: check_project
service: db
action: query_one
params:
query: "SELECT id FROM projects WHERE id = ? AND user_id = ?"
args: ["{{input.id}}", "{{auth.user_id}}"]
on_error: abort
error_message: "Projet non trouvé"
error_status: 404
- id: delete_project
service: db
action: delete
params:
table: projects
where:
id: "{{input.id}}"
output:
status: 200
body:
success: true
message: "Projet supprimé"

View File

@@ -0,0 +1,27 @@
# Scénario: Liste des projets (arborescence)
name: projects_list
version: "1.0"
description: Retourne tous les projets de l'utilisateur en arborescence
steps:
- id: get_projects
service: db
action: query
params:
query: |
SELECT p.*,
GROUP_CONCAT(t.id) as tag_ids,
GROUP_CONCAT(t.name) as tag_names
FROM projects p
LEFT JOIN project_tags pt ON p.id = pt.project_id
LEFT JOIN tags t ON pt.tag_id = t.id
WHERE p.user_id = ?
GROUP BY p.id
ORDER BY p.parent_id ASC, p.position ASC, p.name ASC
args: ["{{auth.user_id}}"]
output:
status: 200
body:
success: true
data: "{{steps.get_projects.result | tree}}"

View File

@@ -0,0 +1,57 @@
# Scénario: Détail d'un projet
name: projects_show
version: "1.0"
description: Retourne un projet avec ses tags et sous-projets
input:
required:
- id
validation:
id:
type: int
steps:
- id: get_project
service: db
action: query_one
params:
query: "SELECT * FROM projects WHERE id = ? AND user_id = ?"
args: ["{{input.id}}", "{{auth.user_id}}"]
on_error: abort
error_message: "Projet non trouvé"
error_status: 404
- id: get_tags
service: db
action: query
params:
query: |
SELECT t.id, t.name, t.color
FROM tags t
JOIN project_tags pt ON t.id = pt.tag_id
WHERE pt.project_id = ?
args: ["{{input.id}}"]
- id: get_children
service: db
action: query
params:
query: |
SELECT * FROM projects
WHERE parent_id = ? AND user_id = ?
ORDER BY position ASC, name ASC
args: ["{{input.id}}", "{{auth.user_id}}"]
output:
status: 200
body:
success: true
data:
id: "{{steps.get_project.result.id}}"
name: "{{steps.get_project.result.name}}"
description: "{{steps.get_project.result.description}}"
parent_id: "{{steps.get_project.result.parent_id}}"
position: "{{steps.get_project.result.position}}"
created_at: "{{steps.get_project.result.created_at}}"
tags: "{{steps.get_tags.result}}"
children: "{{steps.get_children.result}}"

View File

@@ -0,0 +1,115 @@
# Scénario: Modifier un projet
name: projects_update
version: "1.0"
description: Met à jour un projet existant
input:
required:
- id
optional:
- name
- description
- parent_id
- position
- tags
validation:
id:
type: int
name:
type: string
min_length: 1
max_length: 100
description:
type: string
max_length: 65535
parent_id:
type: int
position:
type: int
steps:
- id: get_project
service: db
action: query_one
params:
query: "SELECT * FROM projects WHERE id = ? AND user_id = ?"
args: ["{{input.id}}", "{{auth.user_id}}"]
on_error: abort
error_message: "Projet non trouvé"
error_status: 404
- id: check_parent
service: db
action: query_one
condition: "{{input.parent_id != null && input.parent_id != input.id}}"
params:
query: "SELECT id FROM projects WHERE id = ? AND user_id = ?"
args: ["{{input.parent_id}}", "{{auth.user_id}}"]
on_error: abort
error_message: "Projet parent invalide"
error_status: 422
- id: update_project
service: db
action: update
params:
table: projects
where:
id: "{{input.id}}"
data:
name: "{{input.name ?? steps.get_project.result.name}}"
description: "{{input.description ?? steps.get_project.result.description}}"
parent_id: "{{input.parent_id ?? steps.get_project.result.parent_id}}"
position: "{{input.position ?? steps.get_project.result.position}}"
- id: clear_tags
service: db
action: delete
condition: "{{input.tags != null}}"
params:
table: project_tags
where:
project_id: "{{input.id}}"
- id: sync_tags
service: db
action: exec
condition: "{{input.tags != null && input.tags | length > 0}}"
foreach: "{{input.tags}}"
foreach_as: tag_id
params:
query: |
INSERT INTO project_tags (project_id, tag_id)
SELECT ?, id FROM tags WHERE id = ? AND user_id = ?
args: ["{{input.id}}", "{{tag_id}}", "{{auth.user_id}}"]
- id: get_updated
service: db
action: query_one
params:
query: "SELECT * FROM projects WHERE id = ?"
args: ["{{input.id}}"]
- id: get_tags
service: db
action: query
params:
query: |
SELECT t.id, t.name, t.color
FROM tags t
JOIN project_tags pt ON t.id = pt.tag_id
WHERE pt.project_id = ?
args: ["{{input.id}}"]
output:
status: 200
body:
success: true
message: "Projet mis à jour"
data:
id: "{{steps.get_updated.result.id}}"
name: "{{steps.get_updated.result.name}}"
description: "{{steps.get_updated.result.description}}"
parent_id: "{{steps.get_updated.result.parent_id}}"
position: "{{steps.get_updated.result.position}}"
tags: "{{steps.get_tags.result}}"