@friggframework/devtools 2.0.0--canary.545.ae2019f.0 → 2.0.0--canary.549.a579cca.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__/unit/commands/build.test.js +1 -1
  3. package/frigg-cli/__tests__/unit/commands/doctor.test.js +2 -0
  4. package/frigg-cli/__tests__/unit/commands/install.test.js +19 -23
  5. package/frigg-cli/__tests__/unit/dependencies.test.js +2 -2
  6. package/frigg-cli/build-command/index.js +11 -123
  7. package/frigg-cli/deploy-command/index.js +1 -83
  8. package/frigg-cli/doctor-command/index.js +16 -37
  9. package/frigg-cli/generate-iam-command.js +1 -21
  10. package/frigg-cli/index.js +6 -21
  11. package/frigg-cli/index.test.js +2 -7
  12. package/frigg-cli/init-command/backend-first-handler.js +42 -124
  13. package/frigg-cli/init-command/index.js +1 -2
  14. package/frigg-cli/init-command/template-handler.js +3 -13
  15. package/frigg-cli/install-command/backend-js.js +3 -3
  16. package/frigg-cli/install-command/environment-variables.js +19 -16
  17. package/frigg-cli/install-command/environment-variables.test.js +13 -12
  18. package/frigg-cli/install-command/index.js +9 -14
  19. package/frigg-cli/install-command/integration-file.js +3 -3
  20. package/frigg-cli/install-command/logger.js +12 -0
  21. package/frigg-cli/install-command/validate-package.js +9 -5
  22. package/frigg-cli/jest.config.js +1 -4
  23. package/frigg-cli/repair-command/index.js +128 -121
  24. package/frigg-cli/start-command/index.js +2 -324
  25. package/frigg-cli/ui-command/index.js +36 -58
  26. package/frigg-cli/utils/repo-detection.js +37 -85
  27. package/infrastructure/create-frigg-infrastructure.js +0 -93
  28. package/infrastructure/docs/iam-policy-templates.md +1 -1
  29. package/infrastructure/domains/networking/vpc-builder.test.js +4 -2
  30. package/infrastructure/domains/networking/vpc-resolver.test.js +1 -1
  31. package/infrastructure/domains/shared/resource-discovery.js +5 -5
  32. package/infrastructure/domains/shared/types/app-definition.js +0 -21
  33. package/infrastructure/domains/shared/types/discovery-result.test.js +1 -1
  34. package/infrastructure/domains/shared/utilities/base-definition-factory.js +1 -10
  35. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +2 -2
  36. package/infrastructure/infrastructure-composer.js +0 -2
  37. package/infrastructure/infrastructure-composer.test.js +2 -2
  38. package/management-ui/README.md +109 -245
  39. package/package.json +7 -8
  40. package/frigg-cli/__tests__/application/use-cases/AddApiModuleToIntegrationUseCase.test.js +0 -326
  41. package/frigg-cli/__tests__/application/use-cases/CreateApiModuleUseCase.test.js +0 -337
  42. package/frigg-cli/__tests__/domain/entities/ApiModule.test.js +0 -373
  43. package/frigg-cli/__tests__/domain/entities/AppDefinition.test.js +0 -313
  44. package/frigg-cli/__tests__/domain/services/IntegrationValidator.test.js +0 -269
  45. package/frigg-cli/__tests__/domain/value-objects/IntegrationName.test.js +0 -82
  46. package/frigg-cli/__tests__/infrastructure/adapters/IntegrationJsUpdater.test.js +0 -408
  47. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemApiModuleRepository.test.js +0 -583
  48. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemAppDefinitionRepository.test.js +0 -314
  49. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemIntegrationRepository.test.js +0 -383
  50. package/frigg-cli/__tests__/unit/commands/init.test.js +0 -406
  51. package/frigg-cli/__tests__/unit/commands/provider-dispatch.test.js +0 -383
  52. package/frigg-cli/__tests__/unit/commands/repair.test.js +0 -275
  53. package/frigg-cli/__tests__/unit/start-command/application/RunPreflightChecksUseCase.test.js +0 -411
  54. package/frigg-cli/__tests__/unit/start-command/infrastructure/DatabaseAdapter.test.js +0 -405
  55. package/frigg-cli/__tests__/unit/start-command/infrastructure/DockerAdapter.test.js +0 -496
  56. package/frigg-cli/__tests__/unit/start-command/presentation/InteractivePromptAdapter.test.js +0 -474
  57. package/frigg-cli/__tests__/unit/utils/output.test.js +0 -196
  58. package/frigg-cli/application/use-cases/AddApiModuleToIntegrationUseCase.js +0 -93
  59. package/frigg-cli/application/use-cases/CreateApiModuleUseCase.js +0 -93
  60. package/frigg-cli/application/use-cases/CreateIntegrationUseCase.js +0 -103
  61. package/frigg-cli/container.js +0 -172
  62. package/frigg-cli/docs/OUTPUT_MIGRATION_GUIDE.md +0 -286
  63. package/frigg-cli/domain/entities/ApiModule.js +0 -272
  64. package/frigg-cli/domain/entities/AppDefinition.js +0 -227
  65. package/frigg-cli/domain/entities/Integration.js +0 -198
  66. package/frigg-cli/domain/exceptions/DomainException.js +0 -24
  67. package/frigg-cli/domain/ports/IApiModuleRepository.js +0 -53
  68. package/frigg-cli/domain/ports/IAppDefinitionRepository.js +0 -43
  69. package/frigg-cli/domain/ports/IIntegrationRepository.js +0 -61
  70. package/frigg-cli/domain/services/IntegrationValidator.js +0 -185
  71. package/frigg-cli/domain/value-objects/IntegrationId.js +0 -42
  72. package/frigg-cli/domain/value-objects/IntegrationName.js +0 -60
  73. package/frigg-cli/domain/value-objects/SemanticVersion.js +0 -70
  74. package/frigg-cli/infrastructure/UnitOfWork.js +0 -46
  75. package/frigg-cli/infrastructure/adapters/BackendJsUpdater.js +0 -197
  76. package/frigg-cli/infrastructure/adapters/FileSystemAdapter.js +0 -224
  77. package/frigg-cli/infrastructure/adapters/IntegrationJsUpdater.js +0 -249
  78. package/frigg-cli/infrastructure/adapters/SchemaValidator.js +0 -92
  79. package/frigg-cli/infrastructure/repositories/FileSystemApiModuleRepository.js +0 -373
  80. package/frigg-cli/infrastructure/repositories/FileSystemAppDefinitionRepository.js +0 -116
  81. package/frigg-cli/infrastructure/repositories/FileSystemIntegrationRepository.js +0 -277
  82. package/frigg-cli/package-lock.json +0 -16226
  83. package/frigg-cli/start-command/application/RunPreflightChecksUseCase.js +0 -376
  84. package/frigg-cli/start-command/infrastructure/DatabaseAdapter.js +0 -591
  85. package/frigg-cli/start-command/infrastructure/DockerAdapter.js +0 -306
  86. package/frigg-cli/start-command/presentation/InteractivePromptAdapter.js +0 -329
  87. package/frigg-cli/templates/backend/.env.example +0 -62
  88. package/frigg-cli/templates/backend/.eslintrc.json +0 -12
  89. package/frigg-cli/templates/backend/.prettierrc +0 -6
  90. package/frigg-cli/templates/backend/docker-compose.yml +0 -22
  91. package/frigg-cli/templates/backend/index.js +0 -96
  92. package/frigg-cli/templates/backend/infrastructure.js +0 -12
  93. package/frigg-cli/templates/backend/jest.config.js +0 -17
  94. package/frigg-cli/templates/backend/package.json +0 -50
  95. package/frigg-cli/templates/backend/src/api-modules/.gitkeep +0 -10
  96. package/frigg-cli/templates/backend/src/base/.gitkeep +0 -7
  97. package/frigg-cli/templates/backend/src/integrations/.gitkeep +0 -10
  98. package/frigg-cli/templates/backend/src/integrations/ExampleIntegration.js +0 -65
  99. package/frigg-cli/templates/backend/src/utils/.gitkeep +0 -7
  100. package/frigg-cli/templates/backend/test/setup.js +0 -30
  101. package/frigg-cli/templates/backend/ui-extensions/.gitkeep +0 -0
  102. package/frigg-cli/templates/backend/ui-extensions/README.md +0 -77
  103. package/frigg-cli/utils/__tests__/provider-helper.test.js +0 -55
  104. package/frigg-cli/utils/__tests__/repo-detection.test.js +0 -436
  105. package/frigg-cli/utils/output.js +0 -382
  106. package/frigg-cli/utils/provider-helper.js +0 -75
  107. package/frigg-cli/validate-command/__tests__/adapters/validate-command.test.js +0 -205
  108. package/frigg-cli/validate-command/__tests__/application/validate-app-use-case.test.js +0 -104
  109. package/frigg-cli/validate-command/__tests__/domain/fix-suggestion.test.js +0 -153
  110. package/frigg-cli/validate-command/__tests__/domain/validation-error.test.js +0 -162
  111. package/frigg-cli/validate-command/__tests__/domain/validation-result.test.js +0 -152
  112. package/frigg-cli/validate-command/__tests__/infrastructure/api-module-validator.test.js +0 -332
  113. package/frigg-cli/validate-command/__tests__/infrastructure/app-definition-validator.test.js +0 -191
  114. package/frigg-cli/validate-command/__tests__/infrastructure/integration-class-validator.test.js +0 -146
  115. package/frigg-cli/validate-command/__tests__/infrastructure/template-validation.test.js +0 -155
  116. package/frigg-cli/validate-command/adapters/cli/validate-command.js +0 -199
  117. package/frigg-cli/validate-command/application/use-cases/validate-app-use-case.js +0 -35
  118. package/frigg-cli/validate-command/domain/entities/validation-result.js +0 -74
  119. package/frigg-cli/validate-command/domain/value-objects/fix-suggestion.js +0 -74
  120. package/frigg-cli/validate-command/domain/value-objects/validation-error.js +0 -68
  121. package/frigg-cli/validate-command/infrastructure/validators/api-module-validator.js +0 -181
  122. package/frigg-cli/validate-command/infrastructure/validators/app-definition-validator.js +0 -128
  123. package/frigg-cli/validate-command/infrastructure/validators/integration-class-validator.js +0 -113
  124. package/infrastructure/domains/admin-scripts/admin-script-builder.js +0 -200
  125. package/infrastructure/domains/admin-scripts/admin-script-builder.test.js +0 -499
  126. package/infrastructure/domains/admin-scripts/index.js +0 -5
  127. package/infrastructure/jest.config.js +0 -16
@@ -1,583 +0,0 @@
1
- const {FileSystemApiModuleRepository} = require('../../../infrastructure/repositories/FileSystemApiModuleRepository');
2
- const {ApiModule} = require('../../../domain/entities/ApiModule');
3
-
4
- describe('FileSystemApiModuleRepository', () => {
5
- let repository;
6
- let mockFileSystemAdapter;
7
- let mockSchemaValidator;
8
- let projectRoot;
9
-
10
- beforeEach(() => {
11
- projectRoot = '/test/project';
12
-
13
- mockFileSystemAdapter = {
14
- exists: jest.fn(),
15
- ensureDirectory: jest.fn(),
16
- writeFile: jest.fn(),
17
- readFile: jest.fn(),
18
- listDirectories: jest.fn(),
19
- deleteDirectory: jest.fn(),
20
- };
21
-
22
- mockSchemaValidator = {
23
- validate: jest.fn(),
24
- };
25
-
26
- repository = new FileSystemApiModuleRepository(
27
- mockFileSystemAdapter,
28
- projectRoot,
29
- mockSchemaValidator
30
- );
31
- });
32
-
33
- describe('save', () => {
34
- it('should save an API module with all required files', async () => {
35
- const apiModule = ApiModule.create({
36
- name: 'salesforce',
37
- version: '1.0.0',
38
- displayName: 'Salesforce',
39
- description: 'Salesforce API',
40
- apiConfig: {
41
- authType: 'oauth2',
42
- baseUrl: 'https://api.salesforce.com',
43
- },
44
- });
45
-
46
- await repository.save(apiModule);
47
-
48
- // Verify directories created
49
- expect(mockFileSystemAdapter.ensureDirectory).toHaveBeenCalledWith(
50
- '/test/project/backend/src/api-modules/salesforce'
51
- );
52
- expect(mockFileSystemAdapter.ensureDirectory).toHaveBeenCalledWith(
53
- '/test/project/backend/src/api-modules/salesforce/tests'
54
- );
55
-
56
- // Verify files written (4 files without Entity.js)
57
- expect(mockFileSystemAdapter.writeFile).toHaveBeenCalledTimes(4);
58
- });
59
-
60
- it('should generate Entity.js if module has entities', async () => {
61
- const apiModule = ApiModule.create({
62
- name: 'salesforce',
63
- version: '1.0.0',
64
- displayName: 'Salesforce',
65
- apiConfig: {
66
- authType: 'oauth2',
67
- },
68
- });
69
-
70
- apiModule.addEntity('credential', {
71
- label: 'Credential',
72
- type: 'credential',
73
- required: true,
74
- fields: ['accessToken', 'refreshToken'],
75
- });
76
-
77
- await repository.save(apiModule);
78
-
79
- // Verify Entity.js written (5 files with Entity.js)
80
- expect(mockFileSystemAdapter.writeFile).toHaveBeenCalledTimes(5);
81
-
82
- const entityCall = mockFileSystemAdapter.writeFile.mock.calls.find(
83
- call => call[0].endsWith('Entity.js')
84
- );
85
- expect(entityCall).toBeDefined();
86
- expect(entityCall[1]).toContain('class SalesforceEntity extends EntityBase');
87
- });
88
-
89
- it('should generate Api.js class file correctly', async () => {
90
- const apiModule = ApiModule.create({
91
- name: 'my-test-api',
92
- version: '1.0.0',
93
- displayName: 'My Test API',
94
- description: 'Test API description',
95
- apiConfig: {
96
- authType: 'oauth2',
97
- baseUrl: 'https://api.test.com',
98
- },
99
- });
100
-
101
- await repository.save(apiModule);
102
-
103
- const apiCall = mockFileSystemAdapter.writeFile.mock.calls.find(
104
- call => call[0].endsWith('Api.js')
105
- );
106
-
107
- expect(apiCall).toBeDefined();
108
- expect(apiCall[1]).toContain('class MyTestApiApi extends ApiBase');
109
- expect(apiCall[1]).toContain("this.baseUrl = 'https://api.test.com'");
110
- expect(apiCall[1]).toContain("this.authType = 'oauth2'");
111
- expect(apiCall[1]).toContain('module.exports = MyTestApiApi');
112
- });
113
-
114
- it('should generate definition.js file correctly', async () => {
115
- const apiModule = ApiModule.create({
116
- name: 'salesforce',
117
- version: '1.0.0',
118
- displayName: 'Salesforce',
119
- apiConfig: {
120
- authType: 'oauth2',
121
- },
122
- });
123
-
124
- await repository.save(apiModule);
125
-
126
- const definitionCall = mockFileSystemAdapter.writeFile.mock.calls.find(
127
- call => call[0].endsWith('definition.js')
128
- );
129
-
130
- expect(definitionCall).toBeDefined();
131
- expect(definitionCall[1]).toContain('module.exports = {');
132
- expect(definitionCall[1]).toContain('"name": "salesforce"');
133
- });
134
-
135
- it('should generate config.json file correctly', async () => {
136
- const apiModule = ApiModule.create({
137
- name: 'salesforce',
138
- version: '1.0.0',
139
- displayName: 'Salesforce',
140
- apiConfig: {
141
- authType: 'oauth2',
142
- },
143
- });
144
-
145
- await repository.save(apiModule);
146
-
147
- const configCall = mockFileSystemAdapter.writeFile.mock.calls.find(
148
- call => call[0].endsWith('config.json')
149
- );
150
-
151
- expect(configCall).toBeDefined();
152
- const config = JSON.parse(configCall[1]);
153
- expect(config.name).toBe('salesforce');
154
- expect(config.version).toBe('1.0.0');
155
- expect(config.authType).toBe('oauth2');
156
- });
157
-
158
- it('should generate README.md file correctly', async () => {
159
- const apiModule = ApiModule.create({
160
- name: 'salesforce',
161
- version: '1.0.0',
162
- displayName: 'Salesforce',
163
- description: 'Salesforce API client',
164
- apiConfig: {
165
- authType: 'oauth2',
166
- baseUrl: 'https://api.salesforce.com',
167
- },
168
- });
169
-
170
- await repository.save(apiModule);
171
-
172
- const readmeCall = mockFileSystemAdapter.writeFile.mock.calls.find(
173
- call => call[0].endsWith('README.md')
174
- );
175
-
176
- expect(readmeCall).toBeDefined();
177
- expect(readmeCall[1]).toContain('# Salesforce');
178
- expect(readmeCall[1]).toContain('Salesforce API client');
179
- expect(readmeCall[1]).toContain('https://api.salesforce.com');
180
- });
181
-
182
- it('should throw error if API module validation fails', async () => {
183
- const apiModule = ApiModule.create({
184
- name: 'test-api',
185
- version: '1.0.0',
186
- displayName: 'Test API',
187
- apiConfig: {
188
- authType: 'oauth2',
189
- },
190
- });
191
-
192
- // Mock validate to return errors
193
- jest.spyOn(apiModule, 'validate').mockReturnValue({
194
- isValid: false,
195
- errors: ['Invalid configuration'],
196
- });
197
-
198
- await expect(repository.save(apiModule)).rejects.toThrow(
199
- 'ApiModule validation failed'
200
- );
201
- });
202
-
203
- it('should handle endpoints in Api.js generation', async () => {
204
- const apiModule = ApiModule.create({
205
- name: 'salesforce',
206
- version: '1.0.0',
207
- displayName: 'Salesforce',
208
- apiConfig: {
209
- authType: 'oauth2',
210
- },
211
- });
212
-
213
- apiModule.addEndpoint('getUser', {
214
- method: 'GET',
215
- path: '/user',
216
- description: 'Get user information',
217
- });
218
-
219
- await repository.save(apiModule);
220
-
221
- const apiCall = mockFileSystemAdapter.writeFile.mock.calls.find(
222
- call => call[0].endsWith('Api.js')
223
- );
224
-
225
- expect(apiCall[1]).toContain('async getUser()');
226
- expect(apiCall[1]).toContain('return await this.get(\'/user\')');
227
- });
228
-
229
- it('should handle OAuth scopes in README', async () => {
230
- const apiModule = ApiModule.create({
231
- name: 'salesforce',
232
- version: '1.0.0',
233
- displayName: 'Salesforce',
234
- apiConfig: {
235
- authType: 'oauth2',
236
- },
237
- });
238
-
239
- apiModule.addScope('read:users');
240
- apiModule.addScope('write:users');
241
-
242
- await repository.save(apiModule);
243
-
244
- const readmeCall = mockFileSystemAdapter.writeFile.mock.calls.find(
245
- call => call[0].endsWith('README.md')
246
- );
247
-
248
- expect(readmeCall[1]).toContain('read:users');
249
- expect(readmeCall[1]).toContain('write:users');
250
- });
251
-
252
- it('should handle credentials in README', async () => {
253
- const apiModule = ApiModule.create({
254
- name: 'salesforce',
255
- version: '1.0.0',
256
- displayName: 'Salesforce',
257
- apiConfig: {
258
- authType: 'oauth2',
259
- },
260
- });
261
-
262
- apiModule.addCredential('clientId', {
263
- type: 'string',
264
- description: 'OAuth Client ID',
265
- required: true,
266
- });
267
-
268
- await repository.save(apiModule);
269
-
270
- const readmeCall = mockFileSystemAdapter.writeFile.mock.calls.find(
271
- call => call[0].endsWith('README.md')
272
- );
273
-
274
- expect(readmeCall[1]).toContain('clientId');
275
- expect(readmeCall[1]).toContain('OAuth Client ID');
276
- expect(readmeCall[1]).toContain('(Required)');
277
- });
278
- });
279
-
280
- describe('findByName', () => {
281
- it.skip('should find API module by name (TODO: needs full implementation)', async () => {
282
- // Skip this test because findByName is a simple implementation that
283
- // calls ApiModule.create({name}) which requires apiConfig.
284
- // Full implementation would parse the definition.js file.
285
- mockFileSystemAdapter.exists.mockResolvedValue(true);
286
- mockFileSystemAdapter.readFile.mockResolvedValue('module.exports = {}');
287
-
288
- const result = await repository.findByName('salesforce');
289
-
290
- expect(result).toBeInstanceOf(ApiModule);
291
- expect(result.name).toBe('salesforce');
292
- });
293
-
294
- it('should return null if module directory does not exist', async () => {
295
- mockFileSystemAdapter.exists.mockResolvedValueOnce(false);
296
-
297
- const result = await repository.findByName('nonexistent');
298
-
299
- expect(result).toBeNull();
300
- });
301
-
302
- it('should return null if definition file does not exist', async () => {
303
- mockFileSystemAdapter.exists
304
- .mockResolvedValueOnce(true) // Directory exists
305
- .mockResolvedValueOnce(false); // Definition file doesn't exist
306
-
307
- const result = await repository.findByName('salesforce');
308
-
309
- expect(result).toBeNull();
310
- });
311
- });
312
-
313
- describe('exists', () => {
314
- it('should return true if API module exists', async () => {
315
- mockFileSystemAdapter.exists.mockResolvedValue(true);
316
-
317
- const result = await repository.exists('salesforce');
318
-
319
- expect(result).toBe(true);
320
- expect(mockFileSystemAdapter.exists).toHaveBeenCalledWith(
321
- '/test/project/backend/src/api-modules/salesforce'
322
- );
323
- });
324
-
325
- it('should return false if API module does not exist', async () => {
326
- mockFileSystemAdapter.exists.mockResolvedValue(false);
327
-
328
- const result = await repository.exists('nonexistent');
329
-
330
- expect(result).toBe(false);
331
- });
332
- });
333
-
334
- describe('list', () => {
335
- it('should return empty array if api-modules directory does not exist', async () => {
336
- mockFileSystemAdapter.exists.mockResolvedValue(false);
337
-
338
- const result = await repository.list();
339
-
340
- expect(result).toEqual([]);
341
- });
342
-
343
- it.skip('should list all API modules (TODO: needs full findByName implementation)', async () => {
344
- // Skip because list() uses findByName() which needs full implementation
345
- mockFileSystemAdapter.exists.mockResolvedValue(true);
346
- mockFileSystemAdapter.listDirectories.mockResolvedValue([
347
- 'salesforce',
348
- 'stripe',
349
- ]);
350
- mockFileSystemAdapter.readFile.mockResolvedValue('module.exports = {}');
351
-
352
- const result = await repository.list();
353
-
354
- expect(result).toHaveLength(2);
355
- expect(result[0]).toBeInstanceOf(ApiModule);
356
- expect(result[0].name).toBe('salesforce');
357
- expect(result[1].name).toBe('stripe');
358
- });
359
-
360
- it('should skip invalid modules and log warning', async () => {
361
- const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation();
362
-
363
- mockFileSystemAdapter.exists
364
- .mockResolvedValueOnce(true) // Directory exists
365
- .mockResolvedValueOnce(true) // salesforce dir
366
- .mockResolvedValueOnce(true) // salesforce definition
367
- .mockResolvedValueOnce(false); // invalid dir (doesn't exist)
368
-
369
- mockFileSystemAdapter.listDirectories.mockResolvedValue([
370
- 'salesforce',
371
- 'invalid',
372
- ]);
373
- mockFileSystemAdapter.readFile.mockResolvedValue('module.exports = {}');
374
-
375
- const result = await repository.list();
376
-
377
- // Result will be empty because findByName throws errors
378
- expect(result).toEqual([]);
379
- expect(consoleWarnSpy).toHaveBeenCalledWith(
380
- expect.stringContaining('Failed to load API module'),
381
- expect.any(String)
382
- );
383
-
384
- consoleWarnSpy.mockRestore();
385
- });
386
- });
387
-
388
- describe('delete', () => {
389
- it('should delete API module if it exists', async () => {
390
- mockFileSystemAdapter.exists.mockResolvedValue(true);
391
-
392
- const result = await repository.delete('salesforce');
393
-
394
- expect(result).toBe(true);
395
- expect(mockFileSystemAdapter.deleteDirectory).toHaveBeenCalledWith(
396
- '/test/project/backend/src/api-modules/salesforce'
397
- );
398
- });
399
-
400
- it('should return false if API module does not exist', async () => {
401
- mockFileSystemAdapter.exists.mockResolvedValue(false);
402
-
403
- const result = await repository.delete('nonexistent');
404
-
405
- expect(result).toBe(false);
406
- expect(mockFileSystemAdapter.deleteDirectory).not.toHaveBeenCalled();
407
- });
408
- });
409
-
410
- describe('_generateApiClass', () => {
411
- it('should generate API class with OAuth methods', () => {
412
- const apiModule = ApiModule.create({
413
- name: 'salesforce',
414
- version: '1.0.0',
415
- displayName: 'Salesforce',
416
- apiConfig: {
417
- authType: 'oauth2',
418
- baseUrl: 'https://api.salesforce.com',
419
- },
420
- });
421
-
422
- const result = repository._generateApiClass(apiModule);
423
-
424
- expect(result).toContain('class SalesforceApi extends ApiBase');
425
- expect(result).toContain('async getAuthorizationUri()');
426
- expect(result).toContain('async getTokenFromCode(code)');
427
- expect(result).toContain('async setCredential(credential)');
428
- expect(result).toContain('async testAuth()');
429
- });
430
-
431
- it('should handle kebab-case module names', () => {
432
- const apiModule = ApiModule.create({
433
- name: 'my-api-module',
434
- version: '1.0.0',
435
- displayName: 'My API Module',
436
- apiConfig: {
437
- authType: 'api-key',
438
- },
439
- });
440
-
441
- const result = repository._generateApiClass(apiModule);
442
-
443
- expect(result).toContain('class MyApiModuleApi extends ApiBase');
444
- expect(result).toContain('module.exports = MyApiModuleApi');
445
- });
446
-
447
- it('should include credential parameter if entity exists', () => {
448
- const apiModule = ApiModule.create({
449
- name: 'salesforce',
450
- version: '1.0.0',
451
- displayName: 'Salesforce',
452
- apiConfig: {
453
- authType: 'oauth2',
454
- },
455
- });
456
-
457
- apiModule.addEntity('credential', {
458
- label: 'Credential',
459
- type: 'credential',
460
- required: true,
461
- });
462
-
463
- const result = repository._generateApiClass(apiModule);
464
-
465
- expect(result).toContain('this.credential = params.credential');
466
- });
467
- });
468
-
469
- describe('_generateEndpointMethods', () => {
470
- it('should generate methods for each endpoint', () => {
471
- const apiModule = ApiModule.create({
472
- name: 'salesforce',
473
- version: '1.0.0',
474
- displayName: 'Salesforce',
475
- apiConfig: {
476
- authType: 'oauth2',
477
- },
478
- });
479
-
480
- apiModule.addEndpoint('getUser', {
481
- method: 'GET',
482
- path: '/user',
483
- description: 'Get user information',
484
- });
485
-
486
- apiModule.addEndpoint('createContact', {
487
- method: 'POST',
488
- path: '/contacts',
489
- description: 'Create a contact',
490
- parameters: [{name: 'data'}],
491
- });
492
-
493
- const result = repository._generateEndpointMethods(apiModule);
494
-
495
- expect(result).toContain('async getUser()');
496
- expect(result).toContain("return await this.get('/user')");
497
- expect(result).toContain('async createContact(data)');
498
- expect(result).toContain("return await this.post('/contacts', {data})");
499
- });
500
-
501
- it('should return empty string if no endpoints', () => {
502
- const apiModule = ApiModule.create({
503
- name: 'salesforce',
504
- version: '1.0.0',
505
- displayName: 'Salesforce',
506
- apiConfig: {
507
- authType: 'oauth2',
508
- },
509
- });
510
-
511
- const result = repository._generateEndpointMethods(apiModule);
512
-
513
- expect(result).toBe('');
514
- });
515
- });
516
-
517
- describe('_generateEntityClass', () => {
518
- it('should generate Entity class correctly', () => {
519
- const apiModule = ApiModule.create({
520
- name: 'salesforce',
521
- version: '1.0.0',
522
- displayName: 'Salesforce',
523
- apiConfig: {
524
- authType: 'oauth2',
525
- },
526
- });
527
-
528
- apiModule.addEntity('credential', {
529
- label: 'Credential',
530
- type: 'credential',
531
- required: true,
532
- fields: ['accessToken', 'refreshToken'],
533
- });
534
-
535
- const result = repository._generateEntityClass(apiModule);
536
-
537
- expect(result).toContain('class SalesforceEntity extends EntityBase');
538
- expect(result).toContain("return 'credential'");
539
- expect(result).toContain('accessToken');
540
- expect(result).toContain('refreshToken');
541
- expect(result).toContain('module.exports = SalesforceEntity');
542
- });
543
- });
544
-
545
- describe('_generateReadme', () => {
546
- it('should generate comprehensive README', () => {
547
- const apiModule = ApiModule.create({
548
- name: 'salesforce',
549
- version: '1.0.0',
550
- displayName: 'Salesforce',
551
- description: 'Salesforce API client',
552
- apiConfig: {
553
- authType: 'oauth2',
554
- baseUrl: 'https://api.salesforce.com',
555
- },
556
- });
557
-
558
- apiModule.addScope('read:users');
559
- apiModule.addCredential('clientId', {
560
- type: 'string',
561
- description: 'OAuth Client ID',
562
- required: true,
563
- });
564
- apiModule.addEntity('credential', {
565
- label: 'Credential',
566
- type: 'credential',
567
- required: true,
568
- });
569
-
570
- const result = repository._generateReadme(apiModule);
571
-
572
- expect(result).toContain('# Salesforce');
573
- expect(result).toContain('Salesforce API client');
574
- expect(result).toContain('https://api.salesforce.com');
575
- expect(result).toContain('oauth2');
576
- expect(result).toContain('read:users');
577
- expect(result).toContain('clientId');
578
- expect(result).toContain('OAuth Client ID');
579
- expect(result).toContain('## Usage');
580
- expect(result).toContain('## Development');
581
- });
582
- });
583
- });