@friggframework/devtools 2.0.0--canary.398.dd443c7.0 → 2.0.0--canary.402.d2f4ae6.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.
- package/frigg-cli/.eslintrc.js +141 -0
- package/frigg-cli/__tests__/jest.config.js +102 -0
- package/frigg-cli/__tests__/unit/commands/build.test.js +483 -0
- package/frigg-cli/__tests__/unit/commands/install.test.js +418 -0
- package/frigg-cli/__tests__/unit/commands/ui.test.js +592 -0
- package/frigg-cli/__tests__/utils/command-tester.js +170 -0
- package/frigg-cli/__tests__/utils/mock-factory.js +270 -0
- package/frigg-cli/__tests__/utils/test-fixtures.js +463 -0
- package/frigg-cli/__tests__/utils/test-setup.js +286 -0
- package/frigg-cli/generate-command/__tests__/generate-command.test.js +312 -0
- package/frigg-cli/generate-command/azure-generator.js +43 -0
- package/frigg-cli/generate-command/gcp-generator.js +47 -0
- package/frigg-cli/generate-command/index.js +332 -0
- package/frigg-cli/generate-command/terraform-generator.js +555 -0
- package/frigg-cli/index.js +19 -1
- package/frigg-cli/init-command/backend-first-handler.js +756 -0
- package/frigg-cli/init-command/index.js +93 -0
- package/frigg-cli/init-command/template-handler.js +143 -0
- package/frigg-cli/package.json +51 -0
- package/frigg-cli/test/init-command.test.js +180 -0
- package/frigg-cli/test/npm-registry.test.js +319 -0
- package/frigg-cli/ui-command/index.js +154 -0
- package/frigg-cli/utils/app-resolver.js +319 -0
- package/frigg-cli/utils/backend-path.js +25 -0
- package/frigg-cli/utils/npm-registry.js +167 -0
- package/frigg-cli/utils/process-manager.js +199 -0
- package/frigg-cli/utils/repo-detection.js +405 -0
- package/infrastructure/serverless-template.js +177 -292
- package/management-ui/.eslintrc.js +22 -0
- package/management-ui/README.md +203 -0
- package/management-ui/components.json +21 -0
- package/management-ui/docs/phase2-integration-guide.md +320 -0
- package/management-ui/{dist/index.html → index.html} +1 -2
- package/management-ui/package-lock.json +16517 -0
- package/management-ui/package.json +76 -0
- package/management-ui/packages/devtools/frigg-cli/ui-command/index.js +302 -0
- package/management-ui/postcss.config.js +6 -0
- package/management-ui/server/api/backend.js +256 -0
- package/management-ui/server/api/cli.js +315 -0
- package/management-ui/server/api/codegen.js +663 -0
- package/management-ui/server/api/connections.js +857 -0
- package/management-ui/server/api/discovery.js +185 -0
- package/management-ui/server/api/environment/index.js +1 -0
- package/management-ui/server/api/environment/router.js +378 -0
- package/management-ui/server/api/environment.js +328 -0
- package/management-ui/server/api/integrations.js +876 -0
- package/management-ui/server/api/logs.js +248 -0
- package/management-ui/server/api/monitoring.js +282 -0
- package/management-ui/server/api/open-ide.js +31 -0
- package/management-ui/server/api/project.js +1029 -0
- package/management-ui/server/api/users/sessions.js +371 -0
- package/management-ui/server/api/users/simulation.js +254 -0
- package/management-ui/server/api/users.js +362 -0
- package/management-ui/server/api-contract.md +275 -0
- package/management-ui/server/index.js +873 -0
- package/management-ui/server/middleware/errorHandler.js +93 -0
- package/management-ui/server/middleware/security.js +32 -0
- package/management-ui/server/processManager.js +296 -0
- package/management-ui/server/server.js +346 -0
- package/management-ui/server/services/aws-monitor.js +413 -0
- package/management-ui/server/services/npm-registry.js +347 -0
- package/management-ui/server/services/template-engine.js +538 -0
- package/management-ui/server/utils/cliIntegration.js +220 -0
- package/management-ui/server/utils/environment/auditLogger.js +471 -0
- package/management-ui/server/utils/environment/awsParameterStore.js +264 -0
- package/management-ui/server/utils/environment/encryption.js +278 -0
- package/management-ui/server/utils/environment/envFileManager.js +286 -0
- package/management-ui/server/utils/import-commonjs.js +28 -0
- package/management-ui/server/utils/response.js +83 -0
- package/management-ui/server/websocket/handler.js +325 -0
- package/management-ui/src/App.jsx +109 -0
- package/management-ui/src/components/AppRouter.jsx +65 -0
- package/management-ui/src/components/Button.jsx +70 -0
- package/management-ui/src/components/Card.jsx +97 -0
- package/management-ui/src/components/EnvironmentCompare.jsx +400 -0
- package/management-ui/src/components/EnvironmentEditor.jsx +372 -0
- package/management-ui/src/components/EnvironmentImportExport.jsx +469 -0
- package/management-ui/src/components/EnvironmentSchema.jsx +491 -0
- package/management-ui/src/components/EnvironmentSecurity.jsx +463 -0
- package/management-ui/src/components/ErrorBoundary.jsx +73 -0
- package/management-ui/src/components/IntegrationCard.jsx +481 -0
- package/management-ui/src/components/IntegrationCardEnhanced.jsx +770 -0
- package/management-ui/src/components/IntegrationExplorer.jsx +379 -0
- package/management-ui/src/components/IntegrationStatus.jsx +336 -0
- package/management-ui/src/components/Layout.jsx +716 -0
- package/management-ui/src/components/LoadingSpinner.jsx +113 -0
- package/management-ui/src/components/RepositoryPicker.jsx +248 -0
- package/management-ui/src/components/SessionMonitor.jsx +350 -0
- package/management-ui/src/components/StatusBadge.jsx +208 -0
- package/management-ui/src/components/UserContextSwitcher.jsx +212 -0
- package/management-ui/src/components/UserSimulation.jsx +327 -0
- package/management-ui/src/components/Welcome.jsx +434 -0
- package/management-ui/src/components/codegen/APIEndpointGenerator.jsx +637 -0
- package/management-ui/src/components/codegen/APIModuleSelector.jsx +227 -0
- package/management-ui/src/components/codegen/CodeGenerationWizard.jsx +247 -0
- package/management-ui/src/components/codegen/CodePreviewEditor.jsx +316 -0
- package/management-ui/src/components/codegen/DynamicModuleForm.jsx +271 -0
- package/management-ui/src/components/codegen/FormBuilder.jsx +737 -0
- package/management-ui/src/components/codegen/IntegrationGenerator.jsx +855 -0
- package/management-ui/src/components/codegen/ProjectScaffoldWizard.jsx +797 -0
- package/management-ui/src/components/codegen/SchemaBuilder.jsx +303 -0
- package/management-ui/src/components/codegen/TemplateSelector.jsx +586 -0
- package/management-ui/src/components/codegen/index.js +10 -0
- package/management-ui/src/components/connections/ConnectionConfigForm.jsx +362 -0
- package/management-ui/src/components/connections/ConnectionHealthMonitor.jsx +182 -0
- package/management-ui/src/components/connections/ConnectionTester.jsx +200 -0
- package/management-ui/src/components/connections/EntityRelationshipMapper.jsx +292 -0
- package/management-ui/src/components/connections/OAuthFlow.jsx +204 -0
- package/management-ui/src/components/connections/index.js +5 -0
- package/management-ui/src/components/index.js +21 -0
- package/management-ui/src/components/monitoring/APIGatewayMetrics.jsx +222 -0
- package/management-ui/src/components/monitoring/LambdaMetrics.jsx +169 -0
- package/management-ui/src/components/monitoring/MetricsChart.jsx +197 -0
- package/management-ui/src/components/monitoring/MonitoringDashboard.jsx +393 -0
- package/management-ui/src/components/monitoring/SQSMetrics.jsx +246 -0
- package/management-ui/src/components/monitoring/index.js +6 -0
- package/management-ui/src/components/monitoring/monitoring.css +218 -0
- package/management-ui/src/components/theme-provider.jsx +52 -0
- package/management-ui/src/components/theme-toggle.jsx +39 -0
- package/management-ui/src/components/ui/badge.tsx +36 -0
- package/management-ui/src/components/ui/button.test.jsx +56 -0
- package/management-ui/src/components/ui/button.tsx +57 -0
- package/management-ui/src/components/ui/card.tsx +76 -0
- package/management-ui/src/components/ui/dropdown-menu.tsx +199 -0
- package/management-ui/src/components/ui/select.tsx +157 -0
- package/management-ui/src/components/ui/skeleton.jsx +15 -0
- package/management-ui/src/hooks/useFrigg.jsx +601 -0
- package/management-ui/src/hooks/useSocket.jsx +58 -0
- package/management-ui/src/index.css +193 -0
- package/management-ui/src/lib/utils.ts +6 -0
- package/management-ui/src/main.jsx +10 -0
- package/management-ui/src/pages/CodeGeneration.jsx +14 -0
- package/management-ui/src/pages/Connections.jsx +252 -0
- package/management-ui/src/pages/ConnectionsEnhanced.jsx +633 -0
- package/management-ui/src/pages/Dashboard.jsx +311 -0
- package/management-ui/src/pages/Environment.jsx +314 -0
- package/management-ui/src/pages/IntegrationConfigure.jsx +669 -0
- package/management-ui/src/pages/IntegrationDiscovery.jsx +567 -0
- package/management-ui/src/pages/IntegrationTest.jsx +742 -0
- package/management-ui/src/pages/Integrations.jsx +253 -0
- package/management-ui/src/pages/Monitoring.jsx +17 -0
- package/management-ui/src/pages/Simulation.jsx +155 -0
- package/management-ui/src/pages/Users.jsx +492 -0
- package/management-ui/src/services/api.js +41 -0
- package/management-ui/src/services/apiModuleService.js +193 -0
- package/management-ui/src/services/websocket-handlers.js +120 -0
- package/management-ui/src/test/api/project.test.js +273 -0
- package/management-ui/src/test/components/Welcome.test.jsx +378 -0
- package/management-ui/src/test/mocks/server.js +178 -0
- package/management-ui/src/test/setup.js +61 -0
- package/management-ui/src/test/utils/test-utils.jsx +134 -0
- package/management-ui/src/utils/repository.js +98 -0
- package/management-ui/src/utils/repository.test.js +118 -0
- package/management-ui/src/workflows/phase2-integration-workflows.js +884 -0
- package/management-ui/tailwind.config.js +63 -0
- package/management-ui/tsconfig.json +37 -0
- package/management-ui/tsconfig.node.json +10 -0
- package/management-ui/vite.config.js +26 -0
- package/management-ui/vitest.config.js +38 -0
- package/package.json +5 -5
- package/management-ui/dist/assets/index-BA21WgFa.js +0 -1221
- package/management-ui/dist/assets/index-CbM64Oba.js +0 -1221
- package/management-ui/dist/assets/index-CkvseXTC.css +0 -1
- /package/management-ui/{dist/assets/FriggLogo-B7Xx8ZW1.svg → src/assets/FriggLogo.svg} +0 -0
|
@@ -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
|
+
})
|