station-kit 1.0.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 (181) hide show
  1. package/LICENSE +21 -0
  2. package/dist/cli-main.d.ts +2 -0
  3. package/dist/cli-main.d.ts.map +1 -0
  4. package/dist/cli-main.js +58 -0
  5. package/dist/cli-main.js.map +1 -0
  6. package/dist/cli.d.ts +3 -0
  7. package/dist/cli.d.ts.map +1 -0
  8. package/dist/cli.js +25 -0
  9. package/dist/cli.js.map +1 -0
  10. package/dist/config/loader.d.ts +3 -0
  11. package/dist/config/loader.d.ts.map +1 -0
  12. package/dist/config/loader.js +29 -0
  13. package/dist/config/loader.js.map +1 -0
  14. package/dist/config/schema.d.ts +36 -0
  15. package/dist/config/schema.d.ts.map +1 -0
  16. package/dist/config/schema.js +40 -0
  17. package/dist/config/schema.js.map +1 -0
  18. package/dist/index.d.ts +4 -0
  19. package/dist/index.d.ts.map +1 -0
  20. package/dist/index.js +4 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/server/auth/keys.d.ts +28 -0
  23. package/dist/server/auth/keys.d.ts.map +1 -0
  24. package/dist/server/auth/keys.js +91 -0
  25. package/dist/server/auth/keys.js.map +1 -0
  26. package/dist/server/auth/session.d.ts +9 -0
  27. package/dist/server/auth/session.d.ts.map +1 -0
  28. package/dist/server/auth/session.js +42 -0
  29. package/dist/server/auth/session.js.map +1 -0
  30. package/dist/server/index.d.ts +7 -0
  31. package/dist/server/index.d.ts.map +1 -0
  32. package/dist/server/index.js +253 -0
  33. package/dist/server/index.js.map +1 -0
  34. package/dist/server/log-buffer.d.ts +20 -0
  35. package/dist/server/log-buffer.d.ts.map +1 -0
  36. package/dist/server/log-buffer.js +33 -0
  37. package/dist/server/log-buffer.js.map +1 -0
  38. package/dist/server/log-store.d.ts +11 -0
  39. package/dist/server/log-store.d.ts.map +1 -0
  40. package/dist/server/log-store.js +40 -0
  41. package/dist/server/log-store.js.map +1 -0
  42. package/dist/server/metadata.d.ts +38 -0
  43. package/dist/server/metadata.d.ts.map +1 -0
  44. package/dist/server/metadata.js +130 -0
  45. package/dist/server/metadata.js.map +1 -0
  46. package/dist/server/middleware/auth.d.ts +12 -0
  47. package/dist/server/middleware/auth.d.ts.map +1 -0
  48. package/dist/server/middleware/auth.js +42 -0
  49. package/dist/server/middleware/auth.js.map +1 -0
  50. package/dist/server/middleware/rate-limit.d.ts +15 -0
  51. package/dist/server/middleware/rate-limit.d.ts.map +1 -0
  52. package/dist/server/middleware/rate-limit.js +36 -0
  53. package/dist/server/middleware/rate-limit.js.map +1 -0
  54. package/dist/server/middleware/scope-guard.d.ts +9 -0
  55. package/dist/server/middleware/scope-guard.d.ts.map +1 -0
  56. package/dist/server/middleware/scope-guard.js +17 -0
  57. package/dist/server/middleware/scope-guard.js.map +1 -0
  58. package/dist/server/routes/broadcasts.d.ts +12 -0
  59. package/dist/server/routes/broadcasts.d.ts.map +1 -0
  60. package/dist/server/routes/broadcasts.js +135 -0
  61. package/dist/server/routes/broadcasts.js.map +1 -0
  62. package/dist/server/routes/health.d.ts +9 -0
  63. package/dist/server/routes/health.d.ts.map +1 -0
  64. package/dist/server/routes/health.js +27 -0
  65. package/dist/server/routes/health.js.map +1 -0
  66. package/dist/server/routes/runs.d.ts +12 -0
  67. package/dist/server/routes/runs.d.ts.map +1 -0
  68. package/dist/server/routes/runs.js +122 -0
  69. package/dist/server/routes/runs.js.map +1 -0
  70. package/dist/server/routes/signals.d.ts +10 -0
  71. package/dist/server/routes/signals.d.ts.map +1 -0
  72. package/dist/server/routes/signals.js +120 -0
  73. package/dist/server/routes/signals.js.map +1 -0
  74. package/dist/server/routes/v1/auth.d.ts +7 -0
  75. package/dist/server/routes/v1/auth.d.ts.map +1 -0
  76. package/dist/server/routes/v1/auth.js +28 -0
  77. package/dist/server/routes/v1/auth.js.map +1 -0
  78. package/dist/server/routes/v1/broadcasts.d.ts +10 -0
  79. package/dist/server/routes/v1/broadcasts.d.ts.map +1 -0
  80. package/dist/server/routes/v1/broadcasts.js +68 -0
  81. package/dist/server/routes/v1/broadcasts.js.map +1 -0
  82. package/dist/server/routes/v1/events.d.ts +7 -0
  83. package/dist/server/routes/v1/events.d.ts.map +1 -0
  84. package/dist/server/routes/v1/events.js +57 -0
  85. package/dist/server/routes/v1/events.js.map +1 -0
  86. package/dist/server/routes/v1/health.d.ts +9 -0
  87. package/dist/server/routes/v1/health.d.ts.map +1 -0
  88. package/dist/server/routes/v1/health.js +31 -0
  89. package/dist/server/routes/v1/health.js.map +1 -0
  90. package/dist/server/routes/v1/keys.d.ts +7 -0
  91. package/dist/server/routes/v1/keys.d.ts.map +1 -0
  92. package/dist/server/routes/v1/keys.js +43 -0
  93. package/dist/server/routes/v1/keys.js.map +1 -0
  94. package/dist/server/routes/v1/runs.d.ts +12 -0
  95. package/dist/server/routes/v1/runs.d.ts.map +1 -0
  96. package/dist/server/routes/v1/runs.js +76 -0
  97. package/dist/server/routes/v1/runs.js.map +1 -0
  98. package/dist/server/routes/v1/signals.d.ts +9 -0
  99. package/dist/server/routes/v1/signals.d.ts.map +1 -0
  100. package/dist/server/routes/v1/signals.js +33 -0
  101. package/dist/server/routes/v1/signals.js.map +1 -0
  102. package/dist/server/routes/v1/trigger.d.ts +12 -0
  103. package/dist/server/routes/v1/trigger.d.ts.map +1 -0
  104. package/dist/server/routes/v1/trigger.js +73 -0
  105. package/dist/server/routes/v1/trigger.js.map +1 -0
  106. package/dist/server/sse.d.ts +19 -0
  107. package/dist/server/sse.d.ts.map +1 -0
  108. package/dist/server/sse.js +51 -0
  109. package/dist/server/sse.js.map +1 -0
  110. package/dist/server/subscriber.d.ts +128 -0
  111. package/dist/server/subscriber.d.ts.map +1 -0
  112. package/dist/server/subscriber.js +246 -0
  113. package/dist/server/subscriber.js.map +1 -0
  114. package/dist/server/ws.d.ts +15 -0
  115. package/dist/server/ws.d.ts.map +1 -0
  116. package/dist/server/ws.js +32 -0
  117. package/dist/server/ws.js.map +1 -0
  118. package/next-env.d.ts +6 -0
  119. package/next.config.ts +10 -0
  120. package/package.json +49 -0
  121. package/src/app/broadcasts/[id]/page.tsx +511 -0
  122. package/src/app/broadcasts/page.tsx +158 -0
  123. package/src/app/components/auth-provider.tsx +75 -0
  124. package/src/app/components/breadcrumb-provider.tsx +18 -0
  125. package/src/app/components/dag-view.tsx +380 -0
  126. package/src/app/components/empty-state.tsx +7 -0
  127. package/src/app/components/json-viewer.tsx +153 -0
  128. package/src/app/components/login-page.tsx +78 -0
  129. package/src/app/components/node-detail.tsx +158 -0
  130. package/src/app/components/pulse-dot.tsx +8 -0
  131. package/src/app/components/relative-time.tsx +34 -0
  132. package/src/app/components/run-table.tsx +96 -0
  133. package/src/app/components/schema-form.tsx +121 -0
  134. package/src/app/components/shell.tsx +203 -0
  135. package/src/app/components/status-badge.tsx +10 -0
  136. package/src/app/components/step-timeline.tsx +134 -0
  137. package/src/app/components/theme-provider.tsx +45 -0
  138. package/src/app/components/workflow-node-sidebar.tsx +68 -0
  139. package/src/app/globals.css +1523 -0
  140. package/src/app/hooks/use-api.ts +129 -0
  141. package/src/app/hooks/use-breadcrumb.ts +37 -0
  142. package/src/app/hooks/use-realtime.ts +68 -0
  143. package/src/app/hooks/use-station.tsx +34 -0
  144. package/src/app/layout.tsx +42 -0
  145. package/src/app/page.tsx +275 -0
  146. package/src/app/runs/[id]/page.tsx +277 -0
  147. package/src/app/signals/[name]/page.tsx +250 -0
  148. package/src/app/signals/page.tsx +99 -0
  149. package/src/cli-main.ts +70 -0
  150. package/src/cli.ts +27 -0
  151. package/src/config/loader.ts +33 -0
  152. package/src/config/schema.ts +80 -0
  153. package/src/index.ts +7 -0
  154. package/src/server/auth/keys.ts +112 -0
  155. package/src/server/auth/session.ts +48 -0
  156. package/src/server/index.ts +296 -0
  157. package/src/server/log-buffer.ts +43 -0
  158. package/src/server/log-store.ts +56 -0
  159. package/src/server/metadata.ts +180 -0
  160. package/src/server/middleware/auth.ts +50 -0
  161. package/src/server/middleware/rate-limit.ts +61 -0
  162. package/src/server/middleware/scope-guard.ts +20 -0
  163. package/src/server/routes/broadcasts.ts +160 -0
  164. package/src/server/routes/health.ts +37 -0
  165. package/src/server/routes/runs.ts +149 -0
  166. package/src/server/routes/signals.ts +153 -0
  167. package/src/server/routes/v1/auth.ts +47 -0
  168. package/src/server/routes/v1/broadcasts.ts +84 -0
  169. package/src/server/routes/v1/events.ts +71 -0
  170. package/src/server/routes/v1/health.ts +41 -0
  171. package/src/server/routes/v1/keys.ts +57 -0
  172. package/src/server/routes/v1/runs.ts +97 -0
  173. package/src/server/routes/v1/signals.ts +44 -0
  174. package/src/server/routes/v1/trigger.ts +111 -0
  175. package/src/server/sse.ts +70 -0
  176. package/src/server/subscriber.ts +288 -0
  177. package/src/server/ws.ts +44 -0
  178. package/station.config.example.ts +16 -0
  179. package/tsconfig.json +12 -0
  180. package/tsconfig.next.json +15 -0
  181. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,1523 @@
1
+ /* ─── Brand Design Tokens ─────────────────────────────────── */
2
+
3
+ :root {
4
+ /* Colors — warm, material-based */
5
+ --rust: #8B5A2B;
6
+ --rust-light: #A0622D;
7
+ --rust-pale: #C4834A;
8
+ --patina: #4A6741;
9
+ --patina-light: #5C7C52;
10
+ --signal-green: #6B9962;
11
+ --concrete: #F4F1EC;
12
+ --concrete-dark: #E8E4DC;
13
+ --concrete-mid: #D4CEBF;
14
+ --charcoal: #1C1C1E;
15
+ --charcoal-mid: #2C2C2E;
16
+ --wire: #3A3A3C;
17
+ --muted: #8A8A8E;
18
+ --muted-light: #AEAEB2;
19
+ --sky: #7FB2CC;
20
+ --surface: #FFFFFF;
21
+
22
+ /* Code blocks (theme-invariant) */
23
+ --code-bg: #1C1C1E;
24
+ --code-text: #F4F1EC;
25
+ --code-key: #AEAEB2;
26
+ --code-string: #C4834A;
27
+ --code-number: #7FB2CC;
28
+ --code-bool: #5C7C52;
29
+ --code-null: #8A8A8E;
30
+
31
+ /* Typography */
32
+ --font-display: 'Instrument Serif', serif;
33
+ --font-mono: 'IBM Plex Mono', monospace;
34
+ --font-body: 'Space Grotesk', sans-serif;
35
+
36
+ /* Motion */
37
+ --ease-standard: cubic-bezier(0.25, 0.46, 0.45, 0.94);
38
+ --ease-enter: cubic-bezier(0.0, 0.0, 0.2, 1.0);
39
+ --ease-exit: cubic-bezier(0.4, 0.0, 1.0, 1.0);
40
+ --duration-pulse: 2.4s;
41
+ --duration-blink: 3s;
42
+ --duration-reveal: 500ms;
43
+ --delay-stagger: 120ms;
44
+
45
+ /* Spacing */
46
+ --border-width: 1.5px;
47
+ --sidebar-width: 200px;
48
+ --header-height: 48px;
49
+ }
50
+
51
+ /* ─── Dark Theme ─────────────────────────────────────────── */
52
+
53
+ [data-theme="dark"] {
54
+ --rust: #C4834A;
55
+ --rust-light: #D4975C;
56
+ --rust-pale: #E0A86A;
57
+ --patina: #6B9962;
58
+ --patina-light: #7DAF72;
59
+ --signal-green: #8BB882;
60
+ --concrete: #1A1A1C;
61
+ --concrete-dark: #252527;
62
+ --concrete-mid: #3A3A3C;
63
+ --charcoal: #E8E4DC;
64
+ --charcoal-mid: #C4C0B8;
65
+ --wire: #4A4A4C;
66
+ --muted: #8A8A8E;
67
+ --muted-light: #6A6A6E;
68
+ --sky: #8FC2DC;
69
+ --surface: #222224;
70
+ --code-bg: #111113;
71
+ }
72
+
73
+ /* Sidebar stays dark in both themes */
74
+ [data-theme="dark"] .station-sidebar {
75
+ --charcoal: #1C1C1E;
76
+ --charcoal-mid: #2C2C2E;
77
+ --wire: #3A3A3C;
78
+ --concrete: #F4F1EC;
79
+ --concrete-dark: #E8E4DC;
80
+ --muted: #8A8A8E;
81
+ --muted-light: #AEAEB2;
82
+ --rust: #8B5A2B;
83
+ }
84
+
85
+ /* Status badges: slightly higher alpha on dark backgrounds */
86
+ [data-theme="dark"] .status-badge--running {
87
+ background: rgba(107, 153, 98, 0.2);
88
+ }
89
+ [data-theme="dark"] .status-badge--completed {
90
+ background: rgba(74, 103, 65, 0.18);
91
+ }
92
+ [data-theme="dark"] .status-badge--failed {
93
+ background: rgba(196, 131, 74, 0.15);
94
+ }
95
+
96
+ /* ─── Animations ──────────────────────────────────────────── */
97
+
98
+ @keyframes signal-pulse {
99
+ 0% {
100
+ box-shadow: 0 0 0 0 rgba(107, 153, 98, 0.4);
101
+ }
102
+ 70% {
103
+ box-shadow: 0 0 0 6px rgba(107, 153, 98, 0);
104
+ }
105
+ 100% {
106
+ box-shadow: 0 0 0 0 rgba(107, 153, 98, 0);
107
+ }
108
+ }
109
+
110
+ @keyframes status-blink {
111
+ 0%, 45%, 55%, 100% {
112
+ opacity: 1;
113
+ }
114
+ 50% {
115
+ opacity: 0.15;
116
+ }
117
+ }
118
+
119
+ @keyframes stagger-reveal {
120
+ from {
121
+ opacity: 0;
122
+ transform: translateY(8px);
123
+ }
124
+ to {
125
+ opacity: 1;
126
+ transform: translateY(0);
127
+ }
128
+ }
129
+
130
+ @keyframes load-sweep {
131
+ 0% { width: 0; }
132
+ 15% { width: 15%; }
133
+ 40% { width: 45%; }
134
+ 70% { width: 70%; }
135
+ 90% { width: 90%; }
136
+ 100% { width: 100%; }
137
+ }
138
+
139
+ /* ─── Reset & Base ────────────────────────────────────────── */
140
+
141
+ *, *::before, *::after {
142
+ box-sizing: border-box;
143
+ margin: 0;
144
+ padding: 0;
145
+ }
146
+
147
+ html {
148
+ font-size: 16px;
149
+ -webkit-font-smoothing: antialiased;
150
+ -moz-osx-font-smoothing: grayscale;
151
+ }
152
+
153
+ body {
154
+ font-family: var(--font-body);
155
+ font-weight: 300;
156
+ color: var(--charcoal);
157
+ background-color: var(--concrete);
158
+ line-height: 1.5;
159
+ }
160
+
161
+ a {
162
+ color: var(--rust);
163
+ text-decoration: none;
164
+ }
165
+
166
+ a:hover {
167
+ color: var(--rust-light);
168
+ }
169
+
170
+ /* ─── Typography Utilities ────────────────────────────────── */
171
+
172
+ .font-display {
173
+ font-family: var(--font-display);
174
+ font-weight: 400;
175
+ }
176
+
177
+ .font-mono {
178
+ font-family: var(--font-mono);
179
+ font-size: 0.875rem;
180
+ }
181
+
182
+ .font-body {
183
+ font-family: var(--font-body);
184
+ font-weight: 300;
185
+ }
186
+
187
+ .section-label {
188
+ font-family: var(--font-mono);
189
+ font-size: 0.65rem;
190
+ text-transform: uppercase;
191
+ letter-spacing: 0.22em;
192
+ color: var(--rust);
193
+ display: flex;
194
+ align-items: center;
195
+ gap: 0.5rem;
196
+ }
197
+
198
+ .section-label::before {
199
+ content: '';
200
+ display: block;
201
+ width: 1.5rem;
202
+ height: var(--border-width);
203
+ background: var(--rust);
204
+ }
205
+
206
+ /* ─── Layout ──────────────────────────────────────────────── */
207
+
208
+ .station-layout {
209
+ display: flex;
210
+ min-height: 100vh;
211
+ }
212
+
213
+ .station-sidebar {
214
+ width: var(--sidebar-width);
215
+ background: var(--charcoal);
216
+ color: var(--concrete);
217
+ display: flex;
218
+ flex-direction: column;
219
+ position: fixed;
220
+ top: 0;
221
+ left: 0;
222
+ bottom: 0;
223
+ z-index: 10;
224
+ transition: width 200ms var(--ease-standard);
225
+ overflow: hidden;
226
+ }
227
+
228
+ .station-sidebar-logo {
229
+ padding: 1rem 1rem 0.75rem;
230
+ border-bottom: var(--border-width) solid var(--wire);
231
+ }
232
+
233
+ .station-sidebar-mark {
234
+ display: flex;
235
+ align-items: center;
236
+ gap: 0.5rem;
237
+ color: var(--concrete);
238
+ }
239
+
240
+ .station-sidebar-mark svg {
241
+ flex-shrink: 0;
242
+ opacity: 0.85;
243
+ }
244
+
245
+ .station-sidebar-logo h1 {
246
+ font-family: var(--font-display);
247
+ font-size: 1.125rem;
248
+ font-weight: 400;
249
+ color: var(--concrete);
250
+ line-height: 1;
251
+ }
252
+
253
+ .station-sidebar-logo span {
254
+ display: block;
255
+ font-family: var(--font-mono);
256
+ font-size: 0.5625rem;
257
+ color: var(--muted);
258
+ text-transform: uppercase;
259
+ letter-spacing: 0.15em;
260
+ margin-top: 0.375rem;
261
+ }
262
+
263
+ .station-sidebar-nav {
264
+ padding: 0.5rem 0;
265
+ flex: 1;
266
+ }
267
+
268
+ .station-sidebar-nav-label {
269
+ font-family: var(--font-mono);
270
+ font-size: 0.5625rem;
271
+ text-transform: uppercase;
272
+ letter-spacing: 0.2em;
273
+ color: var(--muted);
274
+ padding: 0.5rem 1rem 0.25rem;
275
+ }
276
+
277
+ .station-sidebar-nav a {
278
+ display: flex;
279
+ align-items: center;
280
+ gap: 0.5rem;
281
+ padding: 0.375rem 1rem;
282
+ font-family: var(--font-mono);
283
+ font-size: 0.8125rem;
284
+ color: var(--muted-light);
285
+ border-left: 2px solid transparent;
286
+ transition: color 200ms var(--ease-standard), background 200ms var(--ease-standard);
287
+ }
288
+
289
+ .station-sidebar-nav a svg {
290
+ flex-shrink: 0;
291
+ opacity: 0.7;
292
+ transition: opacity 200ms var(--ease-standard);
293
+ }
294
+
295
+ .station-sidebar-nav a:hover {
296
+ color: var(--concrete);
297
+ background: var(--charcoal-mid);
298
+ }
299
+
300
+ .station-sidebar-nav a:hover svg {
301
+ opacity: 1;
302
+ }
303
+
304
+ .station-sidebar-nav a.active {
305
+ color: var(--concrete);
306
+ background: var(--wire);
307
+ border-left: 2px solid var(--rust);
308
+ }
309
+
310
+ .station-sidebar-nav a.active svg {
311
+ opacity: 1;
312
+ }
313
+
314
+ /* ─── Sidebar Collapse Button ────────────────────────────── */
315
+
316
+ .sidebar-collapse-btn {
317
+ display: flex;
318
+ align-items: center;
319
+ justify-content: center;
320
+ width: 100%;
321
+ padding: 0.5rem;
322
+ border: none;
323
+ border-top: var(--border-width) solid var(--wire);
324
+ background: transparent;
325
+ color: var(--muted);
326
+ cursor: pointer;
327
+ transition: color 200ms var(--ease-standard), background 200ms var(--ease-standard);
328
+ }
329
+
330
+ .sidebar-collapse-btn:hover {
331
+ color: var(--concrete);
332
+ background: var(--charcoal-mid);
333
+ }
334
+
335
+ /* ─── Collapsed Sidebar ──────────────────────────────────── */
336
+
337
+ [data-collapsed="true"] .station-sidebar {
338
+ width: 48px;
339
+ }
340
+
341
+ [data-collapsed="true"] .station-main {
342
+ margin-left: 48px;
343
+ }
344
+
345
+ [data-collapsed="true"] .station-sidebar-logo h1,
346
+ [data-collapsed="true"] .station-sidebar-logo > span,
347
+ [data-collapsed="true"] .station-sidebar-nav-label,
348
+ [data-collapsed="true"] .nav-label {
349
+ display: none;
350
+ }
351
+
352
+ [data-collapsed="true"] .station-sidebar-logo {
353
+ padding: 0.75rem;
354
+ display: flex;
355
+ justify-content: center;
356
+ }
357
+
358
+ [data-collapsed="true"] .station-sidebar-mark {
359
+ justify-content: center;
360
+ }
361
+
362
+ [data-collapsed="true"] .station-sidebar-nav a {
363
+ justify-content: center;
364
+ padding: 0.5rem;
365
+ border-left-color: transparent;
366
+ }
367
+
368
+ [data-collapsed="true"] .station-sidebar-nav a.active {
369
+ border-left-color: transparent;
370
+ }
371
+
372
+ .station-main {
373
+ margin-left: var(--sidebar-width);
374
+ flex: 1;
375
+ display: flex;
376
+ flex-direction: column;
377
+ transition: margin-left 200ms var(--ease-standard);
378
+ }
379
+
380
+ .station-header {
381
+ height: var(--header-height);
382
+ background: var(--concrete);
383
+ border-bottom: var(--border-width) solid var(--concrete-dark);
384
+ display: flex;
385
+ align-items: center;
386
+ justify-content: space-between;
387
+ padding: 0 1.5rem;
388
+ position: sticky;
389
+ top: 0;
390
+ z-index: 5;
391
+ }
392
+
393
+ /* ─── Breadcrumb ─────────────────────────────────────────── */
394
+
395
+ .breadcrumb {
396
+ display: flex;
397
+ align-items: center;
398
+ font-family: var(--font-mono);
399
+ font-size: 0.75rem;
400
+ color: var(--muted);
401
+ gap: 0;
402
+ }
403
+
404
+ .breadcrumb-item {
405
+ display: inline-flex;
406
+ align-items: center;
407
+ }
408
+
409
+ .breadcrumb-sep {
410
+ margin: 0 0.375rem;
411
+ color: var(--concrete-mid);
412
+ user-select: none;
413
+ }
414
+
415
+ .breadcrumb-link {
416
+ color: var(--muted);
417
+ text-decoration: none;
418
+ transition: color 150ms var(--ease-standard);
419
+ max-width: 200px;
420
+ overflow: hidden;
421
+ text-overflow: ellipsis;
422
+ white-space: nowrap;
423
+ }
424
+
425
+ .breadcrumb-link:hover {
426
+ color: var(--rust);
427
+ }
428
+
429
+ .breadcrumb-segment {
430
+ color: var(--muted);
431
+ }
432
+
433
+ .breadcrumb-segment--current {
434
+ color: var(--charcoal);
435
+ max-width: 200px;
436
+ overflow: hidden;
437
+ text-overflow: ellipsis;
438
+ white-space: nowrap;
439
+ }
440
+
441
+ .station-content {
442
+ flex: 1;
443
+ padding: 1.5rem;
444
+ max-width: 1100px;
445
+ }
446
+
447
+ /* ─── Status Badges ───────────────────────────────────────── */
448
+
449
+ .status-badge {
450
+ display: inline-flex;
451
+ align-items: center;
452
+ gap: 0.375rem;
453
+ padding: 0.125rem 0.5rem;
454
+ border-radius: 3px;
455
+ font-family: var(--font-mono);
456
+ font-size: 0.6875rem;
457
+ text-transform: uppercase;
458
+ letter-spacing: 0.05em;
459
+ line-height: 1.6;
460
+ }
461
+
462
+ .status-badge-dot {
463
+ width: 6px;
464
+ height: 6px;
465
+ border-radius: 50%;
466
+ flex-shrink: 0;
467
+ }
468
+
469
+ .status-badge--pending {
470
+ background: var(--concrete-dark);
471
+ color: var(--muted);
472
+ }
473
+
474
+ .status-badge--pending .status-badge-dot {
475
+ background: var(--muted);
476
+ }
477
+
478
+ .status-badge--running {
479
+ background: rgba(74, 103, 65, 0.12);
480
+ color: var(--patina);
481
+ }
482
+
483
+ .status-badge--running .status-badge-dot {
484
+ background: var(--signal-green);
485
+ animation: status-blink var(--duration-blink) ease-in-out infinite;
486
+ }
487
+
488
+ .status-badge--completed {
489
+ background: rgba(74, 103, 65, 0.1);
490
+ color: var(--patina);
491
+ }
492
+
493
+ .status-badge--completed .status-badge-dot {
494
+ background: var(--patina);
495
+ }
496
+
497
+ .status-badge--failed {
498
+ background: rgba(139, 90, 43, 0.1);
499
+ color: var(--rust);
500
+ }
501
+
502
+ .status-badge--failed .status-badge-dot {
503
+ background: var(--rust);
504
+ }
505
+
506
+ .status-badge--cancelled {
507
+ background: rgba(58, 58, 60, 0.08);
508
+ color: var(--muted);
509
+ }
510
+
511
+ .status-badge--cancelled .status-badge-dot {
512
+ background: var(--wire);
513
+ }
514
+
515
+ .status-badge--skipped {
516
+ background: transparent;
517
+ color: var(--muted);
518
+ border: var(--border-width) dashed var(--muted-light);
519
+ }
520
+
521
+ .status-badge--skipped .status-badge-dot {
522
+ background: var(--muted-light);
523
+ }
524
+
525
+ /* ─── Tables ──────────────────────────────────────────────── */
526
+
527
+ .station-table {
528
+ width: 100%;
529
+ border-collapse: collapse;
530
+ font-family: var(--font-body);
531
+ font-size: 0.875rem;
532
+ }
533
+
534
+ .station-table th {
535
+ font-family: var(--font-mono);
536
+ font-size: 0.6875rem;
537
+ text-transform: uppercase;
538
+ letter-spacing: 0.1em;
539
+ color: var(--muted);
540
+ text-align: left;
541
+ padding: 0.5rem 0.75rem;
542
+ border-bottom: var(--border-width) solid var(--concrete-dark);
543
+ font-weight: 400;
544
+ }
545
+
546
+ .station-table td {
547
+ padding: 0.625rem 0.75rem;
548
+ border-bottom: 1px solid var(--concrete-dark);
549
+ vertical-align: middle;
550
+ }
551
+
552
+ .station-table tr:hover td {
553
+ background: var(--concrete-dark);
554
+ }
555
+
556
+ .station-table .mono {
557
+ font-family: var(--font-mono);
558
+ font-size: 0.8125rem;
559
+ }
560
+
561
+ .station-table .truncate {
562
+ max-width: 120px;
563
+ overflow: hidden;
564
+ text-overflow: ellipsis;
565
+ white-space: nowrap;
566
+ }
567
+
568
+ /* ─── Cards ───────────────────────────────────────────────── */
569
+
570
+ .station-card {
571
+ background: var(--surface);
572
+ border: var(--border-width) solid var(--concrete-dark);
573
+ border-radius: 4px;
574
+ padding: 1.25rem;
575
+ }
576
+
577
+ .station-card h3 {
578
+ font-family: var(--font-display);
579
+ font-size: 1.125rem;
580
+ margin-bottom: 0.75rem;
581
+ }
582
+
583
+ /* ─── Stat Cards ──────────────────────────────────────────── */
584
+
585
+ .stat-grid {
586
+ display: grid;
587
+ grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
588
+ gap: 0.75rem;
589
+ margin-bottom: 1.5rem;
590
+ }
591
+
592
+ .stat-card {
593
+ background: var(--surface);
594
+ border: var(--border-width) solid var(--concrete-dark);
595
+ border-radius: 4px;
596
+ padding: 1rem;
597
+ }
598
+
599
+ .stat-card-label {
600
+ font-family: var(--font-mono);
601
+ font-size: 0.625rem;
602
+ text-transform: uppercase;
603
+ letter-spacing: 0.15em;
604
+ color: var(--muted);
605
+ margin-bottom: 0.25rem;
606
+ }
607
+
608
+ .stat-card-value {
609
+ font-family: var(--font-display);
610
+ font-size: 2rem;
611
+ color: var(--charcoal);
612
+ line-height: 1.1;
613
+ }
614
+
615
+ /* ─── Pulse Dot (Connection Indicator) ────────────────────── */
616
+
617
+ .pulse-dot {
618
+ width: 8px;
619
+ height: 8px;
620
+ border-radius: 50%;
621
+ background: var(--signal-green);
622
+ animation: signal-pulse var(--duration-pulse) ease-out infinite;
623
+ }
624
+
625
+ .pulse-dot--disconnected {
626
+ background: var(--muted);
627
+ animation: none;
628
+ }
629
+
630
+ /* ─── Step Timeline ───────────────────────────────────────── */
631
+
632
+ .step-timeline {
633
+ position: relative;
634
+ padding-left: 1.5rem;
635
+ }
636
+
637
+ .step-timeline::before {
638
+ content: '';
639
+ position: absolute;
640
+ left: 5px;
641
+ top: 0;
642
+ bottom: 0;
643
+ width: var(--border-width);
644
+ background: var(--concrete-dark);
645
+ }
646
+
647
+ .step-timeline-item {
648
+ position: relative;
649
+ padding: 0.75rem 0;
650
+ }
651
+
652
+ .step-timeline-dot {
653
+ position: absolute;
654
+ left: -1.5rem;
655
+ top: 1rem;
656
+ width: 10px;
657
+ height: 10px;
658
+ border-radius: 50%;
659
+ border: var(--border-width) solid var(--concrete-dark);
660
+ background: var(--surface);
661
+ transform: translateX(-50%);
662
+ left: calc(-1.5rem + 5px);
663
+ }
664
+
665
+ .step-timeline-dot--completed {
666
+ background: var(--patina);
667
+ border-color: var(--patina);
668
+ }
669
+
670
+ .step-timeline-dot--running {
671
+ background: var(--signal-green);
672
+ border-color: var(--signal-green);
673
+ animation: signal-pulse var(--duration-pulse) ease-out infinite;
674
+ }
675
+
676
+ .step-timeline-dot--failed {
677
+ background: var(--rust);
678
+ border-color: var(--rust);
679
+ }
680
+
681
+ .step-timeline-name {
682
+ font-family: var(--font-mono);
683
+ font-size: 0.8125rem;
684
+ color: var(--charcoal);
685
+ }
686
+
687
+ .step-timeline-meta {
688
+ font-family: var(--font-mono);
689
+ font-size: 0.6875rem;
690
+ color: var(--muted);
691
+ margin-top: 0.125rem;
692
+ }
693
+
694
+ /* ─── JSON Viewer ─────────────────────────────────────────── */
695
+
696
+ .json-viewer {
697
+ font-family: var(--font-mono);
698
+ font-size: 0.8125rem;
699
+ line-height: 1.5;
700
+ background: var(--code-bg);
701
+ color: var(--code-text);
702
+ border-radius: 4px;
703
+ padding: 1rem;
704
+ overflow-x: auto;
705
+ }
706
+
707
+ .json-viewer .json-key {
708
+ color: var(--code-key);
709
+ }
710
+
711
+ .json-viewer .json-string {
712
+ color: var(--code-string);
713
+ }
714
+
715
+ .json-viewer .json-number {
716
+ color: var(--code-number);
717
+ }
718
+
719
+ .json-viewer .json-boolean {
720
+ color: var(--code-bool);
721
+ }
722
+
723
+ .json-viewer .json-null {
724
+ color: var(--code-null);
725
+ }
726
+
727
+ /* ─── Empty States ────────────────────────────────────────── */
728
+
729
+ .empty-state {
730
+ text-align: center;
731
+ padding: 3rem 1rem;
732
+ color: var(--muted);
733
+ }
734
+
735
+ .empty-state-text {
736
+ font-family: var(--font-mono);
737
+ font-size: 0.8125rem;
738
+ }
739
+
740
+ /* ─── Loading Bar ─────────────────────────────────────────── */
741
+
742
+ .loading-bar {
743
+ height: var(--border-width);
744
+ background: var(--concrete-dark);
745
+ position: relative;
746
+ overflow: hidden;
747
+ border-radius: 1px;
748
+ }
749
+
750
+ .loading-bar-fill {
751
+ height: 100%;
752
+ background: var(--rust);
753
+ animation: load-sweep 3.5s cubic-bezier(0.4, 0, 0.2, 1) infinite;
754
+ }
755
+
756
+ /* ─── DAG View ────────────────────────────────────────────── */
757
+
758
+ .dag-container {
759
+ background: var(--surface);
760
+ border: var(--border-width) solid var(--concrete-dark);
761
+ border-radius: 4px;
762
+ padding: 1.5rem;
763
+ overflow-x: auto;
764
+ }
765
+
766
+ .dag-node {
767
+ cursor: pointer;
768
+ }
769
+
770
+ .dag-node rect {
771
+ stroke-width: 1.5;
772
+ rx: 4;
773
+ ry: 4;
774
+ }
775
+
776
+ .dag-node text {
777
+ font-family: var(--font-mono);
778
+ font-size: 11px;
779
+ }
780
+
781
+ .dag-edge {
782
+ fill: none;
783
+ stroke: var(--concrete-mid);
784
+ stroke-width: 1.5;
785
+ transition: stroke 300ms var(--ease-standard);
786
+ }
787
+
788
+ .dag-edge--active {
789
+ stroke: var(--patina);
790
+ stroke-width: 2;
791
+ }
792
+
793
+ /* ─── Page Title ──────────────────────────────────────────── */
794
+
795
+ .page-title {
796
+ font-family: var(--font-display);
797
+ font-size: 1.75rem;
798
+ font-weight: 400;
799
+ margin-bottom: 1.5rem;
800
+ color: var(--charcoal);
801
+ }
802
+
803
+ /* ─── Buttons ─────────────────────────────────────────────── */
804
+
805
+ .btn {
806
+ display: inline-flex;
807
+ align-items: center;
808
+ gap: 0.375rem;
809
+ font-family: var(--font-mono);
810
+ font-size: 0.75rem;
811
+ padding: 0.375rem 0.75rem;
812
+ border: var(--border-width) solid var(--charcoal);
813
+ border-radius: 3px;
814
+ background: var(--charcoal);
815
+ color: var(--concrete);
816
+ cursor: pointer;
817
+ transition: transform 200ms var(--ease-standard), background 200ms var(--ease-standard);
818
+ }
819
+
820
+ .btn:hover {
821
+ transform: translateY(-2px);
822
+ background: var(--rust);
823
+ border-color: var(--rust);
824
+ }
825
+
826
+ .btn--primary {
827
+ background: var(--patina);
828
+ border-color: var(--patina);
829
+ }
830
+
831
+ .btn--primary:hover {
832
+ background: var(--patina-light);
833
+ border-color: var(--patina-light);
834
+ }
835
+
836
+ .btn--primary:disabled {
837
+ opacity: 0.5;
838
+ cursor: not-allowed;
839
+ transform: none;
840
+ }
841
+
842
+ .btn--sm {
843
+ font-size: 0.6875rem;
844
+ padding: 0.25rem 0.5rem;
845
+ }
846
+
847
+ .btn--danger {
848
+ background: var(--rust);
849
+ border-color: var(--rust);
850
+ }
851
+
852
+ .btn--danger:hover {
853
+ background: var(--rust-light);
854
+ border-color: var(--rust-light);
855
+ }
856
+
857
+ .btn:disabled {
858
+ opacity: 0.5;
859
+ cursor: not-allowed;
860
+ transform: none;
861
+ }
862
+
863
+ /* ─── Detail Row ──────────────────────────────────────────── */
864
+
865
+ .detail-grid {
866
+ display: grid;
867
+ grid-template-columns: auto 1fr;
868
+ gap: 0.25rem 1rem;
869
+ font-size: 0.875rem;
870
+ }
871
+
872
+ .detail-label {
873
+ font-family: var(--font-mono);
874
+ font-size: 0.75rem;
875
+ color: var(--muted);
876
+ text-transform: uppercase;
877
+ letter-spacing: 0.05em;
878
+ }
879
+
880
+ .detail-value {
881
+ font-family: var(--font-body);
882
+ color: var(--charcoal);
883
+ }
884
+
885
+ .detail-value.mono {
886
+ font-family: var(--font-mono);
887
+ font-size: 0.8125rem;
888
+ }
889
+
890
+ /* ─── Stagger Reveal ──────────────────────────────────────── */
891
+
892
+ .reveal-item {
893
+ animation: stagger-reveal var(--duration-reveal) var(--ease-standard) both;
894
+ }
895
+
896
+ /* ─── Input Textarea ─────────────────────────────────────── */
897
+
898
+ .input-textarea {
899
+ width: 100%;
900
+ margin-top: 0.5rem;
901
+ padding: 0.75rem;
902
+ font-family: var(--font-mono);
903
+ font-size: 0.8125rem;
904
+ line-height: 1.5;
905
+ color: var(--code-text);
906
+ background: var(--code-bg);
907
+ border: var(--border-width) solid var(--wire);
908
+ border-radius: 4px;
909
+ resize: vertical;
910
+ outline: none;
911
+ }
912
+
913
+ .input-textarea:focus {
914
+ border-color: var(--patina);
915
+ }
916
+
917
+ /* ─── Log Container ──────────────────────────────────────── */
918
+
919
+ .log-container {
920
+ background: var(--code-bg);
921
+ border-radius: 4px;
922
+ max-height: 400px;
923
+ overflow-y: auto;
924
+ overflow-x: auto;
925
+ }
926
+
927
+ .log-line {
928
+ display: flex;
929
+ gap: 0.625rem;
930
+ padding: 0.125rem 0.75rem;
931
+ font-family: var(--font-mono);
932
+ font-size: 0.75rem;
933
+ line-height: 1.6;
934
+ border-bottom: 1px solid var(--wire);
935
+ }
936
+
937
+ .log-line:last-of-type {
938
+ border-bottom: none;
939
+ }
940
+
941
+ .log-timestamp {
942
+ color: var(--muted);
943
+ flex-shrink: 0;
944
+ min-width: 5.5rem;
945
+ }
946
+
947
+ .log-level {
948
+ flex-shrink: 0;
949
+ min-width: 2rem;
950
+ color: var(--muted);
951
+ }
952
+
953
+ .log-level[data-level="stderr"] {
954
+ color: var(--rust-pale);
955
+ }
956
+
957
+ .log-message {
958
+ color: var(--code-text);
959
+ white-space: pre-wrap;
960
+ word-break: break-all;
961
+ }
962
+
963
+ /* ─── Schema Reference ───────────────────────────────────── */
964
+
965
+ .schema-ref {
966
+ border: var(--border-width) solid var(--concrete-dark);
967
+ border-radius: 4px;
968
+ padding: 0.75rem;
969
+ background: var(--surface);
970
+ margin-bottom: 0.75rem;
971
+ }
972
+
973
+ .schema-ref-title {
974
+ font-family: var(--font-mono);
975
+ font-size: 0.625rem;
976
+ text-transform: uppercase;
977
+ letter-spacing: 0.15em;
978
+ color: var(--muted);
979
+ margin-bottom: 0.5rem;
980
+ }
981
+
982
+ .schema-field {
983
+ display: flex;
984
+ justify-content: space-between;
985
+ padding: 0.25rem 0;
986
+ font-family: var(--font-mono);
987
+ font-size: 0.8125rem;
988
+ border-bottom: 1px solid var(--concrete-dark);
989
+ }
990
+
991
+ .schema-field:last-child {
992
+ border-bottom: none;
993
+ }
994
+
995
+ .schema-field-name {
996
+ color: var(--charcoal);
997
+ }
998
+
999
+ .schema-field-type {
1000
+ color: var(--muted);
1001
+ }
1002
+
1003
+ .schema-field-type--optional::after {
1004
+ content: '?';
1005
+ color: var(--rust-pale);
1006
+ }
1007
+
1008
+ /* ─── Error Block ────────────────────────────────────────── */
1009
+
1010
+ .error-block {
1011
+ background: rgba(139, 90, 43, 0.08);
1012
+ border: var(--border-width) solid rgba(139, 90, 43, 0.2);
1013
+ border-radius: 4px;
1014
+ padding: 0.75rem 1rem;
1015
+ font-family: var(--font-mono);
1016
+ font-size: 0.8125rem;
1017
+ color: var(--rust);
1018
+ white-space: pre-wrap;
1019
+ word-break: break-word;
1020
+ line-height: 1.5;
1021
+ }
1022
+
1023
+ /* ─── Copy Button ────────────────────────────────────────── */
1024
+
1025
+ .copy-btn {
1026
+ position: absolute;
1027
+ top: 0.5rem;
1028
+ right: 0.5rem;
1029
+ background: var(--wire);
1030
+ border: none;
1031
+ border-radius: 3px;
1032
+ padding: 0.25rem 0.5rem;
1033
+ font-family: var(--font-mono);
1034
+ font-size: 0.625rem;
1035
+ color: var(--muted-light);
1036
+ cursor: pointer;
1037
+ opacity: 0;
1038
+ transition: opacity 200ms var(--ease-standard);
1039
+ text-transform: uppercase;
1040
+ letter-spacing: 0.1em;
1041
+ }
1042
+
1043
+ .json-viewer:hover .copy-btn,
1044
+ .copy-btn:focus {
1045
+ opacity: 1;
1046
+ }
1047
+
1048
+ .copy-btn:hover {
1049
+ color: var(--concrete);
1050
+ background: var(--muted);
1051
+ }
1052
+
1053
+ /* ─── Filter Bar ─────────────────────────────────────────── */
1054
+
1055
+ .filter-bar {
1056
+ display: flex;
1057
+ gap: 0.375rem;
1058
+ margin-bottom: 1rem;
1059
+ }
1060
+
1061
+ .filter-btn {
1062
+ font-family: var(--font-mono);
1063
+ font-size: 0.6875rem;
1064
+ padding: 0.25rem 0.625rem;
1065
+ border: var(--border-width) solid var(--concrete-dark);
1066
+ border-radius: 3px;
1067
+ background: var(--surface);
1068
+ color: var(--muted);
1069
+ cursor: pointer;
1070
+ text-transform: uppercase;
1071
+ letter-spacing: 0.05em;
1072
+ transition: all 150ms var(--ease-standard);
1073
+ }
1074
+
1075
+ .filter-btn:hover {
1076
+ border-color: var(--charcoal);
1077
+ color: var(--charcoal);
1078
+ }
1079
+
1080
+ .filter-btn--active {
1081
+ background: var(--charcoal);
1082
+ border-color: var(--charcoal);
1083
+ color: var(--concrete);
1084
+ }
1085
+
1086
+ /* ─── Node Detail Panel ──────────────────────────────────── */
1087
+
1088
+ .node-detail {
1089
+ background: var(--surface);
1090
+ border: var(--border-width) solid var(--concrete-dark);
1091
+ border-radius: 4px;
1092
+ padding: 1.25rem;
1093
+ margin-top: 1rem;
1094
+ }
1095
+
1096
+ .node-detail-header {
1097
+ display: flex;
1098
+ align-items: center;
1099
+ justify-content: space-between;
1100
+ margin-bottom: 1rem;
1101
+ }
1102
+
1103
+ .node-detail-title {
1104
+ font-family: var(--font-mono);
1105
+ font-size: 0.875rem;
1106
+ font-weight: 500;
1107
+ }
1108
+
1109
+ /* ─── Detail Section ─────────────────────────────────────── */
1110
+
1111
+ .detail-section {
1112
+ margin-bottom: 1.5rem;
1113
+ }
1114
+
1115
+ .detail-section:last-child {
1116
+ margin-bottom: 0;
1117
+ }
1118
+
1119
+ .detail-section-label {
1120
+ font-family: var(--font-mono);
1121
+ font-size: 0.65rem;
1122
+ text-transform: uppercase;
1123
+ letter-spacing: 0.22em;
1124
+ color: var(--rust);
1125
+ margin-bottom: 0.5rem;
1126
+ display: flex;
1127
+ align-items: center;
1128
+ gap: 0.5rem;
1129
+ }
1130
+
1131
+ .detail-section-label::before {
1132
+ content: '';
1133
+ display: block;
1134
+ width: 1rem;
1135
+ height: var(--border-width);
1136
+ background: var(--rust);
1137
+ }
1138
+
1139
+ /* ─── Meta Value Link ────────────────────────────────────── */
1140
+
1141
+ .meta-value--link {
1142
+ color: var(--rust);
1143
+ cursor: pointer;
1144
+ }
1145
+
1146
+ .meta-value--link:hover {
1147
+ color: var(--rust-light);
1148
+ text-decoration: underline;
1149
+ }
1150
+
1151
+ /* ─── DAG Node Selected ──────────────────────────────────── */
1152
+
1153
+ .dag-node--selected rect {
1154
+ stroke: var(--rust) !important;
1155
+ stroke-width: 2.5 !important;
1156
+ }
1157
+
1158
+ /* ─── JSON Parse Error ───────────────────────────────────── */
1159
+
1160
+ .json-parse-error {
1161
+ font-family: var(--font-mono);
1162
+ font-size: 0.6875rem;
1163
+ color: var(--rust);
1164
+ margin-top: 0.25rem;
1165
+ }
1166
+
1167
+ /* ─── Clickable Table Rows ───────────────────────────────── */
1168
+
1169
+ .station-table tr.clickable-row {
1170
+ cursor: pointer;
1171
+ }
1172
+
1173
+ .station-table tr.clickable-row:hover td {
1174
+ background: var(--concrete-dark);
1175
+ }
1176
+
1177
+ /* ─── Page Header ────────────────────────────────────────── */
1178
+
1179
+ .page-header {
1180
+ display: flex;
1181
+ align-items: center;
1182
+ justify-content: space-between;
1183
+ margin-bottom: 1.5rem;
1184
+ }
1185
+
1186
+ .page-header-actions {
1187
+ display: flex;
1188
+ gap: 0.5rem;
1189
+ align-items: center;
1190
+ }
1191
+
1192
+ /* ─── Schema Side by Side ────────────────────────────────── */
1193
+
1194
+ .schema-pair {
1195
+ display: grid;
1196
+ grid-template-columns: 1fr 1fr;
1197
+ gap: 1rem;
1198
+ }
1199
+
1200
+ /* ─── Inline Config Grid ─────────────────────────────────── */
1201
+
1202
+ .config-grid {
1203
+ display: flex;
1204
+ flex-wrap: wrap;
1205
+ gap: 1.5rem;
1206
+ margin-bottom: 0.5rem;
1207
+ }
1208
+
1209
+ .config-item {
1210
+ display: flex;
1211
+ flex-direction: column;
1212
+ gap: 0.125rem;
1213
+ }
1214
+
1215
+ .config-item-label {
1216
+ font-family: var(--font-mono);
1217
+ font-size: 0.625rem;
1218
+ text-transform: uppercase;
1219
+ letter-spacing: 0.15em;
1220
+ color: var(--muted);
1221
+ }
1222
+
1223
+ .config-item-value {
1224
+ font-family: var(--font-mono);
1225
+ font-size: 0.875rem;
1226
+ color: var(--charcoal);
1227
+ }
1228
+
1229
+ /* ─── Workflow Layout (sidebar + panel) ──────────────────── */
1230
+
1231
+ .workflow-layout {
1232
+ display: flex;
1233
+ gap: 0;
1234
+ align-items: flex-start;
1235
+ border: var(--border-width) solid var(--concrete-dark);
1236
+ border-radius: 4px;
1237
+ background: var(--surface);
1238
+ min-height: 400px;
1239
+ }
1240
+
1241
+ .workflow-sidebar {
1242
+ width: 220px;
1243
+ flex-shrink: 0;
1244
+ border-right: var(--border-width) solid var(--concrete-dark);
1245
+ padding: 0.5rem 0;
1246
+ overflow-y: auto;
1247
+ max-height: calc(100vh - var(--header-height) - 10rem);
1248
+ position: sticky;
1249
+ top: calc(var(--header-height) + 1.5rem);
1250
+ }
1251
+
1252
+ .workflow-sidebar-label {
1253
+ font-family: var(--font-mono);
1254
+ font-size: 0.6rem;
1255
+ text-transform: uppercase;
1256
+ letter-spacing: 0.2em;
1257
+ color: var(--muted);
1258
+ padding: 0.375rem 0.75rem 0.5rem;
1259
+ }
1260
+
1261
+ .workflow-node-item {
1262
+ display: flex;
1263
+ align-items: center;
1264
+ gap: 0.5rem;
1265
+ width: 100%;
1266
+ padding: 0.5rem 0.75rem;
1267
+ border: none;
1268
+ background: none;
1269
+ cursor: pointer;
1270
+ font-family: var(--font-mono);
1271
+ font-size: 0.8125rem;
1272
+ color: var(--charcoal);
1273
+ text-align: left;
1274
+ transition: background 150ms var(--ease-standard);
1275
+ border-left: 2px solid transparent;
1276
+ }
1277
+
1278
+ .workflow-node-item:hover {
1279
+ background: var(--concrete);
1280
+ }
1281
+
1282
+ .workflow-node-item--active {
1283
+ background: var(--concrete);
1284
+ border-left-color: var(--patina);
1285
+ }
1286
+
1287
+ .workflow-node-item--failed {
1288
+ color: var(--rust);
1289
+ }
1290
+
1291
+ .workflow-node-dot {
1292
+ width: 8px;
1293
+ height: 8px;
1294
+ border-radius: 50%;
1295
+ flex-shrink: 0;
1296
+ }
1297
+
1298
+ .workflow-node-dot--running {
1299
+ animation: signal-pulse var(--duration-pulse) ease-out infinite;
1300
+ }
1301
+
1302
+ .workflow-node-name {
1303
+ flex: 1;
1304
+ overflow: hidden;
1305
+ text-overflow: ellipsis;
1306
+ white-space: nowrap;
1307
+ }
1308
+
1309
+ .workflow-node-duration {
1310
+ font-size: 0.6875rem;
1311
+ color: var(--muted);
1312
+ flex-shrink: 0;
1313
+ }
1314
+
1315
+ .workflow-panel {
1316
+ flex: 1;
1317
+ min-width: 0;
1318
+ padding: 1.25rem;
1319
+ overflow-y: auto;
1320
+ max-height: calc(100vh - var(--header-height) - 10rem);
1321
+ }
1322
+
1323
+ .node-detail-meta {
1324
+ font-family: var(--font-mono);
1325
+ font-size: 0.75rem;
1326
+ color: var(--muted);
1327
+ }
1328
+
1329
+ @media (max-width: 768px) {
1330
+ .workflow-layout {
1331
+ flex-direction: column;
1332
+ min-height: auto;
1333
+ }
1334
+ .workflow-sidebar {
1335
+ width: 100%;
1336
+ max-height: none;
1337
+ position: static;
1338
+ border-right: none;
1339
+ border-bottom: var(--border-width) solid var(--concrete-dark);
1340
+ display: flex;
1341
+ overflow-x: auto;
1342
+ padding: 0.25rem;
1343
+ gap: 0.25rem;
1344
+ }
1345
+ .workflow-sidebar-label {
1346
+ display: none;
1347
+ }
1348
+ .workflow-node-item {
1349
+ white-space: nowrap;
1350
+ padding: 0.375rem 0.5rem;
1351
+ border-left: none;
1352
+ border-bottom: 2px solid transparent;
1353
+ font-size: 0.75rem;
1354
+ }
1355
+ .workflow-node-item--active {
1356
+ border-bottom-color: var(--patina);
1357
+ border-left-color: transparent;
1358
+ }
1359
+ .workflow-panel {
1360
+ max-height: none;
1361
+ }
1362
+ }
1363
+
1364
+ /* ─── Collapsible Sections ───────────────────────────────── */
1365
+
1366
+ .collapsible-header {
1367
+ display: flex;
1368
+ align-items: center;
1369
+ gap: 0.375rem;
1370
+ font-family: var(--font-mono);
1371
+ font-size: 0.75rem;
1372
+ color: var(--muted);
1373
+ cursor: pointer;
1374
+ background: none;
1375
+ border: none;
1376
+ padding: 0.375rem 0;
1377
+ text-transform: uppercase;
1378
+ letter-spacing: 0.1em;
1379
+ transition: color 150ms var(--ease-standard);
1380
+ }
1381
+
1382
+ .collapsible-header:hover {
1383
+ color: var(--charcoal);
1384
+ }
1385
+
1386
+ .collapsible-chevron {
1387
+ font-size: 0.625rem;
1388
+ line-height: 1;
1389
+ }
1390
+
1391
+ .collapsible-content {
1392
+ padding-top: 0.5rem;
1393
+ }
1394
+
1395
+ /* ─── Activity Row (clickable live activity items) ───────── */
1396
+
1397
+ .activity-row {
1398
+ cursor: pointer;
1399
+ border-radius: 2px;
1400
+ transition: background 150ms var(--ease-standard);
1401
+ }
1402
+
1403
+ .activity-row:hover {
1404
+ background: var(--concrete-dark);
1405
+ }
1406
+
1407
+ /* ─── Theme Toggle ───────────────────────────────────────── */
1408
+
1409
+ .theme-toggle {
1410
+ display: flex;
1411
+ align-items: center;
1412
+ justify-content: center;
1413
+ width: 28px;
1414
+ height: 28px;
1415
+ border: none;
1416
+ border-radius: 4px;
1417
+ background: transparent;
1418
+ color: var(--muted);
1419
+ cursor: pointer;
1420
+ transition: color 200ms var(--ease-standard), background 200ms var(--ease-standard);
1421
+ }
1422
+
1423
+ .theme-toggle:hover {
1424
+ color: var(--charcoal);
1425
+ background: var(--concrete-dark);
1426
+ }
1427
+
1428
+ /* ─── Login Page ─────────────────────────────────────────── */
1429
+
1430
+ .login-page {
1431
+ display: flex;
1432
+ align-items: center;
1433
+ justify-content: center;
1434
+ min-height: 100vh;
1435
+ background: var(--concrete);
1436
+ }
1437
+
1438
+ .login-card {
1439
+ width: 320px;
1440
+ background: var(--surface);
1441
+ border: var(--border-width) solid var(--concrete-dark);
1442
+ border-radius: 4px;
1443
+ padding: 2rem 1.5rem;
1444
+ }
1445
+
1446
+ .login-header {
1447
+ display: flex;
1448
+ align-items: center;
1449
+ gap: 0.5rem;
1450
+ margin-bottom: 1.5rem;
1451
+ color: var(--charcoal);
1452
+ }
1453
+
1454
+ .login-header h1 {
1455
+ font-family: var(--font-display);
1456
+ font-size: 1.25rem;
1457
+ font-weight: 400;
1458
+ }
1459
+
1460
+ .login-field {
1461
+ margin-bottom: 0.75rem;
1462
+ }
1463
+
1464
+ .login-field label {
1465
+ display: block;
1466
+ font-family: var(--font-mono);
1467
+ font-size: 0.625rem;
1468
+ text-transform: uppercase;
1469
+ letter-spacing: 0.15em;
1470
+ color: var(--muted);
1471
+ margin-bottom: 0.25rem;
1472
+ }
1473
+
1474
+ .login-field input {
1475
+ width: 100%;
1476
+ padding: 0.5rem 0.625rem;
1477
+ font-family: var(--font-mono);
1478
+ font-size: 0.8125rem;
1479
+ color: var(--charcoal);
1480
+ background: var(--concrete);
1481
+ border: var(--border-width) solid var(--concrete-dark);
1482
+ border-radius: 3px;
1483
+ outline: none;
1484
+ transition: border-color 150ms var(--ease-standard);
1485
+ }
1486
+
1487
+ .login-field input:focus {
1488
+ border-color: var(--patina);
1489
+ }
1490
+
1491
+ .login-error {
1492
+ font-family: var(--font-mono);
1493
+ font-size: 0.75rem;
1494
+ color: var(--rust);
1495
+ margin-bottom: 0.75rem;
1496
+ }
1497
+
1498
+ .login-btn {
1499
+ width: 100%;
1500
+ justify-content: center;
1501
+ margin-top: 0.25rem;
1502
+ }
1503
+
1504
+ /* ─── Logout Button ──────────────────────────────────────── */
1505
+
1506
+ .logout-btn {
1507
+ display: flex;
1508
+ align-items: center;
1509
+ justify-content: center;
1510
+ width: 28px;
1511
+ height: 28px;
1512
+ border: none;
1513
+ border-radius: 4px;
1514
+ background: transparent;
1515
+ color: var(--muted);
1516
+ cursor: pointer;
1517
+ transition: color 200ms var(--ease-standard), background 200ms var(--ease-standard);
1518
+ }
1519
+
1520
+ .logout-btn:hover {
1521
+ color: var(--rust);
1522
+ background: var(--concrete-dark);
1523
+ }