@manojkmfsi/monodog 1.1.37 → 1.1.39

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 (200) hide show
  1. package/CHANGELOG.md +12 -0
  2. package/monodog-dashboard/dist/assets/index-C08ciT3A.js +20 -0
  3. package/monodog-dashboard/dist/assets/index-C08ciT3A.js.map +1 -0
  4. package/monodog-dashboard/dist/assets/index-Dc2vaUOq.css +1 -0
  5. package/monodog-dashboard/{index.html → dist/index.html} +2 -1
  6. package/monodog-dashboard/package.json +0 -3
  7. package/package.json +2 -2
  8. package/monodog-dashboard/THEME_SYSTEM.md +0 -309
  9. package/monodog-dashboard/__tests__/AuthCallbackPage.test.ts +0 -138
  10. package/monodog-dashboard/__tests__/ConfigInspector.test.ts +0 -33
  11. package/monodog-dashboard/__tests__/DependencyGraph.test.ts +0 -72
  12. package/monodog-dashboard/__tests__/HealthStatus.test.ts +0 -69
  13. package/monodog-dashboard/__tests__/LoginPage.test.ts +0 -79
  14. package/monodog-dashboard/__tests__/PackageDetail.test.ts +0 -22
  15. package/monodog-dashboard/__tests__/ReleaseManager.test.tsx +0 -510
  16. package/monodog-dashboard/__tests__/api-client.test.ts +0 -475
  17. package/monodog-dashboard/__tests__/components.test.ts +0 -117
  18. package/monodog-dashboard/__tests__/dashboard.utils.test.ts +0 -144
  19. package/monodog-dashboard/__tests__/dependency.utils.test.ts +0 -125
  20. package/monodog-dashboard/__tests__/monorepoService.unit.test.ts +0 -103
  21. package/monodog-dashboard/__tests__/packages.utils.test.ts +0 -37
  22. package/monodog-dashboard/__tests__/pages.test.ts +0 -103
  23. package/monodog-dashboard/package-lock.json +0 -4587
  24. package/monodog-dashboard/postcss.config.js +0 -6
  25. package/monodog-dashboard/src/components/App.tsx +0 -36
  26. package/monodog-dashboard/src/components/LoadingState.tsx +0 -18
  27. package/monodog-dashboard/src/components/PermissionGuard.tsx +0 -92
  28. package/monodog-dashboard/src/components/ProtectedRoute.tsx +0 -24
  29. package/monodog-dashboard/src/components/configuration/Configuration.tsx +0 -91
  30. package/monodog-dashboard/src/components/configuration/components/BrandingSettings.tsx +0 -128
  31. package/monodog-dashboard/src/components/configuration/components/ConfigurationHeader.tsx +0 -34
  32. package/monodog-dashboard/src/components/configuration/components/ConfigurationModal.tsx +0 -20
  33. package/monodog-dashboard/src/components/configuration/components/ConfigurationTabs.tsx +0 -30
  34. package/monodog-dashboard/src/components/configuration/components/FeatureToggles.tsx +0 -66
  35. package/monodog-dashboard/src/components/configuration/components/GeneralSettings.tsx +0 -77
  36. package/monodog-dashboard/src/components/configuration/components/MonorepoSettings.tsx +0 -179
  37. package/monodog-dashboard/src/components/configuration/components/index.ts +0 -8
  38. package/monodog-dashboard/src/components/configuration/types/config.types.ts +0 -52
  39. package/monodog-dashboard/src/components/configuration/utils/config.utils.tsx +0 -115
  40. package/monodog-dashboard/src/components/main-dashboard/Dashboard.tsx +0 -242
  41. package/monodog-dashboard/src/components/main-dashboard/Layout.tsx +0 -112
  42. package/monodog-dashboard/src/components/main-dashboard/components/Header.tsx +0 -47
  43. package/monodog-dashboard/src/components/main-dashboard/components/PackageDistribution.tsx +0 -35
  44. package/monodog-dashboard/src/components/main-dashboard/components/PackageSearchFilter.tsx +0 -47
  45. package/monodog-dashboard/src/components/main-dashboard/components/PackageTable.tsx +0 -87
  46. package/monodog-dashboard/src/components/main-dashboard/components/QuickActions.tsx +0 -60
  47. package/monodog-dashboard/src/components/main-dashboard/components/StatsCards.tsx +0 -73
  48. package/monodog-dashboard/src/components/main-dashboard/components/index.ts +0 -7
  49. package/monodog-dashboard/src/components/main-dashboard/types/dashboard.types.ts +0 -33
  50. package/monodog-dashboard/src/components/main-dashboard/utils/dashboard.utils.tsx +0 -70
  51. package/monodog-dashboard/src/components/modules/ci-integration/CIIntegration.tsx +0 -277
  52. package/monodog-dashboard/src/components/modules/ci-integration/components/BuildDetails.tsx +0 -228
  53. package/monodog-dashboard/src/components/modules/ci-integration/components/BuildList.tsx +0 -212
  54. package/monodog-dashboard/src/components/modules/ci-integration/components/BuildOverview.tsx +0 -158
  55. package/monodog-dashboard/src/components/modules/ci-integration/components/CIIntegrationHeader.tsx +0 -38
  56. package/monodog-dashboard/src/components/modules/ci-integration/components/ErrorState.tsx +0 -25
  57. package/monodog-dashboard/src/components/modules/ci-integration/components/LoadingState.tsx +0 -16
  58. package/monodog-dashboard/src/components/modules/ci-integration/components/PipelineStatus.tsx +0 -223
  59. package/monodog-dashboard/src/components/modules/ci-integration/components/index.ts +0 -8
  60. package/monodog-dashboard/src/components/modules/ci-integration/types/ci.types.ts +0 -97
  61. package/monodog-dashboard/src/components/modules/ci-integration/utils/ci.utils.tsx +0 -264
  62. package/monodog-dashboard/src/components/modules/config-inspector/ConfigInspector.tsx +0 -324
  63. package/monodog-dashboard/src/components/modules/config-inspector/components/ConfigEditor.tsx +0 -93
  64. package/monodog-dashboard/src/components/modules/config-inspector/components/ConfigInspectorHeader.tsx +0 -36
  65. package/monodog-dashboard/src/components/modules/config-inspector/components/ConfigPreview.tsx +0 -89
  66. package/monodog-dashboard/src/components/modules/config-inspector/components/ConfigSidebar.tsx +0 -141
  67. package/monodog-dashboard/src/components/modules/config-inspector/components/ConfigToolbar.tsx +0 -184
  68. package/monodog-dashboard/src/components/modules/config-inspector/components/ErrorState.tsx +0 -25
  69. package/monodog-dashboard/src/components/modules/config-inspector/components/LoadingState.tsx +0 -16
  70. package/monodog-dashboard/src/components/modules/config-inspector/components/ValidationPanel.tsx +0 -155
  71. package/monodog-dashboard/src/components/modules/config-inspector/components/index.ts +0 -9
  72. package/monodog-dashboard/src/components/modules/config-inspector/types/config.types.ts +0 -100
  73. package/monodog-dashboard/src/components/modules/config-inspector/utils/config.utils.tsx +0 -704
  74. package/monodog-dashboard/src/components/modules/dependency-graph/DependencyGraph.tsx +0 -224
  75. package/monodog-dashboard/src/components/modules/dependency-graph/components/CircularDependencies.tsx +0 -177
  76. package/monodog-dashboard/src/components/modules/dependency-graph/components/DependencyDetails.tsx +0 -192
  77. package/monodog-dashboard/src/components/modules/dependency-graph/components/DependencyGraphHeader.tsx +0 -30
  78. package/monodog-dashboard/src/components/modules/dependency-graph/components/DependencyList.tsx +0 -177
  79. package/monodog-dashboard/src/components/modules/dependency-graph/components/ErrorState.tsx +0 -25
  80. package/monodog-dashboard/src/components/modules/dependency-graph/components/GraphLegend.tsx +0 -89
  81. package/monodog-dashboard/src/components/modules/dependency-graph/components/GraphStats.tsx +0 -141
  82. package/monodog-dashboard/src/components/modules/dependency-graph/components/GraphToolbar.tsx +0 -107
  83. package/monodog-dashboard/src/components/modules/dependency-graph/components/GraphVisualization.tsx +0 -179
  84. package/monodog-dashboard/src/components/modules/dependency-graph/components/LoadingState.tsx +0 -16
  85. package/monodog-dashboard/src/components/modules/dependency-graph/components/index.ts +0 -11
  86. package/monodog-dashboard/src/components/modules/dependency-graph/types/dependency.types.ts +0 -105
  87. package/monodog-dashboard/src/components/modules/dependency-graph/utils/dependency.utils.tsx +0 -433
  88. package/monodog-dashboard/src/components/modules/health-status/HealthStatus.tsx +0 -739
  89. package/monodog-dashboard/src/components/modules/health-status/components/ErrorState.tsx +0 -25
  90. package/monodog-dashboard/src/components/modules/health-status/components/HealthActions.tsx +0 -122
  91. package/monodog-dashboard/src/components/modules/health-status/components/HealthAlerts.tsx +0 -151
  92. package/monodog-dashboard/src/components/modules/health-status/components/HealthMetrics.tsx +0 -132
  93. package/monodog-dashboard/src/components/modules/health-status/components/HealthStatusHeader.tsx +0 -30
  94. package/monodog-dashboard/src/components/modules/health-status/components/LoadingState.tsx +0 -16
  95. package/monodog-dashboard/src/components/modules/health-status/components/OverallHealthScore.tsx +0 -122
  96. package/monodog-dashboard/src/components/modules/health-status/components/PackageHealthList.tsx +0 -195
  97. package/monodog-dashboard/src/components/modules/health-status/types/health.types.ts +0 -80
  98. package/monodog-dashboard/src/components/modules/health-status/utils/health.utils.tsx +0 -220
  99. package/monodog-dashboard/src/components/modules/packages/PackageDetail.tsx +0 -255
  100. package/monodog-dashboard/src/components/modules/packages/PackagesOverview.tsx +0 -166
  101. package/monodog-dashboard/src/components/modules/packages/components/ConfigurationTab.tsx +0 -311
  102. package/monodog-dashboard/src/components/modules/packages/components/DependenciesTab.tsx +0 -154
  103. package/monodog-dashboard/src/components/modules/packages/components/ErrorState.tsx +0 -21
  104. package/monodog-dashboard/src/components/modules/packages/components/HealthMetricsTab.tsx +0 -275
  105. package/monodog-dashboard/src/components/modules/packages/components/LoadingState.tsx +0 -14
  106. package/monodog-dashboard/src/components/modules/packages/components/PackageDetailHeader.tsx +0 -167
  107. package/monodog-dashboard/src/components/modules/packages/components/PackageDetailTabs.tsx +0 -49
  108. package/monodog-dashboard/src/components/modules/packages/components/PackageStats.tsx +0 -70
  109. package/monodog-dashboard/src/components/modules/packages/components/PackagesTable.tsx +0 -163
  110. package/monodog-dashboard/src/components/modules/packages/components/RecentCommitsTab.tsx +0 -90
  111. package/monodog-dashboard/src/components/modules/packages/components/SearchAndFilter.tsx +0 -66
  112. package/monodog-dashboard/src/components/modules/packages/components/index.ts +0 -12
  113. package/monodog-dashboard/src/components/modules/packages/types/packages.types.ts +0 -101
  114. package/monodog-dashboard/src/components/modules/packages/utils/packages.utils.tsx +0 -178
  115. package/monodog-dashboard/src/components/pipeline/JobsList.tsx +0 -83
  116. package/monodog-dashboard/src/components/pipeline/LogViewer.tsx +0 -392
  117. package/monodog-dashboard/src/components/pipeline/PipelineManager.tsx +0 -562
  118. package/monodog-dashboard/src/components/pipeline/WorkflowRunsList.tsx +0 -272
  119. package/monodog-dashboard/src/components/pipeline/WorkflowTrigger.tsx +0 -182
  120. package/monodog-dashboard/src/components/pipeline/utils/pipeline.utils.tsx +0 -11
  121. package/monodog-dashboard/src/components/publish-control/PublishControl.tsx +0 -229
  122. package/monodog-dashboard/src/components/publish-control/components/ChangelogViewer.tsx +0 -103
  123. package/monodog-dashboard/src/components/publish-control/components/ErrorState.tsx +0 -23
  124. package/monodog-dashboard/src/components/publish-control/components/LoadingState.tsx +0 -10
  125. package/monodog-dashboard/src/components/publish-control/components/PackageReleaseTable.tsx +0 -140
  126. package/monodog-dashboard/src/components/publish-control/components/PublishHeader.tsx +0 -30
  127. package/monodog-dashboard/src/components/publish-control/components/QuickActionCards.tsx +0 -56
  128. package/monodog-dashboard/src/components/publish-control/components/ReleaseSchedule.tsx +0 -104
  129. package/monodog-dashboard/src/components/publish-control/components/index.ts +0 -8
  130. package/monodog-dashboard/src/components/publish-control/types/publish.types.ts +0 -39
  131. package/monodog-dashboard/src/components/publish-control/utils/publish.utils.ts +0 -59
  132. package/monodog-dashboard/src/components/release-manager/ReleaseManager.tsx +0 -342
  133. package/monodog-dashboard/src/components/release-manager/components/ChangesetPreview.tsx +0 -123
  134. package/monodog-dashboard/src/components/release-manager/components/ErrorState.tsx +0 -38
  135. package/monodog-dashboard/src/components/release-manager/components/LoadingState.tsx +0 -18
  136. package/monodog-dashboard/src/components/release-manager/components/PackageSelector.tsx +0 -137
  137. package/monodog-dashboard/src/components/release-manager/components/PublishConfirmation.tsx +0 -150
  138. package/monodog-dashboard/src/components/release-manager/components/ReleaseValidation.tsx +0 -138
  139. package/monodog-dashboard/src/components/release-manager/components/VersionBumpSelector.tsx +0 -142
  140. package/monodog-dashboard/src/components/release-manager/types/index.ts +0 -9
  141. package/monodog-dashboard/src/components/release-manager/types/release-manager.types.ts +0 -29
  142. package/monodog-dashboard/src/components/setup-guide/SetupGuide.tsx +0 -96
  143. package/monodog-dashboard/src/components/setup-guide/components/SetupHeader.tsx +0 -24
  144. package/monodog-dashboard/src/components/setup-guide/components/SetupModal.tsx +0 -24
  145. package/monodog-dashboard/src/components/setup-guide/components/SetupProgress.tsx +0 -29
  146. package/monodog-dashboard/src/components/setup-guide/components/StepContent.tsx +0 -272
  147. package/monodog-dashboard/src/components/setup-guide/components/StepNavigation.tsx +0 -62
  148. package/monodog-dashboard/src/components/setup-guide/components/index.ts +0 -6
  149. package/monodog-dashboard/src/components/setup-guide/types/setup.types.ts +0 -38
  150. package/monodog-dashboard/src/components/setup-guide/utils/setup.utils.ts +0 -79
  151. package/monodog-dashboard/src/constants/api-config.ts +0 -81
  152. package/monodog-dashboard/src/constants/index.ts +0 -9
  153. package/monodog-dashboard/src/constants/messages.ts +0 -40
  154. package/monodog-dashboard/src/icons/heroicons.ts +0 -65
  155. package/monodog-dashboard/src/icons/index.tsx +0 -1788
  156. package/monodog-dashboard/src/index.css +0 -175
  157. package/monodog-dashboard/src/main.tsx +0 -9
  158. package/monodog-dashboard/src/pages/AuthCallbackPage.tsx +0 -103
  159. package/monodog-dashboard/src/pages/CIPage.tsx +0 -6
  160. package/monodog-dashboard/src/pages/ConfigPage.tsx +0 -6
  161. package/monodog-dashboard/src/pages/DashboardPage.tsx +0 -6
  162. package/monodog-dashboard/src/pages/DependenciesPage.tsx +0 -6
  163. package/monodog-dashboard/src/pages/HealthPage.tsx +0 -6
  164. package/monodog-dashboard/src/pages/LoginPage.tsx +0 -77
  165. package/monodog-dashboard/src/pages/PackageDetailPage.tsx +0 -6
  166. package/monodog-dashboard/src/pages/PackagesPage.tsx +0 -6
  167. package/monodog-dashboard/src/pages/PipelinePage.tsx +0 -22
  168. package/monodog-dashboard/src/pages/PublishPage.tsx +0 -6
  169. package/monodog-dashboard/src/pages/ReleaseManagerPage.tsx +0 -9
  170. package/monodog-dashboard/src/pages/index.ts +0 -28
  171. package/monodog-dashboard/src/routes/AppRouter.tsx +0 -89
  172. package/monodog-dashboard/src/routes/AppRouterPages.tsx +0 -56
  173. package/monodog-dashboard/src/routes/index.ts +0 -10
  174. package/monodog-dashboard/src/routes/routes.config.ts +0 -94
  175. package/monodog-dashboard/src/services/api/api-client.ts +0 -189
  176. package/monodog-dashboard/src/services/api/index.ts +0 -30
  177. package/monodog-dashboard/src/services/api/types/api.types.ts +0 -63
  178. package/monodog-dashboard/src/services/auth-context.tsx +0 -344
  179. package/monodog-dashboard/src/services/monorepoService.ts +0 -733
  180. package/monodog-dashboard/src/services/permission-context.tsx +0 -193
  181. package/monodog-dashboard/src/theme/COMPONENT_UPDATES.md +0 -273
  182. package/monodog-dashboard/src/theme/INTEGRATION.md +0 -381
  183. package/monodog-dashboard/src/theme/README.md +0 -239
  184. package/monodog-dashboard/src/theme/examples.tsx +0 -386
  185. package/monodog-dashboard/src/theme/index.ts +0 -402
  186. package/monodog-dashboard/src/theme/migration-guide.md +0 -335
  187. package/monodog-dashboard/src/types/auth-context.types.ts +0 -51
  188. package/monodog-dashboard/src/types/component.types.ts +0 -96
  189. package/monodog-dashboard/src/types/icons.types.ts +0 -13
  190. package/monodog-dashboard/src/types/index.ts +0 -68
  191. package/monodog-dashboard/src/types/monorepo-service.types.ts +0 -53
  192. package/monodog-dashboard/src/types/permission-context.types.ts +0 -49
  193. package/monodog-dashboard/src/types/pipeline.types.ts +0 -89
  194. package/monodog-dashboard/src/types/routes.types.ts +0 -12
  195. package/monodog-dashboard/src/utils/cookies.ts +0 -47
  196. package/monodog-dashboard/tailwind.config.js +0 -68
  197. package/monodog-dashboard/tsconfig.app.json +0 -14
  198. package/monodog-dashboard/tsconfig.json +0 -17
  199. package/monodog-dashboard/tsconfig.node.json +0 -10
  200. package/monodog-dashboard/vite.config.ts +0 -9
@@ -1,83 +0,0 @@
1
- import React from 'react';
2
- import {
3
- ClockIcon,
4
- ExclamationCircleIcon,
5
- CheckCircleIcon,
6
- XCircleIcon,
7
- } from '../../icons/index';
8
-
9
- import type { JobsListProps } from '../../types';
10
-
11
- function getStatusIcon(
12
- status: string,
13
- conclusion: string | null,
14
- isUpdating: boolean = false
15
- ) {
16
- if (isUpdating) {
17
- return <ClockIcon className="h-6 w-6 text-yellow-500 animate-spin" />;
18
- }
19
- if (status === 'completed') {
20
- if (conclusion === 'success') {
21
- return <CheckCircleIcon className="h-6 w-6 text-green-600" />;
22
- } else if (conclusion === 'failure') {
23
- return <ExclamationCircleIcon className="h-6 w-6 text-red-600" />;
24
- } else if (conclusion === 'cancelled') {
25
- return <XCircleIcon className="h-6 w-6 text-gray-600" />;
26
- } else if (conclusion === 'skipped') {
27
- return <ClockIcon className="h-6 w-6 text-gray-600" />;
28
- }
29
- }
30
- if (status === 'in_progress' || status === 'queued') {
31
- return <ClockIcon className="h-6 w-6 text-blue-600" />;
32
- }
33
- return <ClockIcon className="h-6 w-6 text-gray-600" />;
34
- }
35
-
36
- export default function JobsList({
37
- jobs,
38
- selectedJob,
39
- onSelectJob,
40
- }: JobsListProps) {
41
- return (
42
- <>
43
- {jobs.map(job => {
44
- const jobDuration =
45
- job.startedAt && job.completedAt
46
- ? Math.round(
47
- (new Date(job.completedAt).getTime() -
48
- new Date(job.startedAt).getTime()) /
49
- 1000
50
- )
51
- : 0;
52
-
53
- return (
54
- <div
55
- key={job.id}
56
- className={`rounded-lg border transition-colors ${
57
- selectedJob?.id === job.id
58
- ? 'border-blue-500 bg-blue-50'
59
- : 'border-gray-200 hover:bg-gray-50'
60
- }`}
61
- >
62
- <button
63
- onClick={() => onSelectJob(job)}
64
- className="w-full text-left p-3"
65
- >
66
- <div className="flex items-start gap-2">
67
- {getStatusIcon(job.status, job.conclusion)}
68
- <div className="min-w-0 flex-1">
69
- <p className="font-medium text-sm text-gray-900 truncate">
70
- {job.name}
71
- </p>
72
- <p className="text-xs text-gray-500 mt-1">
73
- {job.status} • {jobDuration}s
74
- </p>
75
- </div>
76
- </div>
77
- </button>
78
- </div>
79
- );
80
- })}
81
- </>
82
- );
83
- }
@@ -1,392 +0,0 @@
1
- import React, { useState, useRef, useEffect, useMemo } from 'react';
2
- import { ChevronDownIcon, ChevronRightIcon , CheckCircleIcon, ExclamationCircleIcon, ClockIcon, XCircleIcon} from '../../icons/index';
3
- import DOMPurify from 'dompurify';
4
- import type { HierarchicalStep, LogViewerProps, StepItemProps } from '../../types';
5
-
6
- interface LogLine {
7
- lineNumber: number;
8
- timestamp: string;
9
- content: string;
10
- ansiContent: string;
11
- }
12
-
13
- /**
14
- * Convert ANSI escape codes to HTML
15
- * Supports colors, bold, dim, italic, etc.
16
- */
17
- function ansiToHtml(text: string): string {
18
- const ansiRegex =
19
- // eslint-disable-next-line no-control-regex
20
- /\u001b\[([0-9;]*m|K)|(\u001b\(B|\u001b\[m)/g;
21
-
22
- let html = '';
23
- let styles: Record<string, string> = {};
24
-
25
- const matches = Array.from(text.matchAll(ansiRegex));
26
-
27
- const colorMap: Record<number, string> = {
28
- 30: '#000000',
29
- 31: '#e06c75', // red
30
- 32: '#98c379', // green
31
- 33: '#e5c07b', // yellow
32
- 34: '#61afef', // blue
33
- 35: '#c678dd', // magenta
34
- 36: '#56b6c2', // cyan
35
- 37: '#abb2bf', // white
36
- };
37
-
38
- const brightColorMap: Record<number, string> = {
39
- 90: '#5c6370',
40
- 91: '#ff7b86',
41
- 92: '#98c379',
42
- 93: '#e5c07b',
43
- 94: '#61afef',
44
- 95: '#c678dd',
45
- 96: '#56b6c2',
46
- 97: '#ffffff',
47
- };
48
-
49
- let currentIndex = 0;
50
- let buffer = '';
51
-
52
- for (const match of matches) {
53
- // Add text before this match
54
- const textBefore = text.substring(currentIndex, match.index);
55
- buffer += escapeHtml(textBefore);
56
-
57
- const code = match[1];
58
- currentIndex = match.index! + match[0].length;
59
-
60
- if (code === 'm' || code === '0m') {
61
- // Reset
62
- if (buffer) {
63
- html += wrapWithStyles(buffer, styles);
64
- buffer = '';
65
- }
66
- styles = {};
67
- } else if (code === '1m') {
68
- styles.fontWeight = 'bold';
69
- } else if (code === '2m') {
70
- styles.opacity = '0.5';
71
- } else if (code === '3m') {
72
- styles.fontStyle = 'italic';
73
- } else if (code === '4m') {
74
- styles.textDecoration = 'underline';
75
- } else if (code.includes(';')) {
76
- // Multiple codes
77
- const codes = code.split(';').filter((c) => c);
78
- for (const c of codes) {
79
- const num = parseInt(c);
80
- if (num === 0) {
81
- if (buffer) {
82
- html += wrapWithStyles(buffer, styles);
83
- buffer = '';
84
- }
85
- styles = {};
86
- } else if (num === 1) {
87
- styles.fontWeight = 'bold';
88
- } else if (num === 2) {
89
- styles.opacity = '0.5';
90
- } else if (num === 3) {
91
- styles.fontStyle = 'italic';
92
- } else if (num === 4) {
93
- styles.textDecoration = 'underline';
94
- } else if (colorMap[num]) {
95
- styles.color = colorMap[num];
96
- } else if (brightColorMap[num]) {
97
- styles.color = brightColorMap[num];
98
- } else if (num >= 40 && num <= 47) {
99
- const bgColor = colorMap[num - 10];
100
- if (bgColor) styles.backgroundColor = bgColor;
101
- } else if (num >= 100 && num <= 107) {
102
- const bgColor = brightColorMap[num - 60];
103
- if (bgColor) styles.backgroundColor = bgColor;
104
- }
105
- }
106
- } else {
107
- const num = parseInt(code);
108
- if (colorMap[num]) {
109
- styles.color = colorMap[num];
110
- } else if (brightColorMap[num]) {
111
- styles.color = brightColorMap[num];
112
- }
113
- }
114
- }
115
-
116
- // Add remaining text
117
- const remaining = text.substring(currentIndex);
118
- buffer += escapeHtml(remaining);
119
- if (buffer) {
120
- html += wrapWithStyles(buffer, styles);
121
- }
122
-
123
- return html;
124
- }
125
-
126
- function escapeHtml(text: string): string {
127
- const map: Record<string, string> = {
128
- '&': '&amp;',
129
- '<': '&lt;',
130
- '>': '&gt;',
131
- '"': '&quot;',
132
- "'": '&#039;',
133
- };
134
- return text.replace(/[&<>"']/g, (char) => map[char]);
135
- }
136
-
137
- function wrapWithStyles(
138
- text: string,
139
- styles: Record<string, string>
140
- ): string {
141
- if (!text || Object.keys(styles).length === 0) {
142
- return text;
143
- }
144
-
145
- const styleStr = Object.entries(styles)
146
- .map(([key, value]) => {
147
- const cssKey = key.replace(/([A-Z])/g, '-$1').toLowerCase();
148
- return `${cssKey}: ${value}`;
149
- })
150
- .join('; ');
151
-
152
- return `<span style="${styleStr}">${text}</span>`;
153
- }
154
-
155
- function getStatusIcon(status: string, conclusion: string | null) {
156
- if (status === 'completed') {
157
- if (conclusion === 'success') {
158
- return <CheckCircleIcon className="h-6 w-6 text-green-600" />;
159
- } else if (conclusion === 'failure') {
160
- return <ExclamationCircleIcon className="h-6 w-6 text-red-600" />;
161
- } else if (conclusion === 'cancelled') {
162
- return <XCircleIcon className="h-6 w-6 text-gray-600" />;
163
- }
164
- }
165
- return <ClockIcon className="h-6 w-6 text-blue-600" />;
166
- }
167
-
168
-
169
- interface StepItemProps {
170
- step: HierarchicalStep;
171
- onToggle: (stepNumber: number) => void;
172
- expandedSteps: Set<number>;
173
- stepIndex: number;
174
- }
175
-
176
- function StepItem({ step, onToggle, expandedSteps, stepIndex }: StepItemProps) {
177
- const logContainerRef = useRef<HTMLDivElement>(null);
178
- const [showAll, setShowAll] = useState(false);
179
- const isExpanded = expandedSteps.has(step.stepNumber);
180
- const hasChildren = (step.children && step.children.length > 0) || step.logs.length > 0;
181
-
182
- const displayedLines = useMemo(() => {
183
- if (showAll) return step.logs;
184
- return step.logs.slice(0, 1000);
185
- }, [step.logs, showAll]);
186
-
187
- useEffect(() => {
188
- if (step.status === 'in_progress' && isExpanded && logContainerRef.current) {
189
- const container = logContainerRef.current;
190
- container.scrollTop = container.scrollHeight;
191
- }
192
- }, [step.logs.length, isExpanded, step.status]);
193
-
194
- const durationSec = useMemo(() => {
195
- if (!step.startedAt || !step.completedAt) return 0;
196
- return Math.round((new Date(step.completedAt).getTime() - new Date(step.startedAt).getTime())/100)/10;
197
- }, [step.startedAt, step.completedAt]);
198
-
199
- const indentStyle = {
200
- marginLeft: `${step.level * 20}px`,
201
- };
202
-
203
- return (
204
- <>
205
- <div className="border-b border-gray-200 bg-white" style={indentStyle}>
206
- {/* GitHub-style Step Header */}
207
- <button
208
- onClick={() => onToggle(step.stepNumber)}
209
- className="w-full flex items-center gap-3 px-4 py-3 hover:bg-gray-50 transition-colors group"
210
- >
211
- {/* Expand/Collapse Icon */}
212
- <span className="text-gray-400 group-hover:text-gray-600 flex-shrink-0">
213
- {hasChildren ? (
214
- isExpanded ? (
215
- <ChevronDownIcon className="h-5 w-5" />
216
- ) : (
217
- <ChevronRightIcon className="h-5 w-5" />
218
- )
219
- ) : (
220
- <span className="h-5 w-5" />
221
- )}
222
- </span>
223
-
224
- {/* Step Number Badge */}
225
- <span className="text-xs font-bold text-gray-500 bg-gray-100 rounded px-2 py-1 min-w-fit">
226
- {stepIndex + 1}
227
- </span>
228
-
229
- {/* Step Name */}
230
- <span className="flex-1 text-left">
231
- <span className="text-sm font-semibold text-gray-900">{step.stepName}</span>
232
- </span>
233
-
234
- {/* Duration */}
235
- <div className="flex items-center gap-2 ml-2">
236
- {step.status === 'in_progress' && (
237
- <span className="flex h-2 w-2 rounded-full bg-blue-500 animate-pulse" />
238
- )}
239
- <span className="text-xs text-gray-500 font-mono">{durationSec}s</span>
240
- </div>
241
- </button>
242
-
243
- {/* Expanded Log Content */}
244
- {isExpanded && step.logs.length > 0 && (
245
- <div className="border-t border-gray-200 bg-[#0d1117]">
246
- {/* Log Container */}
247
- <div
248
- ref={logContainerRef}
249
- className="text-gray-300 font-mono text-xs max-h-[600px] overflow-y-auto"
250
- >
251
- {displayedLines.length === 0 ? (
252
- <div className="px-10 py-6 italic text-gray-500 text-center">No logs available yet</div>
253
- ) : (
254
- <div className="py-2 px-0">
255
- {displayedLines.map((line, idx) => (
256
- <div
257
- key={`${step.stepNumber}-${line.lineNumber}`}
258
- className="flex hover:bg-gray-800/40 transition-colors leading-6 group"
259
- >
260
- {/* Line Number Column */}
261
- <span className="w-12 text-right pr-4 text-gray-600 select-none border-r border-gray-800 flex-shrink-0 group-hover:bg-gray-800/20">
262
- {line.lineNumber}
263
- </span>
264
-
265
- {/* Timestamp Column */}
266
- <span className="pl-4 pr-3 text-gray-500 select-none opacity-50 text-xs flex-shrink-0">
267
- {line.timestamp.split('T')[1]?.slice(0, 8)}
268
- </span>
269
-
270
- {/* Log Content Column */}
271
- <span
272
- className="flex-1 whitespace-pre break-all pr-4 text-gray-300"
273
- dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(ansiToHtml(line.ansiContent)) }}
274
- />
275
- </div>
276
- ))}
277
- </div>
278
- )}
279
-
280
- {/* Show More Button */}
281
- {!showAll && step.logs.length > 1000 && (
282
- <div className="px-4 py-2 border-t border-gray-800 text-center">
283
- <button
284
- onClick={() => setShowAll(true)}
285
- className="text-blue-400 hover:text-blue-300 text-xs font-medium transition-colors"
286
- >
287
- Show all {step.logs.length} lines
288
- </button>
289
- </div>
290
- )}
291
- </div>
292
- </div>
293
- )}
294
- </div>
295
-
296
- {/* Render Children */}
297
- {isExpanded && step.children && step.children.length > 0 && (
298
- <>
299
- {step.children.map((childStep, childIdx) => (
300
- <StepItem
301
- key={childStep.stepNumber}
302
- step={childStep}
303
- onToggle={onToggle}
304
- expandedSteps={expandedSteps}
305
- stepIndex={childIdx}
306
- />
307
- ))}
308
- </>
309
- )}
310
- </>
311
- );
312
- }
313
- export default function LogViewer({
314
- steps,
315
- jobName,
316
- jobStatus,
317
- jobConclusion,
318
- gitHubLogsUrl,
319
- }: LogViewerProps) {
320
- const [expandedSteps, setExpandedSteps] = useState<Set<number>>(
321
- new Set([steps[0]?.stepNumber])
322
- );
323
-
324
- const handleToggleStep = (stepNumber: number) => {
325
- setExpandedSteps((prev) => {
326
- const next = new Set(prev);
327
- if (next.has(stepNumber)) {
328
- next.delete(stepNumber);
329
- } else {
330
- next.add(stepNumber);
331
- }
332
- return next;
333
- });
334
- };
335
-
336
- // Count total steps recursively
337
- const countSteps = (stepList: HierarchicalStep[]): number => {
338
- return stepList.reduce((count, step) => {
339
- let total = 1;
340
- if (step.children && step.children.length > 0) {
341
- total += countSteps(step.children);
342
- }
343
- return count + total;
344
- }, 0);
345
- };
346
-
347
- return (
348
- <div className="flex flex-col h-full bg-white border border-gray-200 rounded-lg overflow-hidden">
349
- {/* Header */}
350
- <div className="px-4 py-3 bg-gray-50 border-b border-gray-200 flex items-center justify-between">
351
- <div className="flex items-center gap-3">
352
- <span className={`text-lg font-semibold `}>
353
- {getStatusIcon(jobStatus, jobConclusion)}
354
- </span>
355
- <div>
356
- <h3 className="text-sm font-semibold text-gray-900">{jobName}</h3>
357
- <p className="text-xs text-gray-500">
358
- {countSteps(steps)} step
359
- {countSteps(steps) !== 1 ? 's' : ''}
360
- </p>
361
- </div>
362
- </div>
363
-
364
- <a
365
- href={gitHubLogsUrl}
366
- target="_blank"
367
- rel="noopener noreferrer"
368
- className="text-xs px-3 py-2 bg-blue-50 text-blue-600 hover:bg-blue-100 rounded border border-blue-200 transition-colors"
369
- >
370
- View on GitHub
371
- </a>
372
- </div>
373
-
374
- {/* Steps */}
375
- <div className="flex-1 overflow-y-auto">
376
- {steps.length === 0 ? (
377
- <div className="p-4 text-center text-gray-500">No steps available</div>
378
- ) : (
379
- steps.map((step, idx) => (
380
- <StepItem
381
- key={step.stepNumber}
382
- step={step}
383
- onToggle={handleToggleStep}
384
- expandedSteps={expandedSteps}
385
- stepIndex={idx}
386
- />
387
- ))
388
- )}
389
- </div>
390
- </div>
391
- );
392
- }