aios-core 4.1.0 → 4.2.0

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.
Files changed (145) hide show
  1. package/.aios-core/.session/current-session.json +14 -0
  2. package/.aios-core/core/registry/registry-schema.json +166 -166
  3. package/.aios-core/core/registry/service-registry.json +6585 -6585
  4. package/.aios-core/data/entity-registry.yaml +208 -8
  5. package/.aios-core/data/registry-update-log.jsonl +165 -0
  6. package/.aios-core/development/scripts/approval-workflow.js +642 -642
  7. package/.aios-core/development/scripts/backup-manager.js +606 -606
  8. package/.aios-core/development/scripts/branch-manager.js +389 -389
  9. package/.aios-core/development/scripts/code-quality-improver.js +1311 -1311
  10. package/.aios-core/development/scripts/commit-message-generator.js +849 -849
  11. package/.aios-core/development/scripts/conflict-resolver.js +674 -674
  12. package/.aios-core/development/scripts/dependency-analyzer.js +637 -637
  13. package/.aios-core/development/scripts/diff-generator.js +351 -351
  14. package/.aios-core/development/scripts/elicitation-engine.js +384 -384
  15. package/.aios-core/development/scripts/elicitation-session-manager.js +299 -299
  16. package/.aios-core/development/scripts/git-wrapper.js +461 -461
  17. package/.aios-core/development/scripts/manifest-preview.js +244 -244
  18. package/.aios-core/development/scripts/metrics-tracker.js +775 -775
  19. package/.aios-core/development/scripts/modification-validator.js +554 -554
  20. package/.aios-core/development/scripts/pattern-learner.js +1224 -1224
  21. package/.aios-core/development/scripts/performance-analyzer.js +757 -757
  22. package/.aios-core/development/scripts/refactoring-suggester.js +1138 -1138
  23. package/.aios-core/development/scripts/rollback-handler.js +530 -530
  24. package/.aios-core/development/scripts/security-checker.js +358 -358
  25. package/.aios-core/development/scripts/template-engine.js +239 -239
  26. package/.aios-core/development/scripts/template-validator.js +278 -278
  27. package/.aios-core/development/scripts/test-generator.js +843 -843
  28. package/.aios-core/development/scripts/transaction-manager.js +589 -589
  29. package/.aios-core/development/scripts/usage-tracker.js +673 -673
  30. package/.aios-core/development/scripts/validate-filenames.js +226 -226
  31. package/.aios-core/development/scripts/version-tracker.js +526 -526
  32. package/.aios-core/development/scripts/yaml-validator.js +396 -396
  33. package/.aios-core/development/tasks/validate-next-story.md +99 -2
  34. package/.aios-core/development/templates/service-template/README.md.hbs +158 -158
  35. package/.aios-core/development/templates/service-template/__tests__/index.test.ts.hbs +237 -237
  36. package/.aios-core/development/templates/service-template/client.ts.hbs +403 -403
  37. package/.aios-core/development/templates/service-template/errors.ts.hbs +182 -182
  38. package/.aios-core/development/templates/service-template/index.ts.hbs +120 -120
  39. package/.aios-core/development/templates/service-template/package.json.hbs +87 -87
  40. package/.aios-core/development/templates/service-template/types.ts.hbs +145 -145
  41. package/.aios-core/development/templates/squad-template/LICENSE +21 -21
  42. package/.aios-core/docs/SHARD-TRANSLATION-GUIDE.md +335 -0
  43. package/.aios-core/docs/component-creation-guide.md +458 -0
  44. package/.aios-core/docs/session-update-pattern.md +307 -0
  45. package/.aios-core/docs/standards/AIOS-FRAMEWORK-MASTER.md +1963 -0
  46. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1-SUMMARY.md +1190 -0
  47. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1.md +439 -0
  48. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO.md +5398 -0
  49. package/.aios-core/docs/standards/V3-ARCHITECTURAL-DECISIONS.md +523 -0
  50. package/.aios-core/docs/template-syntax.md +267 -0
  51. package/.aios-core/docs/troubleshooting-guide.md +625 -0
  52. package/.aios-core/infrastructure/templates/aios-sync.yaml.template +193 -193
  53. package/.aios-core/infrastructure/templates/coderabbit.yaml.template +279 -279
  54. package/.aios-core/infrastructure/templates/github-workflows/ci.yml.template +169 -169
  55. package/.aios-core/infrastructure/templates/github-workflows/pr-automation.yml.template +330 -330
  56. package/.aios-core/infrastructure/templates/github-workflows/release.yml.template +196 -196
  57. package/.aios-core/infrastructure/templates/gitignore/gitignore-aios-base.tmpl +63 -63
  58. package/.aios-core/infrastructure/templates/gitignore/gitignore-brownfield-merge.tmpl +18 -18
  59. package/.aios-core/infrastructure/templates/gitignore/gitignore-node.tmpl +85 -85
  60. package/.aios-core/infrastructure/templates/gitignore/gitignore-python.tmpl +145 -145
  61. package/.aios-core/infrastructure/tests/utilities-audit-results.json +501 -0
  62. package/.aios-core/install-manifest.yaml +101 -101
  63. package/.aios-core/local-config.yaml.template +70 -70
  64. package/.aios-core/manifests/agents.csv +29 -0
  65. package/.aios-core/manifests/schema/manifest-schema.json +190 -190
  66. package/.aios-core/manifests/tasks.csv +198 -0
  67. package/.aios-core/manifests/workers.csv +204 -0
  68. package/.aios-core/monitor/hooks/lib/__init__.py +1 -1
  69. package/.aios-core/monitor/hooks/lib/enrich.py +58 -58
  70. package/.aios-core/monitor/hooks/lib/send_event.py +47 -47
  71. package/.aios-core/monitor/hooks/notification.py +29 -29
  72. package/.aios-core/monitor/hooks/post_tool_use.py +45 -45
  73. package/.aios-core/monitor/hooks/pre_compact.py +29 -29
  74. package/.aios-core/monitor/hooks/pre_tool_use.py +40 -40
  75. package/.aios-core/monitor/hooks/stop.py +29 -29
  76. package/.aios-core/monitor/hooks/subagent_stop.py +29 -29
  77. package/.aios-core/monitor/hooks/user_prompt_submit.py +38 -38
  78. package/.aios-core/product/templates/adr.hbs +125 -125
  79. package/.aios-core/product/templates/component-react-tmpl.tsx +98 -98
  80. package/.aios-core/product/templates/dbdr.hbs +241 -241
  81. package/.aios-core/product/templates/engine/schemas/adr.schema.json +102 -102
  82. package/.aios-core/product/templates/engine/schemas/dbdr.schema.json +205 -205
  83. package/.aios-core/product/templates/engine/schemas/epic.schema.json +175 -175
  84. package/.aios-core/product/templates/engine/schemas/pmdr.schema.json +175 -175
  85. package/.aios-core/product/templates/engine/schemas/prd-v2.schema.json +300 -300
  86. package/.aios-core/product/templates/engine/schemas/prd.schema.json +152 -152
  87. package/.aios-core/product/templates/engine/schemas/story.schema.json +222 -222
  88. package/.aios-core/product/templates/engine/schemas/task.schema.json +154 -154
  89. package/.aios-core/product/templates/epic.hbs +212 -212
  90. package/.aios-core/product/templates/eslintrc-security.json +32 -32
  91. package/.aios-core/product/templates/github-actions-cd.yml +212 -212
  92. package/.aios-core/product/templates/github-actions-ci.yml +172 -172
  93. package/.aios-core/product/templates/pmdr.hbs +186 -186
  94. package/.aios-core/product/templates/prd-v2.0.hbs +216 -216
  95. package/.aios-core/product/templates/prd.hbs +201 -201
  96. package/.aios-core/product/templates/shock-report-tmpl.html +502 -502
  97. package/.aios-core/product/templates/story.hbs +263 -263
  98. package/.aios-core/product/templates/task.hbs +170 -170
  99. package/.aios-core/product/templates/tmpl-comment-on-examples.sql +158 -158
  100. package/.aios-core/product/templates/tmpl-migration-script.sql +91 -91
  101. package/.aios-core/product/templates/tmpl-rls-granular-policies.sql +104 -104
  102. package/.aios-core/product/templates/tmpl-rls-kiss-policy.sql +10 -10
  103. package/.aios-core/product/templates/tmpl-rls-roles.sql +135 -135
  104. package/.aios-core/product/templates/tmpl-rls-simple.sql +77 -77
  105. package/.aios-core/product/templates/tmpl-rls-tenant.sql +152 -152
  106. package/.aios-core/product/templates/tmpl-rollback-script.sql +77 -77
  107. package/.aios-core/product/templates/tmpl-seed-data.sql +140 -140
  108. package/.aios-core/product/templates/tmpl-smoke-test.sql +16 -16
  109. package/.aios-core/product/templates/tmpl-staging-copy-merge.sql +139 -139
  110. package/.aios-core/product/templates/tmpl-stored-proc.sql +140 -140
  111. package/.aios-core/product/templates/tmpl-trigger.sql +152 -152
  112. package/.aios-core/product/templates/tmpl-view-materialized.sql +133 -133
  113. package/.aios-core/product/templates/tmpl-view.sql +177 -177
  114. package/.aios-core/product/templates/token-exports-css-tmpl.css +240 -240
  115. package/.aios-core/quality/schemas/quality-metrics.schema.json +233 -233
  116. package/.aios-core/scripts/migrate-framework-docs.sh +300 -300
  117. package/.aios-core/scripts/pm.sh +0 -0
  118. package/.claude/hooks/enforce-architecture-first.py +196 -196
  119. package/.claude/hooks/mind-clone-governance.py +192 -192
  120. package/.claude/hooks/read-protection.py +151 -151
  121. package/.claude/hooks/slug-validation.py +176 -176
  122. package/.claude/hooks/sql-governance.py +182 -182
  123. package/.claude/hooks/write-path-validation.py +194 -194
  124. package/.claude/rules/agent-authority.md +105 -0
  125. package/.claude/rules/coderabbit-integration.md +93 -0
  126. package/.claude/rules/ids-principles.md +112 -0
  127. package/.claude/rules/story-lifecycle.md +139 -0
  128. package/.claude/rules/workflow-execution.md +150 -0
  129. package/LICENSE +48 -48
  130. package/bin/aios-minimal.js +0 -0
  131. package/bin/aios.js +0 -0
  132. package/package.json +1 -1
  133. package/packages/aios-install/bin/aios-install.js +0 -0
  134. package/packages/aios-install/bin/edmcp.js +0 -0
  135. package/packages/aios-pro-cli/bin/aios-pro.js +0 -0
  136. package/packages/installer/src/wizard/pro-setup.js +433 -49
  137. package/scripts/check-markdown-links.py +352 -352
  138. package/scripts/code-intel-health-check.js +343 -0
  139. package/scripts/dashboard-parallel-dev.sh +0 -0
  140. package/scripts/dashboard-parallel-phase3.sh +0 -0
  141. package/scripts/dashboard-parallel-phase4.sh +0 -0
  142. package/scripts/glue/README.md +355 -0
  143. package/scripts/glue/compose-agent-prompt.cjs +362 -0
  144. package/scripts/install-monitor-hooks.sh +0 -0
  145. package/.aios-core/lib/build.json +0 -1
@@ -1,279 +1,279 @@
1
- /**
2
- * Template Validator for AIOS-FULLSTACK
3
- * Validates component templates for required structure and placeholders
4
- * @module template-validator
5
- */
6
-
7
- const fs = require('fs-extra');
8
- const _path = require('path');
9
- const _yaml = require('js-yaml');
10
- const TemplateEngine = require('./template-engine');
11
-
12
- class TemplateValidator {
13
- constructor() {
14
- this.engine = new TemplateEngine();
15
- this.requiredVariables = {
16
- agent: [
17
- 'AGENT_NAME',
18
- 'AGENT_ID',
19
- 'AGENT_TITLE',
20
- 'AGENT_ICON',
21
- 'WHEN_TO_USE',
22
- 'PERSONA_ROLE',
23
- 'PERSONA_STYLE',
24
- 'PERSONA_IDENTITY',
25
- 'PERSONA_FOCUS'
26
- ],
27
- task: [
28
- 'TASK_TITLE',
29
- 'TASK_ID',
30
- 'AGENT_NAME',
31
- 'VERSION',
32
- 'TASK_DESCRIPTION',
33
- 'OUTPUT_DESCRIPTION'
34
- ],
35
- workflow: [
36
- 'WORKFLOW_ID',
37
- 'WORKFLOW_NAME',
38
- 'WORKFLOW_DESCRIPTION',
39
- 'WORKFLOW_VERSION',
40
- 'WORKFLOW_TYPE',
41
- 'AUTHOR',
42
- 'CREATED_DATE',
43
- 'LAST_MODIFIED'
44
- ]
45
- };
46
- }
47
-
48
- /**
49
- * Validate a template file
50
- * @param {string} templatePath - Path to template file
51
- * @param {string} templateType - Type of template (agent, task, workflow)
52
- * @returns {Promise<Object>} Validation result
53
- */
54
- async validateTemplateFile(templatePath, templateType) {
55
- try {
56
- const template = await fs.readFile(templatePath, 'utf8');
57
- return this.validateTemplate(template, templateType);
58
- } catch (error) {
59
- return {
60
- valid: false,
61
- errors: [`Failed to read template file: ${error.message}`]
62
- };
63
- }
64
- }
65
-
66
- /**
67
- * Validate a template string
68
- * @param {string} template - Template content
69
- * @param {string} templateType - Type of template
70
- * @returns {Object} Validation result
71
- */
72
- validateTemplate(template, templateType) {
73
- const errors = [];
74
- const warnings = [];
75
-
76
- // Check template type is valid
77
- if (!this.requiredVariables[templateType]) {
78
- return {
79
- valid: false,
80
- errors: [`Unknown template type: ${templateType}`]
81
- };
82
- }
83
-
84
- // Validate required variables
85
- const requiredVars = this.requiredVariables[templateType];
86
- const validation = this.engine.validateTemplate(template, requiredVars);
87
-
88
- if (!validation.valid) {
89
- errors.push(`Missing required variables: ${validation.missing.join(', ')}`);
90
- }
91
-
92
- // Check for balanced conditionals
93
- const conditionalCheck = this.checkBalancedConditionals(template);
94
- if (!conditionalCheck.valid) {
95
- errors.push(...conditionalCheck.errors);
96
- }
97
-
98
- // Check for balanced loops
99
- const loopCheck = this.checkBalancedLoops(template);
100
- if (!loopCheck.valid) {
101
- errors.push(...loopCheck.errors);
102
- }
103
-
104
- // Template-specific validation
105
- const specificCheck = this.validateSpecificTemplate(template, templateType);
106
- if (!specificCheck.valid) {
107
- errors.push(...specificCheck.errors);
108
- }
109
- warnings.push(...specificCheck.warnings);
110
-
111
- // Check for potential security issues
112
- const securityCheck = this.checkSecurityIssues(template);
113
- if (!securityCheck.valid) {
114
- errors.push(...securityCheck.errors);
115
- }
116
- warnings.push(...securityCheck.warnings);
117
-
118
- return {
119
- valid: errors.length === 0,
120
- errors,
121
- warnings,
122
- variables: this.engine.getTemplateVariables(template)
123
- };
124
- }
125
-
126
- /**
127
- * Check for balanced conditional blocks
128
- * @private
129
- */
130
- checkBalancedConditionals(template) {
131
- const errors = [];
132
- const openTags = template.match(/\{\{#IF_([^}]+)\}\}/g) || [];
133
- const closeTags = template.match(/\{\{\/IF_([^}]+)\}\}/g) || [];
134
-
135
- const openConditions = openTags.map(tag => tag.match(/IF_([^}]+)/)[1]);
136
- const closeConditions = closeTags.map(tag => tag.match(/IF_([^}]+)/)[1]);
137
-
138
- // Check each open has a close
139
- for (const condition of openConditions) {
140
- if (!closeConditions.includes(condition)) {
141
- errors.push(`Unclosed conditional: {{#IF_${condition}}}`);
142
- }
143
- }
144
-
145
- // Check each close has an open
146
- for (const condition of closeConditions) {
147
- if (!openConditions.includes(condition)) {
148
- errors.push(`Unexpected closing tag: {{/IF_${condition}}}`);
149
- }
150
- }
151
-
152
- return { valid: errors.length === 0, errors };
153
- }
154
-
155
- /**
156
- * Check for balanced loop blocks
157
- * @private
158
- */
159
- checkBalancedLoops(template) {
160
- const errors = [];
161
- const openTags = template.match(/\{\{#EACH_([^}]+)\}\}/g) || [];
162
- const closeTags = template.match(/\{\{\/EACH_([^}]+)\}\}/g) || [];
163
-
164
- const openLoops = openTags.map(tag => tag.match(/EACH_([^}]+)/)[1]);
165
- const closeLoops = closeTags.map(tag => tag.match(/EACH_([^}]+)/)[1]);
166
-
167
- // Check each open has a close
168
- for (const loop of openLoops) {
169
- if (!closeLoops.includes(loop)) {
170
- errors.push(`Unclosed loop: {{#EACH_${loop}}}`);
171
- }
172
- }
173
-
174
- // Check each close has an open
175
- for (const loop of closeLoops) {
176
- if (!openLoops.includes(loop)) {
177
- errors.push(`Unexpected closing tag: {{/EACH_${loop}}}`);
178
- }
179
- }
180
-
181
- return { valid: errors.length === 0, errors };
182
- }
183
-
184
- /**
185
- * Template-specific validation
186
- * @private
187
- */
188
- validateSpecificTemplate(template, templateType) {
189
- const errors = [];
190
- const warnings = [];
191
-
192
- switch (templateType) {
193
- case 'agent':
194
- // Check for YAML structure
195
- if (!template.includes('agent:') || !template.includes('persona:')) {
196
- errors.push('Agent template must include agent: and persona: sections');
197
- }
198
- if (!template.includes('commands:')) {
199
- warnings.push('Agent template should include commands: section');
200
- }
201
- break;
202
-
203
- case 'task':
204
- // Check for markdown headers
205
- if (!template.includes('# {{TASK_TITLE}}')) {
206
- warnings.push('Task template should start with # {{TASK_TITLE}}');
207
- }
208
- if (!template.includes('## Workflow')) {
209
- warnings.push('Task template should include ## Workflow section');
210
- }
211
- break;
212
-
213
- case 'workflow':
214
- // Check for YAML structure
215
- if (!template.includes('workflow:') || !template.includes('steps:')) {
216
- errors.push('Workflow template must include workflow: and steps: sections');
217
- }
218
- if (!template.includes('metadata:')) {
219
- warnings.push('Workflow template should include metadata: section');
220
- }
221
- break;
222
- }
223
-
224
- return { valid: errors.length === 0, errors, warnings };
225
- }
226
-
227
- /**
228
- * Check for potential security issues in template
229
- * @private
230
- */
231
- checkSecurityIssues(template) {
232
- const errors = [];
233
- const warnings = [];
234
-
235
- // Check for dangerous patterns
236
- const dangerousPatterns = [
237
- { pattern: /eval\s*\(/, message: 'Template contains eval() - security risk' },
238
- { pattern: /Function\s*\(/, message: 'Template contains Function() - security risk' },
239
- { pattern: /require\s*\([^'"]+\)/, message: 'Dynamic require detected - potential security risk' },
240
- { pattern: /<script/i, message: 'Script tags detected in template' }
241
- ];
242
-
243
- for (const { pattern, message } of dangerousPatterns) {
244
- if (pattern.test(template)) {
245
- errors.push(message);
246
- }
247
- }
248
-
249
- // Check for suspicious patterns
250
- if (template.includes('__proto__') || template.includes('constructor')) {
251
- warnings.push('Template contains potentially dangerous property access');
252
- }
253
-
254
- return { valid: errors.length === 0, errors, warnings };
255
- }
256
-
257
- /**
258
- * Get required variables for a template type
259
- * @param {string} templateType - Type of template
260
- * @returns {string[]} Array of required variable names
261
- */
262
- getRequiredVariables(templateType) {
263
- return this.requiredVariables[templateType] || [];
264
- }
265
-
266
- /**
267
- * Add custom required variables for a template type
268
- * @param {string} templateType - Type of template
269
- * @param {string[]} variables - Additional required variables
270
- */
271
- addRequiredVariables(templateType, variables) {
272
- if (!this.requiredVariables[templateType]) {
273
- this.requiredVariables[templateType] = [];
274
- }
275
- this.requiredVariables[templateType].push(...variables);
276
- }
277
- }
278
-
1
+ /**
2
+ * Template Validator for AIOS-FULLSTACK
3
+ * Validates component templates for required structure and placeholders
4
+ * @module template-validator
5
+ */
6
+
7
+ const fs = require('fs-extra');
8
+ const _path = require('path');
9
+ const _yaml = require('js-yaml');
10
+ const TemplateEngine = require('./template-engine');
11
+
12
+ class TemplateValidator {
13
+ constructor() {
14
+ this.engine = new TemplateEngine();
15
+ this.requiredVariables = {
16
+ agent: [
17
+ 'AGENT_NAME',
18
+ 'AGENT_ID',
19
+ 'AGENT_TITLE',
20
+ 'AGENT_ICON',
21
+ 'WHEN_TO_USE',
22
+ 'PERSONA_ROLE',
23
+ 'PERSONA_STYLE',
24
+ 'PERSONA_IDENTITY',
25
+ 'PERSONA_FOCUS'
26
+ ],
27
+ task: [
28
+ 'TASK_TITLE',
29
+ 'TASK_ID',
30
+ 'AGENT_NAME',
31
+ 'VERSION',
32
+ 'TASK_DESCRIPTION',
33
+ 'OUTPUT_DESCRIPTION'
34
+ ],
35
+ workflow: [
36
+ 'WORKFLOW_ID',
37
+ 'WORKFLOW_NAME',
38
+ 'WORKFLOW_DESCRIPTION',
39
+ 'WORKFLOW_VERSION',
40
+ 'WORKFLOW_TYPE',
41
+ 'AUTHOR',
42
+ 'CREATED_DATE',
43
+ 'LAST_MODIFIED'
44
+ ]
45
+ };
46
+ }
47
+
48
+ /**
49
+ * Validate a template file
50
+ * @param {string} templatePath - Path to template file
51
+ * @param {string} templateType - Type of template (agent, task, workflow)
52
+ * @returns {Promise<Object>} Validation result
53
+ */
54
+ async validateTemplateFile(templatePath, templateType) {
55
+ try {
56
+ const template = await fs.readFile(templatePath, 'utf8');
57
+ return this.validateTemplate(template, templateType);
58
+ } catch (error) {
59
+ return {
60
+ valid: false,
61
+ errors: [`Failed to read template file: ${error.message}`]
62
+ };
63
+ }
64
+ }
65
+
66
+ /**
67
+ * Validate a template string
68
+ * @param {string} template - Template content
69
+ * @param {string} templateType - Type of template
70
+ * @returns {Object} Validation result
71
+ */
72
+ validateTemplate(template, templateType) {
73
+ const errors = [];
74
+ const warnings = [];
75
+
76
+ // Check template type is valid
77
+ if (!this.requiredVariables[templateType]) {
78
+ return {
79
+ valid: false,
80
+ errors: [`Unknown template type: ${templateType}`]
81
+ };
82
+ }
83
+
84
+ // Validate required variables
85
+ const requiredVars = this.requiredVariables[templateType];
86
+ const validation = this.engine.validateTemplate(template, requiredVars);
87
+
88
+ if (!validation.valid) {
89
+ errors.push(`Missing required variables: ${validation.missing.join(', ')}`);
90
+ }
91
+
92
+ // Check for balanced conditionals
93
+ const conditionalCheck = this.checkBalancedConditionals(template);
94
+ if (!conditionalCheck.valid) {
95
+ errors.push(...conditionalCheck.errors);
96
+ }
97
+
98
+ // Check for balanced loops
99
+ const loopCheck = this.checkBalancedLoops(template);
100
+ if (!loopCheck.valid) {
101
+ errors.push(...loopCheck.errors);
102
+ }
103
+
104
+ // Template-specific validation
105
+ const specificCheck = this.validateSpecificTemplate(template, templateType);
106
+ if (!specificCheck.valid) {
107
+ errors.push(...specificCheck.errors);
108
+ }
109
+ warnings.push(...specificCheck.warnings);
110
+
111
+ // Check for potential security issues
112
+ const securityCheck = this.checkSecurityIssues(template);
113
+ if (!securityCheck.valid) {
114
+ errors.push(...securityCheck.errors);
115
+ }
116
+ warnings.push(...securityCheck.warnings);
117
+
118
+ return {
119
+ valid: errors.length === 0,
120
+ errors,
121
+ warnings,
122
+ variables: this.engine.getTemplateVariables(template)
123
+ };
124
+ }
125
+
126
+ /**
127
+ * Check for balanced conditional blocks
128
+ * @private
129
+ */
130
+ checkBalancedConditionals(template) {
131
+ const errors = [];
132
+ const openTags = template.match(/\{\{#IF_([^}]+)\}\}/g) || [];
133
+ const closeTags = template.match(/\{\{\/IF_([^}]+)\}\}/g) || [];
134
+
135
+ const openConditions = openTags.map(tag => tag.match(/IF_([^}]+)/)[1]);
136
+ const closeConditions = closeTags.map(tag => tag.match(/IF_([^}]+)/)[1]);
137
+
138
+ // Check each open has a close
139
+ for (const condition of openConditions) {
140
+ if (!closeConditions.includes(condition)) {
141
+ errors.push(`Unclosed conditional: {{#IF_${condition}}}`);
142
+ }
143
+ }
144
+
145
+ // Check each close has an open
146
+ for (const condition of closeConditions) {
147
+ if (!openConditions.includes(condition)) {
148
+ errors.push(`Unexpected closing tag: {{/IF_${condition}}}`);
149
+ }
150
+ }
151
+
152
+ return { valid: errors.length === 0, errors };
153
+ }
154
+
155
+ /**
156
+ * Check for balanced loop blocks
157
+ * @private
158
+ */
159
+ checkBalancedLoops(template) {
160
+ const errors = [];
161
+ const openTags = template.match(/\{\{#EACH_([^}]+)\}\}/g) || [];
162
+ const closeTags = template.match(/\{\{\/EACH_([^}]+)\}\}/g) || [];
163
+
164
+ const openLoops = openTags.map(tag => tag.match(/EACH_([^}]+)/)[1]);
165
+ const closeLoops = closeTags.map(tag => tag.match(/EACH_([^}]+)/)[1]);
166
+
167
+ // Check each open has a close
168
+ for (const loop of openLoops) {
169
+ if (!closeLoops.includes(loop)) {
170
+ errors.push(`Unclosed loop: {{#EACH_${loop}}}`);
171
+ }
172
+ }
173
+
174
+ // Check each close has an open
175
+ for (const loop of closeLoops) {
176
+ if (!openLoops.includes(loop)) {
177
+ errors.push(`Unexpected closing tag: {{/EACH_${loop}}}`);
178
+ }
179
+ }
180
+
181
+ return { valid: errors.length === 0, errors };
182
+ }
183
+
184
+ /**
185
+ * Template-specific validation
186
+ * @private
187
+ */
188
+ validateSpecificTemplate(template, templateType) {
189
+ const errors = [];
190
+ const warnings = [];
191
+
192
+ switch (templateType) {
193
+ case 'agent':
194
+ // Check for YAML structure
195
+ if (!template.includes('agent:') || !template.includes('persona:')) {
196
+ errors.push('Agent template must include agent: and persona: sections');
197
+ }
198
+ if (!template.includes('commands:')) {
199
+ warnings.push('Agent template should include commands: section');
200
+ }
201
+ break;
202
+
203
+ case 'task':
204
+ // Check for markdown headers
205
+ if (!template.includes('# {{TASK_TITLE}}')) {
206
+ warnings.push('Task template should start with # {{TASK_TITLE}}');
207
+ }
208
+ if (!template.includes('## Workflow')) {
209
+ warnings.push('Task template should include ## Workflow section');
210
+ }
211
+ break;
212
+
213
+ case 'workflow':
214
+ // Check for YAML structure
215
+ if (!template.includes('workflow:') || !template.includes('steps:')) {
216
+ errors.push('Workflow template must include workflow: and steps: sections');
217
+ }
218
+ if (!template.includes('metadata:')) {
219
+ warnings.push('Workflow template should include metadata: section');
220
+ }
221
+ break;
222
+ }
223
+
224
+ return { valid: errors.length === 0, errors, warnings };
225
+ }
226
+
227
+ /**
228
+ * Check for potential security issues in template
229
+ * @private
230
+ */
231
+ checkSecurityIssues(template) {
232
+ const errors = [];
233
+ const warnings = [];
234
+
235
+ // Check for dangerous patterns
236
+ const dangerousPatterns = [
237
+ { pattern: /eval\s*\(/, message: 'Template contains eval() - security risk' },
238
+ { pattern: /Function\s*\(/, message: 'Template contains Function() - security risk' },
239
+ { pattern: /require\s*\([^'"]+\)/, message: 'Dynamic require detected - potential security risk' },
240
+ { pattern: /<script/i, message: 'Script tags detected in template' }
241
+ ];
242
+
243
+ for (const { pattern, message } of dangerousPatterns) {
244
+ if (pattern.test(template)) {
245
+ errors.push(message);
246
+ }
247
+ }
248
+
249
+ // Check for suspicious patterns
250
+ if (template.includes('__proto__') || template.includes('constructor')) {
251
+ warnings.push('Template contains potentially dangerous property access');
252
+ }
253
+
254
+ return { valid: errors.length === 0, errors, warnings };
255
+ }
256
+
257
+ /**
258
+ * Get required variables for a template type
259
+ * @param {string} templateType - Type of template
260
+ * @returns {string[]} Array of required variable names
261
+ */
262
+ getRequiredVariables(templateType) {
263
+ return this.requiredVariables[templateType] || [];
264
+ }
265
+
266
+ /**
267
+ * Add custom required variables for a template type
268
+ * @param {string} templateType - Type of template
269
+ * @param {string[]} variables - Additional required variables
270
+ */
271
+ addRequiredVariables(templateType, variables) {
272
+ if (!this.requiredVariables[templateType]) {
273
+ this.requiredVariables[templateType] = [];
274
+ }
275
+ this.requiredVariables[templateType].push(...variables);
276
+ }
277
+ }
278
+
279
279
  module.exports = TemplateValidator;