@friggframework/devtools 2.0.0--canary.548.c8ae0ca.0 → 2.0.0--canary.545.dba001a.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 (127) hide show
  1. package/frigg-cli/README.md +1 -1
  2. package/frigg-cli/__tests__/application/use-cases/AddApiModuleToIntegrationUseCase.test.js +326 -0
  3. package/frigg-cli/__tests__/application/use-cases/CreateApiModuleUseCase.test.js +337 -0
  4. package/frigg-cli/__tests__/domain/entities/ApiModule.test.js +373 -0
  5. package/frigg-cli/__tests__/domain/entities/AppDefinition.test.js +313 -0
  6. package/frigg-cli/__tests__/domain/services/IntegrationValidator.test.js +269 -0
  7. package/frigg-cli/__tests__/domain/value-objects/IntegrationName.test.js +82 -0
  8. package/frigg-cli/__tests__/infrastructure/adapters/IntegrationJsUpdater.test.js +408 -0
  9. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemApiModuleRepository.test.js +583 -0
  10. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemAppDefinitionRepository.test.js +314 -0
  11. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemIntegrationRepository.test.js +383 -0
  12. package/frigg-cli/__tests__/unit/commands/build.test.js +1 -1
  13. package/frigg-cli/__tests__/unit/commands/doctor.test.js +0 -2
  14. package/frigg-cli/__tests__/unit/commands/init.test.js +406 -0
  15. package/frigg-cli/__tests__/unit/commands/install.test.js +23 -19
  16. package/frigg-cli/__tests__/unit/commands/provider-dispatch.test.js +383 -0
  17. package/frigg-cli/__tests__/unit/commands/repair.test.js +275 -0
  18. package/frigg-cli/__tests__/unit/dependencies.test.js +2 -2
  19. package/frigg-cli/__tests__/unit/start-command/application/RunPreflightChecksUseCase.test.js +411 -0
  20. package/frigg-cli/__tests__/unit/start-command/infrastructure/DatabaseAdapter.test.js +405 -0
  21. package/frigg-cli/__tests__/unit/start-command/infrastructure/DockerAdapter.test.js +496 -0
  22. package/frigg-cli/__tests__/unit/start-command/presentation/InteractivePromptAdapter.test.js +474 -0
  23. package/frigg-cli/__tests__/unit/utils/output.test.js +196 -0
  24. package/frigg-cli/application/use-cases/AddApiModuleToIntegrationUseCase.js +93 -0
  25. package/frigg-cli/application/use-cases/CreateApiModuleUseCase.js +93 -0
  26. package/frigg-cli/application/use-cases/CreateIntegrationUseCase.js +103 -0
  27. package/frigg-cli/build-command/index.js +123 -11
  28. package/frigg-cli/container.js +172 -0
  29. package/frigg-cli/deploy-command/index.js +83 -1
  30. package/frigg-cli/docs/OUTPUT_MIGRATION_GUIDE.md +286 -0
  31. package/frigg-cli/doctor-command/index.js +37 -16
  32. package/frigg-cli/domain/entities/ApiModule.js +272 -0
  33. package/frigg-cli/domain/entities/AppDefinition.js +227 -0
  34. package/frigg-cli/domain/entities/Integration.js +198 -0
  35. package/frigg-cli/domain/exceptions/DomainException.js +24 -0
  36. package/frigg-cli/domain/ports/IApiModuleRepository.js +53 -0
  37. package/frigg-cli/domain/ports/IAppDefinitionRepository.js +43 -0
  38. package/frigg-cli/domain/ports/IIntegrationRepository.js +61 -0
  39. package/frigg-cli/domain/services/IntegrationValidator.js +185 -0
  40. package/frigg-cli/domain/value-objects/IntegrationId.js +42 -0
  41. package/frigg-cli/domain/value-objects/IntegrationName.js +60 -0
  42. package/frigg-cli/domain/value-objects/SemanticVersion.js +70 -0
  43. package/frigg-cli/generate-iam-command.js +21 -1
  44. package/frigg-cli/index.js +21 -6
  45. package/frigg-cli/index.test.js +7 -2
  46. package/frigg-cli/infrastructure/UnitOfWork.js +46 -0
  47. package/frigg-cli/infrastructure/adapters/BackendJsUpdater.js +197 -0
  48. package/frigg-cli/infrastructure/adapters/FileSystemAdapter.js +224 -0
  49. package/frigg-cli/infrastructure/adapters/IntegrationJsUpdater.js +249 -0
  50. package/frigg-cli/infrastructure/adapters/SchemaValidator.js +92 -0
  51. package/frigg-cli/infrastructure/repositories/FileSystemApiModuleRepository.js +373 -0
  52. package/frigg-cli/infrastructure/repositories/FileSystemAppDefinitionRepository.js +116 -0
  53. package/frigg-cli/infrastructure/repositories/FileSystemIntegrationRepository.js +277 -0
  54. package/frigg-cli/init-command/backend-first-handler.js +124 -42
  55. package/frigg-cli/init-command/index.js +2 -1
  56. package/frigg-cli/init-command/template-handler.js +13 -3
  57. package/frigg-cli/install-command/backend-js.js +3 -3
  58. package/frigg-cli/install-command/environment-variables.js +16 -19
  59. package/frigg-cli/install-command/environment-variables.test.js +12 -13
  60. package/frigg-cli/install-command/index.js +14 -9
  61. package/frigg-cli/install-command/integration-file.js +3 -3
  62. package/frigg-cli/install-command/validate-package.js +5 -9
  63. package/frigg-cli/jest.config.js +4 -1
  64. package/frigg-cli/package-lock.json +16226 -0
  65. package/frigg-cli/repair-command/index.js +121 -128
  66. package/frigg-cli/start-command/application/RunPreflightChecksUseCase.js +376 -0
  67. package/frigg-cli/start-command/index.js +324 -2
  68. package/frigg-cli/start-command/infrastructure/DatabaseAdapter.js +591 -0
  69. package/frigg-cli/start-command/infrastructure/DockerAdapter.js +306 -0
  70. package/frigg-cli/start-command/presentation/InteractivePromptAdapter.js +329 -0
  71. package/frigg-cli/templates/backend/.env.example +62 -0
  72. package/frigg-cli/templates/backend/.eslintrc.json +12 -0
  73. package/frigg-cli/templates/backend/.prettierrc +6 -0
  74. package/frigg-cli/templates/backend/docker-compose.yml +22 -0
  75. package/frigg-cli/templates/backend/index.js +96 -0
  76. package/frigg-cli/templates/backend/infrastructure.js +12 -0
  77. package/frigg-cli/templates/backend/jest.config.js +17 -0
  78. package/frigg-cli/templates/backend/package.json +50 -0
  79. package/frigg-cli/templates/backend/src/api-modules/.gitkeep +10 -0
  80. package/frigg-cli/templates/backend/src/base/.gitkeep +7 -0
  81. package/frigg-cli/templates/backend/src/integrations/.gitkeep +10 -0
  82. package/frigg-cli/templates/backend/src/integrations/ExampleIntegration.js +65 -0
  83. package/frigg-cli/templates/backend/src/utils/.gitkeep +7 -0
  84. package/frigg-cli/templates/backend/test/setup.js +30 -0
  85. package/frigg-cli/templates/backend/ui-extensions/.gitkeep +0 -0
  86. package/frigg-cli/templates/backend/ui-extensions/README.md +77 -0
  87. package/frigg-cli/ui-command/index.js +58 -36
  88. package/frigg-cli/utils/__tests__/provider-helper.test.js +55 -0
  89. package/frigg-cli/utils/__tests__/repo-detection.test.js +436 -0
  90. package/frigg-cli/utils/output.js +382 -0
  91. package/frigg-cli/utils/provider-helper.js +75 -0
  92. package/frigg-cli/utils/repo-detection.js +85 -37
  93. package/frigg-cli/validate-command/__tests__/adapters/validate-command.test.js +205 -0
  94. package/frigg-cli/validate-command/__tests__/application/validate-app-use-case.test.js +104 -0
  95. package/frigg-cli/validate-command/__tests__/domain/fix-suggestion.test.js +153 -0
  96. package/frigg-cli/validate-command/__tests__/domain/validation-error.test.js +162 -0
  97. package/frigg-cli/validate-command/__tests__/domain/validation-result.test.js +152 -0
  98. package/frigg-cli/validate-command/__tests__/infrastructure/api-module-validator.test.js +332 -0
  99. package/frigg-cli/validate-command/__tests__/infrastructure/app-definition-validator.test.js +191 -0
  100. package/frigg-cli/validate-command/__tests__/infrastructure/integration-class-validator.test.js +146 -0
  101. package/frigg-cli/validate-command/__tests__/infrastructure/template-validation.test.js +155 -0
  102. package/frigg-cli/validate-command/adapters/cli/validate-command.js +199 -0
  103. package/frigg-cli/validate-command/application/use-cases/validate-app-use-case.js +35 -0
  104. package/frigg-cli/validate-command/domain/entities/validation-result.js +74 -0
  105. package/frigg-cli/validate-command/domain/value-objects/fix-suggestion.js +74 -0
  106. package/frigg-cli/validate-command/domain/value-objects/validation-error.js +68 -0
  107. package/frigg-cli/validate-command/infrastructure/validators/api-module-validator.js +181 -0
  108. package/frigg-cli/validate-command/infrastructure/validators/app-definition-validator.js +128 -0
  109. package/frigg-cli/validate-command/infrastructure/validators/integration-class-validator.js +113 -0
  110. package/infrastructure/create-frigg-infrastructure.js +93 -0
  111. package/infrastructure/docs/iam-policy-templates.md +1 -1
  112. package/infrastructure/domains/admin-scripts/admin-script-builder.js +200 -0
  113. package/infrastructure/domains/admin-scripts/admin-script-builder.test.js +499 -0
  114. package/infrastructure/domains/admin-scripts/index.js +5 -0
  115. package/infrastructure/domains/networking/vpc-builder.test.js +2 -4
  116. package/infrastructure/domains/networking/vpc-resolver.test.js +1 -1
  117. package/infrastructure/domains/shared/resource-discovery.js +5 -5
  118. package/infrastructure/domains/shared/types/app-definition.js +21 -0
  119. package/infrastructure/domains/shared/types/discovery-result.test.js +1 -1
  120. package/infrastructure/domains/shared/utilities/base-definition-factory.js +10 -1
  121. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +2 -2
  122. package/infrastructure/infrastructure-composer.js +2 -0
  123. package/infrastructure/infrastructure-composer.test.js +2 -2
  124. package/infrastructure/jest.config.js +16 -0
  125. package/management-ui/README.md +245 -109
  126. package/package.json +8 -7
  127. package/frigg-cli/install-command/logger.js +0 -12
@@ -0,0 +1,408 @@
1
+ const {IntegrationJsUpdater} = require('../../../infrastructure/adapters/IntegrationJsUpdater');
2
+
3
+ describe('IntegrationJsUpdater', () => {
4
+ let updater;
5
+ let mockFileSystemAdapter;
6
+ let backendPath;
7
+
8
+ beforeEach(() => {
9
+ backendPath = '/test/project/backend';
10
+ mockFileSystemAdapter = {
11
+ exists: jest.fn(),
12
+ updateFile: jest.fn(),
13
+ };
14
+ updater = new IntegrationJsUpdater(mockFileSystemAdapter, backendPath);
15
+ });
16
+
17
+ describe('addModuleToIntegration', () => {
18
+ it('should add a local module with correct import and definition', async () => {
19
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
20
+
21
+ class TestIntegration extends IntegrationBase {
22
+ static Definition = {
23
+ name: 'test',
24
+ modules: {
25
+ // Add your API modules here
26
+ },
27
+ };
28
+ }
29
+
30
+ module.exports = TestIntegration;
31
+ `;
32
+
33
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
34
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
35
+ const result = callback(initialContent);
36
+ return result;
37
+ });
38
+
39
+ await updater.addModuleToIntegration('test-integration', 'salesforce', 'local');
40
+
41
+ expect(mockFileSystemAdapter.updateFile).toHaveBeenCalledWith(
42
+ '/test/project/backend/src/integrations/TestIntegrationIntegration.js',
43
+ expect.any(Function)
44
+ );
45
+
46
+ // Verify the callback produces correct output
47
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
48
+ const result = callback(initialContent);
49
+
50
+ expect(result).toContain("const salesforce = require('../api-modules/salesforce');");
51
+ expect(result).toContain('salesforce: {');
52
+ expect(result).toContain('definition: salesforce.Definition,');
53
+ });
54
+
55
+ it('should add an npm module with correct import path', async () => {
56
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
57
+
58
+ class TestIntegration extends IntegrationBase {
59
+ static Definition = {
60
+ name: 'test',
61
+ modules: {
62
+ },
63
+ };
64
+ }
65
+
66
+ module.exports = TestIntegration;
67
+ `;
68
+
69
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
70
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
71
+ const result = callback(initialContent);
72
+ return result;
73
+ });
74
+
75
+ await updater.addModuleToIntegration('test-integration', 'stripe', 'npm');
76
+
77
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
78
+ const result = callback(initialContent);
79
+
80
+ expect(result).toContain("const stripe = require('@friggframework/api-module-stripe');");
81
+ expect(result).toContain('stripe: {');
82
+ expect(result).toContain('definition: stripe.Definition,');
83
+ });
84
+
85
+ it('should handle kebab-case module names and convert to camelCase', async () => {
86
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
87
+
88
+ class TestIntegration extends IntegrationBase {
89
+ static Definition = {
90
+ name: 'test',
91
+ modules: {
92
+ },
93
+ };
94
+ }
95
+
96
+ module.exports = TestIntegration;
97
+ `;
98
+
99
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
100
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
101
+ const result = callback(initialContent);
102
+ return result;
103
+ });
104
+
105
+ await updater.addModuleToIntegration('test-integration', 'my-api-module', 'local');
106
+
107
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
108
+ const result = callback(initialContent);
109
+
110
+ expect(result).toContain("const myApiModule = require('../api-modules/my-api-module');");
111
+ expect(result).toContain('myApiModule: {');
112
+ expect(result).toContain('definition: myApiModule.Definition,');
113
+ });
114
+
115
+ it('should not add duplicate import if already exists', async () => {
116
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
117
+ const salesforce = require('../api-modules/salesforce');
118
+
119
+ class TestIntegration extends IntegrationBase {
120
+ static Definition = {
121
+ name: 'test',
122
+ modules: {
123
+ salesforce: {
124
+ definition: salesforce.Definition,
125
+ },
126
+ },
127
+ };
128
+ }
129
+
130
+ module.exports = TestIntegration;
131
+ `;
132
+
133
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
134
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
135
+ const result = callback(initialContent);
136
+ return result;
137
+ });
138
+
139
+ await updater.addModuleToIntegration('test-integration', 'salesforce', 'local');
140
+
141
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
142
+ const result = callback(initialContent);
143
+
144
+ // Should not add duplicate
145
+ const importCount = (result.match(/const salesforce = require/g) || []).length;
146
+ expect(importCount).toBe(1);
147
+
148
+ const definitionCount = (result.match(/salesforce: {/g) || []).length;
149
+ expect(definitionCount).toBe(1);
150
+ });
151
+
152
+ it('should throw error if Integration.js does not exist', async () => {
153
+ mockFileSystemAdapter.exists.mockResolvedValue(false);
154
+
155
+ await expect(
156
+ updater.addModuleToIntegration('test-integration', 'salesforce', 'local')
157
+ ).rejects.toThrow('Integration.js not found');
158
+ });
159
+
160
+ it('should insert import after existing requires', async () => {
161
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
162
+ const existingModule = require('../api-modules/existing');
163
+
164
+ class TestIntegration extends IntegrationBase {
165
+ static Definition = {
166
+ modules: {
167
+ existingModule: {
168
+ definition: existingModule.Definition,
169
+ },
170
+ },
171
+ };
172
+ }
173
+
174
+ module.exports = TestIntegration;
175
+ `;
176
+
177
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
178
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
179
+ const result = callback(initialContent);
180
+ return result;
181
+ });
182
+
183
+ await updater.addModuleToIntegration('test-integration', 'salesforce', 'local');
184
+
185
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
186
+ const result = callback(initialContent);
187
+
188
+ const lines = result.split('\n');
189
+ const salesforceImportLine = lines.findIndex(l => l.includes('const salesforce'));
190
+ const classDefLine = lines.findIndex(l => l.includes('class TestIntegration'));
191
+
192
+ expect(salesforceImportLine).toBeGreaterThan(-1);
193
+ expect(salesforceImportLine).toBeLessThan(classDefLine);
194
+ });
195
+ });
196
+
197
+ describe('addModulesToIntegration', () => {
198
+ it('should add multiple modules in single operation', async () => {
199
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
200
+
201
+ class TestIntegration extends IntegrationBase {
202
+ static Definition = {
203
+ name: 'test',
204
+ modules: {
205
+ },
206
+ };
207
+ }
208
+
209
+ module.exports = TestIntegration;
210
+ `;
211
+
212
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
213
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
214
+ const result = callback(initialContent);
215
+ return result;
216
+ });
217
+
218
+ await updater.addModulesToIntegration('test-integration', [
219
+ {name: 'salesforce', source: 'local'},
220
+ {name: 'stripe', source: 'npm'},
221
+ ]);
222
+
223
+ expect(mockFileSystemAdapter.updateFile).toHaveBeenCalledTimes(1);
224
+
225
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
226
+ const result = callback(initialContent);
227
+
228
+ // Verify both modules added
229
+ expect(result).toContain("const salesforce = require('../api-modules/salesforce');");
230
+ expect(result).toContain("const stripe = require('@friggframework/api-module-stripe');");
231
+ expect(result).toContain('salesforce: {');
232
+ expect(result).toContain('stripe: {');
233
+ });
234
+
235
+ it('should handle empty modules array', async () => {
236
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
237
+
238
+ class TestIntegration extends IntegrationBase {
239
+ static Definition = {
240
+ name: 'test',
241
+ modules: {},
242
+ };
243
+ }
244
+
245
+ module.exports = TestIntegration;
246
+ `;
247
+
248
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
249
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
250
+ const result = callback(initialContent);
251
+ return result;
252
+ });
253
+
254
+ await updater.addModulesToIntegration('test-integration', []);
255
+
256
+ expect(mockFileSystemAdapter.updateFile).toHaveBeenCalled();
257
+
258
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
259
+ const result = callback(initialContent);
260
+
261
+ // Content should be unchanged
262
+ expect(result).toBe(initialContent);
263
+ });
264
+
265
+ it('should default source to local if not specified', async () => {
266
+ const initialContent = `const { IntegrationBase } = require('@friggframework/core');
267
+
268
+ class TestIntegration extends IntegrationBase {
269
+ static Definition = {
270
+ modules: {},
271
+ };
272
+ }
273
+
274
+ module.exports = TestIntegration;
275
+ `;
276
+
277
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
278
+ mockFileSystemAdapter.updateFile.mockImplementation(async (path, callback) => {
279
+ const result = callback(initialContent);
280
+ return result;
281
+ });
282
+
283
+ await updater.addModulesToIntegration('test-integration', [
284
+ {name: 'salesforce'}, // No source specified
285
+ ]);
286
+
287
+ const callback = mockFileSystemAdapter.updateFile.mock.calls[0][1];
288
+ const result = callback(initialContent);
289
+
290
+ expect(result).toContain("const salesforce = require('../api-modules/salesforce');");
291
+ });
292
+ });
293
+
294
+ describe('exists', () => {
295
+ it('should check if Integration.js exists', async () => {
296
+ mockFileSystemAdapter.exists.mockResolvedValue(true);
297
+
298
+ const result = await updater.exists('test-integration');
299
+
300
+ expect(result).toBe(true);
301
+ expect(mockFileSystemAdapter.exists).toHaveBeenCalledWith(
302
+ '/test/project/backend/src/integrations/TestIntegrationIntegration.js'
303
+ );
304
+ });
305
+
306
+ it('should return false if Integration.js does not exist', async () => {
307
+ mockFileSystemAdapter.exists.mockResolvedValue(false);
308
+
309
+ const result = await updater.exists('test-integration');
310
+
311
+ expect(result).toBe(false);
312
+ });
313
+ });
314
+
315
+ describe('_toCamelCase', () => {
316
+ it('should convert kebab-case to camelCase', () => {
317
+ expect(updater._toCamelCase('my-api-module')).toBe('myApiModule');
318
+ expect(updater._toCamelCase('salesforce')).toBe('salesforce');
319
+ expect(updater._toCamelCase('stripe-payments')).toBe('stripePayments');
320
+ expect(updater._toCamelCase('my-long-module-name')).toBe('myLongModuleName');
321
+ });
322
+ });
323
+
324
+ describe('_addModuleImport', () => {
325
+ it('should add local import correctly', () => {
326
+ const content = `const { IntegrationBase } = require('@friggframework/core');
327
+
328
+ class Test extends IntegrationBase {}`;
329
+
330
+ const result = updater._addModuleImport(content, 'salesforce', 'local');
331
+
332
+ expect(result).toContain("const salesforce = require('../api-modules/salesforce');");
333
+ });
334
+
335
+ it('should add npm import correctly', () => {
336
+ const content = `const { IntegrationBase } = require('@friggframework/core');
337
+
338
+ class Test extends IntegrationBase {}`;
339
+
340
+ const result = updater._addModuleImport(content, 'stripe', 'npm');
341
+
342
+ expect(result).toContain("const stripe = require('@friggframework/api-module-stripe');");
343
+ });
344
+
345
+ it('should treat git source as local', () => {
346
+ const content = `const { IntegrationBase } = require('@friggframework/core');
347
+
348
+ class Test extends IntegrationBase {}`;
349
+
350
+ const result = updater._addModuleImport(content, 'custom-module', 'git');
351
+
352
+ expect(result).toContain("const customModule = require('../api-modules/custom-module');");
353
+ });
354
+ });
355
+
356
+ describe('_addModuleToDefinition', () => {
357
+ it('should add module to existing modules object', () => {
358
+ const content = `class Test extends IntegrationBase {
359
+ static Definition = {
360
+ name: 'test',
361
+ modules: {
362
+ // Add modules here
363
+ },
364
+ };
365
+ }`;
366
+
367
+ const result = updater._addModuleToDefinition(content, 'salesforce');
368
+
369
+ expect(result).toContain('salesforce: {');
370
+ expect(result).toContain('definition: salesforce.Definition,');
371
+ });
372
+
373
+ it('should not add duplicate module', () => {
374
+ const content = `class Test extends IntegrationBase {
375
+ static Definition = {
376
+ name: 'test',
377
+ modules: {
378
+ salesforce: {
379
+ definition: salesforce.Definition,
380
+ },
381
+ },
382
+ };
383
+ }`;
384
+
385
+ const result = updater._addModuleToDefinition(content, 'salesforce');
386
+
387
+ const occurrences = (result.match(/salesforce: {/g) || []).length;
388
+ expect(occurrences).toBe(1);
389
+ });
390
+
391
+ it('should preserve existing modules when adding new one', () => {
392
+ const content = `class Test extends IntegrationBase {
393
+ static Definition = {
394
+ modules: {
395
+ existing: {
396
+ definition: existing.Definition,
397
+ },
398
+ },
399
+ };
400
+ }`;
401
+
402
+ const result = updater._addModuleToDefinition(content, 'salesforce');
403
+
404
+ expect(result).toContain('existing: {');
405
+ expect(result).toContain('salesforce: {');
406
+ });
407
+ });
408
+ });