@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,145 @@
1
+ import { Component, ComponentChildren, ComponentType } from 'preact';
2
+ import type { LayoutErrorInfo } from '../types/layout.ts';
3
+
4
+ export interface IslandErrorBoundaryProps {
5
+ children: ComponentChildren;
6
+ islandId: string;
7
+ onError?: (error: Error, errorInfo: LayoutErrorInfo) => void;
8
+ fallback?: (error: Error, islandId: string) => ComponentChildren;
9
+ isolateError?: boolean;
10
+ }
11
+
12
+ export interface IslandErrorBoundaryState {
13
+ hasError: boolean;
14
+ error: Error | null;
15
+ errorInfo: LayoutErrorInfo | null;
16
+ }
17
+
18
+ /**
19
+ * Specialized error boundary for island components
20
+ * Provides error isolation to prevent island errors from affecting the main layout
21
+ */
22
+ export class IslandErrorBoundary extends Component<IslandErrorBoundaryProps, IslandErrorBoundaryState> {
23
+ constructor(props: IslandErrorBoundaryProps) {
24
+ super(props);
25
+ this.state = { hasError: false, error: null, errorInfo: null };
26
+ }
27
+
28
+ static override getDerivedStateFromError(error: Error): Partial<IslandErrorBoundaryState> {
29
+ return { hasError: true, error };
30
+ }
31
+
32
+ override componentDidCatch(error: Error, errorInfo: { componentStack?: string }): void {
33
+ const layoutErrorInfo: LayoutErrorInfo = {
34
+ layoutPath: `island:${this.props.islandId}`,
35
+ errorType: 'island',
36
+ timestamp: Date.now(),
37
+ componentStack: errorInfo.componentStack,
38
+ errorBoundary: 'IslandErrorBoundary',
39
+ };
40
+
41
+ this.setState({ errorInfo: layoutErrorInfo });
42
+
43
+ if (this.props.onError) {
44
+ this.props.onError(error, layoutErrorInfo);
45
+ }
46
+
47
+ const isDevelopment = typeof Deno !== 'undefined' && Deno.env.get('NODE_ENV') === 'development';
48
+ if (isDevelopment) {
49
+ console.error(`Island Error [${this.props.islandId}]:`, error);
50
+ }
51
+
52
+ if (!this.props.isolateError) {
53
+ throw error;
54
+ }
55
+ }
56
+
57
+ private handleRemoveIsland = (): void => {
58
+ const islandElement = document.querySelector(`[data-island-id="${this.props.islandId}"]`);
59
+ if (islandElement) {
60
+ islandElement.remove();
61
+ }
62
+ };
63
+
64
+ private handleReloadIsland = (): void => {
65
+ this.setState({ hasError: false, error: null, errorInfo: null });
66
+ };
67
+
68
+ private renderFallback(): ComponentChildren {
69
+ const { error } = this.state;
70
+ const { fallback, islandId } = this.props;
71
+
72
+ if (fallback && error) {
73
+ return fallback(error, islandId);
74
+ }
75
+
76
+ const isDevelopment = typeof Deno !== 'undefined' && Deno.env.get('NODE_ENV') === 'development';
77
+
78
+ return (
79
+ <div class="island-error-boundary" data-island-error={islandId}>
80
+ <div class="island-error-container">
81
+ <div class="island-error-header">
82
+ <span class="island-error-icon">⚠️</span>
83
+ <span class="island-error-title">Island Error</span>
84
+ </div>
85
+
86
+ <p class="island-error-message">
87
+ An error occurred in island "{islandId}". The rest of the page should work normally.
88
+ </p>
89
+
90
+ <div class="island-error-actions">
91
+ <button onClick={this.handleReloadIsland} class="island-reload-button">
92
+ Reload Island
93
+ </button>
94
+ <button onClick={this.handleRemoveIsland} class="island-remove-button">
95
+ Remove Island
96
+ </button>
97
+ </div>
98
+
99
+ {isDevelopment && error && (
100
+ <details class="island-error-details">
101
+ <summary>Error Details (Development)</summary>
102
+ <div class="island-error-info">
103
+ <p><strong>Island ID:</strong> {islandId}</p>
104
+ <p><strong>Error:</strong> {error.message}</p>
105
+ </div>
106
+ <pre class="island-error-stack">{error.stack}</pre>
107
+ </details>
108
+ )}
109
+ </div>
110
+ </div>
111
+ );
112
+ }
113
+
114
+ render() {
115
+ if (this.state.hasError) {
116
+ return this.renderFallback();
117
+ }
118
+ return this.props.children;
119
+ }
120
+ }
121
+
122
+ /**
123
+ * Higher-order component to wrap islands with error boundaries
124
+ */
125
+ export function withIslandErrorBoundary<P extends object>(
126
+ WrappedComponent: ComponentType<P>,
127
+ islandId: string,
128
+ options?: {
129
+ fallback?: (error: Error, islandId: string) => ComponentChildren;
130
+ isolateError?: boolean;
131
+ onError?: (error: Error, errorInfo: LayoutErrorInfo) => void;
132
+ }
133
+ ) {
134
+ return function IslandWithErrorBoundary(props: P) {
135
+ return (
136
+ <IslandErrorBoundary
137
+ islandId={islandId}
138
+ fallback={options?.fallback}
139
+ isolateError={options?.isolateError ?? true}
140
+ onError={options?.onError}>
141
+ <WrappedComponent {...props} />
142
+ </IslandErrorBoundary>
143
+ );
144
+ };
145
+ }
@@ -0,0 +1,141 @@
1
+ import { Component, ComponentChildren } from 'preact';
2
+ import type { LayoutErrorInfo, LayoutContext, LayoutData } from '../types/layout.ts';
3
+
4
+ export interface LayoutDataErrorBoundaryProps {
5
+ children: ComponentChildren;
6
+ layoutPath: string;
7
+ context: LayoutContext;
8
+ onError?: (error: Error, errorInfo: LayoutErrorInfo) => void;
9
+ fallbackData?: LayoutData;
10
+ retryLoader?: () => Promise<LayoutData>;
11
+ }
12
+
13
+ export interface LayoutDataErrorBoundaryState {
14
+ hasError: boolean;
15
+ error: Error | null;
16
+ errorInfo: LayoutErrorInfo | null;
17
+ retryCount: number;
18
+ isRetrying: boolean;
19
+ fallbackData: LayoutData | null;
20
+ }
21
+
22
+ /**
23
+ * Specialized error boundary for layout data loading errors
24
+ * Provides specific handling for data loader failures with retry and fallback mechanisms
25
+ */
26
+ export class LayoutDataErrorBoundary extends Component<LayoutDataErrorBoundaryProps, LayoutDataErrorBoundaryState> {
27
+ private maxRetries = 3;
28
+
29
+ constructor(props: LayoutDataErrorBoundaryProps) {
30
+ super(props);
31
+ this.state = {
32
+ hasError: false,
33
+ error: null,
34
+ errorInfo: null,
35
+ retryCount: 0,
36
+ isRetrying: false,
37
+ fallbackData: props.fallbackData || null,
38
+ };
39
+ }
40
+
41
+ static override getDerivedStateFromError(error: Error): Partial<LayoutDataErrorBoundaryState> {
42
+ return { hasError: true, error };
43
+ }
44
+
45
+ override componentDidCatch(error: Error, errorInfo: { componentStack?: string }): void {
46
+ const layoutErrorInfo: LayoutErrorInfo = {
47
+ layoutPath: this.props.layoutPath,
48
+ errorType: 'loader',
49
+ timestamp: Date.now(),
50
+ componentStack: errorInfo.componentStack,
51
+ errorBoundary: 'LayoutDataErrorBoundary',
52
+ };
53
+
54
+ this.setState({ errorInfo: layoutErrorInfo });
55
+
56
+ if (this.props.onError) {
57
+ this.props.onError(error, layoutErrorInfo);
58
+ }
59
+ }
60
+
61
+ private handleRetry = async (): Promise<void> => {
62
+ if (this.state.retryCount >= this.maxRetries || !this.props.retryLoader) {
63
+ return;
64
+ }
65
+
66
+ this.setState({ isRetrying: true });
67
+
68
+ try {
69
+ const data = await this.props.retryLoader();
70
+ this.setState({
71
+ hasError: false,
72
+ error: null,
73
+ errorInfo: null,
74
+ retryCount: this.state.retryCount + 1,
75
+ isRetrying: false,
76
+ fallbackData: data,
77
+ });
78
+ } catch (retryError) {
79
+ this.setState({
80
+ retryCount: this.state.retryCount + 1,
81
+ isRetrying: false,
82
+ error: retryError instanceof Error ? retryError : new Error(String(retryError)),
83
+ });
84
+ }
85
+ };
86
+
87
+ private handleUseFallback = (): void => {
88
+ if (this.state.fallbackData) {
89
+ this.setState({ hasError: false, error: null, errorInfo: null });
90
+ }
91
+ };
92
+
93
+ private renderErrorUI(): ComponentChildren {
94
+ const { error, retryCount, isRetrying, fallbackData } = this.state;
95
+ const canRetry = retryCount < this.maxRetries && this.props.retryLoader;
96
+ const hasFallback = fallbackData !== null;
97
+ const isDevelopment = typeof Deno !== 'undefined' && Deno.env.get('NODE_ENV') === 'development';
98
+
99
+ return (
100
+ <div class="layout-data-error-boundary">
101
+ <div class="error-container">
102
+ <h3>Data Loading Error</h3>
103
+ <p>Failed to load data for layout: {this.props.layoutPath}</p>
104
+
105
+ <div class="error-actions">
106
+ {canRetry && (
107
+ <button onClick={this.handleRetry} disabled={isRetrying} class="retry-button">
108
+ {isRetrying ? 'Retrying...' : `Retry (${this.maxRetries - retryCount} left)`}
109
+ </button>
110
+ )}
111
+
112
+ {hasFallback && (
113
+ <button onClick={this.handleUseFallback} class="fallback-button">
114
+ Use Cached Data
115
+ </button>
116
+ )}
117
+ </div>
118
+
119
+ {isDevelopment && error && (
120
+ <details class="error-details">
121
+ <summary>Error Details (Development)</summary>
122
+ <div class="error-info">
123
+ <p><strong>Error:</strong> {error.message}</p>
124
+ <p><strong>Layout:</strong> {this.props.layoutPath}</p>
125
+ <p><strong>Retry Count:</strong> {retryCount}</p>
126
+ </div>
127
+ <pre class="error-stack">{error.stack}</pre>
128
+ </details>
129
+ )}
130
+ </div>
131
+ </div>
132
+ );
133
+ }
134
+
135
+ render() {
136
+ if (this.state.hasError) {
137
+ return this.renderErrorUI();
138
+ }
139
+ return this.props.children;
140
+ }
141
+ }
@@ -0,0 +1,127 @@
1
+ import { Component, ComponentChildren } from 'preact';
2
+ import type { LayoutErrorInfo, ErrorRecoveryStrategy } from '../types/layout.ts';
3
+
4
+ export interface LayoutErrorBoundaryProps {
5
+ children: ComponentChildren;
6
+ fallback?: (error: Error, retry: () => void) => ComponentChildren;
7
+ onError?: (error: Error, errorInfo: LayoutErrorInfo) => void;
8
+ recoveryStrategy?: ErrorRecoveryStrategy;
9
+ layoutPath?: string;
10
+ errorType?: 'component' | 'loader' | 'rendering' | 'island';
11
+ }
12
+
13
+ export interface LayoutErrorBoundaryState {
14
+ hasError: boolean;
15
+ error: Error | null;
16
+ errorInfo: LayoutErrorInfo | null;
17
+ retryCount: number;
18
+ }
19
+
20
+ export class LayoutErrorBoundary extends Component<LayoutErrorBoundaryProps, LayoutErrorBoundaryState> {
21
+ private maxRetries = 3;
22
+
23
+ constructor(props: LayoutErrorBoundaryProps) {
24
+ super(props);
25
+ this.state = {
26
+ hasError: false,
27
+ error: null,
28
+ errorInfo: null,
29
+ retryCount: 0,
30
+ };
31
+ }
32
+
33
+ static override getDerivedStateFromError(error: Error): Partial<LayoutErrorBoundaryState> {
34
+ return {
35
+ hasError: true,
36
+ error,
37
+ };
38
+ }
39
+
40
+ override componentDidCatch(error: Error, errorInfo: any): void {
41
+ const layoutErrorInfo: LayoutErrorInfo = {
42
+ layoutPath: this.props.layoutPath || 'unknown',
43
+ errorType: this.props.errorType || 'component',
44
+ timestamp: Date.now(),
45
+ componentStack: errorInfo.componentStack,
46
+ errorBoundary: this.constructor.name,
47
+ };
48
+
49
+ this.setState({
50
+ errorInfo: layoutErrorInfo,
51
+ });
52
+
53
+ // Call onError callback if provided
54
+ if (this.props.onError) {
55
+ this.props.onError(error, layoutErrorInfo);
56
+ }
57
+
58
+ // Log error in development mode
59
+ if (typeof Deno !== 'undefined' && Deno.env.get('NODE_ENV') === 'development') {
60
+ console.error('Layout Error Boundary caught an error:', error);
61
+ console.error('Error Info:', layoutErrorInfo);
62
+ console.error('Component Stack:', errorInfo.componentStack);
63
+ }
64
+ }
65
+
66
+ private handleRetry = (): void => {
67
+ if (this.state.retryCount < this.maxRetries) {
68
+ this.setState({
69
+ hasError: false,
70
+ error: null,
71
+ errorInfo: null,
72
+ retryCount: this.state.retryCount + 1,
73
+ });
74
+ }
75
+ };
76
+
77
+ private renderFallback(): ComponentChildren {
78
+ const { error } = this.state;
79
+ const { fallback } = this.props;
80
+
81
+ if (fallback && error) {
82
+ return fallback(error, this.handleRetry);
83
+ }
84
+
85
+ // Default fallback UI
86
+ return (
87
+ <div class="layout-error-boundary">
88
+ <div class="error-container">
89
+ <h2>Something went wrong</h2>
90
+ <p>An error occurred while rendering this layout.</p>
91
+ {this.state.retryCount < this.maxRetries && (
92
+ <button onClick={this.handleRetry} class="retry-button">
93
+ Try Again ({this.maxRetries - this.state.retryCount} attempts left)
94
+ </button>
95
+ )}
96
+ {typeof Deno !== 'undefined' && Deno.env.get('NODE_ENV') === 'development' && (
97
+ <details class="error-details">
98
+ <summary>Error Details (Development)</summary>
99
+ <pre>{error?.stack}</pre>
100
+ {this.state.errorInfo && (
101
+ <div>
102
+ <p>
103
+ <strong>Layout Path:</strong> {this.state.errorInfo.layoutPath}
104
+ </p>
105
+ <p>
106
+ <strong>Error Type:</strong> {this.state.errorInfo.errorType}
107
+ </p>
108
+ <p>
109
+ <strong>Timestamp:</strong> {new Date(this.state.errorInfo.timestamp).toISOString()}
110
+ </p>
111
+ </div>
112
+ )}
113
+ </details>
114
+ )}
115
+ </div>
116
+ </div>
117
+ );
118
+ }
119
+
120
+ render() {
121
+ if (this.state.hasError) {
122
+ return this.renderFallback();
123
+ }
124
+
125
+ return this.props.children;
126
+ }
127
+ }
@@ -0,0 +1,52 @@
1
+ /** @jsxImportSource preact */
2
+ import type { ComponentChildren } from 'preact';
3
+ import { PersistentIslandProvider } from '../core/islands/persistent-island-context.tsx';
4
+ import { defaultIslandPersistence, type IslandPersistence } from '../core/islands/island-persistence.ts';
5
+
6
+ interface PersistentIslandProps {
7
+ /** Unique ID used as the storage key for this island's state */
8
+ persistentId: string;
9
+ /** The island component(s) to wrap with persistence context */
10
+ children: ComponentChildren;
11
+ /** Custom persistence instance (defaults to sessionStorage-backed) */
12
+ persistence?: IslandPersistence;
13
+ }
14
+
15
+ /**
16
+ * PersistentIsland — provides automatic state persistence context to child islands.
17
+ *
18
+ * Wrap any island with this component and use `usePersistentIslandContext()` inside
19
+ * the island to get `saveState`, `loadState`, and `clearState` functions.
20
+ *
21
+ * Usage in a page:
22
+ * ```tsx
23
+ * <PersistentIsland persistentId="my-counter" island={{ condition: 'on:client' }}>
24
+ * <MyCounter />
25
+ * </PersistentIsland>
26
+ * ```
27
+ *
28
+ * Usage inside the island:
29
+ * ```tsx
30
+ * import { usePersistentIslandContext } from '@useavalon/avalon';
31
+ *
32
+ * function MyCounter() {
33
+ * const { saveState, loadState, clearState } = usePersistentIslandContext();
34
+ * // ...
35
+ * }
36
+ * ```
37
+ */
38
+ export function PersistentIsland({
39
+ persistentId,
40
+ children,
41
+ persistence = defaultIslandPersistence,
42
+ }: Readonly<PersistentIslandProps>) {
43
+ return (
44
+ <PersistentIslandProvider persistentId={persistentId} persistence={persistence}>
45
+ <div data-persistent-id={persistentId}>
46
+ {children}
47
+ </div>
48
+ </PersistentIslandProvider>
49
+ );
50
+ }
51
+
52
+ export default PersistentIsland;
@@ -0,0 +1,233 @@
1
+ /**
2
+ * StreamingErrorBoundary - Error boundary component for streaming contexts
3
+ *
4
+ * This component provides error isolation for Suspense boundaries in streaming SSR.
5
+ * It ensures that errors in one component don't break the entire page.
6
+ */
7
+
8
+ import { Component, type ComponentChildren } from 'preact';
9
+
10
+ export interface StreamingErrorBoundaryProps {
11
+ children: ComponentChildren;
12
+ fallback?: (error: Error, retry: () => void) => ComponentChildren;
13
+ onError?: (error: Error, errorInfo: any) => void;
14
+ componentId?: string;
15
+ isolateError?: boolean; // If true, error won't propagate to parent
16
+ }
17
+
18
+ export interface StreamingErrorBoundaryState {
19
+ hasError: boolean;
20
+ error: Error | null;
21
+ errorInfo: any;
22
+ }
23
+
24
+ /**
25
+ * Error boundary component for streaming contexts
26
+ *
27
+ * Wraps Suspense boundaries to provide error isolation and recovery.
28
+ * Prevents errors in one component from breaking the entire page.
29
+ */
30
+ export class StreamingErrorBoundary extends Component<
31
+ StreamingErrorBoundaryProps,
32
+ StreamingErrorBoundaryState
33
+ > {
34
+ constructor(props: StreamingErrorBoundaryProps) {
35
+ super(props);
36
+ this.state = {
37
+ hasError: false,
38
+ error: null,
39
+ errorInfo: null,
40
+ };
41
+ }
42
+
43
+ static override getDerivedStateFromError(error: Error): Partial<StreamingErrorBoundaryState> {
44
+ return {
45
+ hasError: true,
46
+ error,
47
+ };
48
+ }
49
+
50
+ override componentDidCatch(error: Error, errorInfo: any): void {
51
+ this.setState({
52
+ errorInfo,
53
+ });
54
+
55
+ // Log the error
56
+ console.error('[StreamingErrorBoundary] Caught error:', {
57
+ componentId: this.props.componentId,
58
+ error: error.message,
59
+ stack: error.stack,
60
+ componentStack: errorInfo.componentStack,
61
+ });
62
+
63
+ // Call onError callback if provided
64
+ if (this.props.onError) {
65
+ this.props.onError(error, errorInfo);
66
+ }
67
+
68
+ // If isolateError is false, re-throw to propagate to parent
69
+ if (!this.props.isolateError) {
70
+ throw error;
71
+ }
72
+ }
73
+
74
+ private handleRetry = (): void => {
75
+ this.setState({
76
+ hasError: false,
77
+ error: null,
78
+ errorInfo: null,
79
+ });
80
+ };
81
+
82
+ private renderFallback(): ComponentChildren {
83
+ const { error } = this.state;
84
+ const { fallback, componentId } = this.props;
85
+
86
+ if (fallback && error) {
87
+ return fallback(error, this.handleRetry);
88
+ }
89
+
90
+ // Default fallback UI
91
+ const isDevelopment = typeof Deno !== 'undefined' && Deno.env.get('DENO_ENV') !== 'production';
92
+
93
+ return (
94
+ <div
95
+ class="streaming-error-boundary"
96
+ data-error-boundary="true"
97
+ data-component-id={componentId}
98
+ style={{
99
+ background: '#fff3cd',
100
+ border: '2px solid #ffc107',
101
+ borderRadius: '8px',
102
+ padding: '20px',
103
+ margin: '20px 0',
104
+ fontFamily: 'system-ui, -apple-system, sans-serif',
105
+ }}
106
+ >
107
+ <div
108
+ class="error-boundary-header"
109
+ style={{
110
+ display: 'flex',
111
+ alignItems: 'center',
112
+ gap: '10px',
113
+ marginBottom: '10px',
114
+ }}
115
+ >
116
+ <span style={{ fontSize: '24px' }}>⚠️</span>
117
+ <h3 style={{ margin: 0, color: '#856404' }}>Component Error</h3>
118
+ </div>
119
+
120
+ <p style={{ margin: '10px 0', color: '#856404' }}>
121
+ An error occurred while rendering this component. The rest of the page should work normally.
122
+ </p>
123
+
124
+ <button
125
+ onClick={this.handleRetry}
126
+ style={{
127
+ background: '#ffc107',
128
+ border: 'none',
129
+ borderRadius: '4px',
130
+ padding: '8px 16px',
131
+ cursor: 'pointer',
132
+ fontWeight: 'bold',
133
+ color: '#856404',
134
+ marginTop: '10px',
135
+ }}
136
+ >
137
+ Retry
138
+ </button>
139
+
140
+ {isDevelopment && error && (
141
+ <details style={{ marginTop: '15px' }}>
142
+ <summary
143
+ style={{
144
+ cursor: 'pointer',
145
+ color: '#856404',
146
+ fontWeight: 'bold',
147
+ }}
148
+ >
149
+ Error Details (Development Mode)
150
+ </summary>
151
+ <div style={{ marginTop: '10px' }}>
152
+ {componentId && (
153
+ <p>
154
+ <strong>Component ID:</strong> {componentId}
155
+ </p>
156
+ )}
157
+ <p>
158
+ <strong>Error:</strong> {error.message}
159
+ </p>
160
+ {error.stack && (
161
+ <pre
162
+ style={{
163
+ background: '#f5f5f5',
164
+ padding: '10px',
165
+ borderRadius: '4px',
166
+ overflowX: 'auto',
167
+ fontSize: '12px',
168
+ marginTop: '10px',
169
+ }}
170
+ >
171
+ {error.stack}
172
+ </pre>
173
+ )}
174
+ {this.state.errorInfo?.componentStack && (
175
+ <div>
176
+ <p>
177
+ <strong>Component Stack:</strong>
178
+ </p>
179
+ <pre
180
+ style={{
181
+ background: '#f5f5f5',
182
+ padding: '10px',
183
+ borderRadius: '4px',
184
+ overflowX: 'auto',
185
+ fontSize: '12px',
186
+ marginTop: '10px',
187
+ }}
188
+ >
189
+ {this.state.errorInfo.componentStack}
190
+ </pre>
191
+ </div>
192
+ )}
193
+ </div>
194
+ </details>
195
+ )}
196
+ </div>
197
+ );
198
+ }
199
+
200
+ render() {
201
+ if (this.state.hasError) {
202
+ return this.renderFallback();
203
+ }
204
+
205
+ return this.props.children;
206
+ }
207
+ }
208
+
209
+ /**
210
+ * Higher-order component to wrap components with streaming error boundaries
211
+ */
212
+ export function withStreamingErrorBoundary<P extends object>(
213
+ WrappedComponent: (props: P) => ComponentChildren,
214
+ options?: {
215
+ fallback?: (error: Error, retry: () => void) => ComponentChildren;
216
+ componentId?: string;
217
+ isolateError?: boolean;
218
+ onError?: (error: Error, errorInfo: any) => void;
219
+ }
220
+ ) {
221
+ return function ComponentWithErrorBoundary(props: P) {
222
+ return (
223
+ <StreamingErrorBoundary
224
+ componentId={options?.componentId}
225
+ fallback={options?.fallback}
226
+ isolateError={options?.isolateError ?? true}
227
+ onError={options?.onError}
228
+ >
229
+ <WrappedComponent {...props} />
230
+ </StreamingErrorBoundary>
231
+ );
232
+ };
233
+ }