@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,430 +0,0 @@
1
- const {FileSystemIntegrationRepository} = require('../../../infrastructure/repositories/FileSystemIntegrationRepository');
2
- const {Integration} = require('../../../domain/entities/Integration');
3
- const {IntegrationName} = require('../../../domain/value-objects/IntegrationName');
4
-
5
- describe('FileSystemIntegrationRepository', () => {
6
- let repository;
7
- let mockFileSystemAdapter;
8
- let mockSchemaValidator;
9
- let backendPath;
10
-
11
- beforeEach(() => {
12
- backendPath = '/test/project/backend';
13
-
14
- mockFileSystemAdapter = {
15
- exists: jest.fn(),
16
- ensureDirectory: jest.fn(),
17
- writeFile: jest.fn(),
18
- readFile: jest.fn(),
19
- listFiles: jest.fn(),
20
- };
21
-
22
- mockSchemaValidator = {
23
- validate: jest.fn(),
24
- };
25
-
26
- repository = new FileSystemIntegrationRepository(
27
- mockFileSystemAdapter,
28
- backendPath,
29
- mockSchemaValidator
30
- );
31
- });
32
-
33
- describe('save', () => {
34
- it('should save a new integration as a single file', async () => {
35
- const integration = new Integration({
36
- name: 'test-integration',
37
- version: '1.0.0',
38
- displayName: 'Test Integration',
39
- description: 'Test description',
40
- type: 'sync',
41
- category: 'CRM',
42
- });
43
-
44
- mockSchemaValidator.validate.mockResolvedValue({valid: true, errors: []});
45
- mockFileSystemAdapter.exists.mockResolvedValue(false); // Integration.js doesn't exist yet
46
-
47
- await repository.save(integration);
48
-
49
- // Verify integrations directory created
50
- expect(mockFileSystemAdapter.ensureDirectory).toHaveBeenCalledWith(
51
- '/test/project/backend/src/integrations'
52
- );
53
-
54
- // Verify schema validation
55
- expect(mockSchemaValidator.validate).toHaveBeenCalledWith(
56
- 'integration-definition',
57
- expect.any(Object)
58
- );
59
-
60
- // Verify only Integration.js written
61
- expect(mockFileSystemAdapter.writeFile).toHaveBeenCalledTimes(1);
62
-
63
- // Verify Integration.js content
64
- const writeCall = mockFileSystemAdapter.writeFile.mock.calls[0];
65
- expect(writeCall[0]).toBe('/test/project/backend/src/integrations/TestIntegrationIntegration.js');
66
- expect(writeCall[1]).toContain('class TestIntegrationIntegration extends IntegrationBase');
67
- expect(writeCall[1]).toContain('static Definition = {');
68
- expect(writeCall[1]).toContain("name: 'test-integration'");
69
- });
70
-
71
- it('should NOT write Integration.js if it already exists', async () => {
72
- const integration = new Integration({
73
- name: 'test-integration',
74
- version: '1.0.0',
75
- displayName: 'Test Integration',
76
- description: 'Test description',
77
- });
78
-
79
- mockSchemaValidator.validate.mockResolvedValue({valid: true, errors: []});
80
- mockFileSystemAdapter.exists.mockResolvedValue(true); // Integration.js already exists
81
-
82
- await repository.save(integration);
83
-
84
- // Verify Integration.js NOT written
85
- expect(mockFileSystemAdapter.writeFile).not.toHaveBeenCalled();
86
- });
87
-
88
- it('should throw error if integration is invalid', async () => {
89
- // Create invalid integration (invalid type)
90
- const integration = new Integration({
91
- name: 'test-integration',
92
- version: '1.0.0',
93
- displayName: 'Test Integration',
94
- type: 'invalid-type',
95
- });
96
-
97
- await expect(repository.save(integration)).rejects.toThrow('Invalid integration');
98
- });
99
-
100
- it('should throw error if schema validation fails', async () => {
101
- const integration = new Integration({
102
- name: 'test-integration',
103
- version: '1.0.0',
104
- displayName: 'Test Integration',
105
- });
106
-
107
- mockSchemaValidator.validate.mockResolvedValue({
108
- valid: false,
109
- errors: ['Invalid schema'],
110
- });
111
-
112
- await expect(repository.save(integration)).rejects.toThrow('Schema validation failed');
113
- });
114
-
115
- it('should handle kebab-case to PascalCase conversion correctly', async () => {
116
- const integration = new Integration({
117
- name: 'my-awesome-api',
118
- version: '1.0.0',
119
- displayName: 'My Awesome API',
120
- });
121
-
122
- mockSchemaValidator.validate.mockResolvedValue({valid: true, errors: []});
123
- mockFileSystemAdapter.exists.mockResolvedValue(false);
124
-
125
- await repository.save(integration);
126
-
127
- const writeCall = mockFileSystemAdapter.writeFile.mock.calls[0];
128
- expect(writeCall[0]).toBe('/test/project/backend/src/integrations/MyAwesomeApiIntegration.js');
129
- expect(writeCall[1]).toContain('class MyAwesomeApiIntegration extends IntegrationBase');
130
- });
131
- });
132
-
133
- describe('findByName', () => {
134
- it('should find integration by name string', async () => {
135
- const integrationJsContent = `
136
- class TestIntegrationIntegration extends IntegrationBase {
137
- static Definition = {
138
- name: 'test-integration',
139
- version: '1.0.0',
140
- display: {
141
- label: 'Test Integration',
142
- description: 'Test description',
143
- },
144
- };
145
- }
146
- module.exports = TestIntegrationIntegration;
147
- `;
148
-
149
- mockFileSystemAdapter.exists.mockResolvedValue(true);
150
- mockFileSystemAdapter.readFile.mockResolvedValue(integrationJsContent);
151
-
152
- const result = await repository.findByName('test-integration');
153
-
154
- expect(result).toBeInstanceOf(Integration);
155
- expect(result.name.value).toBe('test-integration');
156
- expect(result.version.value).toBe('1.0.0');
157
- });
158
-
159
- it('should find integration by IntegrationName value object', async () => {
160
- const integrationJsContent = `
161
- class TestIntegrationIntegration extends IntegrationBase {
162
- static Definition = {
163
- name: 'test-integration',
164
- version: '1.0.0',
165
- display: {},
166
- };
167
- }
168
- `;
169
-
170
- mockFileSystemAdapter.exists.mockResolvedValue(true);
171
- mockFileSystemAdapter.readFile.mockResolvedValue(integrationJsContent);
172
-
173
- const name = new IntegrationName('test-integration');
174
- const result = await repository.findByName(name);
175
-
176
- expect(result).toBeInstanceOf(Integration);
177
- expect(result.name.value).toBe('test-integration');
178
- });
179
-
180
- it('should return null if integration file does not exist', async () => {
181
- mockFileSystemAdapter.exists.mockResolvedValue(false);
182
-
183
- const result = await repository.findByName('nonexistent');
184
-
185
- expect(result).toBeNull();
186
- });
187
- });
188
-
189
- describe('exists', () => {
190
- it('should return true if integration exists', async () => {
191
- mockFileSystemAdapter.exists.mockResolvedValue(true);
192
-
193
- const result = await repository.exists('test-integration');
194
-
195
- expect(result).toBe(true);
196
- expect(mockFileSystemAdapter.exists).toHaveBeenCalledWith(
197
- '/test/project/backend/src/integrations/TestIntegrationIntegration.js'
198
- );
199
- });
200
-
201
- it('should return false if integration does not exist', async () => {
202
- mockFileSystemAdapter.exists.mockResolvedValue(false);
203
-
204
- const result = await repository.exists('nonexistent');
205
-
206
- expect(result).toBe(false);
207
- });
208
-
209
- it('should work with IntegrationName value object', async () => {
210
- mockFileSystemAdapter.exists.mockResolvedValue(true);
211
-
212
- const name = new IntegrationName('test-integration');
213
- const result = await repository.exists(name);
214
-
215
- expect(result).toBe(true);
216
- });
217
- });
218
-
219
- describe('list', () => {
220
- it('should return empty array if integrations directory does not exist', async () => {
221
- mockFileSystemAdapter.exists.mockResolvedValue(false);
222
-
223
- const result = await repository.list();
224
-
225
- expect(result).toEqual([]);
226
- });
227
-
228
- it('should return list of all integrations', async () => {
229
- mockFileSystemAdapter.exists.mockResolvedValue(true);
230
- mockFileSystemAdapter.listFiles.mockResolvedValue([
231
- 'Integration1Integration.js',
232
- 'Integration2Integration.js',
233
- ]);
234
-
235
- const integration1Content = `
236
- class Integration1Integration extends IntegrationBase {
237
- static Definition = {
238
- name: 'integration-1',
239
- version: '1.0.0',
240
- display: {},
241
- };
242
- }
243
- `;
244
-
245
- const integration2Content = `
246
- class Integration2Integration extends IntegrationBase {
247
- static Definition = {
248
- name: 'integration-2',
249
- version: '2.0.0',
250
- display: {},
251
- };
252
- }
253
- `;
254
-
255
- mockFileSystemAdapter.readFile
256
- .mockResolvedValueOnce(integration1Content)
257
- .mockResolvedValueOnce(integration2Content);
258
-
259
- const result = await repository.list();
260
-
261
- expect(result).toHaveLength(2);
262
- expect(result[0]).toBeInstanceOf(Integration);
263
- expect(result[0].name.value).toBe('integration-1');
264
- expect(result[1].name.value).toBe('integration-2');
265
- });
266
-
267
- it('should skip invalid integrations and log warning', async () => {
268
- const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation();
269
-
270
- mockFileSystemAdapter.exists.mockResolvedValue(true);
271
- mockFileSystemAdapter.listFiles.mockResolvedValue([
272
- 'ValidIntegration.js',
273
- 'InvalidIntegration.js',
274
- ]);
275
-
276
- const validContent = `
277
- class ValidIntegration extends IntegrationBase {
278
- static Definition = {
279
- name: 'valid-integration',
280
- version: '1.0.0',
281
- display: {},
282
- };
283
- }
284
- `;
285
-
286
- mockFileSystemAdapter.readFile
287
- .mockResolvedValueOnce(validContent)
288
- .mockRejectedValueOnce(new Error('Read error'));
289
-
290
- const result = await repository.list();
291
-
292
- expect(result).toHaveLength(1);
293
- expect(result[0].name.value).toBe('valid-integration');
294
- expect(consoleWarnSpy).toHaveBeenCalledWith(
295
- expect.stringContaining('Failed to load integration'),
296
- expect.any(String)
297
- );
298
-
299
- consoleWarnSpy.mockRestore();
300
- });
301
-
302
- it('should filter out non-Integration files', async () => {
303
- mockFileSystemAdapter.exists.mockResolvedValue(true);
304
- mockFileSystemAdapter.listFiles.mockResolvedValue([
305
- 'TestIntegration.js',
306
- 'helper.js',
307
- 'utils.js',
308
- ]);
309
-
310
- const integrationContent = `
311
- class TestIntegration extends IntegrationBase {
312
- static Definition = {
313
- name: 'test',
314
- version: '1.0.0',
315
- display: {},
316
- };
317
- }
318
- `;
319
-
320
- mockFileSystemAdapter.readFile.mockResolvedValue(integrationContent);
321
-
322
- const result = await repository.list();
323
-
324
- // Should only process TestIntegration.js (ends with Integration.js)
325
- expect(mockFileSystemAdapter.readFile).toHaveBeenCalledTimes(1);
326
- expect(result).toHaveLength(1);
327
- });
328
- });
329
-
330
- describe('_generateIntegrationClass', () => {
331
- it('should generate valid Integration.js class file', () => {
332
- const integration = new Integration({
333
- name: 'my-test-integration',
334
- version: '1.0.0',
335
- displayName: 'My Test Integration',
336
- description: 'Test description',
337
- category: 'CRM',
338
- });
339
-
340
- const result = repository._generateIntegrationClass(integration);
341
-
342
- expect(result).toContain("const { IntegrationBase } = require('@friggframework/core');");
343
- expect(result).toContain('class MyTestIntegrationIntegration extends IntegrationBase');
344
- expect(result).toContain('static Definition = {');
345
- expect(result).toContain("name: 'my-test-integration'");
346
- expect(result).toContain("version: '1.0.0'");
347
- expect(result).toContain('modules: {');
348
- expect(result).toContain('routes: [');
349
- expect(result).toContain('module.exports = MyTestIntegrationIntegration');
350
- });
351
-
352
- it('should handle single-word integration names', () => {
353
- const integration = new Integration({
354
- name: 'salesforce',
355
- version: '1.0.0',
356
- displayName: 'Salesforce',
357
- description: 'Salesforce integration',
358
- });
359
-
360
- const result = repository._generateIntegrationClass(integration);
361
-
362
- expect(result).toContain('class SalesforceIntegration extends IntegrationBase');
363
- expect(result).toContain('module.exports = SalesforceIntegration');
364
- });
365
-
366
- it('should include proper JSDoc comments', () => {
367
- const integration = new Integration({
368
- name: 'test-integration',
369
- version: '1.0.0',
370
- displayName: 'Test Integration',
371
- description: 'Test description',
372
- });
373
-
374
- const result = repository._generateIntegrationClass(integration);
375
-
376
- expect(result).toContain('/**');
377
- expect(result).toContain('* Test Integration');
378
- expect(result).toContain('* Test description');
379
- expect(result).toContain('*/');
380
- });
381
- });
382
-
383
- describe('_parseStaticDefinition', () => {
384
- it('should parse static Definition from Integration.js content', () => {
385
- const content = `
386
- class TestIntegration extends IntegrationBase {
387
- static Definition = {
388
- name: 'test',
389
- version: '1.0.0',
390
- display: {
391
- label: 'Test',
392
- description: 'Test integration',
393
- },
394
- };
395
- }
396
- `;
397
-
398
- const result = repository._parseStaticDefinition(content);
399
-
400
- expect(result.name).toBe('test');
401
- expect(result.version).toBe('1.0.0');
402
- expect(result.display.label).toBe('Test');
403
- });
404
-
405
- it('should handle multi-line definition objects', () => {
406
- const content = `
407
- class ComplexIntegration extends IntegrationBase {
408
- static Definition = {
409
- name: 'complex',
410
- version: '2.0.0',
411
- modules: {
412
- module1: { definition: module1.Definition },
413
- module2: { definition: module2.Definition },
414
- },
415
- routes: [
416
- { path: '/auth', method: 'GET' },
417
- ],
418
- };
419
- }
420
- `;
421
-
422
- const result = repository._parseStaticDefinition(content);
423
-
424
- expect(result.name).toBe('complex');
425
- expect(result.version).toBe('2.0.0');
426
- expect(result.modules).toBeDefined();
427
- expect(result.routes).toBeDefined();
428
- });
429
- });
430
- });