@friggframework/devtools 2.0.0-next.4 → 2.0.0-next.41

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 (198) hide show
  1. package/frigg-cli/.eslintrc.js +141 -0
  2. package/frigg-cli/__tests__/jest.config.js +102 -0
  3. package/frigg-cli/__tests__/unit/commands/build.test.js +483 -0
  4. package/frigg-cli/__tests__/unit/commands/install.test.js +418 -0
  5. package/frigg-cli/__tests__/unit/commands/ui.test.js +592 -0
  6. package/frigg-cli/__tests__/utils/command-tester.js +170 -0
  7. package/frigg-cli/__tests__/utils/mock-factory.js +270 -0
  8. package/frigg-cli/__tests__/utils/test-fixtures.js +463 -0
  9. package/frigg-cli/__tests__/utils/test-setup.js +286 -0
  10. package/frigg-cli/build-command/index.js +54 -0
  11. package/frigg-cli/deploy-command/index.js +175 -0
  12. package/frigg-cli/generate-command/__tests__/generate-command.test.js +312 -0
  13. package/frigg-cli/generate-command/azure-generator.js +43 -0
  14. package/frigg-cli/generate-command/gcp-generator.js +47 -0
  15. package/frigg-cli/generate-command/index.js +332 -0
  16. package/frigg-cli/generate-command/terraform-generator.js +555 -0
  17. package/frigg-cli/generate-iam-command.js +115 -0
  18. package/frigg-cli/index.js +47 -1
  19. package/frigg-cli/index.test.js +1 -4
  20. package/frigg-cli/init-command/backend-first-handler.js +756 -0
  21. package/frigg-cli/init-command/index.js +93 -0
  22. package/frigg-cli/init-command/template-handler.js +143 -0
  23. package/frigg-cli/install-command/index.js +1 -4
  24. package/frigg-cli/package.json +51 -0
  25. package/frigg-cli/start-command/index.js +30 -4
  26. package/frigg-cli/start-command/start-command.test.js +155 -0
  27. package/frigg-cli/test/init-command.test.js +180 -0
  28. package/frigg-cli/test/npm-registry.test.js +319 -0
  29. package/frigg-cli/ui-command/index.js +154 -0
  30. package/frigg-cli/utils/app-resolver.js +319 -0
  31. package/frigg-cli/utils/backend-path.js +16 -17
  32. package/frigg-cli/utils/npm-registry.js +167 -0
  33. package/frigg-cli/utils/process-manager.js +199 -0
  34. package/frigg-cli/utils/repo-detection.js +405 -0
  35. package/infrastructure/DEPLOYMENT-INSTRUCTIONS.md +268 -0
  36. package/infrastructure/GENERATE-IAM-DOCS.md +278 -0
  37. package/infrastructure/IAM-POLICY-TEMPLATES.md +176 -0
  38. package/infrastructure/README.md +443 -0
  39. package/infrastructure/WEBSOCKET-CONFIGURATION.md +105 -0
  40. package/infrastructure/__tests__/fixtures/mock-aws-resources.js +391 -0
  41. package/infrastructure/__tests__/helpers/test-utils.js +277 -0
  42. package/infrastructure/aws-discovery.js +1176 -0
  43. package/infrastructure/aws-discovery.test.js +1220 -0
  44. package/infrastructure/build-time-discovery.js +206 -0
  45. package/infrastructure/build-time-discovery.test.js +378 -0
  46. package/infrastructure/create-frigg-infrastructure.js +3 -5
  47. package/infrastructure/env-validator.js +77 -0
  48. package/infrastructure/frigg-deployment-iam-stack.yaml +401 -0
  49. package/infrastructure/iam-generator.js +836 -0
  50. package/infrastructure/iam-generator.test.js +172 -0
  51. package/infrastructure/iam-policy-basic.json +218 -0
  52. package/infrastructure/iam-policy-full.json +288 -0
  53. package/infrastructure/integration.test.js +383 -0
  54. package/infrastructure/run-discovery.js +110 -0
  55. package/infrastructure/serverless-template.js +1493 -138
  56. package/infrastructure/serverless-template.test.js +1804 -0
  57. package/management-ui/.eslintrc.js +22 -0
  58. package/management-ui/README.md +203 -0
  59. package/management-ui/components.json +21 -0
  60. package/management-ui/docs/phase2-integration-guide.md +320 -0
  61. package/management-ui/index.html +13 -0
  62. package/management-ui/package-lock.json +16517 -0
  63. package/management-ui/package.json +76 -0
  64. package/management-ui/packages/devtools/frigg-cli/ui-command/index.js +302 -0
  65. package/management-ui/postcss.config.js +6 -0
  66. package/management-ui/server/api/backend.js +256 -0
  67. package/management-ui/server/api/cli.js +315 -0
  68. package/management-ui/server/api/codegen.js +663 -0
  69. package/management-ui/server/api/connections.js +857 -0
  70. package/management-ui/server/api/discovery.js +185 -0
  71. package/management-ui/server/api/environment/index.js +1 -0
  72. package/management-ui/server/api/environment/router.js +378 -0
  73. package/management-ui/server/api/environment.js +328 -0
  74. package/management-ui/server/api/integrations.js +876 -0
  75. package/management-ui/server/api/logs.js +248 -0
  76. package/management-ui/server/api/monitoring.js +282 -0
  77. package/management-ui/server/api/open-ide.js +31 -0
  78. package/management-ui/server/api/project.js +1029 -0
  79. package/management-ui/server/api/users/sessions.js +371 -0
  80. package/management-ui/server/api/users/simulation.js +254 -0
  81. package/management-ui/server/api/users.js +362 -0
  82. package/management-ui/server/api-contract.md +275 -0
  83. package/management-ui/server/index.js +873 -0
  84. package/management-ui/server/middleware/errorHandler.js +93 -0
  85. package/management-ui/server/middleware/security.js +32 -0
  86. package/management-ui/server/processManager.js +296 -0
  87. package/management-ui/server/server.js +346 -0
  88. package/management-ui/server/services/aws-monitor.js +413 -0
  89. package/management-ui/server/services/npm-registry.js +347 -0
  90. package/management-ui/server/services/template-engine.js +538 -0
  91. package/management-ui/server/utils/cliIntegration.js +220 -0
  92. package/management-ui/server/utils/environment/auditLogger.js +471 -0
  93. package/management-ui/server/utils/environment/awsParameterStore.js +264 -0
  94. package/management-ui/server/utils/environment/encryption.js +278 -0
  95. package/management-ui/server/utils/environment/envFileManager.js +286 -0
  96. package/management-ui/server/utils/import-commonjs.js +28 -0
  97. package/management-ui/server/utils/response.js +83 -0
  98. package/management-ui/server/websocket/handler.js +325 -0
  99. package/management-ui/src/App.jsx +109 -0
  100. package/management-ui/src/assets/FriggLogo.svg +1 -0
  101. package/management-ui/src/components/AppRouter.jsx +65 -0
  102. package/management-ui/src/components/Button.jsx +70 -0
  103. package/management-ui/src/components/Card.jsx +97 -0
  104. package/management-ui/src/components/EnvironmentCompare.jsx +400 -0
  105. package/management-ui/src/components/EnvironmentEditor.jsx +372 -0
  106. package/management-ui/src/components/EnvironmentImportExport.jsx +469 -0
  107. package/management-ui/src/components/EnvironmentSchema.jsx +491 -0
  108. package/management-ui/src/components/EnvironmentSecurity.jsx +463 -0
  109. package/management-ui/src/components/ErrorBoundary.jsx +73 -0
  110. package/management-ui/src/components/IntegrationCard.jsx +481 -0
  111. package/management-ui/src/components/IntegrationCardEnhanced.jsx +770 -0
  112. package/management-ui/src/components/IntegrationExplorer.jsx +379 -0
  113. package/management-ui/src/components/IntegrationStatus.jsx +336 -0
  114. package/management-ui/src/components/Layout.jsx +716 -0
  115. package/management-ui/src/components/LoadingSpinner.jsx +113 -0
  116. package/management-ui/src/components/RepositoryPicker.jsx +248 -0
  117. package/management-ui/src/components/SessionMonitor.jsx +350 -0
  118. package/management-ui/src/components/StatusBadge.jsx +208 -0
  119. package/management-ui/src/components/UserContextSwitcher.jsx +212 -0
  120. package/management-ui/src/components/UserSimulation.jsx +327 -0
  121. package/management-ui/src/components/Welcome.jsx +434 -0
  122. package/management-ui/src/components/codegen/APIEndpointGenerator.jsx +637 -0
  123. package/management-ui/src/components/codegen/APIModuleSelector.jsx +227 -0
  124. package/management-ui/src/components/codegen/CodeGenerationWizard.jsx +247 -0
  125. package/management-ui/src/components/codegen/CodePreviewEditor.jsx +316 -0
  126. package/management-ui/src/components/codegen/DynamicModuleForm.jsx +271 -0
  127. package/management-ui/src/components/codegen/FormBuilder.jsx +737 -0
  128. package/management-ui/src/components/codegen/IntegrationGenerator.jsx +855 -0
  129. package/management-ui/src/components/codegen/ProjectScaffoldWizard.jsx +797 -0
  130. package/management-ui/src/components/codegen/SchemaBuilder.jsx +303 -0
  131. package/management-ui/src/components/codegen/TemplateSelector.jsx +586 -0
  132. package/management-ui/src/components/codegen/index.js +10 -0
  133. package/management-ui/src/components/connections/ConnectionConfigForm.jsx +362 -0
  134. package/management-ui/src/components/connections/ConnectionHealthMonitor.jsx +182 -0
  135. package/management-ui/src/components/connections/ConnectionTester.jsx +200 -0
  136. package/management-ui/src/components/connections/EntityRelationshipMapper.jsx +292 -0
  137. package/management-ui/src/components/connections/OAuthFlow.jsx +204 -0
  138. package/management-ui/src/components/connections/index.js +5 -0
  139. package/management-ui/src/components/index.js +21 -0
  140. package/management-ui/src/components/monitoring/APIGatewayMetrics.jsx +222 -0
  141. package/management-ui/src/components/monitoring/LambdaMetrics.jsx +169 -0
  142. package/management-ui/src/components/monitoring/MetricsChart.jsx +197 -0
  143. package/management-ui/src/components/monitoring/MonitoringDashboard.jsx +393 -0
  144. package/management-ui/src/components/monitoring/SQSMetrics.jsx +246 -0
  145. package/management-ui/src/components/monitoring/index.js +6 -0
  146. package/management-ui/src/components/monitoring/monitoring.css +218 -0
  147. package/management-ui/src/components/theme-provider.jsx +52 -0
  148. package/management-ui/src/components/theme-toggle.jsx +39 -0
  149. package/management-ui/src/components/ui/badge.tsx +36 -0
  150. package/management-ui/src/components/ui/button.test.jsx +56 -0
  151. package/management-ui/src/components/ui/button.tsx +57 -0
  152. package/management-ui/src/components/ui/card.tsx +76 -0
  153. package/management-ui/src/components/ui/dropdown-menu.tsx +199 -0
  154. package/management-ui/src/components/ui/select.tsx +157 -0
  155. package/management-ui/src/components/ui/skeleton.jsx +15 -0
  156. package/management-ui/src/hooks/useFrigg.jsx +601 -0
  157. package/management-ui/src/hooks/useSocket.jsx +58 -0
  158. package/management-ui/src/index.css +193 -0
  159. package/management-ui/src/lib/utils.ts +6 -0
  160. package/management-ui/src/main.jsx +10 -0
  161. package/management-ui/src/pages/CodeGeneration.jsx +14 -0
  162. package/management-ui/src/pages/Connections.jsx +252 -0
  163. package/management-ui/src/pages/ConnectionsEnhanced.jsx +633 -0
  164. package/management-ui/src/pages/Dashboard.jsx +311 -0
  165. package/management-ui/src/pages/Environment.jsx +314 -0
  166. package/management-ui/src/pages/IntegrationConfigure.jsx +669 -0
  167. package/management-ui/src/pages/IntegrationDiscovery.jsx +567 -0
  168. package/management-ui/src/pages/IntegrationTest.jsx +742 -0
  169. package/management-ui/src/pages/Integrations.jsx +253 -0
  170. package/management-ui/src/pages/Monitoring.jsx +17 -0
  171. package/management-ui/src/pages/Simulation.jsx +155 -0
  172. package/management-ui/src/pages/Users.jsx +492 -0
  173. package/management-ui/src/services/api.js +41 -0
  174. package/management-ui/src/services/apiModuleService.js +193 -0
  175. package/management-ui/src/services/websocket-handlers.js +120 -0
  176. package/management-ui/src/test/api/project.test.js +273 -0
  177. package/management-ui/src/test/components/Welcome.test.jsx +378 -0
  178. package/management-ui/src/test/mocks/server.js +178 -0
  179. package/management-ui/src/test/setup.js +61 -0
  180. package/management-ui/src/test/utils/test-utils.jsx +134 -0
  181. package/management-ui/src/utils/repository.js +98 -0
  182. package/management-ui/src/utils/repository.test.js +118 -0
  183. package/management-ui/src/workflows/phase2-integration-workflows.js +884 -0
  184. package/management-ui/tailwind.config.js +63 -0
  185. package/management-ui/tsconfig.json +37 -0
  186. package/management-ui/tsconfig.node.json +10 -0
  187. package/management-ui/vite.config.js +26 -0
  188. package/management-ui/vitest.config.js +38 -0
  189. package/package.json +20 -9
  190. package/infrastructure/app-handler-helpers.js +0 -57
  191. package/infrastructure/backend-utils.js +0 -90
  192. package/infrastructure/routers/auth.js +0 -26
  193. package/infrastructure/routers/integration-defined-routers.js +0 -37
  194. package/infrastructure/routers/middleware/loadUser.js +0 -15
  195. package/infrastructure/routers/middleware/requireLoggedInUser.js +0 -12
  196. package/infrastructure/routers/user.js +0 -41
  197. package/infrastructure/routers/websocket.js +0 -55
  198. package/infrastructure/workers/integration-defined-workers.js +0 -24
@@ -0,0 +1,134 @@
1
+ import React from 'react'
2
+ import { render } from '@testing-library/react'
3
+ import { BrowserRouter } from 'react-router-dom'
4
+ import { vi } from 'vitest'
5
+ import { ThemeProvider } from '../../components/theme-provider'
6
+
7
+ // Mock providers for testing
8
+ const MockSocketProvider = ({ children }) => children
9
+ const MockFriggProvider = ({ children, initialUser, initialRepositories, initialCurrentRepository }) => {
10
+ // Create a minimal mock context
11
+ const mockContext = {
12
+ user: initialUser,
13
+ repositories: initialRepositories || [],
14
+ currentRepository: initialCurrentRepository,
15
+ isLoading: false,
16
+ loading: false,
17
+ error: null,
18
+ status: 'running',
19
+ integrations: [],
20
+ envVariables: {},
21
+ users: [],
22
+ connections: [],
23
+ currentUser: initialUser,
24
+ switchRepository: vi.fn().mockResolvedValue(),
25
+ fetchRepositories: vi.fn().mockResolvedValue([]),
26
+ startFrigg: vi.fn().mockResolvedValue(),
27
+ stopFrigg: vi.fn().mockResolvedValue(),
28
+ restartFrigg: vi.fn().mockResolvedValue(),
29
+ installIntegration: vi.fn().mockResolvedValue(),
30
+ createUser: vi.fn().mockResolvedValue(),
31
+ refreshData: vi.fn().mockResolvedValue(),
32
+ }
33
+
34
+ return (
35
+ <div data-testid="mock-frigg-provider" data-context={JSON.stringify(mockContext)}>
36
+ {children}
37
+ </div>
38
+ )
39
+ }
40
+
41
+ // Custom render function that includes all providers
42
+ export function renderWithProviders(ui, options = {}) {
43
+ const {
44
+ initialEntries = ['/'],
45
+ user = null,
46
+ repositories = [],
47
+ currentRepository = null,
48
+ ...renderOptions
49
+ } = options
50
+
51
+ function Wrapper({ children }) {
52
+ return (
53
+ <BrowserRouter>
54
+ <ThemeProvider defaultTheme="light">
55
+ <MockSocketProvider>
56
+ <MockFriggProvider
57
+ initialUser={user}
58
+ initialRepositories={repositories}
59
+ initialCurrentRepository={currentRepository}
60
+ >
61
+ {children}
62
+ </MockFriggProvider>
63
+ </MockSocketProvider>
64
+ </ThemeProvider>
65
+ </BrowserRouter>
66
+ )
67
+ }
68
+
69
+ return render(ui, { wrapper: Wrapper, ...renderOptions })
70
+ }
71
+
72
+ // Mock data factories
73
+ export const createMockRepository = (overrides = {}) => ({
74
+ name: 'test-repo',
75
+ path: '/test/path',
76
+ framework: 'React',
77
+ hasBackend: true,
78
+ version: '1.0.0',
79
+ detectionReasons: ['frigg dependencies'],
80
+ ...overrides
81
+ })
82
+
83
+ export const createMockUser = (overrides = {}) => ({
84
+ id: '1',
85
+ email: 'test@example.com',
86
+ name: 'Test User',
87
+ createdAt: '2023-01-01T00:00:00Z',
88
+ ...overrides
89
+ })
90
+
91
+ export const createMockIntegration = (overrides = {}) => ({
92
+ name: 'test-integration',
93
+ version: '1.0.0',
94
+ installed: true,
95
+ configured: true,
96
+ status: 'active',
97
+ ...overrides
98
+ })
99
+
100
+ export const createMockConnection = (overrides = {}) => ({
101
+ id: '1',
102
+ type: 'slack',
103
+ name: 'Test Connection',
104
+ status: 'active',
105
+ lastUsed: '2023-01-01T00:00:00Z',
106
+ config: {},
107
+ ...overrides
108
+ })
109
+
110
+ // Common test scenarios
111
+ export const mockApiSuccess = (endpoint, data) => {
112
+ return {
113
+ endpoint,
114
+ response: {
115
+ data: { data }
116
+ }
117
+ }
118
+ }
119
+
120
+ export const mockApiError = (endpoint, error = 'Test error') => {
121
+ return {
122
+ endpoint,
123
+ error: new Error(error)
124
+ }
125
+ }
126
+
127
+ // Async utilities
128
+ export const waitForLoading = () => new Promise(resolve => setTimeout(resolve, 0))
129
+
130
+ export const waitForAnimation = () => new Promise(resolve => setTimeout(resolve, 100))
131
+
132
+ // Re-export everything from testing library
133
+ export * from '@testing-library/react'
134
+ export { default as userEvent } from '@testing-library/user-event'
@@ -0,0 +1,98 @@
1
+ import crypto from 'crypto'
2
+
3
+ /**
4
+ * Validates if a repository path is valid
5
+ * @param {string} path - Repository path to validate
6
+ * @returns {boolean} - True if valid, false otherwise
7
+ */
8
+ export function validateRepositoryPath(path) {
9
+ if (!path || typeof path !== 'string') {
10
+ return false
11
+ }
12
+
13
+ // Must be absolute path and have reasonable length
14
+ return path.startsWith('/') && path.length > 2
15
+ }
16
+
17
+ /**
18
+ * Formats a repository name for display
19
+ * @param {string} name - Raw repository name
20
+ * @returns {string} - Formatted display name
21
+ */
22
+ export function formatRepositoryName(name) {
23
+ if (!name || typeof name !== 'string') {
24
+ return ''
25
+ }
26
+
27
+ let formatted = name
28
+
29
+ // Remove common prefixes and suffixes
30
+ formatted = formatted
31
+ .replace(/^frigg-(.+)-api$/, '$1-api') // frigg-slack-api -> slack-api
32
+ .replace(/-backend$/, '')
33
+ .replace(/^frontend-/, '')
34
+
35
+ // Convert to title case
36
+ formatted = formatted
37
+ .split(/[-_]/)
38
+ .map(word => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase())
39
+ .join(' ')
40
+
41
+ return formatted
42
+ }
43
+
44
+ /**
45
+ * Parses raw repository info into standardized format
46
+ * @param {object} rawRepo - Raw repository data
47
+ * @returns {object} - Parsed repository info
48
+ */
49
+ export function parseRepositoryInfo(rawRepo) {
50
+ const {
51
+ name = '',
52
+ path = '',
53
+ framework = null,
54
+ hasBackend = false,
55
+ detectionReasons = []
56
+ } = rawRepo
57
+
58
+ // Generate consistent ID based on path
59
+ const id = crypto
60
+ .createHash('sha256')
61
+ .update(path)
62
+ .digest('hex')
63
+ .substring(0, 16)
64
+
65
+ return {
66
+ id,
67
+ name,
68
+ displayName: formatRepositoryName(name),
69
+ path,
70
+ framework,
71
+ hasBackend,
72
+ detectionReasons: Array.isArray(detectionReasons) ? detectionReasons : [],
73
+ status: 'unknown'
74
+ }
75
+ }
76
+
77
+ /**
78
+ * Gets the current status of a repository
79
+ * @param {object} repo - Repository info
80
+ * @returns {Promise<string>} - Status: 'active', 'inactive', or 'error'
81
+ */
82
+ export async function getRepositoryStatus(repo) {
83
+ try {
84
+ if (!validateRepositoryPath(repo?.path) || !repo?.name) {
85
+ return 'error'
86
+ }
87
+
88
+ // In a real implementation, this would check:
89
+ // - If development servers are running
90
+ // - If the repository is accessible
91
+ // - If there are any errors in logs
92
+
93
+ // For now, return a mock status
94
+ return 'inactive'
95
+ } catch (error) {
96
+ return 'error'
97
+ }
98
+ }
@@ -0,0 +1,118 @@
1
+ import { describe, it, expect } from 'vitest'
2
+ import {
3
+ validateRepositoryPath,
4
+ formatRepositoryName,
5
+ parseRepositoryInfo,
6
+ getRepositoryStatus
7
+ } from './repository'
8
+
9
+ describe('Repository Utilities', () => {
10
+ describe('validateRepositoryPath', () => {
11
+ it('returns true for valid repository paths', () => {
12
+ expect(validateRepositoryPath('/valid/path/to/repo')).toBe(true)
13
+ expect(validateRepositoryPath('/Users/user/project')).toBe(true)
14
+ })
15
+
16
+ it('returns false for invalid repository paths', () => {
17
+ expect(validateRepositoryPath('')).toBe(false)
18
+ expect(validateRepositoryPath(null)).toBe(false)
19
+ expect(validateRepositoryPath(undefined)).toBe(false)
20
+ expect(validateRepositoryPath('relative/path')).toBe(false)
21
+ })
22
+
23
+ it('returns false for paths that are too short', () => {
24
+ expect(validateRepositoryPath('/')).toBe(false)
25
+ expect(validateRepositoryPath('/a')).toBe(false)
26
+ })
27
+ })
28
+
29
+ describe('formatRepositoryName', () => {
30
+ it('formats repository names correctly', () => {
31
+ expect(formatRepositoryName('my-awesome-app')).toBe('My Awesome App')
32
+ expect(formatRepositoryName('frigg-integration')).toBe('Frigg Integration')
33
+ expect(formatRepositoryName('test_repo')).toBe('Test Repo')
34
+ })
35
+
36
+ it('handles edge cases', () => {
37
+ expect(formatRepositoryName('')).toBe('')
38
+ expect(formatRepositoryName('single')).toBe('Single')
39
+ expect(formatRepositoryName('UPPERCASE')).toBe('Uppercase')
40
+ })
41
+
42
+ it('removes common prefixes and suffixes', () => {
43
+ expect(formatRepositoryName('frigg-slack-api')).toBe('Slack Api')
44
+ expect(formatRepositoryName('my-app-backend')).toBe('My App')
45
+ expect(formatRepositoryName('frontend-react-app')).toBe('React App')
46
+ })
47
+ })
48
+
49
+ describe('parseRepositoryInfo', () => {
50
+ it('parses complete repository information', () => {
51
+ const rawRepo = {
52
+ name: 'test-repo',
53
+ path: '/test/path',
54
+ framework: 'React',
55
+ hasBackend: true,
56
+ detectionReasons: ['frigg dependencies', 'config file']
57
+ }
58
+
59
+ const parsed = parseRepositoryInfo(rawRepo)
60
+
61
+ expect(parsed).toEqual({
62
+ id: expect.any(String),
63
+ name: 'test-repo',
64
+ displayName: 'Test Repo',
65
+ path: '/test/path',
66
+ framework: 'React',
67
+ hasBackend: true,
68
+ detectionReasons: ['frigg dependencies', 'config file'],
69
+ status: 'unknown'
70
+ })
71
+ })
72
+
73
+ it('handles minimal repository information', () => {
74
+ const rawRepo = {
75
+ name: 'minimal',
76
+ path: '/minimal/path'
77
+ }
78
+
79
+ const parsed = parseRepositoryInfo(rawRepo)
80
+
81
+ expect(parsed).toEqual({
82
+ id: expect.any(String),
83
+ name: 'minimal',
84
+ displayName: 'Minimal',
85
+ path: '/minimal/path',
86
+ framework: null,
87
+ hasBackend: false,
88
+ detectionReasons: [],
89
+ status: 'unknown'
90
+ })
91
+ })
92
+
93
+ it('generates consistent IDs for same repository', () => {
94
+ const repo = { name: 'test', path: '/test' }
95
+ const parsed1 = parseRepositoryInfo(repo)
96
+ const parsed2 = parseRepositoryInfo(repo)
97
+
98
+ expect(parsed1.id).toBe(parsed2.id)
99
+ })
100
+ })
101
+
102
+ describe('getRepositoryStatus', () => {
103
+ it('returns active for running repositories', async () => {
104
+ const mockRepo = { name: 'test', path: '/test' }
105
+ const status = await getRepositoryStatus(mockRepo)
106
+
107
+ // This would normally check if servers are running, etc.
108
+ expect(['active', 'inactive', 'error']).toContain(status)
109
+ })
110
+
111
+ it('handles repository status check errors', async () => {
112
+ const invalidRepo = { name: '', path: '' }
113
+ const status = await getRepositoryStatus(invalidRepo)
114
+
115
+ expect(status).toBe('error')
116
+ })
117
+ })
118
+ })