@useavalon/avalon 0.1.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 (159) hide show
  1. package/README.md +54 -0
  2. package/mod.ts +301 -0
  3. package/package.json +85 -0
  4. package/src/build/README.md +310 -0
  5. package/src/build/integration-bundler-plugin.ts +116 -0
  6. package/src/build/integration-config.ts +168 -0
  7. package/src/build/integration-detection-plugin.ts +117 -0
  8. package/src/build/integration-resolver-plugin.ts +90 -0
  9. package/src/build/island-manifest.ts +269 -0
  10. package/src/build/island-types-generator.ts +476 -0
  11. package/src/build/mdx-island-transform.ts +464 -0
  12. package/src/build/mdx-plugin.ts +98 -0
  13. package/src/build/page-island-transform.ts +598 -0
  14. package/src/build/prop-extractors/index.ts +21 -0
  15. package/src/build/prop-extractors/lit.ts +140 -0
  16. package/src/build/prop-extractors/qwik.ts +16 -0
  17. package/src/build/prop-extractors/solid.ts +125 -0
  18. package/src/build/prop-extractors/svelte.ts +194 -0
  19. package/src/build/prop-extractors/vue.ts +111 -0
  20. package/src/build/sidecar-file-manager.ts +104 -0
  21. package/src/build/sidecar-renderer.ts +30 -0
  22. package/src/client/adapters/index.ts +13 -0
  23. package/src/client/adapters/lit-adapter.ts +654 -0
  24. package/src/client/adapters/preact-adapter.ts +331 -0
  25. package/src/client/adapters/qwik-adapter.ts +345 -0
  26. package/src/client/adapters/react-adapter.ts +353 -0
  27. package/src/client/adapters/solid-adapter.ts +451 -0
  28. package/src/client/adapters/svelte-adapter.ts +524 -0
  29. package/src/client/adapters/vue-adapter.ts +467 -0
  30. package/src/client/components.ts +35 -0
  31. package/src/client/css-hmr-handler.ts +344 -0
  32. package/src/client/framework-adapter.ts +462 -0
  33. package/src/client/hmr-coordinator.ts +396 -0
  34. package/src/client/hmr-error-overlay.js +533 -0
  35. package/src/client/main.js +816 -0
  36. package/src/client/tests/css-hmr-handler.test.ts +360 -0
  37. package/src/client/tests/framework-adapter.test.ts +519 -0
  38. package/src/client/tests/hmr-coordinator.test.ts +176 -0
  39. package/src/client/tests/hydration-option-parsing.test.ts +107 -0
  40. package/src/client/tests/lit-adapter.test.ts +427 -0
  41. package/src/client/tests/preact-adapter.test.ts +353 -0
  42. package/src/client/tests/qwik-adapter.test.ts +343 -0
  43. package/src/client/tests/react-adapter.test.ts +317 -0
  44. package/src/client/tests/solid-adapter.test.ts +396 -0
  45. package/src/client/tests/svelte-adapter.test.ts +387 -0
  46. package/src/client/tests/vue-adapter.test.ts +407 -0
  47. package/src/client/types/framework-runtime.d.ts +68 -0
  48. package/src/client/types/vite-hmr.d.ts +46 -0
  49. package/src/client/types/vite-virtual-modules.d.ts +60 -0
  50. package/src/components/Image.tsx +123 -0
  51. package/src/components/IslandErrorBoundary.tsx +145 -0
  52. package/src/components/LayoutDataErrorBoundary.tsx +141 -0
  53. package/src/components/LayoutErrorBoundary.tsx +127 -0
  54. package/src/components/PersistentIsland.tsx +52 -0
  55. package/src/components/StreamingErrorBoundary.tsx +233 -0
  56. package/src/components/StreamingLayout.tsx +538 -0
  57. package/src/components/tests/component-analyzer.test.ts +96 -0
  58. package/src/components/tests/component-detection.test.ts +347 -0
  59. package/src/components/tests/persistent-islands.test.ts +398 -0
  60. package/src/core/components/component-analyzer.ts +192 -0
  61. package/src/core/components/component-detection.ts +508 -0
  62. package/src/core/components/enhanced-framework-detector.ts +500 -0
  63. package/src/core/components/framework-registry.ts +563 -0
  64. package/src/core/components/tests/enhanced-framework-detector.test.ts +577 -0
  65. package/src/core/components/tests/framework-registry.test.ts +465 -0
  66. package/src/core/content/mdx-processor.ts +46 -0
  67. package/src/core/integrations/README.md +282 -0
  68. package/src/core/integrations/index.ts +19 -0
  69. package/src/core/integrations/loader.ts +125 -0
  70. package/src/core/integrations/registry.ts +195 -0
  71. package/src/core/islands/island-persistence.ts +325 -0
  72. package/src/core/islands/island-state-serializer.ts +258 -0
  73. package/src/core/islands/persistent-island-context.tsx +80 -0
  74. package/src/core/islands/use-persistent-state.ts +68 -0
  75. package/src/core/layout/enhanced-layout-resolver.ts +322 -0
  76. package/src/core/layout/layout-cache-manager.ts +485 -0
  77. package/src/core/layout/layout-composer.ts +357 -0
  78. package/src/core/layout/layout-data-loader.ts +516 -0
  79. package/src/core/layout/layout-discovery.ts +243 -0
  80. package/src/core/layout/layout-matcher.ts +299 -0
  81. package/src/core/layout/layout-types.ts +110 -0
  82. package/src/core/layout/tests/enhanced-layout-resolver.test.ts +477 -0
  83. package/src/core/layout/tests/layout-cache-optimization.test.ts +149 -0
  84. package/src/core/layout/tests/layout-composer.test.ts +486 -0
  85. package/src/core/layout/tests/layout-data-loader.test.ts +443 -0
  86. package/src/core/layout/tests/layout-discovery.test.ts +253 -0
  87. package/src/core/layout/tests/layout-matcher.test.ts +480 -0
  88. package/src/core/modules/framework-module-resolver.ts +273 -0
  89. package/src/core/modules/tests/framework-module-resolver.test.ts +263 -0
  90. package/src/core/modules/tests/module-resolution-integration.test.ts +117 -0
  91. package/src/islands/component-analysis.ts +213 -0
  92. package/src/islands/css-utils.ts +565 -0
  93. package/src/islands/discovery/index.ts +80 -0
  94. package/src/islands/discovery/registry.ts +340 -0
  95. package/src/islands/discovery/resolver.ts +477 -0
  96. package/src/islands/discovery/scanner.ts +386 -0
  97. package/src/islands/discovery/tests/island-discovery.test.ts +881 -0
  98. package/src/islands/discovery/types.ts +117 -0
  99. package/src/islands/discovery/validator.ts +544 -0
  100. package/src/islands/discovery/watcher.ts +368 -0
  101. package/src/islands/framework-detection.ts +428 -0
  102. package/src/islands/integration-loader.ts +490 -0
  103. package/src/islands/island.tsx +565 -0
  104. package/src/islands/render-cache.ts +550 -0
  105. package/src/islands/types.ts +80 -0
  106. package/src/islands/universal-css-collector.ts +157 -0
  107. package/src/islands/universal-head-collector.ts +137 -0
  108. package/src/layout-system.d.ts +592 -0
  109. package/src/layout-system.ts +218 -0
  110. package/src/middleware/__tests__/discovery.test.ts +107 -0
  111. package/src/middleware/discovery.ts +268 -0
  112. package/src/middleware/executor.ts +315 -0
  113. package/src/middleware/index.ts +76 -0
  114. package/src/middleware/types.ts +99 -0
  115. package/src/nitro/build-config.ts +576 -0
  116. package/src/nitro/config.ts +483 -0
  117. package/src/nitro/error-handler.ts +636 -0
  118. package/src/nitro/index.ts +173 -0
  119. package/src/nitro/island-manifest.ts +584 -0
  120. package/src/nitro/middleware-adapter.ts +260 -0
  121. package/src/nitro/renderer.ts +1458 -0
  122. package/src/nitro/route-discovery.ts +439 -0
  123. package/src/nitro/types.ts +321 -0
  124. package/src/render/collect-css.ts +198 -0
  125. package/src/render/error-pages.ts +79 -0
  126. package/src/render/isolated-ssr-renderer.ts +654 -0
  127. package/src/render/ssr.ts +1030 -0
  128. package/src/schemas/api.ts +30 -0
  129. package/src/schemas/core.ts +64 -0
  130. package/src/schemas/index.ts +212 -0
  131. package/src/schemas/layout.ts +279 -0
  132. package/src/schemas/routing/index.ts +38 -0
  133. package/src/schemas/routing.ts +376 -0
  134. package/src/types/as-island.ts +20 -0
  135. package/src/types/image.d.ts +106 -0
  136. package/src/types/index.d.ts +22 -0
  137. package/src/types/island-jsx.d.ts +33 -0
  138. package/src/types/island-prop.d.ts +20 -0
  139. package/src/types/layout.ts +285 -0
  140. package/src/types/mdx.d.ts +6 -0
  141. package/src/types/routing.ts +555 -0
  142. package/src/types/tests/layout-types.test.ts +197 -0
  143. package/src/types/types.ts +5 -0
  144. package/src/types/urlpattern.d.ts +49 -0
  145. package/src/types/vite-env.d.ts +11 -0
  146. package/src/utils/dev-logger.ts +299 -0
  147. package/src/utils/fs.ts +151 -0
  148. package/src/vite-plugin/auto-discover.ts +551 -0
  149. package/src/vite-plugin/config.ts +266 -0
  150. package/src/vite-plugin/errors.ts +127 -0
  151. package/src/vite-plugin/image-optimization.ts +151 -0
  152. package/src/vite-plugin/integration-activator.ts +126 -0
  153. package/src/vite-plugin/island-sidecar-plugin.ts +176 -0
  154. package/src/vite-plugin/module-discovery.ts +189 -0
  155. package/src/vite-plugin/nitro-integration.ts +1334 -0
  156. package/src/vite-plugin/plugin.ts +329 -0
  157. package/src/vite-plugin/tests/image-optimization.test.ts +54 -0
  158. package/src/vite-plugin/types.ts +327 -0
  159. package/src/vite-plugin/validation.ts +228 -0
@@ -0,0 +1,197 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import {
3
+ LayoutContextSchema,
4
+ LayoutDataSchema,
5
+ LayoutRouteSchema,
6
+ LayoutHandlerSchema,
7
+ LayoutPropsSchema,
8
+ LayoutDiscoveryOptionsSchema,
9
+ LayoutConfigSchema,
10
+ ResolvedLayoutSchema,
11
+ } from '../../schemas/layout.ts';
12
+ import { validators, safeValidators } from '../../schemas/index.ts';
13
+ import type {
14
+ LayoutContext,
15
+ LayoutData,
16
+ LayoutRoute,
17
+ LayoutHandler,
18
+ LayoutProps,
19
+ LayoutDiscoveryOptions,
20
+ LayoutConfig,
21
+ ResolvedLayout,
22
+ } from '../layout.ts';
23
+
24
+ describe('Layout System Types and Schemas', () => {
25
+ it('LayoutContext - should validate valid layout context', () => {
26
+ const mockRequest = new Request('https://example.com/test');
27
+ const mockParams = { id: '123' };
28
+ const mockQuery = new URLSearchParams('?page=1');
29
+ const mockState = new Map();
30
+
31
+ const validContext: LayoutContext = {
32
+ request: mockRequest,
33
+ params: mockParams,
34
+ query: mockQuery,
35
+ state: mockState,
36
+ };
37
+
38
+ const result = safeValidators.layoutContext(validContext);
39
+ expect(result.success).toEqual(true);
40
+ });
41
+
42
+ it('LayoutContext - should reject invalid layout context', () => {
43
+ const invalidContext = {
44
+ request: 'not-a-request',
45
+ params: 'not-an-object',
46
+ query: 'not-urlsearchparams',
47
+ state: 'not-a-map',
48
+ };
49
+
50
+ const result = safeValidators.layoutContext(invalidContext);
51
+ expect(result.success).toEqual(false);
52
+ });
53
+
54
+ it('LayoutData - should validate layout data as record', () => {
55
+ const validData: LayoutData = {
56
+ user: { name: 'John', id: 123 },
57
+ settings: { theme: 'dark' },
58
+ items: [1, 2, 3],
59
+ };
60
+
61
+ const result = safeValidators.layoutData(validData);
62
+ expect(result.success).toEqual(true);
63
+ });
64
+
65
+ it('LayoutData - should accept empty layout data', () => {
66
+ const emptyData: LayoutData = {};
67
+
68
+ const result = safeValidators.layoutData(emptyData);
69
+ expect(result.success).toEqual(true);
70
+ });
71
+
72
+ it('LayoutHandler - should validate valid layout handler', () => {
73
+ const validHandler = {
74
+ component: () => null,
75
+ path: '/src/pages/blog/_layout.tsx',
76
+ priority: 10,
77
+ };
78
+
79
+ const result = safeValidators.layoutHandler(validHandler);
80
+ expect(result.success).toEqual(true);
81
+ });
82
+
83
+ it('LayoutDiscoveryOptions - should validate with defaults', () => {
84
+ const options = {
85
+ baseDirectory: '/src/pages',
86
+ };
87
+
88
+ const result = safeValidators.layoutDiscoveryOptions(options);
89
+ expect(result.success).toEqual(true);
90
+ if (result.success) {
91
+ expect(result.data.filePattern).toEqual('_layout.tsx');
92
+ expect(result.data.excludeDirectories).toEqual([]);
93
+ expect(result.data.enableWatching).toEqual(false);
94
+ expect(result.data.developmentMode).toEqual(false);
95
+ }
96
+ });
97
+
98
+ it('LayoutDiscoveryOptions - should validate with custom options', () => {
99
+ const options: LayoutDiscoveryOptions = {
100
+ baseDirectory: '/src/pages',
101
+ filePattern: 'layout.tsx',
102
+ excludeDirectories: ['node_modules', '.git'],
103
+ enableWatching: true,
104
+ developmentMode: true,
105
+ };
106
+
107
+ const result = safeValidators.layoutDiscoveryOptions(options);
108
+ expect(result.success).toEqual(true);
109
+ if (result.success) {
110
+ expect(result.data.filePattern).toEqual('layout.tsx');
111
+ expect(result.data.excludeDirectories).toEqual(['node_modules', '.git']);
112
+ expect(result.data.enableWatching).toEqual(true);
113
+ expect(result.data.developmentMode).toEqual(true);
114
+ }
115
+ });
116
+
117
+ it('LayoutConfig - should validate layout config with all options', () => {
118
+ const config: LayoutConfig = {
119
+ skipLayouts: ['root', 'admin'],
120
+ replaceLayout: true,
121
+ onlyLayouts: ['custom'],
122
+ customLayout: '/custom/layout.tsx',
123
+ };
124
+
125
+ const result = safeValidators.layoutConfig(config);
126
+ expect(result.success).toEqual(true);
127
+ });
128
+
129
+ it('LayoutConfig - should validate empty layout config', () => {
130
+ const config: LayoutConfig = {};
131
+
132
+ const result = safeValidators.layoutConfig(config);
133
+ expect(result.success).toEqual(true);
134
+ });
135
+
136
+ it('ResolvedLayout - should validate complete resolved layout', () => {
137
+ const resolvedLayout: ResolvedLayout = {
138
+ handlers: [],
139
+ dataLoaders: [],
140
+ errorBoundaries: [],
141
+ streamingComponents: [],
142
+ metadata: {
143
+ totalLayouts: 2,
144
+ resolutionTime: 15.5,
145
+ cacheHit: false,
146
+ },
147
+ };
148
+
149
+ const result = safeValidators.resolvedLayout(resolvedLayout);
150
+ expect(result.success).toEqual(true);
151
+ });
152
+
153
+ it('ResolvedLayout - should require all metadata fields', () => {
154
+ const incompleteLayout = {
155
+ handlers: [],
156
+ dataLoaders: [],
157
+ errorBoundaries: [],
158
+ streamingComponents: [],
159
+ metadata: {
160
+ totalLayouts: 2,
161
+ },
162
+ };
163
+
164
+ const result = safeValidators.resolvedLayout(incompleteLayout);
165
+ expect(result.success).toEqual(false);
166
+ });
167
+
168
+ it('Type compatibility - should ensure TypeScript types match Zod schemas', () => {
169
+ const mockRequest = new Request('https://example.com');
170
+ const layoutContext: LayoutContext = {
171
+ request: mockRequest,
172
+ params: { id: '123' },
173
+ query: new URLSearchParams(),
174
+ state: new Map(),
175
+ };
176
+
177
+ const validatedContext = validators.layoutContext(layoutContext);
178
+ expect(validatedContext).toBeDefined();
179
+ expect(validatedContext.request).toEqual(mockRequest);
180
+ });
181
+
182
+ it('Error handling - should provide meaningful error messages', () => {
183
+ const invalidData = {
184
+ request: null,
185
+ params: null,
186
+ query: null,
187
+ state: null,
188
+ };
189
+
190
+ const result = safeValidators.layoutContext(invalidData);
191
+ expect(result.success).toEqual(false);
192
+ if (!result.success) {
193
+ expect(result.error.message).toContain('Invalid layout context');
194
+ expect(result.error.getFormattedErrors().length > 0).toEqual(true);
195
+ }
196
+ });
197
+ });
@@ -0,0 +1,5 @@
1
+ export type ImportConfig = { names: string[]; from: string };
2
+ export type CleanupFunction = () => void;
3
+
4
+ // Re-export layout types for convenience
5
+ export * from './layout.ts';
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Global URLPattern type declaration.
3
+ * URLPattern is available at runtime in Node 22+, Bun, and Deno,
4
+ * but TypeScript's lib.dom.d.ts may not include it in all configurations.
5
+ */
6
+ declare class URLPattern {
7
+ constructor(init?: URLPatternInit | string, baseURL?: string);
8
+ readonly protocol: string;
9
+ readonly username: string;
10
+ readonly password: string;
11
+ readonly hostname: string;
12
+ readonly port: string;
13
+ readonly pathname: string;
14
+ readonly search: string;
15
+ readonly hash: string;
16
+ test(input?: URLPatternInput, baseURL?: string): boolean;
17
+ exec(input?: URLPatternInput, baseURL?: string): URLPatternResult | null;
18
+ }
19
+
20
+ interface URLPatternInit {
21
+ baseURL?: string;
22
+ username?: string;
23
+ password?: string;
24
+ hostname?: string;
25
+ port?: string;
26
+ pathname?: string;
27
+ search?: string;
28
+ hash?: string;
29
+ protocol?: string;
30
+ }
31
+
32
+ type URLPatternInput = URLPatternInit | string;
33
+
34
+ interface URLPatternComponentResult {
35
+ input: string;
36
+ groups: Record<string, string | undefined>;
37
+ }
38
+
39
+ interface URLPatternResult {
40
+ inputs: [URLPatternInput, string?];
41
+ protocol: URLPatternComponentResult;
42
+ username: URLPatternComponentResult;
43
+ password: URLPatternComponentResult;
44
+ hostname: URLPatternComponentResult;
45
+ port: URLPatternComponentResult;
46
+ pathname: URLPatternComponentResult;
47
+ search: URLPatternComponentResult;
48
+ hash: URLPatternComponentResult;
49
+ }
@@ -0,0 +1,11 @@
1
+ /// <reference types="vite/client" />
2
+
3
+ interface ImportMetaEnv {
4
+ readonly VITE_APP_TITLE: string;
5
+ // more env variables...
6
+ }
7
+
8
+ interface ImportMeta {
9
+ readonly env: ImportMetaEnv;
10
+ readonly hot?: import('vite/types/hot').ViteHotContext;
11
+ }
@@ -0,0 +1,299 @@
1
+ /**
2
+ * Dev-Only Logging Utilities
3
+ *
4
+ * These functions provide environment-aware logging that only outputs in development mode.
5
+ * In production (NODE_ENV=production), all logging is suppressed for better performance.
6
+ *
7
+ * @module dev-logger
8
+ */
9
+
10
+ // ============================================================================
11
+ // Environment Detection
12
+ // ============================================================================
13
+
14
+ /**
15
+ * Check if we're in development mode
16
+ * Returns true if NODE_ENV is not set to "production"
17
+ */
18
+ export function isDev(): boolean {
19
+ try {
20
+ return process.env.NODE_ENV !== "production";
21
+ } catch {
22
+ return true; // Default to dev mode if we can't check
23
+ }
24
+ }
25
+
26
+ /**
27
+ * Check if verbose logging is enabled
28
+ * Returns true only if AVALON_VERBOSE=1 is set
29
+ */
30
+ export function isVerbose(): boolean {
31
+ try {
32
+ return process.env.AVALON_VERBOSE === "1";
33
+ } catch {
34
+ return false;
35
+ }
36
+ }
37
+
38
+ // ============================================================================
39
+ // Dev-Only Logging Functions
40
+ // ============================================================================
41
+
42
+ /**
43
+ * Log a message only in development mode AND when AVALON_VERBOSE=1 is set.
44
+ * By default this is a no-op — set AVALON_VERBOSE=1 to enable diagnostic output.
45
+ *
46
+ * @param args - Arguments to pass to console.log
47
+ */
48
+ export function devLog(...args: unknown[]): void {
49
+ if (isDev() && isVerbose()) {
50
+ console.log(...args);
51
+ }
52
+ }
53
+
54
+ /**
55
+ * Log a warning only in development mode
56
+ * In production, this is a no-op for performance
57
+ *
58
+ * @param args - Arguments to pass to console.warn
59
+ */
60
+ export function devWarn(...args: unknown[]): void {
61
+ if (isDev()) {
62
+ console.warn(...args);
63
+ }
64
+ }
65
+
66
+ /**
67
+ * Log an error only in development mode
68
+ * In production, this is a no-op for performance
69
+ *
70
+ * @param args - Arguments to pass to console.error
71
+ */
72
+ export function devError(...args: unknown[]): void {
73
+ if (isDev()) {
74
+ console.error(...args);
75
+ }
76
+ }
77
+
78
+ // ============================================================================
79
+ // Performance Tracking
80
+ // ============================================================================
81
+
82
+ /** Default threshold for slow render warnings (in milliseconds) */
83
+ const DEFAULT_SLOW_RENDER_THRESHOLD = 100;
84
+
85
+ /**
86
+ * Log render timing information for an island component
87
+ * Only logs in development mode AND when AVALON_VERBOSE=1 is set
88
+ * Warns when render time exceeds the threshold
89
+ *
90
+ * @param src - The island source path
91
+ * @param durationMs - The render duration in milliseconds
92
+ * @param threshold - Optional threshold for slow render warning (default: 100ms)
93
+ */
94
+ export function logRenderTiming(
95
+ src: string,
96
+ durationMs: number,
97
+ threshold: number = DEFAULT_SLOW_RENDER_THRESHOLD
98
+ ): void {
99
+ if (!isDev() || !isVerbose()) return;
100
+
101
+ if (durationMs > threshold) {
102
+ console.warn(`⚠️ Slow island render: ${src} took ${durationMs.toFixed(2)}ms`);
103
+ } else {
104
+ console.log(`🏝️ ${src} rendered in ${durationMs.toFixed(2)}ms`);
105
+ }
106
+ }
107
+
108
+ /**
109
+ * Log a cache hit event (dev mode only)
110
+ *
111
+ * @param cacheType - The type of cache (e.g., 'analysis', 'path', 'framework')
112
+ * @param key - The cache key that was hit
113
+ */
114
+ export function logCacheHit(cacheType: string, key: string): void {
115
+ if (!isDev() || !isVerbose()) return;
116
+ console.log(`📦 Cache HIT [${cacheType}]: ${key}`);
117
+ }
118
+
119
+ /**
120
+ * Log a cache miss event (dev mode only)
121
+ *
122
+ * @param cacheType - The type of cache (e.g., 'analysis', 'path', 'framework')
123
+ * @param key - The cache key that was missed
124
+ */
125
+ export function logCacheMiss(cacheType: string, key: string): void {
126
+ if (!isDev() || !isVerbose()) return;
127
+ console.log(`📭 Cache MISS [${cacheType}]: ${key}`);
128
+ }
129
+
130
+ // ============================================================================
131
+ // DevLogger Class (for Development Server UI)
132
+ // ============================================================================
133
+
134
+ const AVALON_ASCII = `
135
+ █████╗ ██╗ ██╗ █████╗ ██╗ ██████╗ ███╗ ██╗
136
+ ██╔══██╗██║ ██║██╔══██╗██║ ██╔═══██╗████╗ ██║
137
+ ███████║██║ ██║███████║██║ ██║ ██║██╔██╗ ██║
138
+ ██╔══██║╚██╗ ██╔╝██╔══██║██║ ██║ ██║██║╚██╗██║
139
+ ██║ ██║ ╚████╔╝ ██║ ██║███████╗╚██████╔╝██║ ╚████║
140
+ ╚═╝ ╚═╝ ╚═══╝ ╚═╝ ╚═╝╚══════╝ ╚═════╝ ╚═╝ ╚═══╝
141
+ `;
142
+
143
+ const COLORS = {
144
+ reset: '\x1b[0m',
145
+ cyan: '\x1b[36m',
146
+ green: '\x1b[32m',
147
+ yellow: '\x1b[33m',
148
+ blue: '\x1b[34m',
149
+ magenta: '\x1b[35m',
150
+ gray: '\x1b[90m',
151
+ bold: '\x1b[1m',
152
+ };
153
+
154
+ const SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
155
+
156
+ interface DevTask {
157
+ name: string;
158
+ status: 'pending' | 'running' | 'done';
159
+ startTime?: number;
160
+ }
161
+
162
+ export class DevLogger {
163
+ private readonly tasks: Map<string, DevTask> = new Map();
164
+ private spinnerInterval?: ReturnType<typeof setInterval>;
165
+ private currentFrame = 0;
166
+ private readonly startTime = Date.now();
167
+ private readonly originalConsoleLog: typeof console.log;
168
+ private readonly originalConsoleWarn: typeof console.warn;
169
+ private readonly originalConsoleError: typeof console.error;
170
+ private readonly suppressedLogs: string[] = [];
171
+ private readonly headerLines: number = 0;
172
+
173
+ constructor() {
174
+ this.originalConsoleLog = console.log;
175
+ this.originalConsoleWarn = console.warn;
176
+ this.originalConsoleError = console.error;
177
+
178
+ this.suppressConsole();
179
+
180
+ // Clear screen once at start
181
+ process.stdout.write('\x1b[2J\x1b[H');
182
+
183
+ const header = COLORS.cyan + AVALON_ASCII + COLORS.reset + '\n' +
184
+ COLORS.gray + ' Development Server' + COLORS.reset + '\n\n';
185
+ this.headerLines = AVALON_ASCII.split('\n').length + 2;
186
+
187
+ process.stdout.write(header);
188
+ }
189
+
190
+ private suppressConsole() {
191
+ console.log = (...args: unknown[]) => {
192
+ this.suppressedLogs.push(args.map(String).join(' '));
193
+ };
194
+ console.warn = (...args: unknown[]) => {
195
+ this.suppressedLogs.push('[WARN] ' + args.map(String).join(' '));
196
+ };
197
+ console.error = this.originalConsoleError;
198
+ }
199
+
200
+ restoreConsole() {
201
+ console.log = this.originalConsoleLog;
202
+ console.warn = this.originalConsoleWarn;
203
+ console.error = this.originalConsoleError;
204
+ }
205
+
206
+
207
+
208
+ addTask(id: string, name: string) {
209
+ this.tasks.set(id, { name, status: 'pending' });
210
+ this.render();
211
+ }
212
+
213
+ startTask(id: string) {
214
+ const task = this.tasks.get(id);
215
+ if (task) {
216
+ task.status = 'running';
217
+ task.startTime = Date.now();
218
+ this.render();
219
+ }
220
+ }
221
+
222
+ completeTask(id: string) {
223
+ const task = this.tasks.get(id);
224
+ if (task) {
225
+ task.status = 'done';
226
+ this.render();
227
+ }
228
+ }
229
+
230
+
231
+ private render() {
232
+ let currentTask: DevTask | null = null;
233
+ for (const task of this.tasks.values()) {
234
+ if (task.status === 'running') {
235
+ currentTask = task;
236
+ break;
237
+ }
238
+ }
239
+
240
+ // Use carriage return to overwrite the same line
241
+ let output = '\r'; // Return to start of line
242
+
243
+ if (currentTask) {
244
+ const spinner = COLORS.cyan + SPINNER_FRAMES[this.currentFrame] + COLORS.reset;
245
+ output += `${spinner} ${currentTask.name}`;
246
+ } else {
247
+ output += `${COLORS.gray}Initializing...${COLORS.reset}`;
248
+ }
249
+
250
+ // Clear to end of line
251
+ output += '\x1b[K';
252
+
253
+ process.stdout.write(output);
254
+ }
255
+
256
+ startSpinner() {
257
+ this.spinnerInterval = setInterval(() => {
258
+ this.currentFrame = (this.currentFrame + 1) % SPINNER_FRAMES.length;
259
+ this.render();
260
+ }, 100);
261
+ }
262
+
263
+ stopSpinner() {
264
+ if (this.spinnerInterval) {
265
+ clearInterval(this.spinnerInterval);
266
+ }
267
+ }
268
+
269
+ finish(serverUrl: string, viteUrl?: string, hmrUrl?: string) {
270
+ this.stopSpinner();
271
+
272
+ // Clear the status line
273
+ let output = `\x1b[${this.headerLines + 1};0H`;
274
+ output += '\x1b[J'; // Clear from cursor down
275
+ process.stdout.write(output);
276
+
277
+ const elapsed = ((Date.now() - this.startTime) / 1000).toFixed(1);
278
+
279
+ this.originalConsoleLog('');
280
+ this.originalConsoleLog(COLORS.green + COLORS.bold + '✨ Server ready!' + COLORS.reset);
281
+ this.originalConsoleLog('');
282
+ this.originalConsoleLog(COLORS.cyan + ' ➜ ' + COLORS.reset + COLORS.bold + 'Local: ' + COLORS.reset + COLORS.cyan + serverUrl + COLORS.reset);
283
+
284
+ if (viteUrl) {
285
+ this.originalConsoleLog(COLORS.cyan + ' ➜ ' + COLORS.reset + COLORS.bold + 'Vite: ' + COLORS.reset + COLORS.gray + viteUrl + COLORS.reset);
286
+ }
287
+
288
+ if (hmrUrl) {
289
+ this.originalConsoleLog(COLORS.cyan + ' ➜ ' + COLORS.reset + COLORS.bold + 'HMR: ' + COLORS.reset + COLORS.gray + hmrUrl + COLORS.reset);
290
+ }
291
+
292
+ this.originalConsoleLog('');
293
+ this.originalConsoleLog(COLORS.gray + ` Ready in ${elapsed}s` + COLORS.reset);
294
+ this.originalConsoleLog('');
295
+ this.originalConsoleLog(COLORS.gray + ' Press Ctrl+C to stop' + COLORS.reset);
296
+ this.originalConsoleLog('');
297
+
298
+ }
299
+ }
@@ -0,0 +1,151 @@
1
+ /**
2
+ * Filesystem utilities for Avalon.
3
+ * Provides Node.js-compatible implementations of common filesystem
4
+ * functions used throughout the codebase.
5
+ */
6
+
7
+ import { readdirSync, statSync } from "node:fs";
8
+ import { join } from "node:path";
9
+
10
+ export interface WalkEntry {
11
+ path: string;
12
+ name: string;
13
+ isFile: boolean;
14
+ isDirectory: boolean;
15
+ isSymlink: boolean;
16
+ }
17
+
18
+ export interface WalkOptions {
19
+ maxDepth?: number;
20
+ includeFiles?: boolean;
21
+ includeDirs?: boolean;
22
+ includeSymlinks?: boolean;
23
+ match?: RegExp[];
24
+ skip?: RegExp[];
25
+ exts?: string[];
26
+ }
27
+
28
+ /**
29
+ * Walks a directory tree yielding entries, compatible with @std/fs walk().
30
+ */
31
+ export async function* walk(
32
+ root: string,
33
+ options: WalkOptions = {},
34
+ ): AsyncIterableIterator<WalkEntry> {
35
+ const {
36
+ maxDepth = Infinity,
37
+ includeFiles = true,
38
+ includeDirs = true,
39
+ exts,
40
+ match,
41
+ skip,
42
+ } = options;
43
+
44
+ yield* walkSync(root, 0, { maxDepth, includeFiles, includeDirs, exts, match, skip });
45
+ }
46
+
47
+ interface WalkSyncState {
48
+ maxDepth: number;
49
+ includeFiles: boolean;
50
+ includeDirs: boolean;
51
+ exts?: string[];
52
+ match?: RegExp[];
53
+ skip?: RegExp[];
54
+ }
55
+
56
+ function matchesFile(name: string, fullPath: string, opts: WalkSyncState): boolean {
57
+ if (opts.exts && !opts.exts.some((ext) => name.endsWith(ext))) return false;
58
+ if (opts.match && !opts.match.some((r) => r.test(fullPath))) return false;
59
+ return true;
60
+ }
61
+
62
+ function matchesDir(fullPath: string, match?: RegExp[]): boolean {
63
+ return !match || match.some((r) => r.test(fullPath));
64
+ }
65
+
66
+ function* walkSync(
67
+ dir: string,
68
+ depth: number,
69
+ opts: WalkSyncState,
70
+ ): Generator<WalkEntry> {
71
+ if (depth > opts.maxDepth) return;
72
+
73
+ let entries;
74
+ try {
75
+ entries = readdirSync(dir, { withFileTypes: true });
76
+ } catch {
77
+ return;
78
+ }
79
+
80
+ for (const entry of entries) {
81
+ const fullPath = join(dir, entry.name);
82
+
83
+ if (opts.skip?.some((r) => r.test(fullPath))) continue;
84
+
85
+ const walkEntry: WalkEntry = {
86
+ path: fullPath,
87
+ name: entry.name,
88
+ isFile: entry.isFile(),
89
+ isDirectory: entry.isDirectory(),
90
+ isSymlink: entry.isSymbolicLink(),
91
+ };
92
+
93
+ if (entry.isDirectory()) {
94
+ if (opts.includeDirs && matchesDir(fullPath, opts.match)) {
95
+ yield walkEntry;
96
+ }
97
+ yield* walkSync(fullPath, depth + 1, opts);
98
+ } else if (entry.isFile() && opts.includeFiles && matchesFile(entry.name, fullPath, opts)) {
99
+ yield walkEntry;
100
+ }
101
+ }
102
+ }
103
+
104
+ /**
105
+ * Ensures a directory exists, creating it recursively if needed.
106
+ */
107
+ export async function ensureDir(dir: string): Promise<void> {
108
+ const { mkdirSync } = await import("node:fs");
109
+ try {
110
+ mkdirSync(dir, { recursive: true });
111
+ } catch {
112
+ // Directory already exists
113
+ }
114
+ }
115
+
116
+ /**
117
+ * Check if a file or directory exists.
118
+ */
119
+ export function exists(path: string): Promise<boolean> {
120
+ try {
121
+ statSync(path);
122
+ return Promise.resolve(true);
123
+ } catch {
124
+ return Promise.resolve(false);
125
+ }
126
+ }
127
+
128
+ /**
129
+ * Synchronous check if a file or directory exists.
130
+ */
131
+ export function existsSync(path: string): boolean {
132
+ try {
133
+ statSync(path);
134
+ return true;
135
+ } catch {
136
+ return false;
137
+ }
138
+ }
139
+
140
+ /**
141
+ * Empties a directory by removing all contents.
142
+ */
143
+ export async function emptyDir(dir: string): Promise<void> {
144
+ const { rmSync, mkdirSync } = await import("node:fs");
145
+ try {
146
+ rmSync(dir, { recursive: true, force: true });
147
+ } catch {
148
+ // Directory doesn't exist
149
+ }
150
+ mkdirSync(dir, { recursive: true });
151
+ }