@friggframework/devtools 2.0.0--canary.522.cbd3d5a.0 → 2.0.0--canary.517.21b69ac.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 (247) hide show
  1. package/.eslintrc.json +3 -0
  2. package/CHANGELOG.md +132 -0
  3. package/frigg-cli/README.md +1 -1
  4. package/frigg-cli/__tests__/unit/commands/doctor.test.js +2 -0
  5. package/frigg-cli/__tests__/unit/commands/install.test.js +17 -21
  6. package/frigg-cli/doctor-command/index.js +16 -17
  7. package/frigg-cli/index.js +6 -21
  8. package/frigg-cli/index.test.js +1 -7
  9. package/frigg-cli/init-command/backend-first-handler.js +42 -124
  10. package/frigg-cli/init-command/index.js +1 -2
  11. package/frigg-cli/init-command/template-handler.js +3 -13
  12. package/frigg-cli/install-command/backend-js.js +3 -3
  13. package/frigg-cli/install-command/environment-variables.js +19 -16
  14. package/frigg-cli/install-command/environment-variables.test.js +13 -12
  15. package/frigg-cli/install-command/index.js +9 -14
  16. package/frigg-cli/install-command/integration-file.js +3 -3
  17. package/frigg-cli/install-command/logger.js +12 -0
  18. package/frigg-cli/install-command/validate-package.js +9 -5
  19. package/frigg-cli/jest.config.js +1 -4
  20. package/frigg-cli/repair-command/index.js +128 -101
  21. package/frigg-cli/start-command/index.js +2 -246
  22. package/frigg-cli/ui-command/index.js +36 -58
  23. package/frigg-cli/utils/repo-detection.js +37 -85
  24. package/infrastructure/docs/iam-policy-templates.md +1 -1
  25. package/infrastructure/domains/networking/vpc-builder.test.js +4 -2
  26. package/infrastructure/domains/networking/vpc-resolver.test.js +1 -1
  27. package/infrastructure/domains/shared/cloudformation-discovery.test.js +7 -4
  28. package/infrastructure/domains/shared/resource-discovery.js +5 -5
  29. package/infrastructure/domains/shared/types/discovery-result.test.js +1 -1
  30. package/infrastructure/domains/shared/utilities/base-definition-factory.js +2 -25
  31. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +2 -2
  32. package/infrastructure/infrastructure-composer.test.js +2 -2
  33. package/layers/prisma/.build-complete +3 -0
  34. package/layers/prisma/nodejs/package.json +8 -0
  35. package/management-ui/.eslintrc.js +22 -0
  36. package/management-ui/README.md +109 -245
  37. package/management-ui/components.json +21 -0
  38. package/management-ui/docs/phase2-integration-guide.md +320 -0
  39. package/management-ui/index.html +13 -0
  40. package/management-ui/package.json +76 -0
  41. package/management-ui/packages/devtools/frigg-cli/ui-command/index.js +302 -0
  42. package/management-ui/postcss.config.js +6 -0
  43. package/management-ui/server/api/backend.js +256 -0
  44. package/management-ui/server/api/cli.js +315 -0
  45. package/management-ui/server/api/codegen.js +663 -0
  46. package/management-ui/server/api/connections.js +857 -0
  47. package/management-ui/server/api/discovery.js +185 -0
  48. package/management-ui/server/api/environment/index.js +1 -0
  49. package/management-ui/server/api/environment/router.js +378 -0
  50. package/management-ui/server/api/environment.js +328 -0
  51. package/management-ui/server/api/integrations.js +876 -0
  52. package/management-ui/server/api/logs.js +248 -0
  53. package/management-ui/server/api/monitoring.js +282 -0
  54. package/management-ui/server/api/open-ide.js +31 -0
  55. package/management-ui/server/api/project.js +1029 -0
  56. package/management-ui/server/api/users/sessions.js +371 -0
  57. package/management-ui/server/api/users/simulation.js +254 -0
  58. package/management-ui/server/api/users.js +362 -0
  59. package/management-ui/server/api-contract.md +275 -0
  60. package/management-ui/server/index.js +873 -0
  61. package/management-ui/server/middleware/errorHandler.js +93 -0
  62. package/management-ui/server/middleware/security.js +32 -0
  63. package/management-ui/server/processManager.js +296 -0
  64. package/management-ui/server/server.js +346 -0
  65. package/management-ui/server/services/aws-monitor.js +413 -0
  66. package/management-ui/server/services/npm-registry.js +347 -0
  67. package/management-ui/server/services/template-engine.js +538 -0
  68. package/management-ui/server/utils/cliIntegration.js +220 -0
  69. package/management-ui/server/utils/environment/auditLogger.js +471 -0
  70. package/management-ui/server/utils/environment/awsParameterStore.js +275 -0
  71. package/management-ui/server/utils/environment/encryption.js +278 -0
  72. package/management-ui/server/utils/environment/envFileManager.js +286 -0
  73. package/management-ui/server/utils/import-commonjs.js +28 -0
  74. package/management-ui/server/utils/response.js +83 -0
  75. package/management-ui/server/websocket/handler.js +325 -0
  76. package/management-ui/src/App.jsx +25 -0
  77. package/management-ui/src/assets/FriggLogo.svg +1 -0
  78. package/management-ui/src/components/AppRouter.jsx +65 -0
  79. package/management-ui/src/components/Button.jsx +70 -0
  80. package/management-ui/src/components/Card.jsx +97 -0
  81. package/management-ui/src/components/EnvironmentCompare.jsx +400 -0
  82. package/management-ui/src/components/EnvironmentEditor.jsx +372 -0
  83. package/management-ui/src/components/EnvironmentImportExport.jsx +469 -0
  84. package/management-ui/src/components/EnvironmentSchema.jsx +491 -0
  85. package/management-ui/src/components/EnvironmentSecurity.jsx +463 -0
  86. package/management-ui/src/components/ErrorBoundary.jsx +73 -0
  87. package/management-ui/src/components/IntegrationCard.jsx +481 -0
  88. package/management-ui/src/components/IntegrationCardEnhanced.jsx +770 -0
  89. package/management-ui/src/components/IntegrationExplorer.jsx +379 -0
  90. package/management-ui/src/components/IntegrationStatus.jsx +336 -0
  91. package/management-ui/src/components/Layout.jsx +716 -0
  92. package/management-ui/src/components/LoadingSpinner.jsx +113 -0
  93. package/management-ui/src/components/RepositoryPicker.jsx +248 -0
  94. package/management-ui/src/components/SessionMonitor.jsx +350 -0
  95. package/management-ui/src/components/StatusBadge.jsx +208 -0
  96. package/management-ui/src/components/UserContextSwitcher.jsx +212 -0
  97. package/management-ui/src/components/UserSimulation.jsx +327 -0
  98. package/management-ui/src/components/Welcome.jsx +434 -0
  99. package/management-ui/src/components/codegen/APIEndpointGenerator.jsx +637 -0
  100. package/management-ui/src/components/codegen/APIModuleSelector.jsx +227 -0
  101. package/management-ui/src/components/codegen/CodeGenerationWizard.jsx +247 -0
  102. package/management-ui/src/components/codegen/CodePreviewEditor.jsx +316 -0
  103. package/management-ui/src/components/codegen/DynamicModuleForm.jsx +271 -0
  104. package/management-ui/src/components/codegen/FormBuilder.jsx +737 -0
  105. package/management-ui/src/components/codegen/IntegrationGenerator.jsx +855 -0
  106. package/management-ui/src/components/codegen/ProjectScaffoldWizard.jsx +797 -0
  107. package/management-ui/src/components/codegen/SchemaBuilder.jsx +303 -0
  108. package/management-ui/src/components/codegen/TemplateSelector.jsx +586 -0
  109. package/management-ui/src/components/codegen/index.js +10 -0
  110. package/management-ui/src/components/connections/ConnectionConfigForm.jsx +362 -0
  111. package/management-ui/src/components/connections/ConnectionHealthMonitor.jsx +182 -0
  112. package/management-ui/src/components/connections/ConnectionTester.jsx +200 -0
  113. package/management-ui/src/components/connections/EntityRelationshipMapper.jsx +292 -0
  114. package/management-ui/src/components/connections/OAuthFlow.jsx +204 -0
  115. package/management-ui/src/components/connections/index.js +5 -0
  116. package/management-ui/src/components/index.js +21 -0
  117. package/management-ui/src/components/monitoring/APIGatewayMetrics.jsx +222 -0
  118. package/management-ui/src/components/monitoring/LambdaMetrics.jsx +169 -0
  119. package/management-ui/src/components/monitoring/MetricsChart.jsx +197 -0
  120. package/management-ui/src/components/monitoring/MonitoringDashboard.jsx +393 -0
  121. package/management-ui/src/components/monitoring/SQSMetrics.jsx +246 -0
  122. package/management-ui/src/components/monitoring/index.js +6 -0
  123. package/management-ui/src/components/monitoring/monitoring.css +218 -0
  124. package/management-ui/src/components/theme-provider.jsx +52 -0
  125. package/management-ui/src/components/theme-toggle.jsx +39 -0
  126. package/management-ui/src/components/ui/badge.tsx +36 -0
  127. package/management-ui/src/components/ui/button.test.jsx +56 -0
  128. package/management-ui/src/components/ui/button.tsx +57 -0
  129. package/management-ui/src/components/ui/card.tsx +76 -0
  130. package/management-ui/src/components/ui/dropdown-menu.tsx +199 -0
  131. package/management-ui/src/components/ui/select.tsx +157 -0
  132. package/management-ui/src/components/ui/skeleton.jsx +15 -0
  133. package/management-ui/src/hooks/useFrigg.jsx +387 -0
  134. package/management-ui/src/hooks/useSocket.jsx +58 -0
  135. package/management-ui/src/index.css +193 -0
  136. package/management-ui/src/lib/utils.ts +6 -0
  137. package/management-ui/src/main.jsx +10 -0
  138. package/management-ui/src/pages/CodeGeneration.jsx +14 -0
  139. package/management-ui/src/pages/Connections.jsx +252 -0
  140. package/management-ui/src/pages/ConnectionsEnhanced.jsx +633 -0
  141. package/management-ui/src/pages/Dashboard.jsx +311 -0
  142. package/management-ui/src/pages/Environment.jsx +314 -0
  143. package/management-ui/src/pages/IntegrationConfigure.jsx +669 -0
  144. package/management-ui/src/pages/IntegrationDiscovery.jsx +567 -0
  145. package/management-ui/src/pages/IntegrationTest.jsx +742 -0
  146. package/management-ui/src/pages/Integrations.jsx +253 -0
  147. package/management-ui/src/pages/Monitoring.jsx +17 -0
  148. package/management-ui/src/pages/Simulation.jsx +155 -0
  149. package/management-ui/src/pages/Users.jsx +492 -0
  150. package/management-ui/src/services/api.js +41 -0
  151. package/management-ui/src/services/apiModuleService.js +193 -0
  152. package/management-ui/src/services/websocket-handlers.js +120 -0
  153. package/management-ui/src/test/api/project.test.js +273 -0
  154. package/management-ui/src/test/components/Welcome.test.jsx +378 -0
  155. package/management-ui/src/test/mocks/server.js +178 -0
  156. package/management-ui/src/test/setup.js +61 -0
  157. package/management-ui/src/test/utils/test-utils.jsx +134 -0
  158. package/management-ui/src/utils/repository.js +98 -0
  159. package/management-ui/src/utils/repository.test.js +118 -0
  160. package/management-ui/src/workflows/phase2-integration-workflows.js +884 -0
  161. package/management-ui/tailwind.config.js +63 -0
  162. package/management-ui/tsconfig.json +37 -0
  163. package/management-ui/tsconfig.node.json +10 -0
  164. package/management-ui/vite.config.js +26 -0
  165. package/management-ui/vitest.config.js +38 -0
  166. package/package.json +7 -17
  167. package/frigg-cli/__tests__/application/use-cases/AddApiModuleToIntegrationUseCase.test.js +0 -326
  168. package/frigg-cli/__tests__/application/use-cases/CreateApiModuleUseCase.test.js +0 -337
  169. package/frigg-cli/__tests__/domain/entities/ApiModule.test.js +0 -373
  170. package/frigg-cli/__tests__/domain/entities/AppDefinition.test.js +0 -313
  171. package/frigg-cli/__tests__/domain/services/IntegrationValidator.test.js +0 -269
  172. package/frigg-cli/__tests__/domain/value-objects/IntegrationName.test.js +0 -82
  173. package/frigg-cli/__tests__/infrastructure/adapters/IntegrationJsUpdater.test.js +0 -408
  174. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemApiModuleRepository.test.js +0 -583
  175. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemAppDefinitionRepository.test.js +0 -314
  176. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemIntegrationRepository.test.js +0 -430
  177. package/frigg-cli/__tests__/unit/commands/init.test.js +0 -406
  178. package/frigg-cli/__tests__/unit/commands/repair.test.js +0 -275
  179. package/frigg-cli/__tests__/unit/start-command/application/RunPreflightChecksUseCase.test.js +0 -411
  180. package/frigg-cli/__tests__/unit/start-command/infrastructure/DatabaseAdapter.test.js +0 -405
  181. package/frigg-cli/__tests__/unit/start-command/infrastructure/DockerAdapter.test.js +0 -496
  182. package/frigg-cli/__tests__/unit/start-command/presentation/InteractivePromptAdapter.test.js +0 -474
  183. package/frigg-cli/__tests__/unit/utils/output.test.js +0 -196
  184. package/frigg-cli/application/use-cases/AddApiModuleToIntegrationUseCase.js +0 -93
  185. package/frigg-cli/application/use-cases/CreateApiModuleUseCase.js +0 -93
  186. package/frigg-cli/application/use-cases/CreateIntegrationUseCase.js +0 -103
  187. package/frigg-cli/container.js +0 -172
  188. package/frigg-cli/docs/OUTPUT_MIGRATION_GUIDE.md +0 -286
  189. package/frigg-cli/domain/entities/ApiModule.js +0 -272
  190. package/frigg-cli/domain/entities/AppDefinition.js +0 -227
  191. package/frigg-cli/domain/entities/Integration.js +0 -198
  192. package/frigg-cli/domain/exceptions/DomainException.js +0 -24
  193. package/frigg-cli/domain/ports/IApiModuleRepository.js +0 -53
  194. package/frigg-cli/domain/ports/IAppDefinitionRepository.js +0 -43
  195. package/frigg-cli/domain/ports/IIntegrationRepository.js +0 -61
  196. package/frigg-cli/domain/services/IntegrationValidator.js +0 -185
  197. package/frigg-cli/domain/value-objects/IntegrationId.js +0 -42
  198. package/frigg-cli/domain/value-objects/IntegrationName.js +0 -60
  199. package/frigg-cli/domain/value-objects/SemanticVersion.js +0 -70
  200. package/frigg-cli/infrastructure/UnitOfWork.js +0 -46
  201. package/frigg-cli/infrastructure/adapters/BackendJsUpdater.js +0 -197
  202. package/frigg-cli/infrastructure/adapters/FileSystemAdapter.js +0 -224
  203. package/frigg-cli/infrastructure/adapters/IntegrationJsUpdater.js +0 -249
  204. package/frigg-cli/infrastructure/adapters/SchemaValidator.js +0 -92
  205. package/frigg-cli/infrastructure/repositories/FileSystemApiModuleRepository.js +0 -373
  206. package/frigg-cli/infrastructure/repositories/FileSystemAppDefinitionRepository.js +0 -116
  207. package/frigg-cli/infrastructure/repositories/FileSystemIntegrationRepository.js +0 -277
  208. package/frigg-cli/package-lock.json +0 -16226
  209. package/frigg-cli/start-command/application/RunPreflightChecksUseCase.js +0 -376
  210. package/frigg-cli/start-command/infrastructure/DatabaseAdapter.js +0 -591
  211. package/frigg-cli/start-command/infrastructure/DockerAdapter.js +0 -306
  212. package/frigg-cli/start-command/presentation/InteractivePromptAdapter.js +0 -329
  213. package/frigg-cli/templates/backend/.env.example +0 -62
  214. package/frigg-cli/templates/backend/.eslintrc.json +0 -12
  215. package/frigg-cli/templates/backend/.prettierrc +0 -6
  216. package/frigg-cli/templates/backend/docker-compose.yml +0 -22
  217. package/frigg-cli/templates/backend/index.js +0 -96
  218. package/frigg-cli/templates/backend/infrastructure.js +0 -12
  219. package/frigg-cli/templates/backend/jest.config.js +0 -17
  220. package/frigg-cli/templates/backend/package.json +0 -50
  221. package/frigg-cli/templates/backend/src/api-modules/.gitkeep +0 -10
  222. package/frigg-cli/templates/backend/src/base/.gitkeep +0 -7
  223. package/frigg-cli/templates/backend/src/integrations/.gitkeep +0 -10
  224. package/frigg-cli/templates/backend/src/integrations/ExampleIntegration.js +0 -65
  225. package/frigg-cli/templates/backend/src/utils/.gitkeep +0 -7
  226. package/frigg-cli/templates/backend/test/setup.js +0 -30
  227. package/frigg-cli/templates/backend/ui-extensions/.gitkeep +0 -0
  228. package/frigg-cli/templates/backend/ui-extensions/README.md +0 -77
  229. package/frigg-cli/utils/__tests__/repo-detection.test.js +0 -436
  230. package/frigg-cli/utils/output.js +0 -382
  231. package/frigg-cli/validate-command/__tests__/adapters/validate-command.test.js +0 -205
  232. package/frigg-cli/validate-command/__tests__/application/validate-app-use-case.test.js +0 -104
  233. package/frigg-cli/validate-command/__tests__/domain/fix-suggestion.test.js +0 -153
  234. package/frigg-cli/validate-command/__tests__/domain/validation-error.test.js +0 -162
  235. package/frigg-cli/validate-command/__tests__/domain/validation-result.test.js +0 -152
  236. package/frigg-cli/validate-command/__tests__/infrastructure/api-module-validator.test.js +0 -332
  237. package/frigg-cli/validate-command/__tests__/infrastructure/app-definition-validator.test.js +0 -191
  238. package/frigg-cli/validate-command/__tests__/infrastructure/integration-class-validator.test.js +0 -146
  239. package/frigg-cli/validate-command/__tests__/infrastructure/template-validation.test.js +0 -155
  240. package/frigg-cli/validate-command/adapters/cli/validate-command.js +0 -199
  241. package/frigg-cli/validate-command/application/use-cases/validate-app-use-case.js +0 -35
  242. package/frigg-cli/validate-command/domain/entities/validation-result.js +0 -74
  243. package/frigg-cli/validate-command/domain/value-objects/fix-suggestion.js +0 -74
  244. package/frigg-cli/validate-command/domain/value-objects/validation-error.js +0 -68
  245. package/frigg-cli/validate-command/infrastructure/validators/api-module-validator.js +0 -181
  246. package/frigg-cli/validate-command/infrastructure/validators/app-definition-validator.js +0 -128
  247. package/frigg-cli/validate-command/infrastructure/validators/integration-class-validator.js +0 -113
@@ -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
- });