@lyrra/mcp-server 1.1.3 → 1.1.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +59 -242
- package/dist/auth-session.js +160 -0
- package/dist/eduflow-block-docs.js +438 -0
- package/dist/index.js +179 -12
- package/dist/lyrra-http.js +80 -0
- package/dist/openapi-parse.js +61 -0
- package/dist/register-eduflow-block-tools.js +31 -0
- package/package.json +36 -13
- package/Dockerfile +0 -16
- package/dist/client.d.ts +0 -23
- package/dist/client.d.ts.map +0 -1
- package/dist/client.js +0 -92
- package/dist/client.js.map +0 -1
- package/dist/config.d.ts +0 -8
- package/dist/config.d.ts.map +0 -1
- package/dist/config.js +0 -8
- package/dist/config.js.map +0 -1
- package/dist/http-server.d.ts +0 -8
- package/dist/http-server.d.ts.map +0 -1
- package/dist/http-server.js +0 -481
- package/dist/http-server.js.map +0 -1
- package/dist/index.d.ts +0 -3
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/resources/block-types.d.ts +0 -318
- package/dist/resources/block-types.d.ts.map +0 -1
- package/dist/resources/block-types.js +0 -297
- package/dist/resources/block-types.js.map +0 -1
- package/dist/resources/flow-schema.d.ts +0 -147
- package/dist/resources/flow-schema.d.ts.map +0 -1
- package/dist/resources/flow-schema.js +0 -143
- package/dist/resources/flow-schema.js.map +0 -1
- package/dist/server-factory.d.ts +0 -8
- package/dist/server-factory.d.ts.map +0 -1
- package/dist/server-factory.js +0 -82
- package/dist/server-factory.js.map +0 -1
- package/dist/tools/admin.d.ts +0 -265
- package/dist/tools/admin.d.ts.map +0 -1
- package/dist/tools/admin.js +0 -118
- package/dist/tools/admin.js.map +0 -1
- package/dist/tools/ai-designer.d.ts +0 -297
- package/dist/tools/ai-designer.d.ts.map +0 -1
- package/dist/tools/ai-designer.js +0 -89
- package/dist/tools/ai-designer.js.map +0 -1
- package/dist/tools/analytics.d.ts +0 -95
- package/dist/tools/analytics.d.ts.map +0 -1
- package/dist/tools/analytics.js +0 -44
- package/dist/tools/analytics.js.map +0 -1
- package/dist/tools/auth.d.ts +0 -61
- package/dist/tools/auth.d.ts.map +0 -1
- package/dist/tools/auth.js +0 -36
- package/dist/tools/auth.js.map +0 -1
- package/dist/tools/blocks.d.ts +0 -457
- package/dist/tools/blocks.d.ts.map +0 -1
- package/dist/tools/blocks.js +0 -173
- package/dist/tools/blocks.js.map +0 -1
- package/dist/tools/connections.d.ts +0 -173
- package/dist/tools/connections.d.ts.map +0 -1
- package/dist/tools/connections.js +0 -81
- package/dist/tools/connections.js.map +0 -1
- package/dist/tools/eduflow.d.ts +0 -409
- package/dist/tools/eduflow.d.ts.map +0 -1
- package/dist/tools/eduflow.js +0 -139
- package/dist/tools/eduflow.js.map +0 -1
- package/dist/tools/participants.d.ts +0 -221
- package/dist/tools/participants.d.ts.map +0 -1
- package/dist/tools/participants.js +0 -70
- package/dist/tools/participants.js.map +0 -1
- package/dist/tools/presentation.d.ts +0 -233
- package/dist/tools/presentation.d.ts.map +0 -1
- package/dist/tools/presentation.js +0 -57
- package/dist/tools/presentation.js.map +0 -1
- package/dist/tools/projects.d.ts +0 -131
- package/dist/tools/projects.d.ts.map +0 -1
- package/dist/tools/projects.js +0 -55
- package/dist/tools/projects.js.map +0 -1
- package/dist/tools/resources.d.ts +0 -93
- package/dist/tools/resources.d.ts.map +0 -1
- package/dist/tools/resources.js +0 -37
- package/dist/tools/resources.js.map +0 -1
- package/dist/tools/store.d.ts +0 -125
- package/dist/tools/store.d.ts.map +0 -1
- package/dist/tools/store.js +0 -66
- package/dist/tools/store.js.map +0 -1
- package/mcp-config.example.json +0 -14
- package/src/client.ts +0 -106
- package/src/config.ts +0 -7
- package/src/http-server.ts +0 -591
- package/src/index.ts +0 -23
- package/src/resources/block-types.ts +0 -298
- package/src/resources/flow-schema.ts +0 -148
- package/src/server-factory.ts +0 -109
- package/src/tools/admin.ts +0 -128
- package/src/tools/ai-designer.ts +0 -97
- package/src/tools/analytics.ts +0 -49
- package/src/tools/auth.ts +0 -39
- package/src/tools/blocks.ts +0 -186
- package/src/tools/connections.ts +0 -83
- package/src/tools/eduflow.ts +0 -150
- package/src/tools/participants.ts +0 -77
- package/src/tools/presentation.ts +0 -61
- package/src/tools/projects.ts +0 -61
- package/src/tools/resources.ts +0 -41
- package/src/tools/store.ts +0 -67
- package/tsconfig.json +0 -19
|
@@ -1,298 +0,0 @@
|
|
|
1
|
-
export const BLOCK_TYPES_RESOURCE = {
|
|
2
|
-
uri: 'lyrra://block-types',
|
|
3
|
-
name: 'Documentation des types de blocs EduFlow',
|
|
4
|
-
description: 'Documentation complète de tous les types de blocs disponibles dans LYRRA Studio EduFlow, avec la structure de contenu attendue, des exemples et les règles de connexion.',
|
|
5
|
-
mimeType: 'application/json',
|
|
6
|
-
content: {
|
|
7
|
-
overview: `LYRRA Studio EduFlow supporte 29 types de blocs organisés en 7 catégories.
|
|
8
|
-
Chaque parcours est un graphe orienté : les blocs sont des nœuds, reliés par des edges (connexions).
|
|
9
|
-
Un parcours DOIT avoir exactement 1 bloc "start" et au moins 1 bloc "end".`,
|
|
10
|
-
|
|
11
|
-
categories: {
|
|
12
|
-
content: {
|
|
13
|
-
name: 'Contenu',
|
|
14
|
-
types: ['text', 'audio', 'video', 'pdf', 'image', 'presentation', 'browser'],
|
|
15
|
-
description: 'Blocs pour présenter du contenu aux apprenants.',
|
|
16
|
-
},
|
|
17
|
-
assessment: {
|
|
18
|
-
name: 'Évaluation',
|
|
19
|
-
types: ['quiz', 'evaluation', 'voice_assessment', 'self_assessment'],
|
|
20
|
-
description: 'Blocs pour évaluer les connaissances des apprenants.',
|
|
21
|
-
},
|
|
22
|
-
interactive: {
|
|
23
|
-
name: 'Interactif',
|
|
24
|
-
types: ['form', 'mindmap', 'timeline', 'chart', 'glossary'],
|
|
25
|
-
description: 'Blocs interactifs pour engager les apprenants.',
|
|
26
|
-
},
|
|
27
|
-
flow_control: {
|
|
28
|
-
name: 'Contrôle de flux',
|
|
29
|
-
types: ['start', 'end', 'split', 'merge', 'loop', 'timer', 'subflow'],
|
|
30
|
-
description: 'Blocs pour structurer le parcours (branchement, boucles, etc.).',
|
|
31
|
-
},
|
|
32
|
-
accessibility: {
|
|
33
|
-
name: 'Accessibilité (DYS)',
|
|
34
|
-
types: ['dys_reader', 'dys_image_zones', 'dys_reading_practice', 'dys_clock'],
|
|
35
|
-
description: 'Blocs adaptés pour les apprenants dyslexiques/dyscalculiques.',
|
|
36
|
-
},
|
|
37
|
-
output: {
|
|
38
|
-
name: 'Sortie',
|
|
39
|
-
types: ['certification', 'email'],
|
|
40
|
-
description: 'Blocs pour générer des documents ou envoyer des notifications.',
|
|
41
|
-
},
|
|
42
|
-
},
|
|
43
|
-
|
|
44
|
-
blocks: {
|
|
45
|
-
start: {
|
|
46
|
-
name: 'Début',
|
|
47
|
-
description: 'Point de départ du parcours. Chaque flux doit avoir exactement UN bloc Start.',
|
|
48
|
-
content: { message: 'string (optionnel) - Message d\'accueil' },
|
|
49
|
-
rules: ['Un seul bloc start par parcours', 'Pas de connexion entrante', 'Au moins une connexion sortante'],
|
|
50
|
-
example: { type: 'start', title: 'Début du parcours', content: { message: 'Bienvenue !' } },
|
|
51
|
-
},
|
|
52
|
-
end: {
|
|
53
|
-
name: 'Fin',
|
|
54
|
-
description: 'Point de fin du parcours. Peut avoir plusieurs blocs End.',
|
|
55
|
-
content: {
|
|
56
|
-
message: 'string (optionnel) - Message de fin',
|
|
57
|
-
certificate: 'boolean (optionnel) - Génère un certificat',
|
|
58
|
-
redirectUrl: 'string (optionnel) - URL de redirection',
|
|
59
|
-
},
|
|
60
|
-
rules: ['Peut avoir plusieurs blocs end', 'Pas de connexion sortante', 'Au moins une connexion entrante'],
|
|
61
|
-
example: { type: 'end', title: 'Fin', content: { message: 'Félicitations !', certificate: true } },
|
|
62
|
-
},
|
|
63
|
-
text: {
|
|
64
|
-
name: 'Texte',
|
|
65
|
-
description: 'Affiche du contenu HTML enrichi. TOUJOURS utiliser format="html".',
|
|
66
|
-
content: {
|
|
67
|
-
text: 'string (requis) - Contenu HTML (<h1>, <p>, <ul>, <strong>, etc.)',
|
|
68
|
-
format: 'string (requis) - TOUJOURS "html"',
|
|
69
|
-
},
|
|
70
|
-
rules: ['TOUJOURS format="html"', 'JAMAIS de Markdown', 'Utiliser des balises HTML standard'],
|
|
71
|
-
example: {
|
|
72
|
-
type: 'text', title: 'Introduction',
|
|
73
|
-
content: { text: '<h2>Bienvenue</h2><p>Ce module couvre...</p>', format: 'html' },
|
|
74
|
-
},
|
|
75
|
-
},
|
|
76
|
-
audio: {
|
|
77
|
-
name: 'Audio',
|
|
78
|
-
description: 'Intègre un projet audio LYRRA avec texte synchronisé.',
|
|
79
|
-
content: {
|
|
80
|
-
projectId: 'string (requis) - ID du projet LYRRA',
|
|
81
|
-
showTranscript: 'boolean (optionnel, défaut: true)',
|
|
82
|
-
showChapters: 'boolean (optionnel, défaut: true)',
|
|
83
|
-
},
|
|
84
|
-
},
|
|
85
|
-
video: {
|
|
86
|
-
name: 'Vidéo',
|
|
87
|
-
description: 'Intègre une vidéo YouTube, Vimeo ou uploadée.',
|
|
88
|
-
content: {
|
|
89
|
-
videoUrl: 'string (requis) - URL de la vidéo',
|
|
90
|
-
autoplay: 'boolean (optionnel)',
|
|
91
|
-
showControls: 'boolean (optionnel, défaut: true)',
|
|
92
|
-
},
|
|
93
|
-
},
|
|
94
|
-
pdf: {
|
|
95
|
-
name: 'PDF',
|
|
96
|
-
description: 'Affiche un document PDF intégré.',
|
|
97
|
-
content: { pdfUrl: 'string (requis) - URL du fichier PDF' },
|
|
98
|
-
},
|
|
99
|
-
image: {
|
|
100
|
-
name: 'Image',
|
|
101
|
-
description: 'Affiche une image avec légende optionnelle.',
|
|
102
|
-
content: {
|
|
103
|
-
imageUrl: 'string (requis) - URL de l\'image',
|
|
104
|
-
caption: 'string (optionnel) - Légende',
|
|
105
|
-
alt: 'string (optionnel) - Texte alternatif',
|
|
106
|
-
},
|
|
107
|
-
},
|
|
108
|
-
quiz: {
|
|
109
|
-
name: 'Quiz',
|
|
110
|
-
description: 'Question unique avec branchement conditionnel possible.',
|
|
111
|
-
content: {
|
|
112
|
-
quizType: '"multiple_choice" | "true_false" | "ordering"',
|
|
113
|
-
question: 'string (requis) - La question',
|
|
114
|
-
answers: 'array - [{text, isCorrect, feedback?}]',
|
|
115
|
-
answerMode: '"text" | "image"',
|
|
116
|
-
branchingMode: '"binary" | "per-answer" (optionnel)',
|
|
117
|
-
explanation: 'string (optionnel) - Explication après réponse',
|
|
118
|
-
shuffleAnswers: 'boolean (optionnel)',
|
|
119
|
-
maxAttempts: 'number (optionnel)',
|
|
120
|
-
},
|
|
121
|
-
rules: [
|
|
122
|
-
'branchingMode="binary" : 2 sorties (correct/incorrect)',
|
|
123
|
-
'branchingMode="per-answer" : une sortie par réponse',
|
|
124
|
-
'Sans branchement : une seule sortie',
|
|
125
|
-
],
|
|
126
|
-
},
|
|
127
|
-
evaluation: {
|
|
128
|
-
name: 'Évaluation',
|
|
129
|
-
description: 'Évaluation complète avec plusieurs questions et scoring.',
|
|
130
|
-
content: {
|
|
131
|
-
questions: 'array - [{type, question, answers?, correctAnswer?, points}]',
|
|
132
|
-
passingScore: 'number (optionnel) - Score minimum pour réussir (%)',
|
|
133
|
-
showResults: 'boolean (optionnel) - Montrer les résultats',
|
|
134
|
-
maxAttempts: 'number (optionnel)',
|
|
135
|
-
timeLimit: 'number (optionnel) - Temps en secondes',
|
|
136
|
-
},
|
|
137
|
-
},
|
|
138
|
-
form: {
|
|
139
|
-
name: 'Formulaire',
|
|
140
|
-
description: 'Formulaire avec champs personnalisés et logique conditionnelle.',
|
|
141
|
-
content: {
|
|
142
|
-
fields: 'array - [{type, label, required?, options?, placeholder?}]',
|
|
143
|
-
submitLabel: 'string (optionnel)',
|
|
144
|
-
rules: 'array (optionnel) - Logique IF/ELSE pour le routage',
|
|
145
|
-
},
|
|
146
|
-
},
|
|
147
|
-
chart: {
|
|
148
|
-
name: 'Graphique',
|
|
149
|
-
description: 'Graphique interactif.',
|
|
150
|
-
content: {
|
|
151
|
-
chartType: '"bar" | "line" | "pie" | "scatter" | "radar" | "area" | "doughnut"',
|
|
152
|
-
data: 'object - {labels[], datasets[{label, data[]}]}',
|
|
153
|
-
title: 'string (optionnel)',
|
|
154
|
-
},
|
|
155
|
-
},
|
|
156
|
-
timeline: {
|
|
157
|
-
name: 'Frise chronologique',
|
|
158
|
-
description: 'Frise avec événements datés.',
|
|
159
|
-
content: {
|
|
160
|
-
events: 'array - [{date, title, description, imageUrl?}]',
|
|
161
|
-
orientation: '"horizontal" | "vertical" (optionnel)',
|
|
162
|
-
},
|
|
163
|
-
},
|
|
164
|
-
mindmap: {
|
|
165
|
-
name: 'Carte mentale',
|
|
166
|
-
description: 'Carte mentale interactive.',
|
|
167
|
-
content: {
|
|
168
|
-
nodes: 'array - [{id, label, parentId?, color?}]',
|
|
169
|
-
centerLabel: 'string - Label du nœud central',
|
|
170
|
-
},
|
|
171
|
-
},
|
|
172
|
-
glossary: {
|
|
173
|
-
name: 'Glossaire',
|
|
174
|
-
description: 'Liste de termes et définitions.',
|
|
175
|
-
content: {
|
|
176
|
-
terms: 'array - [{term, definition, examples?}]',
|
|
177
|
-
sortAlphabetically: 'boolean (optionnel, défaut: true)',
|
|
178
|
-
},
|
|
179
|
-
},
|
|
180
|
-
split: {
|
|
181
|
-
name: 'Embranchement',
|
|
182
|
-
description: 'Sépare le flux en branches parallèles.',
|
|
183
|
-
content: { branches: 'number (optionnel) - Nombre de branches (défaut: 2)' },
|
|
184
|
-
rules: ['Au moins 2 connexions sortantes', 'Généralement suivi d\'un bloc merge'],
|
|
185
|
-
},
|
|
186
|
-
merge: {
|
|
187
|
-
name: 'Fusion',
|
|
188
|
-
description: 'Rejoint les branches parallèles.',
|
|
189
|
-
content: {},
|
|
190
|
-
rules: ['Au moins 2 connexions entrantes', 'Utilisé après un bloc split'],
|
|
191
|
-
},
|
|
192
|
-
loop: {
|
|
193
|
-
name: 'Boucle',
|
|
194
|
-
description: 'Permet de répéter une séquence de blocs.',
|
|
195
|
-
content: {
|
|
196
|
-
maxIterations: 'number (optionnel) - Nombre max de répétitions',
|
|
197
|
-
condition: 'string (optionnel) - Condition de sortie',
|
|
198
|
-
},
|
|
199
|
-
},
|
|
200
|
-
timer: {
|
|
201
|
-
name: 'Minuteur',
|
|
202
|
-
description: 'Bloc avec contrainte de temps.',
|
|
203
|
-
content: {
|
|
204
|
-
duration: 'number (requis) - Durée en secondes',
|
|
205
|
-
action: '"pause" | "skip" | "redirect" - Action à l\'expiration',
|
|
206
|
-
warningAt: 'number (optionnel) - Avertissement X secondes avant la fin',
|
|
207
|
-
},
|
|
208
|
-
},
|
|
209
|
-
subflow: {
|
|
210
|
-
name: 'Sous-parcours',
|
|
211
|
-
description: 'Intègre un autre parcours EduFlow.',
|
|
212
|
-
content: { subflowId: 'string (requis) - ID du parcours à intégrer' },
|
|
213
|
-
},
|
|
214
|
-
certification: {
|
|
215
|
-
name: 'Certification',
|
|
216
|
-
description: 'Génère un certificat PDF personnalisé.',
|
|
217
|
-
content: {
|
|
218
|
-
templateId: 'string (optionnel) - ID du template Word',
|
|
219
|
-
variables: 'object (optionnel) - Variables de substitution',
|
|
220
|
-
},
|
|
221
|
-
},
|
|
222
|
-
email: {
|
|
223
|
-
name: 'Email',
|
|
224
|
-
description: 'Envoie un email automatique.',
|
|
225
|
-
content: {
|
|
226
|
-
subject: 'string (requis) - Objet de l\'email',
|
|
227
|
-
body: 'string (requis) - Corps HTML de l\'email',
|
|
228
|
-
to: '"participant" | "creator" | string - Destinataire',
|
|
229
|
-
},
|
|
230
|
-
},
|
|
231
|
-
voice_assessment: {
|
|
232
|
-
name: 'Évaluation vocale',
|
|
233
|
-
description: 'L\'étudiant enregistre sa voix pour évaluation.',
|
|
234
|
-
content: {
|
|
235
|
-
prompt: 'string (requis) - Consigne de lecture/parole',
|
|
236
|
-
referenceText: 'string (optionnel) - Texte de référence pour comparaison',
|
|
237
|
-
maxDuration: 'number (optionnel) - Durée max en secondes',
|
|
238
|
-
},
|
|
239
|
-
},
|
|
240
|
-
self_assessment: {
|
|
241
|
-
name: 'Auto-évaluation',
|
|
242
|
-
description: 'L\'étudiant s\'évalue sur des critères définis.',
|
|
243
|
-
content: {
|
|
244
|
-
criteria: 'array - [{label, scale}]',
|
|
245
|
-
instructions: 'string (optionnel)',
|
|
246
|
-
},
|
|
247
|
-
},
|
|
248
|
-
presentation: {
|
|
249
|
-
name: 'Présentation',
|
|
250
|
-
description: 'Diaporama avec slides.',
|
|
251
|
-
content: {
|
|
252
|
-
slides: 'array - [{title, content, imageUrl?}]',
|
|
253
|
-
autoAdvance: 'boolean (optionnel)',
|
|
254
|
-
advanceInterval: 'number (optionnel) - Secondes entre slides',
|
|
255
|
-
},
|
|
256
|
-
},
|
|
257
|
-
browser: {
|
|
258
|
-
name: 'Navigateur',
|
|
259
|
-
description: 'Intègre une page web dans un iframe.',
|
|
260
|
-
content: { url: 'string (requis) - URL de la page à intégrer' },
|
|
261
|
-
},
|
|
262
|
-
dys_reader: {
|
|
263
|
-
name: 'Lecteur DYS',
|
|
264
|
-
description: 'Texte adapté pour la dyslexie.',
|
|
265
|
-
content: {
|
|
266
|
-
text: 'string (requis) - Texte à afficher',
|
|
267
|
-
fontSize: 'number (optionnel) - Taille de police (défaut: 18)',
|
|
268
|
-
lineSpacing: 'number (optionnel) - Interligne (défaut: 2)',
|
|
269
|
-
syllableHighlight: 'boolean (optionnel) - Colorer les syllabes',
|
|
270
|
-
},
|
|
271
|
-
},
|
|
272
|
-
dys_image_zones: {
|
|
273
|
-
name: 'Zones d\'image DYS',
|
|
274
|
-
description: 'Image avec zones cliquables.',
|
|
275
|
-
content: {
|
|
276
|
-
imageUrl: 'string (requis)',
|
|
277
|
-
zones: 'array - [{x, y, width, height, label, description}]',
|
|
278
|
-
},
|
|
279
|
-
},
|
|
280
|
-
dys_reading_practice: {
|
|
281
|
-
name: 'Pratique lecture DYS',
|
|
282
|
-
description: 'Exercice de lecture adapté.',
|
|
283
|
-
content: {
|
|
284
|
-
text: 'string (requis)',
|
|
285
|
-
difficulty: '"easy" | "medium" | "hard"',
|
|
286
|
-
},
|
|
287
|
-
},
|
|
288
|
-
dys_clock: {
|
|
289
|
-
name: 'Horloge DYS',
|
|
290
|
-
description: 'Exercice de lecture de l\'heure.',
|
|
291
|
-
content: {
|
|
292
|
-
targetTime: 'string (requis) - Heure cible (format HH:MM)',
|
|
293
|
-
difficulty: '"easy" | "medium" | "hard"',
|
|
294
|
-
},
|
|
295
|
-
},
|
|
296
|
-
},
|
|
297
|
-
},
|
|
298
|
-
};
|
|
@@ -1,148 +0,0 @@
|
|
|
1
|
-
export const FLOW_SCHEMA_RESOURCE = {
|
|
2
|
-
uri: 'lyrra://flow-construction-guide',
|
|
3
|
-
name: 'Guide de construction d\'un parcours EduFlow',
|
|
4
|
-
description: 'Guide complet pour construire un parcours pédagogique dans LYRRA Studio : logique de construction, règles de connexion, bonnes pratiques.',
|
|
5
|
-
mimeType: 'application/json',
|
|
6
|
-
content: {
|
|
7
|
-
overview: `Un parcours EduFlow est un graphe orienté composé de blocs (nœuds) reliés par des edges (connexions).
|
|
8
|
-
Le parcours définit le chemin que suivra l'apprenant, avec possibilité de branchements conditionnels.`,
|
|
9
|
-
|
|
10
|
-
construction_steps: [
|
|
11
|
-
{
|
|
12
|
-
step: 1,
|
|
13
|
-
title: 'Créer le parcours',
|
|
14
|
-
description: 'Utiliser eduflow_create avec un titre et une description. Un bloc "start" et un bloc "end" sont automatiquement créés.',
|
|
15
|
-
tool: 'eduflow_create',
|
|
16
|
-
},
|
|
17
|
-
{
|
|
18
|
-
step: 2,
|
|
19
|
-
title: 'Ajouter des blocs de contenu',
|
|
20
|
-
description: 'Créer des blocs entre start et end : text, video, audio, quiz, etc. Utiliser block_create pour chaque bloc.',
|
|
21
|
-
tool: 'block_create',
|
|
22
|
-
},
|
|
23
|
-
{
|
|
24
|
-
step: 3,
|
|
25
|
-
title: 'Connecter les blocs',
|
|
26
|
-
description: 'Relier les blocs avec connection_add. Chaque connexion va de source vers target.',
|
|
27
|
-
tool: 'connection_add',
|
|
28
|
-
},
|
|
29
|
-
{
|
|
30
|
-
step: 4,
|
|
31
|
-
title: 'Configurer les objectifs',
|
|
32
|
-
description: 'Définir les objectifs pédagogiques avec ai_generate_objectives ou ai_save_objectives.',
|
|
33
|
-
tool: 'ai_generate_objectives',
|
|
34
|
-
},
|
|
35
|
-
{
|
|
36
|
-
step: 5,
|
|
37
|
-
title: 'Créer la page de présentation',
|
|
38
|
-
description: 'Configurer la page publique avec presentation_update et presentation_toggle.',
|
|
39
|
-
tool: 'presentation_update',
|
|
40
|
-
},
|
|
41
|
-
{
|
|
42
|
-
step: 6,
|
|
43
|
-
title: 'Publier',
|
|
44
|
-
description: 'Changer le statut en "published" avec eduflow_change_status pour rendre le parcours accessible.',
|
|
45
|
-
tool: 'eduflow_change_status',
|
|
46
|
-
},
|
|
47
|
-
],
|
|
48
|
-
|
|
49
|
-
connection_rules: {
|
|
50
|
-
general: [
|
|
51
|
-
'Le graphe doit être connexe : chaque bloc doit être atteignable depuis le bloc start',
|
|
52
|
-
'Pas de boucles infinies sans condition de sortie',
|
|
53
|
-
'Chaque chemin doit mener à au moins un bloc end',
|
|
54
|
-
],
|
|
55
|
-
by_block_type: {
|
|
56
|
-
start: {
|
|
57
|
-
incoming: 0,
|
|
58
|
-
outgoing: '1+',
|
|
59
|
-
notes: 'Exactement 1 par parcours. Point d\'entrée unique.',
|
|
60
|
-
},
|
|
61
|
-
end: {
|
|
62
|
-
incoming: '1+',
|
|
63
|
-
outgoing: 0,
|
|
64
|
-
notes: 'Peut avoir plusieurs blocs end pour différentes fins.',
|
|
65
|
-
},
|
|
66
|
-
text: { incoming: '1+', outgoing: '1', notes: 'Flux linéaire.' },
|
|
67
|
-
audio: { incoming: '1+', outgoing: '1', notes: 'Flux linéaire.' },
|
|
68
|
-
video: { incoming: '1+', outgoing: '1', notes: 'Flux linéaire.' },
|
|
69
|
-
pdf: { incoming: '1+', outgoing: '1', notes: 'Flux linéaire.' },
|
|
70
|
-
image: { incoming: '1+', outgoing: '1', notes: 'Flux linéaire.' },
|
|
71
|
-
quiz: {
|
|
72
|
-
incoming: '1+',
|
|
73
|
-
outgoing: '1-N',
|
|
74
|
-
notes: 'Sans branchement: 1 sortie. binary: 2 sorties (correct/incorrect). per-answer: N sorties.',
|
|
75
|
-
},
|
|
76
|
-
evaluation: { incoming: '1+', outgoing: '1-2', notes: '1 sortie (toujours) ou 2 (réussite/échec).' },
|
|
77
|
-
form: { incoming: '1+', outgoing: '1-N', notes: 'Peut avoir des règles conditionnelles.' },
|
|
78
|
-
split: { incoming: '1', outgoing: '2+', notes: 'Crée des branches parallèles.' },
|
|
79
|
-
merge: { incoming: '2+', outgoing: '1', notes: 'Rejoint les branches. Correspond à un split.' },
|
|
80
|
-
loop: { incoming: '1+', outgoing: '2', notes: '1 sortie "continuer" + 1 sortie "quitter la boucle".' },
|
|
81
|
-
timer: { incoming: '1+', outgoing: '1-2', notes: '1 sortie normale + 1 sortie timeout (optionnel).' },
|
|
82
|
-
subflow: { incoming: '1+', outgoing: '1', notes: 'Le sous-parcours s\'exécute entièrement puis continue.' },
|
|
83
|
-
certification: { incoming: '1+', outgoing: '1', notes: 'Génère le certificat puis continue.' },
|
|
84
|
-
email: { incoming: '1+', outgoing: '1', notes: 'Envoie l\'email puis continue.' },
|
|
85
|
-
},
|
|
86
|
-
},
|
|
87
|
-
|
|
88
|
-
positioning: {
|
|
89
|
-
description: 'Les blocs ont des coordonnées X/Y pour l\'éditeur visuel.',
|
|
90
|
-
recommendations: [
|
|
91
|
-
'Espacer horizontalement de 300px entre les blocs séquentiels',
|
|
92
|
-
'Espacer verticalement de 150px pour les branches parallèles',
|
|
93
|
-
'Le bloc start en position (100, 300)',
|
|
94
|
-
'Flux de gauche à droite',
|
|
95
|
-
],
|
|
96
|
-
},
|
|
97
|
-
|
|
98
|
-
patterns: {
|
|
99
|
-
linear: {
|
|
100
|
-
name: 'Parcours linéaire',
|
|
101
|
-
description: 'start → text → video → quiz → end',
|
|
102
|
-
usage: 'Parcours simple sans branchement.',
|
|
103
|
-
},
|
|
104
|
-
branching: {
|
|
105
|
-
name: 'Parcours avec branchement',
|
|
106
|
-
description: 'start → quiz → [correct: text_avancé → end] [incorrect: text_revision → quiz]',
|
|
107
|
-
usage: 'Adaptation au niveau de l\'apprenant.',
|
|
108
|
-
},
|
|
109
|
-
parallel: {
|
|
110
|
-
name: 'Parcours parallèle',
|
|
111
|
-
description: 'start → split → [branche1: text+video] [branche2: audio+pdf] → merge → end',
|
|
112
|
-
usage: 'L\'apprenant choisit son parcours.',
|
|
113
|
-
},
|
|
114
|
-
loop_pattern: {
|
|
115
|
-
name: 'Boucle d\'apprentissage',
|
|
116
|
-
description: 'start → text → quiz → [échec: loop → text] [succès: end]',
|
|
117
|
-
usage: 'Répétition jusqu\'à maîtrise.',
|
|
118
|
-
},
|
|
119
|
-
certification_pattern: {
|
|
120
|
-
name: 'Parcours certifiant',
|
|
121
|
-
description: 'start → text → evaluation → [réussi: certification → email → end] [échoué: end]',
|
|
122
|
-
usage: 'Parcours avec certificat de réussite.',
|
|
123
|
-
},
|
|
124
|
-
},
|
|
125
|
-
|
|
126
|
-
gamification: {
|
|
127
|
-
description: 'Configuration optionnelle pour engager les apprenants.',
|
|
128
|
-
options: {
|
|
129
|
-
badges: 'Badges débloqués par des actions (complétion, score parfait, etc.)',
|
|
130
|
-
missions: 'Objectifs à atteindre avec récompenses',
|
|
131
|
-
leaderboard: 'Classement entre participants',
|
|
132
|
-
certifications: 'Certificats PDF personnalisés',
|
|
133
|
-
},
|
|
134
|
-
},
|
|
135
|
-
|
|
136
|
-
best_practices: [
|
|
137
|
-
'Commencer par définir les objectifs pédagogiques avant de créer les blocs',
|
|
138
|
-
'Alterner les types de blocs (texte → vidéo → quiz) pour maintenir l\'engagement',
|
|
139
|
-
'Utiliser des quiz formatifs (sans score) entre les blocs de contenu',
|
|
140
|
-
'Limiter à 15-20 blocs par parcours pour ne pas surcharger',
|
|
141
|
-
'Toujours avoir un message d\'accueil dans le bloc start',
|
|
142
|
-
'Toujours avoir un message de félicitations dans le bloc end',
|
|
143
|
-
'Utiliser les labels sur les connexions pour les branchements (ex: "Bonne réponse")',
|
|
144
|
-
'Créer une version avant chaque modification majeure',
|
|
145
|
-
'Tester le parcours avec un participant test avant de publier',
|
|
146
|
-
],
|
|
147
|
-
},
|
|
148
|
-
};
|
package/src/server-factory.ts
DELETED
|
@@ -1,109 +0,0 @@
|
|
|
1
|
-
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
|
-
import { LyrraClient } from './client.js';
|
|
3
|
-
|
|
4
|
-
// Tool factory functions
|
|
5
|
-
import { createAuthTools } from './tools/auth.js';
|
|
6
|
-
import { createEduflowTools } from './tools/eduflow.js';
|
|
7
|
-
import { createBlocksTools } from './tools/blocks.js';
|
|
8
|
-
import { createConnectionsTools } from './tools/connections.js';
|
|
9
|
-
import { createParticipantsTools } from './tools/participants.js';
|
|
10
|
-
import { createAnalyticsTools } from './tools/analytics.js';
|
|
11
|
-
import { createAiDesignerTools } from './tools/ai-designer.js';
|
|
12
|
-
import { createPresentationTools } from './tools/presentation.js';
|
|
13
|
-
import { createStoreTools } from './tools/store.js';
|
|
14
|
-
import { createProjectsTools } from './tools/projects.js';
|
|
15
|
-
import { createResourcesTools } from './tools/resources.js';
|
|
16
|
-
import { createAdminTools } from './tools/admin.js';
|
|
17
|
-
|
|
18
|
-
// Resources
|
|
19
|
-
import { BLOCK_TYPES_RESOURCE } from './resources/block-types.js';
|
|
20
|
-
import { FLOW_SCHEMA_RESOURCE } from './resources/flow-schema.js';
|
|
21
|
-
|
|
22
|
-
type ToolDef = { description: string; inputSchema: any; handler: (args: any) => Promise<any> };
|
|
23
|
-
|
|
24
|
-
function registerToolsOnServer(server: McpServer, tools: Record<string, ToolDef>) {
|
|
25
|
-
for (const [name, tool] of Object.entries(tools)) {
|
|
26
|
-
server.tool(
|
|
27
|
-
name,
|
|
28
|
-
tool.description,
|
|
29
|
-
tool.inputSchema.shape
|
|
30
|
-
? Object.fromEntries(
|
|
31
|
-
Object.entries(tool.inputSchema.shape).map(([key, schema]: [string, any]) => [key, schema])
|
|
32
|
-
)
|
|
33
|
-
: {},
|
|
34
|
-
async (args: any) => {
|
|
35
|
-
try {
|
|
36
|
-
const result = await tool.handler(args);
|
|
37
|
-
return {
|
|
38
|
-
content: [{ type: 'text' as const, text: JSON.stringify(result, null, 2) }],
|
|
39
|
-
};
|
|
40
|
-
} catch (error: any) {
|
|
41
|
-
return {
|
|
42
|
-
content: [{ type: 'text' as const, text: `Erreur: ${error.message}` }],
|
|
43
|
-
isError: true,
|
|
44
|
-
};
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
);
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* Create and configure an MCP server with all tools registered.
|
|
53
|
-
* @param client - LyrraClient to use for API calls. If not provided, uses default (env vars).
|
|
54
|
-
*/
|
|
55
|
-
export function createMcpServer(client?: LyrraClient): McpServer {
|
|
56
|
-
const c = client || new LyrraClient();
|
|
57
|
-
|
|
58
|
-
const server = new McpServer({
|
|
59
|
-
name: 'lyrra-studio',
|
|
60
|
-
version: '1.0.0',
|
|
61
|
-
description: 'Serveur MCP pour piloter LYRRA Studio - Plateforme EdTech de création de parcours pédagogiques, conversion PDF→Audio, et store d\'audiobooks.',
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
const allTools: Record<string, ToolDef> = {
|
|
65
|
-
...createAuthTools(c),
|
|
66
|
-
...createEduflowTools(c),
|
|
67
|
-
...createBlocksTools(c),
|
|
68
|
-
...createConnectionsTools(c),
|
|
69
|
-
...createParticipantsTools(c),
|
|
70
|
-
...createAnalyticsTools(c),
|
|
71
|
-
...createAiDesignerTools(c),
|
|
72
|
-
...createPresentationTools(c),
|
|
73
|
-
...createStoreTools(c),
|
|
74
|
-
...createProjectsTools(c),
|
|
75
|
-
...createResourcesTools(c),
|
|
76
|
-
...createAdminTools(c),
|
|
77
|
-
};
|
|
78
|
-
|
|
79
|
-
registerToolsOnServer(server, allTools);
|
|
80
|
-
|
|
81
|
-
// Resources
|
|
82
|
-
server.resource(
|
|
83
|
-
'block-types',
|
|
84
|
-
BLOCK_TYPES_RESOURCE.uri,
|
|
85
|
-
{ description: BLOCK_TYPES_RESOURCE.description, mimeType: BLOCK_TYPES_RESOURCE.mimeType },
|
|
86
|
-
async () => ({
|
|
87
|
-
contents: [{
|
|
88
|
-
uri: BLOCK_TYPES_RESOURCE.uri,
|
|
89
|
-
mimeType: BLOCK_TYPES_RESOURCE.mimeType,
|
|
90
|
-
text: JSON.stringify(BLOCK_TYPES_RESOURCE.content, null, 2),
|
|
91
|
-
}],
|
|
92
|
-
})
|
|
93
|
-
);
|
|
94
|
-
|
|
95
|
-
server.resource(
|
|
96
|
-
'flow-construction-guide',
|
|
97
|
-
FLOW_SCHEMA_RESOURCE.uri,
|
|
98
|
-
{ description: FLOW_SCHEMA_RESOURCE.description, mimeType: FLOW_SCHEMA_RESOURCE.mimeType },
|
|
99
|
-
async () => ({
|
|
100
|
-
contents: [{
|
|
101
|
-
uri: FLOW_SCHEMA_RESOURCE.uri,
|
|
102
|
-
mimeType: FLOW_SCHEMA_RESOURCE.mimeType,
|
|
103
|
-
text: JSON.stringify(FLOW_SCHEMA_RESOURCE.content, null, 2),
|
|
104
|
-
}],
|
|
105
|
-
})
|
|
106
|
-
);
|
|
107
|
-
|
|
108
|
-
return server;
|
|
109
|
-
}
|
package/src/tools/admin.ts
DELETED
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
import { z } from 'zod';
|
|
2
|
-
import { LyrraClient } from '../client.js';
|
|
3
|
-
import { client as defaultClient } from '../client.js';
|
|
4
|
-
|
|
5
|
-
export function createAdminTools(c: LyrraClient) {
|
|
6
|
-
return {
|
|
7
|
-
// --- Versions ---
|
|
8
|
-
version_list: {
|
|
9
|
-
description: 'Lister toutes les versions d\'un parcours EduFlow.',
|
|
10
|
-
inputSchema: z.object({
|
|
11
|
-
flowId: z.string().uuid().describe('ID du parcours'),
|
|
12
|
-
}),
|
|
13
|
-
handler: async ({ flowId }: { flowId: string }) => {
|
|
14
|
-
return c.get(`/flows/${flowId}/versions`, 'eduflow');
|
|
15
|
-
},
|
|
16
|
-
},
|
|
17
|
-
|
|
18
|
-
version_create: {
|
|
19
|
-
description: 'Créer une nouvelle version (snapshot) du parcours. Permet de revenir en arrière si besoin.',
|
|
20
|
-
inputSchema: z.object({
|
|
21
|
-
flowId: z.string().uuid().describe('ID du parcours'),
|
|
22
|
-
versionLabel: z.string().optional().describe('Label de la version (ex: "v2.0 - Ajout quiz")'),
|
|
23
|
-
}),
|
|
24
|
-
handler: async ({ flowId, versionLabel }: any) => {
|
|
25
|
-
return c.post(`/flows/${flowId}/versions`, { versionLabel }, 'eduflow');
|
|
26
|
-
},
|
|
27
|
-
},
|
|
28
|
-
|
|
29
|
-
version_activate: {
|
|
30
|
-
description: 'Activer une version spécifique comme version publique du parcours.',
|
|
31
|
-
inputSchema: z.object({
|
|
32
|
-
flowId: z.string().uuid().describe('ID du parcours'),
|
|
33
|
-
versionId: z.string().uuid().describe('ID de la version à activer'),
|
|
34
|
-
}),
|
|
35
|
-
handler: async ({ flowId, versionId }: { flowId: string; versionId: string }) => {
|
|
36
|
-
return c.post(`/flows/${flowId}/versions/${versionId}/activate`, {}, 'eduflow');
|
|
37
|
-
},
|
|
38
|
-
},
|
|
39
|
-
|
|
40
|
-
// --- Gamification ---
|
|
41
|
-
gamification_stats: {
|
|
42
|
-
description: 'Récupérer les statistiques de gamification de l\'utilisateur (niveau, streak, badges, missions).',
|
|
43
|
-
inputSchema: z.object({}),
|
|
44
|
-
handler: async () => {
|
|
45
|
-
return c.get('/gamification/stats');
|
|
46
|
-
},
|
|
47
|
-
},
|
|
48
|
-
|
|
49
|
-
gamification_objectives: {
|
|
50
|
-
description: 'Récupérer les prochains objectifs de gamification à atteindre.',
|
|
51
|
-
inputSchema: z.object({}),
|
|
52
|
-
handler: async () => {
|
|
53
|
-
return c.get('/gamification/objectives');
|
|
54
|
-
},
|
|
55
|
-
},
|
|
56
|
-
|
|
57
|
-
// --- Activity ---
|
|
58
|
-
activity_history: {
|
|
59
|
-
description: 'Récupérer l\'historique d\'activité de l\'utilisateur avec pagination.',
|
|
60
|
-
inputSchema: z.object({
|
|
61
|
-
page: z.number().optional().describe('Page (défaut: 1)'),
|
|
62
|
-
limit: z.number().optional().describe('Nombre par page (défaut: 20)'),
|
|
63
|
-
}),
|
|
64
|
-
handler: async ({ page, limit }: any) => {
|
|
65
|
-
const params = new URLSearchParams();
|
|
66
|
-
if (page) params.set('page', String(page));
|
|
67
|
-
if (limit) params.set('limit', String(limit));
|
|
68
|
-
const query = params.toString() ? `?${params}` : '';
|
|
69
|
-
return c.get(`/activity${query}`);
|
|
70
|
-
},
|
|
71
|
-
},
|
|
72
|
-
|
|
73
|
-
activity_stats: {
|
|
74
|
-
description: 'Statistiques d\'utilisation par période (jour/semaine/mois).',
|
|
75
|
-
inputSchema: z.object({
|
|
76
|
-
period: z.enum(['day', 'week', 'month']).optional().describe('Période (défaut: week)'),
|
|
77
|
-
}),
|
|
78
|
-
handler: async ({ period }: { period?: string }) => {
|
|
79
|
-
const query = period ? `?period=${period}` : '';
|
|
80
|
-
return c.get(`/activity/stats${query}`);
|
|
81
|
-
},
|
|
82
|
-
},
|
|
83
|
-
|
|
84
|
-
// --- Webhooks ---
|
|
85
|
-
webhook_list: {
|
|
86
|
-
description: 'Lister les webhooks configurés pour recevoir des événements (block.completed, flow.completed, etc.).',
|
|
87
|
-
inputSchema: z.object({}),
|
|
88
|
-
handler: async () => {
|
|
89
|
-
return c.get('/webhooks');
|
|
90
|
-
},
|
|
91
|
-
},
|
|
92
|
-
|
|
93
|
-
webhook_create: {
|
|
94
|
-
description: 'Créer un webhook pour recevoir des notifications en temps réel sur des événements.',
|
|
95
|
-
inputSchema: z.object({
|
|
96
|
-
url: z.string().url().describe('URL du webhook (doit être accessible publiquement)'),
|
|
97
|
-
events: z.array(z.string()).describe('Événements à écouter (ex: ["block.completed", "flow.completed"])'),
|
|
98
|
-
name: z.string().optional().describe('Nom du webhook'),
|
|
99
|
-
}),
|
|
100
|
-
handler: async (data: any) => {
|
|
101
|
-
return c.post('/webhooks', data);
|
|
102
|
-
},
|
|
103
|
-
},
|
|
104
|
-
|
|
105
|
-
webhook_delete: {
|
|
106
|
-
description: 'Supprimer un webhook.',
|
|
107
|
-
inputSchema: z.object({
|
|
108
|
-
webhookId: z.string().uuid().describe('ID du webhook'),
|
|
109
|
-
}),
|
|
110
|
-
handler: async ({ webhookId }: { webhookId: string }) => {
|
|
111
|
-
return c.delete(`/webhooks/${webhookId}`);
|
|
112
|
-
},
|
|
113
|
-
},
|
|
114
|
-
|
|
115
|
-
webhook_test: {
|
|
116
|
-
description: 'Tester un webhook en envoyant un événement de test.',
|
|
117
|
-
inputSchema: z.object({
|
|
118
|
-
webhookId: z.string().uuid().describe('ID du webhook à tester'),
|
|
119
|
-
}),
|
|
120
|
-
handler: async ({ webhookId }: { webhookId: string }) => {
|
|
121
|
-
return c.post(`/webhooks/${webhookId}/test`, {});
|
|
122
|
-
},
|
|
123
|
-
},
|
|
124
|
-
};
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
// Backward compatibility for stdio mode
|
|
128
|
-
export const adminTools = createAdminTools(defaultClient);
|