aios-core 4.0.2 → 4.0.4

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/registry-update-log.jsonl +113 -0
  5. package/.aios-core/development/scripts/approval-workflow.js +642 -642
  6. package/.aios-core/development/scripts/backup-manager.js +606 -606
  7. package/.aios-core/development/scripts/branch-manager.js +389 -389
  8. package/.aios-core/development/scripts/code-quality-improver.js +1311 -1311
  9. package/.aios-core/development/scripts/commit-message-generator.js +849 -849
  10. package/.aios-core/development/scripts/conflict-resolver.js +674 -674
  11. package/.aios-core/development/scripts/dependency-analyzer.js +637 -637
  12. package/.aios-core/development/scripts/diff-generator.js +351 -351
  13. package/.aios-core/development/scripts/elicitation-engine.js +384 -384
  14. package/.aios-core/development/scripts/elicitation-session-manager.js +299 -299
  15. package/.aios-core/development/scripts/git-wrapper.js +461 -461
  16. package/.aios-core/development/scripts/manifest-preview.js +244 -244
  17. package/.aios-core/development/scripts/metrics-tracker.js +775 -775
  18. package/.aios-core/development/scripts/modification-validator.js +554 -554
  19. package/.aios-core/development/scripts/pattern-learner.js +1224 -1224
  20. package/.aios-core/development/scripts/performance-analyzer.js +757 -757
  21. package/.aios-core/development/scripts/refactoring-suggester.js +1138 -1138
  22. package/.aios-core/development/scripts/rollback-handler.js +530 -530
  23. package/.aios-core/development/scripts/security-checker.js +358 -358
  24. package/.aios-core/development/scripts/template-engine.js +239 -239
  25. package/.aios-core/development/scripts/template-validator.js +278 -278
  26. package/.aios-core/development/scripts/test-generator.js +843 -843
  27. package/.aios-core/development/scripts/transaction-manager.js +589 -589
  28. package/.aios-core/development/scripts/usage-tracker.js +673 -673
  29. package/.aios-core/development/scripts/validate-filenames.js +226 -226
  30. package/.aios-core/development/scripts/version-tracker.js +526 -526
  31. package/.aios-core/development/scripts/yaml-validator.js +396 -396
  32. package/.aios-core/development/templates/service-template/README.md.hbs +158 -158
  33. package/.aios-core/development/templates/service-template/__tests__/index.test.ts.hbs +237 -237
  34. package/.aios-core/development/templates/service-template/client.ts.hbs +403 -403
  35. package/.aios-core/development/templates/service-template/errors.ts.hbs +182 -182
  36. package/.aios-core/development/templates/service-template/index.ts.hbs +120 -120
  37. package/.aios-core/development/templates/service-template/package.json.hbs +87 -87
  38. package/.aios-core/development/templates/service-template/types.ts.hbs +145 -145
  39. package/.aios-core/development/templates/squad-template/LICENSE +21 -21
  40. package/.aios-core/docs/SHARD-TRANSLATION-GUIDE.md +335 -0
  41. package/.aios-core/docs/component-creation-guide.md +458 -0
  42. package/.aios-core/docs/session-update-pattern.md +307 -0
  43. package/.aios-core/docs/standards/AIOS-FRAMEWORK-MASTER.md +1963 -0
  44. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1-SUMMARY.md +1190 -0
  45. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1.md +439 -0
  46. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO.md +5398 -0
  47. package/.aios-core/docs/standards/V3-ARCHITECTURAL-DECISIONS.md +523 -0
  48. package/.aios-core/docs/template-syntax.md +267 -0
  49. package/.aios-core/docs/troubleshooting-guide.md +625 -0
  50. package/.aios-core/infrastructure/templates/aios-sync.yaml.template +193 -193
  51. package/.aios-core/infrastructure/templates/coderabbit.yaml.template +279 -279
  52. package/.aios-core/infrastructure/templates/github-workflows/ci.yml.template +169 -169
  53. package/.aios-core/infrastructure/templates/github-workflows/pr-automation.yml.template +330 -330
  54. package/.aios-core/infrastructure/templates/github-workflows/release.yml.template +196 -196
  55. package/.aios-core/infrastructure/templates/gitignore/gitignore-aios-base.tmpl +63 -63
  56. package/.aios-core/infrastructure/templates/gitignore/gitignore-brownfield-merge.tmpl +18 -18
  57. package/.aios-core/infrastructure/templates/gitignore/gitignore-node.tmpl +85 -85
  58. package/.aios-core/infrastructure/templates/gitignore/gitignore-python.tmpl +145 -145
  59. package/.aios-core/infrastructure/tests/utilities-audit-results.json +501 -0
  60. package/.aios-core/install-manifest.yaml +97 -97
  61. package/.aios-core/local-config.yaml.template +68 -68
  62. package/.aios-core/manifests/agents.csv +1 -0
  63. package/.aios-core/manifests/schema/manifest-schema.json +190 -190
  64. package/.aios-core/manifests/tasks.csv +121 -0
  65. package/.aios-core/manifests/workers.csv +204 -0
  66. package/.aios-core/monitor/hooks/lib/__init__.py +1 -1
  67. package/.aios-core/monitor/hooks/lib/enrich.py +58 -58
  68. package/.aios-core/monitor/hooks/lib/send_event.py +47 -47
  69. package/.aios-core/monitor/hooks/notification.py +29 -29
  70. package/.aios-core/monitor/hooks/post_tool_use.py +45 -45
  71. package/.aios-core/monitor/hooks/pre_compact.py +29 -29
  72. package/.aios-core/monitor/hooks/pre_tool_use.py +40 -40
  73. package/.aios-core/monitor/hooks/stop.py +29 -29
  74. package/.aios-core/monitor/hooks/subagent_stop.py +29 -29
  75. package/.aios-core/monitor/hooks/user_prompt_submit.py +38 -38
  76. package/.aios-core/product/templates/adr.hbs +125 -125
  77. package/.aios-core/product/templates/component-react-tmpl.tsx +98 -98
  78. package/.aios-core/product/templates/dbdr.hbs +241 -241
  79. package/.aios-core/product/templates/engine/schemas/adr.schema.json +102 -102
  80. package/.aios-core/product/templates/engine/schemas/dbdr.schema.json +205 -205
  81. package/.aios-core/product/templates/engine/schemas/epic.schema.json +175 -175
  82. package/.aios-core/product/templates/engine/schemas/pmdr.schema.json +175 -175
  83. package/.aios-core/product/templates/engine/schemas/prd-v2.schema.json +300 -300
  84. package/.aios-core/product/templates/engine/schemas/prd.schema.json +152 -152
  85. package/.aios-core/product/templates/engine/schemas/story.schema.json +222 -222
  86. package/.aios-core/product/templates/engine/schemas/task.schema.json +154 -154
  87. package/.aios-core/product/templates/epic.hbs +212 -212
  88. package/.aios-core/product/templates/eslintrc-security.json +32 -32
  89. package/.aios-core/product/templates/github-actions-cd.yml +212 -212
  90. package/.aios-core/product/templates/github-actions-ci.yml +172 -172
  91. package/.aios-core/product/templates/pmdr.hbs +186 -186
  92. package/.aios-core/product/templates/prd-v2.0.hbs +216 -216
  93. package/.aios-core/product/templates/prd.hbs +201 -201
  94. package/.aios-core/product/templates/shock-report-tmpl.html +502 -502
  95. package/.aios-core/product/templates/story.hbs +263 -263
  96. package/.aios-core/product/templates/task.hbs +170 -170
  97. package/.aios-core/product/templates/tmpl-comment-on-examples.sql +158 -158
  98. package/.aios-core/product/templates/tmpl-migration-script.sql +91 -91
  99. package/.aios-core/product/templates/tmpl-rls-granular-policies.sql +104 -104
  100. package/.aios-core/product/templates/tmpl-rls-kiss-policy.sql +10 -10
  101. package/.aios-core/product/templates/tmpl-rls-roles.sql +135 -135
  102. package/.aios-core/product/templates/tmpl-rls-simple.sql +77 -77
  103. package/.aios-core/product/templates/tmpl-rls-tenant.sql +152 -152
  104. package/.aios-core/product/templates/tmpl-rollback-script.sql +77 -77
  105. package/.aios-core/product/templates/tmpl-seed-data.sql +140 -140
  106. package/.aios-core/product/templates/tmpl-smoke-test.sql +16 -16
  107. package/.aios-core/product/templates/tmpl-staging-copy-merge.sql +139 -139
  108. package/.aios-core/product/templates/tmpl-stored-proc.sql +140 -140
  109. package/.aios-core/product/templates/tmpl-trigger.sql +152 -152
  110. package/.aios-core/product/templates/tmpl-view-materialized.sql +133 -133
  111. package/.aios-core/product/templates/tmpl-view.sql +177 -177
  112. package/.aios-core/product/templates/token-exports-css-tmpl.css +240 -240
  113. package/.aios-core/quality/schemas/quality-metrics.schema.json +233 -233
  114. package/.aios-core/scripts/migrate-framework-docs.sh +300 -300
  115. package/.aios-core/scripts/pm.sh +0 -0
  116. package/.claude/hooks/enforce-architecture-first.py +196 -0
  117. package/.claude/hooks/install-hooks.sh +41 -0
  118. package/.claude/hooks/mind-clone-governance.py +192 -0
  119. package/.claude/hooks/pre-commit-mmos-guard.sh +99 -0
  120. package/.claude/hooks/pre-commit-version-check.sh +156 -0
  121. package/.claude/hooks/read-protection.py +151 -0
  122. package/.claude/hooks/slug-validation.py +176 -0
  123. package/.claude/hooks/sql-governance.py +182 -0
  124. package/.claude/hooks/write-path-validation.py +194 -0
  125. package/.claude/rules/agent-authority.md +105 -0
  126. package/.claude/rules/coderabbit-integration.md +93 -0
  127. package/.claude/rules/ids-principles.md +112 -0
  128. package/.claude/rules/story-lifecycle.md +139 -0
  129. package/.claude/rules/workflow-execution.md +150 -0
  130. package/LICENSE +48 -48
  131. package/README.md +30 -7
  132. package/bin/aios-minimal.js +0 -0
  133. package/bin/aios.js +15 -15
  134. package/package.json +2 -4
  135. package/packages/aios-install/bin/aios-install.js +0 -0
  136. package/packages/aios-install/bin/edmcp.js +0 -0
  137. package/packages/aios-pro-cli/bin/aios-pro.js +0 -0
  138. package/scripts/check-markdown-links.py +352 -352
  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,237 +1,237 @@
1
- /**
2
- * Tests for {{pascalCase serviceName}} service.
3
- * @module @aios/{{kebabCase serviceName}}/tests
4
- * @story {{storyId}}
5
- */
6
-
7
- import {
8
- create{{pascalCase serviceName}}Service,
9
- {{pascalCase serviceName}}Error,
10
- {{pascalCase serviceName}}ErrorCode,
11
- } from '../index';
12
- import type { {{pascalCase serviceName}}Config, {{pascalCase serviceName}}Service } from '../types';
13
-
14
- describe('{{pascalCase serviceName}}Service', () => {
15
- // ─────────────────────────────────────────────────────────────────────────────
16
- // Test Configuration
17
- // ─────────────────────────────────────────────────────────────────────────────
18
-
19
- const validConfig: {{pascalCase serviceName}}Config = {
20
- {{#each envVars}}
21
- {{camelCase this.name}}: 'test-{{kebabCase this.name}}',
22
- {{/each}}
23
- };
24
-
25
- // ─────────────────────────────────────────────────────────────────────────────
26
- // Factory Function Tests
27
- // ─────────────────────────────────────────────────────────────────────────────
28
-
29
- describe('create{{pascalCase serviceName}}Service', () => {
30
- it('should create a service instance with valid config', () => {
31
- const service = create{{pascalCase serviceName}}Service(validConfig);
32
-
33
- expect(service).toBeDefined();
34
- expect(typeof service.execute).toBe('function');
35
- expect(typeof service.getConfig).toBe('function');
36
- expect(typeof service.healthCheck).toBe('function');
37
- });
38
-
39
- it('should throw on null config', () => {
40
- expect(() => {
41
- create{{pascalCase serviceName}}Service(null as unknown as {{pascalCase serviceName}}Config);
42
- }).toThrow({{pascalCase serviceName}}Error);
43
- });
44
-
45
- it('should throw on undefined config', () => {
46
- expect(() => {
47
- create{{pascalCase serviceName}}Service(undefined as unknown as {{pascalCase serviceName}}Config);
48
- }).toThrow({{pascalCase serviceName}}Error);
49
- });
50
-
51
- {{#each envVars}}
52
- {{#if this.required}}
53
- it('should throw when {{this.name}} is missing', () => {
54
- const configWithoutField = { ...validConfig };
55
- delete (configWithoutField as Record<string, unknown>).{{camelCase this.name}};
56
-
57
- expect(() => {
58
- create{{pascalCase serviceName}}Service(configWithoutField as {{pascalCase serviceName}}Config);
59
- }).toThrow(expect.objectContaining({
60
- code: {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
61
- }));
62
- });
63
-
64
- {{/if}}
65
- {{/each}}
66
- });
67
-
68
- // ─────────────────────────────────────────────────────────────────────────────
69
- // Configuration Tests
70
- // ─────────────────────────────────────────────────────────────────────────────
71
-
72
- describe('getConfig', () => {
73
- let service: {{pascalCase serviceName}}Service;
74
-
75
- beforeEach(() => {
76
- service = create{{pascalCase serviceName}}Service(validConfig);
77
- });
78
-
79
- it('should return configuration without sensitive values', () => {
80
- const config = service.getConfig();
81
-
82
- expect(config).toBeDefined();
83
- expect(typeof config).toBe('object');
84
- });
85
-
86
- it('should not expose sensitive configuration', () => {
87
- const config = service.getConfig();
88
-
89
- // Verify sensitive fields are not exposed
90
- {{#each envVars}}
91
- {{#if this.sensitive}}
92
- expect(config.{{camelCase this.name}}).toBeUndefined();
93
- {{/if}}
94
- {{/each}}
95
- });
96
- });
97
-
98
- // ─────────────────────────────────────────────────────────────────────────────
99
- // Service Method Tests
100
- // ─────────────────────────────────────────────────────────────────────────────
101
-
102
- describe('execute', () => {
103
- let service: {{pascalCase serviceName}}Service;
104
-
105
- beforeEach(() => {
106
- service = create{{pascalCase serviceName}}Service(validConfig);
107
- });
108
-
109
- it('should throw NOT_IMPLEMENTED for unimplemented execute', async () => {
110
- await expect(service.execute()).rejects.toThrow(expect.objectContaining({
111
- code: {{pascalCase serviceName}}ErrorCode.NOT_IMPLEMENTED,
112
- }));
113
- });
114
- });
115
-
116
- describe('healthCheck', () => {
117
- let service: {{pascalCase serviceName}}Service;
118
-
119
- beforeEach(() => {
120
- service = create{{pascalCase serviceName}}Service(validConfig);
121
- });
122
-
123
- it('should return boolean', async () => {
124
- const result = await service.healthCheck();
125
-
126
- expect(typeof result).toBe('boolean');
127
- });
128
- });
129
-
130
- // ─────────────────────────────────────────────────────────────────────────────
131
- // Error Handling Tests
132
- // ─────────────────────────────────────────────────────────────────────────────
133
-
134
- describe('{{pascalCase serviceName}}Error', () => {
135
- it('should have correct name', () => {
136
- const error = new {{pascalCase serviceName}}Error('Test error');
137
-
138
- expect(error.name).toBe('{{pascalCase serviceName}}Error');
139
- });
140
-
141
- it('should have default error code', () => {
142
- const error = new {{pascalCase serviceName}}Error('Test error');
143
-
144
- expect(error.code).toBe({{pascalCase serviceName}}ErrorCode.UNKNOWN_ERROR);
145
- });
146
-
147
- it('should accept custom error code', () => {
148
- const error = new {{pascalCase serviceName}}Error(
149
- 'Config error',
150
- {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR
151
- );
152
-
153
- expect(error.code).toBe({{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR);
154
- });
155
-
156
- it('should include error details', () => {
157
- const details = { field: 'test', reason: 'invalid' };
158
- const error = new {{pascalCase serviceName}}Error(
159
- 'Validation error',
160
- {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
161
- { details }
162
- );
163
-
164
- expect(error.details).toEqual(details);
165
- });
166
-
167
- it('should preserve cause error', () => {
168
- const cause = new Error('Original error');
169
- const error = new {{pascalCase serviceName}}Error(
170
- 'Wrapped error',
171
- {{pascalCase serviceName}}ErrorCode.NETWORK_ERROR,
172
- { cause }
173
- );
174
-
175
- expect(error.cause).toBe(cause);
176
- });
177
-
178
- it('should serialize to JSON correctly', () => {
179
- const error = new {{pascalCase serviceName}}Error(
180
- 'Test error',
181
- {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
182
- { details: { key: 'value' } }
183
- );
184
-
185
- const json = error.toJSON();
186
-
187
- expect(json.name).toBe('{{pascalCase serviceName}}Error');
188
- expect(json.code).toBe({{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR);
189
- expect(json.message).toBe('Test error');
190
- expect(json.details).toEqual({ key: 'value' });
191
- });
192
- });
193
- });
194
-
195
- {{#if isApiIntegration}}
196
- // ─────────────────────────────────────────────────────────────────────────────
197
- // API Client Tests (API Integrations Only)
198
- // ─────────────────────────────────────────────────────────────────────────────
199
-
200
- describe('{{pascalCase serviceName}}Client', () => {
201
- // Import client for API integration testing
202
- const { {{pascalCase serviceName}}Client } = require('../client');
203
-
204
- const clientConfig: {{pascalCase serviceName}}Config = {
205
- ...validConfig,
206
- baseUrl: 'https://api.example.com',
207
- timeout: 5000,
208
- maxRetries: 2,
209
- debug: false,
210
- };
211
-
212
- describe('constructor', () => {
213
- it('should create client with config', () => {
214
- const client = new {{pascalCase serviceName}}Client(clientConfig);
215
-
216
- expect(client).toBeDefined();
217
- });
218
-
219
- it('should use default values for optional config', () => {
220
- const client = new {{pascalCase serviceName}}Client(validConfig);
221
-
222
- expect(client).toBeDefined();
223
- });
224
- });
225
-
226
- describe('getRateLimit', () => {
227
- it('should return null initially', () => {
228
- const client = new {{pascalCase serviceName}}Client(clientConfig);
229
-
230
- expect(client.getRateLimit()).toBeNull();
231
- });
232
- });
233
-
234
- // Note: Additional client tests require mocking fetch/network
235
- // These should be added based on specific API requirements
236
- });
237
- {{/if}}
1
+ /**
2
+ * Tests for {{pascalCase serviceName}} service.
3
+ * @module @aios/{{kebabCase serviceName}}/tests
4
+ * @story {{storyId}}
5
+ */
6
+
7
+ import {
8
+ create{{pascalCase serviceName}}Service,
9
+ {{pascalCase serviceName}}Error,
10
+ {{pascalCase serviceName}}ErrorCode,
11
+ } from '../index';
12
+ import type { {{pascalCase serviceName}}Config, {{pascalCase serviceName}}Service } from '../types';
13
+
14
+ describe('{{pascalCase serviceName}}Service', () => {
15
+ // ─────────────────────────────────────────────────────────────────────────────
16
+ // Test Configuration
17
+ // ─────────────────────────────────────────────────────────────────────────────
18
+
19
+ const validConfig: {{pascalCase serviceName}}Config = {
20
+ {{#each envVars}}
21
+ {{camelCase this.name}}: 'test-{{kebabCase this.name}}',
22
+ {{/each}}
23
+ };
24
+
25
+ // ─────────────────────────────────────────────────────────────────────────────
26
+ // Factory Function Tests
27
+ // ─────────────────────────────────────────────────────────────────────────────
28
+
29
+ describe('create{{pascalCase serviceName}}Service', () => {
30
+ it('should create a service instance with valid config', () => {
31
+ const service = create{{pascalCase serviceName}}Service(validConfig);
32
+
33
+ expect(service).toBeDefined();
34
+ expect(typeof service.execute).toBe('function');
35
+ expect(typeof service.getConfig).toBe('function');
36
+ expect(typeof service.healthCheck).toBe('function');
37
+ });
38
+
39
+ it('should throw on null config', () => {
40
+ expect(() => {
41
+ create{{pascalCase serviceName}}Service(null as unknown as {{pascalCase serviceName}}Config);
42
+ }).toThrow({{pascalCase serviceName}}Error);
43
+ });
44
+
45
+ it('should throw on undefined config', () => {
46
+ expect(() => {
47
+ create{{pascalCase serviceName}}Service(undefined as unknown as {{pascalCase serviceName}}Config);
48
+ }).toThrow({{pascalCase serviceName}}Error);
49
+ });
50
+
51
+ {{#each envVars}}
52
+ {{#if this.required}}
53
+ it('should throw when {{this.name}} is missing', () => {
54
+ const configWithoutField = { ...validConfig };
55
+ delete (configWithoutField as Record<string, unknown>).{{camelCase this.name}};
56
+
57
+ expect(() => {
58
+ create{{pascalCase serviceName}}Service(configWithoutField as {{pascalCase serviceName}}Config);
59
+ }).toThrow(expect.objectContaining({
60
+ code: {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
61
+ }));
62
+ });
63
+
64
+ {{/if}}
65
+ {{/each}}
66
+ });
67
+
68
+ // ─────────────────────────────────────────────────────────────────────────────
69
+ // Configuration Tests
70
+ // ─────────────────────────────────────────────────────────────────────────────
71
+
72
+ describe('getConfig', () => {
73
+ let service: {{pascalCase serviceName}}Service;
74
+
75
+ beforeEach(() => {
76
+ service = create{{pascalCase serviceName}}Service(validConfig);
77
+ });
78
+
79
+ it('should return configuration without sensitive values', () => {
80
+ const config = service.getConfig();
81
+
82
+ expect(config).toBeDefined();
83
+ expect(typeof config).toBe('object');
84
+ });
85
+
86
+ it('should not expose sensitive configuration', () => {
87
+ const config = service.getConfig();
88
+
89
+ // Verify sensitive fields are not exposed
90
+ {{#each envVars}}
91
+ {{#if this.sensitive}}
92
+ expect(config.{{camelCase this.name}}).toBeUndefined();
93
+ {{/if}}
94
+ {{/each}}
95
+ });
96
+ });
97
+
98
+ // ─────────────────────────────────────────────────────────────────────────────
99
+ // Service Method Tests
100
+ // ─────────────────────────────────────────────────────────────────────────────
101
+
102
+ describe('execute', () => {
103
+ let service: {{pascalCase serviceName}}Service;
104
+
105
+ beforeEach(() => {
106
+ service = create{{pascalCase serviceName}}Service(validConfig);
107
+ });
108
+
109
+ it('should throw NOT_IMPLEMENTED for unimplemented execute', async () => {
110
+ await expect(service.execute()).rejects.toThrow(expect.objectContaining({
111
+ code: {{pascalCase serviceName}}ErrorCode.NOT_IMPLEMENTED,
112
+ }));
113
+ });
114
+ });
115
+
116
+ describe('healthCheck', () => {
117
+ let service: {{pascalCase serviceName}}Service;
118
+
119
+ beforeEach(() => {
120
+ service = create{{pascalCase serviceName}}Service(validConfig);
121
+ });
122
+
123
+ it('should return boolean', async () => {
124
+ const result = await service.healthCheck();
125
+
126
+ expect(typeof result).toBe('boolean');
127
+ });
128
+ });
129
+
130
+ // ─────────────────────────────────────────────────────────────────────────────
131
+ // Error Handling Tests
132
+ // ─────────────────────────────────────────────────────────────────────────────
133
+
134
+ describe('{{pascalCase serviceName}}Error', () => {
135
+ it('should have correct name', () => {
136
+ const error = new {{pascalCase serviceName}}Error('Test error');
137
+
138
+ expect(error.name).toBe('{{pascalCase serviceName}}Error');
139
+ });
140
+
141
+ it('should have default error code', () => {
142
+ const error = new {{pascalCase serviceName}}Error('Test error');
143
+
144
+ expect(error.code).toBe({{pascalCase serviceName}}ErrorCode.UNKNOWN_ERROR);
145
+ });
146
+
147
+ it('should accept custom error code', () => {
148
+ const error = new {{pascalCase serviceName}}Error(
149
+ 'Config error',
150
+ {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR
151
+ );
152
+
153
+ expect(error.code).toBe({{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR);
154
+ });
155
+
156
+ it('should include error details', () => {
157
+ const details = { field: 'test', reason: 'invalid' };
158
+ const error = new {{pascalCase serviceName}}Error(
159
+ 'Validation error',
160
+ {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
161
+ { details }
162
+ );
163
+
164
+ expect(error.details).toEqual(details);
165
+ });
166
+
167
+ it('should preserve cause error', () => {
168
+ const cause = new Error('Original error');
169
+ const error = new {{pascalCase serviceName}}Error(
170
+ 'Wrapped error',
171
+ {{pascalCase serviceName}}ErrorCode.NETWORK_ERROR,
172
+ { cause }
173
+ );
174
+
175
+ expect(error.cause).toBe(cause);
176
+ });
177
+
178
+ it('should serialize to JSON correctly', () => {
179
+ const error = new {{pascalCase serviceName}}Error(
180
+ 'Test error',
181
+ {{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR,
182
+ { details: { key: 'value' } }
183
+ );
184
+
185
+ const json = error.toJSON();
186
+
187
+ expect(json.name).toBe('{{pascalCase serviceName}}Error');
188
+ expect(json.code).toBe({{pascalCase serviceName}}ErrorCode.CONFIGURATION_ERROR);
189
+ expect(json.message).toBe('Test error');
190
+ expect(json.details).toEqual({ key: 'value' });
191
+ });
192
+ });
193
+ });
194
+
195
+ {{#if isApiIntegration}}
196
+ // ─────────────────────────────────────────────────────────────────────────────
197
+ // API Client Tests (API Integrations Only)
198
+ // ─────────────────────────────────────────────────────────────────────────────
199
+
200
+ describe('{{pascalCase serviceName}}Client', () => {
201
+ // Import client for API integration testing
202
+ const { {{pascalCase serviceName}}Client } = require('../client');
203
+
204
+ const clientConfig: {{pascalCase serviceName}}Config = {
205
+ ...validConfig,
206
+ baseUrl: 'https://api.example.com',
207
+ timeout: 5000,
208
+ maxRetries: 2,
209
+ debug: false,
210
+ };
211
+
212
+ describe('constructor', () => {
213
+ it('should create client with config', () => {
214
+ const client = new {{pascalCase serviceName}}Client(clientConfig);
215
+
216
+ expect(client).toBeDefined();
217
+ });
218
+
219
+ it('should use default values for optional config', () => {
220
+ const client = new {{pascalCase serviceName}}Client(validConfig);
221
+
222
+ expect(client).toBeDefined();
223
+ });
224
+ });
225
+
226
+ describe('getRateLimit', () => {
227
+ it('should return null initially', () => {
228
+ const client = new {{pascalCase serviceName}}Client(clientConfig);
229
+
230
+ expect(client.getRateLimit()).toBeNull();
231
+ });
232
+ });
233
+
234
+ // Note: Additional client tests require mocking fetch/network
235
+ // These should be added based on specific API requirements
236
+ });
237
+ {{/if}}