@useavalon/avalon 0.1.11 → 0.1.13

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 (141) hide show
  1. package/README.md +54 -54
  2. package/mod.ts +302 -302
  3. package/package.json +49 -26
  4. package/src/build/integration-bundler-plugin.ts +116 -116
  5. package/src/build/integration-config.ts +168 -168
  6. package/src/build/integration-detection-plugin.ts +117 -117
  7. package/src/build/integration-resolver-plugin.ts +90 -90
  8. package/src/build/island-manifest.ts +269 -269
  9. package/src/build/island-types-generator.ts +476 -476
  10. package/src/build/mdx-island-transform.ts +464 -464
  11. package/src/build/mdx-plugin.ts +98 -98
  12. package/src/build/page-island-transform.ts +598 -598
  13. package/src/build/prop-extractors/index.ts +21 -21
  14. package/src/build/prop-extractors/lit.ts +140 -140
  15. package/src/build/prop-extractors/qwik.ts +16 -16
  16. package/src/build/prop-extractors/solid.ts +125 -125
  17. package/src/build/prop-extractors/svelte.ts +194 -194
  18. package/src/build/prop-extractors/vue.ts +111 -111
  19. package/src/build/sidecar-file-manager.ts +104 -104
  20. package/src/build/sidecar-renderer.ts +30 -30
  21. package/src/client/adapters/index.ts +21 -13
  22. package/src/client/components.ts +35 -35
  23. package/src/client/css-hmr-handler.ts +344 -344
  24. package/src/client/framework-adapter.ts +462 -462
  25. package/src/client/hmr-coordinator.ts +396 -396
  26. package/src/client/hmr-error-overlay.js +533 -533
  27. package/src/client/main.js +824 -816
  28. package/src/client/types/framework-runtime.d.ts +68 -68
  29. package/src/client/types/vite-hmr.d.ts +46 -46
  30. package/src/client/types/vite-virtual-modules.d.ts +70 -60
  31. package/src/components/Image.tsx +123 -123
  32. package/src/components/IslandErrorBoundary.tsx +145 -145
  33. package/src/components/LayoutDataErrorBoundary.tsx +141 -141
  34. package/src/components/LayoutErrorBoundary.tsx +127 -127
  35. package/src/components/PersistentIsland.tsx +52 -52
  36. package/src/components/StreamingErrorBoundary.tsx +233 -233
  37. package/src/components/StreamingLayout.tsx +538 -538
  38. package/src/core/components/component-analyzer.ts +192 -192
  39. package/src/core/components/component-detection.ts +508 -508
  40. package/src/core/components/enhanced-framework-detector.ts +500 -500
  41. package/src/core/components/framework-registry.ts +563 -563
  42. package/src/core/content/mdx-processor.ts +46 -46
  43. package/src/core/integrations/index.ts +19 -19
  44. package/src/core/integrations/loader.ts +125 -125
  45. package/src/core/integrations/registry.ts +175 -175
  46. package/src/core/islands/island-persistence.ts +325 -325
  47. package/src/core/islands/island-state-serializer.ts +258 -258
  48. package/src/core/islands/persistent-island-context.tsx +80 -80
  49. package/src/core/islands/use-persistent-state.ts +68 -68
  50. package/src/core/layout/enhanced-layout-resolver.ts +322 -322
  51. package/src/core/layout/layout-cache-manager.ts +485 -485
  52. package/src/core/layout/layout-composer.ts +357 -357
  53. package/src/core/layout/layout-data-loader.ts +516 -516
  54. package/src/core/layout/layout-discovery.ts +243 -243
  55. package/src/core/layout/layout-matcher.ts +299 -299
  56. package/src/core/layout/layout-types.ts +110 -110
  57. package/src/core/modules/framework-module-resolver.ts +273 -273
  58. package/src/islands/component-analysis.ts +213 -213
  59. package/src/islands/css-utils.ts +565 -565
  60. package/src/islands/discovery/index.ts +80 -80
  61. package/src/islands/discovery/registry.ts +340 -340
  62. package/src/islands/discovery/resolver.ts +477 -477
  63. package/src/islands/discovery/scanner.ts +386 -386
  64. package/src/islands/discovery/types.ts +117 -117
  65. package/src/islands/discovery/validator.ts +544 -544
  66. package/src/islands/discovery/watcher.ts +368 -368
  67. package/src/islands/framework-detection.ts +428 -428
  68. package/src/islands/integration-loader.ts +490 -490
  69. package/src/islands/island.tsx +565 -565
  70. package/src/islands/render-cache.ts +550 -550
  71. package/src/islands/types.ts +80 -80
  72. package/src/islands/universal-css-collector.ts +157 -157
  73. package/src/islands/universal-head-collector.ts +137 -137
  74. package/src/layout-system.d.ts +592 -592
  75. package/src/layout-system.ts +218 -218
  76. package/src/middleware/discovery.ts +268 -268
  77. package/src/middleware/executor.ts +315 -315
  78. package/src/middleware/index.ts +76 -76
  79. package/src/middleware/types.ts +99 -99
  80. package/src/nitro/build-config.ts +575 -575
  81. package/src/nitro/config.ts +483 -483
  82. package/src/nitro/error-handler.ts +636 -636
  83. package/src/nitro/index.ts +173 -173
  84. package/src/nitro/island-manifest.ts +584 -584
  85. package/src/nitro/middleware-adapter.ts +260 -260
  86. package/src/nitro/renderer.ts +1471 -1471
  87. package/src/nitro/route-discovery.ts +439 -439
  88. package/src/nitro/types.ts +321 -321
  89. package/src/render/collect-css.ts +198 -198
  90. package/src/render/error-pages.ts +79 -79
  91. package/src/render/isolated-ssr-renderer.ts +654 -654
  92. package/src/render/ssr.ts +1030 -1030
  93. package/src/schemas/api.ts +30 -30
  94. package/src/schemas/core.ts +64 -64
  95. package/src/schemas/index.ts +212 -212
  96. package/src/schemas/layout.ts +279 -279
  97. package/src/schemas/routing/index.ts +38 -38
  98. package/src/schemas/routing.ts +376 -376
  99. package/src/types/as-island.ts +20 -20
  100. package/src/types/image.d.ts +106 -106
  101. package/src/types/index.d.ts +22 -22
  102. package/src/types/island-jsx.d.ts +33 -33
  103. package/src/types/island-prop.d.ts +20 -20
  104. package/src/types/layout.ts +285 -285
  105. package/src/types/mdx.d.ts +6 -6
  106. package/src/types/routing.ts +555 -555
  107. package/src/types/types.ts +5 -5
  108. package/src/types/urlpattern.d.ts +49 -49
  109. package/src/types/vite-env.d.ts +11 -11
  110. package/src/utils/dev-logger.ts +299 -299
  111. package/src/utils/fs.ts +151 -151
  112. package/src/vite-plugin/auto-discover.ts +551 -551
  113. package/src/vite-plugin/config.ts +266 -266
  114. package/src/vite-plugin/errors.ts +127 -127
  115. package/src/vite-plugin/image-optimization.ts +156 -156
  116. package/src/vite-plugin/integration-activator.ts +126 -126
  117. package/src/vite-plugin/island-sidecar-plugin.ts +176 -176
  118. package/src/vite-plugin/module-discovery.ts +189 -189
  119. package/src/vite-plugin/nitro-integration.ts +1354 -1354
  120. package/src/vite-plugin/plugin.ts +403 -409
  121. package/src/vite-plugin/types.ts +327 -327
  122. package/src/vite-plugin/validation.ts +228 -228
  123. package/src/client/adapters/index.js +0 -12
  124. package/src/client/adapters/lit-adapter.js +0 -467
  125. package/src/client/adapters/lit-adapter.ts +0 -654
  126. package/src/client/adapters/preact-adapter.js +0 -223
  127. package/src/client/adapters/preact-adapter.ts +0 -331
  128. package/src/client/adapters/qwik-adapter.js +0 -259
  129. package/src/client/adapters/qwik-adapter.ts +0 -345
  130. package/src/client/adapters/react-adapter.js +0 -220
  131. package/src/client/adapters/react-adapter.ts +0 -353
  132. package/src/client/adapters/solid-adapter.js +0 -295
  133. package/src/client/adapters/solid-adapter.ts +0 -451
  134. package/src/client/adapters/svelte-adapter.js +0 -368
  135. package/src/client/adapters/svelte-adapter.ts +0 -524
  136. package/src/client/adapters/vue-adapter.js +0 -278
  137. package/src/client/adapters/vue-adapter.ts +0 -467
  138. package/src/client/components.js +0 -23
  139. package/src/client/css-hmr-handler.js +0 -263
  140. package/src/client/framework-adapter.js +0 -283
  141. package/src/client/hmr-coordinator.js +0 -274
@@ -1,340 +1,340 @@
1
- /**
2
- * Island Registry
3
- *
4
- * Central registry for all discovered islands with resolution capabilities.
5
- * Handles registration, resolution by name/namespace, and collision detection.
6
- */
7
-
8
- import type {
9
- DiscoveredIsland,
10
- IslandDirectory,
11
- IslandCollision,
12
- IslandDiscoveryConfig,
13
- } from "./types.ts";
14
- import {
15
- discoverIslandDirectories,
16
- discoverIslandsInDirectory,
17
- getQualifiedIslandName,
18
- parseQualifiedIslandName,
19
- } from "./scanner.ts";
20
-
21
- /**
22
- * Central registry for all discovered islands.
23
- * Provides registration, resolution, and collision detection capabilities.
24
- */
25
- export class IslandRegistry {
26
- /** All discovered islands indexed by qualified name */
27
- private _islands: Map<string, DiscoveredIsland> = new Map();
28
-
29
- /** All discovered island directories */
30
- private _directories: IslandDirectory[] = [];
31
-
32
- /** Index of islands by name (for fast lookup) */
33
- private _byName: Map<string, DiscoveredIsland[]> = new Map();
34
-
35
- /** Detected collisions */
36
- private _collisions: IslandCollision[] = [];
37
-
38
- /** Project root directory */
39
- private _projectRoot: string;
40
-
41
- /** Discovery configuration */
42
- private _config: IslandDiscoveryConfig;
43
-
44
- constructor(projectRoot: string, config: IslandDiscoveryConfig = {}) {
45
- this._projectRoot = projectRoot;
46
- this._config = config;
47
- }
48
-
49
- /**
50
- * Get all discovered islands as a Map.
51
- */
52
- get islands(): Map<string, DiscoveredIsland> {
53
- return new Map(this._islands);
54
- }
55
-
56
- /**
57
- * Get all discovered island directories.
58
- */
59
- get directories(): IslandDirectory[] {
60
- return [...this._directories];
61
- }
62
-
63
- /**
64
- * Get all detected collisions.
65
- */
66
- get collisions(): IslandCollision[] {
67
- return [...this._collisions];
68
- }
69
-
70
- /**
71
- * Get the number of registered islands.
72
- */
73
- get size(): number {
74
- return this._islands.size;
75
- }
76
-
77
- /**
78
- * Register a discovered island in the registry.
79
- * Updates the name index and detects collisions.
80
- *
81
- * @param island - The island to register
82
- */
83
- register(island: DiscoveredIsland): void {
84
- const qualifiedName = getQualifiedIslandName(island);
85
-
86
- // Add to main registry
87
- this._islands.set(qualifiedName, island);
88
-
89
- // Update name index
90
- const existing = this._byName.get(island.name) || [];
91
- existing.push(island);
92
- this._byName.set(island.name, existing);
93
- }
94
-
95
- /**
96
- * Resolve an island by name with optional namespace.
97
- *
98
- * Resolution priority:
99
- * 1. Exact qualified name match (namespace/name)
100
- * 2. Default islands directory (highest priority for unqualified names)
101
- * 3. First match in alphabetical order by namespace
102
- *
103
- * @param name - Component name or qualified name (namespace/name)
104
- * @param namespace - Optional namespace to narrow search
105
- * @returns The resolved island or null if not found
106
- */
107
- resolve(name: string, namespace?: string): DiscoveredIsland | null {
108
- // If namespace is provided, try exact match first
109
- if (namespace !== undefined) {
110
- const qualifiedName = namespace === "" ? name : `${namespace}/${name}`;
111
- const exact = this._islands.get(qualifiedName);
112
- if (exact) return exact;
113
- }
114
-
115
- // Check if name is already a qualified name
116
- if (name.includes("/")) {
117
- const exact = this._islands.get(name);
118
- if (exact) return exact;
119
-
120
- // Parse and try to resolve
121
- const { namespace: parsedNs, name: parsedName } = parseQualifiedIslandName(name);
122
- const qualifiedName = parsedNs === "" ? parsedName : `${parsedNs}/${parsedName}`;
123
- return this._islands.get(qualifiedName) || null;
124
- }
125
-
126
- // Find all islands with this name
127
- const matches = this._byName.get(name);
128
- if (!matches || matches.length === 0) {
129
- return null;
130
- }
131
-
132
- // If only one match, return it
133
- if (matches.length === 1) {
134
- return matches[0];
135
- }
136
-
137
- // Multiple matches - prioritize default directory
138
- const defaultMatch = matches.find(island => island.directory.isDefault);
139
- if (defaultMatch) {
140
- return defaultMatch;
141
- }
142
-
143
- // Return first match (alphabetically by namespace due to sorting)
144
- return matches[0];
145
- }
146
-
147
- /**
148
- * Find all islands matching a name.
149
- * Useful for collision detection and disambiguation.
150
- *
151
- * @param name - Component name to search for
152
- * @returns Array of all islands with this name
153
- */
154
- findByName(name: string): DiscoveredIsland[] {
155
- return this._byName.get(name) || [];
156
- }
157
-
158
- /**
159
- * Get the qualified name for an island.
160
- *
161
- * @param island - The island to get the qualified name for
162
- * @returns The qualified name (namespace/name or just name for default)
163
- */
164
- getQualifiedName(island: DiscoveredIsland): string {
165
- return getQualifiedIslandName(island);
166
- }
167
-
168
- /**
169
- * Check if an island with the given name or qualified name exists.
170
- *
171
- * @param nameOrQualified - Component name or qualified name
172
- * @returns True if the island exists
173
- */
174
- has(nameOrQualified: string): boolean {
175
- // Check qualified name first
176
- if (this._islands.has(nameOrQualified)) {
177
- return true;
178
- }
179
-
180
- // Check by name
181
- return this._byName.has(nameOrQualified);
182
- }
183
-
184
- /**
185
- * Detect naming collisions across all registered islands.
186
- * A collision occurs when multiple islands share the same component name.
187
- *
188
- * @returns Array of detected collisions
189
- */
190
- detectCollisions(): IslandCollision[] {
191
- const collisions: IslandCollision[] = [];
192
-
193
- for (const [name, islands] of this._byName) {
194
- if (islands.length > 1) {
195
- // Determine resolution strategy
196
- const hasDefault = islands.some(island => island.directory.isDefault);
197
- const resolution: "namespace" | "priority" = hasDefault ? "priority" : "namespace";
198
-
199
- collisions.push({
200
- name,
201
- islands: [...islands],
202
- resolution,
203
- });
204
- }
205
- }
206
-
207
- this._collisions = collisions;
208
- return collisions;
209
- }
210
-
211
- /**
212
- * Clear the registry and rebuild from filesystem.
213
- * Discovers all island directories and their components.
214
- */
215
- async rebuild(): Promise<void> {
216
- // Clear existing data
217
- this._islands.clear();
218
- this._byName.clear();
219
- this._directories = [];
220
- this._collisions = [];
221
-
222
- // Discover directories
223
- this._directories = await discoverIslandDirectories(
224
- this._projectRoot,
225
- this._config
226
- );
227
-
228
- // Discover and register islands from each directory
229
- for (const directory of this._directories) {
230
- const islands = await discoverIslandsInDirectory(
231
- directory,
232
- this._projectRoot
233
- );
234
-
235
- for (const island of islands) {
236
- this.register(island);
237
- }
238
- }
239
-
240
- // Detect collisions
241
- this.detectCollisions();
242
- }
243
-
244
- /**
245
- * Get all islands as an array.
246
- *
247
- * @returns Array of all registered islands
248
- */
249
- getAllIslands(): DiscoveredIsland[] {
250
- return Array.from(this._islands.values());
251
- }
252
-
253
- /**
254
- * Get islands from a specific directory.
255
- *
256
- * @param directory - The directory to get islands from
257
- * @returns Array of islands in the directory
258
- */
259
- getIslandsInDirectory(directory: IslandDirectory): DiscoveredIsland[] {
260
- return this.getAllIslands().filter(
261
- island => island.directory.path === directory.path
262
- );
263
- }
264
-
265
- /**
266
- * Get islands by namespace.
267
- *
268
- * @param namespace - The namespace to filter by (empty string for default)
269
- * @returns Array of islands in the namespace
270
- */
271
- getIslandsByNamespace(namespace: string): DiscoveredIsland[] {
272
- return this.getAllIslands().filter(
273
- island => island.namespace === namespace
274
- );
275
- }
276
-
277
- /**
278
- * Remove an island from the registry.
279
- *
280
- * @param qualifiedName - The qualified name of the island to remove
281
- * @returns True if the island was removed
282
- */
283
- unregister(qualifiedName: string): boolean {
284
- const island = this._islands.get(qualifiedName);
285
- if (!island) {
286
- return false;
287
- }
288
-
289
- // Remove from main registry
290
- this._islands.delete(qualifiedName);
291
-
292
- // Remove from name index
293
- const byName = this._byName.get(island.name);
294
- if (byName) {
295
- const filtered = byName.filter(i =>
296
- getQualifiedIslandName(i) !== qualifiedName
297
- );
298
- if (filtered.length === 0) {
299
- this._byName.delete(island.name);
300
- } else {
301
- this._byName.set(island.name, filtered);
302
- }
303
- }
304
-
305
- return true;
306
- }
307
-
308
- /**
309
- * Create a snapshot of the registry state.
310
- * Useful for debugging and testing.
311
- */
312
- toJSON(): {
313
- islands: Record<string, DiscoveredIsland>;
314
- directories: IslandDirectory[];
315
- collisions: IslandCollision[];
316
- } {
317
- return {
318
- islands: Object.fromEntries(this._islands),
319
- directories: this._directories,
320
- collisions: this._collisions,
321
- };
322
- }
323
- }
324
-
325
- /**
326
- * Create and initialize an island registry.
327
- * Convenience function that creates a registry and rebuilds it.
328
- *
329
- * @param projectRoot - The root directory of the project
330
- * @param config - Optional configuration for discovery
331
- * @returns Initialized island registry
332
- */
333
- export async function createIslandRegistry(
334
- projectRoot: string,
335
- config: IslandDiscoveryConfig = {}
336
- ): Promise<IslandRegistry> {
337
- const registry = new IslandRegistry(projectRoot, config);
338
- await registry.rebuild();
339
- return registry;
340
- }
1
+ /**
2
+ * Island Registry
3
+ *
4
+ * Central registry for all discovered islands with resolution capabilities.
5
+ * Handles registration, resolution by name/namespace, and collision detection.
6
+ */
7
+
8
+ import type {
9
+ DiscoveredIsland,
10
+ IslandDirectory,
11
+ IslandCollision,
12
+ IslandDiscoveryConfig,
13
+ } from "./types.ts";
14
+ import {
15
+ discoverIslandDirectories,
16
+ discoverIslandsInDirectory,
17
+ getQualifiedIslandName,
18
+ parseQualifiedIslandName,
19
+ } from "./scanner.ts";
20
+
21
+ /**
22
+ * Central registry for all discovered islands.
23
+ * Provides registration, resolution, and collision detection capabilities.
24
+ */
25
+ export class IslandRegistry {
26
+ /** All discovered islands indexed by qualified name */
27
+ private _islands: Map<string, DiscoveredIsland> = new Map();
28
+
29
+ /** All discovered island directories */
30
+ private _directories: IslandDirectory[] = [];
31
+
32
+ /** Index of islands by name (for fast lookup) */
33
+ private _byName: Map<string, DiscoveredIsland[]> = new Map();
34
+
35
+ /** Detected collisions */
36
+ private _collisions: IslandCollision[] = [];
37
+
38
+ /** Project root directory */
39
+ private _projectRoot: string;
40
+
41
+ /** Discovery configuration */
42
+ private _config: IslandDiscoveryConfig;
43
+
44
+ constructor(projectRoot: string, config: IslandDiscoveryConfig = {}) {
45
+ this._projectRoot = projectRoot;
46
+ this._config = config;
47
+ }
48
+
49
+ /**
50
+ * Get all discovered islands as a Map.
51
+ */
52
+ get islands(): Map<string, DiscoveredIsland> {
53
+ return new Map(this._islands);
54
+ }
55
+
56
+ /**
57
+ * Get all discovered island directories.
58
+ */
59
+ get directories(): IslandDirectory[] {
60
+ return [...this._directories];
61
+ }
62
+
63
+ /**
64
+ * Get all detected collisions.
65
+ */
66
+ get collisions(): IslandCollision[] {
67
+ return [...this._collisions];
68
+ }
69
+
70
+ /**
71
+ * Get the number of registered islands.
72
+ */
73
+ get size(): number {
74
+ return this._islands.size;
75
+ }
76
+
77
+ /**
78
+ * Register a discovered island in the registry.
79
+ * Updates the name index and detects collisions.
80
+ *
81
+ * @param island - The island to register
82
+ */
83
+ register(island: DiscoveredIsland): void {
84
+ const qualifiedName = getQualifiedIslandName(island);
85
+
86
+ // Add to main registry
87
+ this._islands.set(qualifiedName, island);
88
+
89
+ // Update name index
90
+ const existing = this._byName.get(island.name) || [];
91
+ existing.push(island);
92
+ this._byName.set(island.name, existing);
93
+ }
94
+
95
+ /**
96
+ * Resolve an island by name with optional namespace.
97
+ *
98
+ * Resolution priority:
99
+ * 1. Exact qualified name match (namespace/name)
100
+ * 2. Default islands directory (highest priority for unqualified names)
101
+ * 3. First match in alphabetical order by namespace
102
+ *
103
+ * @param name - Component name or qualified name (namespace/name)
104
+ * @param namespace - Optional namespace to narrow search
105
+ * @returns The resolved island or null if not found
106
+ */
107
+ resolve(name: string, namespace?: string): DiscoveredIsland | null {
108
+ // If namespace is provided, try exact match first
109
+ if (namespace !== undefined) {
110
+ const qualifiedName = namespace === "" ? name : `${namespace}/${name}`;
111
+ const exact = this._islands.get(qualifiedName);
112
+ if (exact) return exact;
113
+ }
114
+
115
+ // Check if name is already a qualified name
116
+ if (name.includes("/")) {
117
+ const exact = this._islands.get(name);
118
+ if (exact) return exact;
119
+
120
+ // Parse and try to resolve
121
+ const { namespace: parsedNs, name: parsedName } = parseQualifiedIslandName(name);
122
+ const qualifiedName = parsedNs === "" ? parsedName : `${parsedNs}/${parsedName}`;
123
+ return this._islands.get(qualifiedName) || null;
124
+ }
125
+
126
+ // Find all islands with this name
127
+ const matches = this._byName.get(name);
128
+ if (!matches || matches.length === 0) {
129
+ return null;
130
+ }
131
+
132
+ // If only one match, return it
133
+ if (matches.length === 1) {
134
+ return matches[0];
135
+ }
136
+
137
+ // Multiple matches - prioritize default directory
138
+ const defaultMatch = matches.find(island => island.directory.isDefault);
139
+ if (defaultMatch) {
140
+ return defaultMatch;
141
+ }
142
+
143
+ // Return first match (alphabetically by namespace due to sorting)
144
+ return matches[0];
145
+ }
146
+
147
+ /**
148
+ * Find all islands matching a name.
149
+ * Useful for collision detection and disambiguation.
150
+ *
151
+ * @param name - Component name to search for
152
+ * @returns Array of all islands with this name
153
+ */
154
+ findByName(name: string): DiscoveredIsland[] {
155
+ return this._byName.get(name) || [];
156
+ }
157
+
158
+ /**
159
+ * Get the qualified name for an island.
160
+ *
161
+ * @param island - The island to get the qualified name for
162
+ * @returns The qualified name (namespace/name or just name for default)
163
+ */
164
+ getQualifiedName(island: DiscoveredIsland): string {
165
+ return getQualifiedIslandName(island);
166
+ }
167
+
168
+ /**
169
+ * Check if an island with the given name or qualified name exists.
170
+ *
171
+ * @param nameOrQualified - Component name or qualified name
172
+ * @returns True if the island exists
173
+ */
174
+ has(nameOrQualified: string): boolean {
175
+ // Check qualified name first
176
+ if (this._islands.has(nameOrQualified)) {
177
+ return true;
178
+ }
179
+
180
+ // Check by name
181
+ return this._byName.has(nameOrQualified);
182
+ }
183
+
184
+ /**
185
+ * Detect naming collisions across all registered islands.
186
+ * A collision occurs when multiple islands share the same component name.
187
+ *
188
+ * @returns Array of detected collisions
189
+ */
190
+ detectCollisions(): IslandCollision[] {
191
+ const collisions: IslandCollision[] = [];
192
+
193
+ for (const [name, islands] of this._byName) {
194
+ if (islands.length > 1) {
195
+ // Determine resolution strategy
196
+ const hasDefault = islands.some(island => island.directory.isDefault);
197
+ const resolution: "namespace" | "priority" = hasDefault ? "priority" : "namespace";
198
+
199
+ collisions.push({
200
+ name,
201
+ islands: [...islands],
202
+ resolution,
203
+ });
204
+ }
205
+ }
206
+
207
+ this._collisions = collisions;
208
+ return collisions;
209
+ }
210
+
211
+ /**
212
+ * Clear the registry and rebuild from filesystem.
213
+ * Discovers all island directories and their components.
214
+ */
215
+ async rebuild(): Promise<void> {
216
+ // Clear existing data
217
+ this._islands.clear();
218
+ this._byName.clear();
219
+ this._directories = [];
220
+ this._collisions = [];
221
+
222
+ // Discover directories
223
+ this._directories = await discoverIslandDirectories(
224
+ this._projectRoot,
225
+ this._config
226
+ );
227
+
228
+ // Discover and register islands from each directory
229
+ for (const directory of this._directories) {
230
+ const islands = await discoverIslandsInDirectory(
231
+ directory,
232
+ this._projectRoot
233
+ );
234
+
235
+ for (const island of islands) {
236
+ this.register(island);
237
+ }
238
+ }
239
+
240
+ // Detect collisions
241
+ this.detectCollisions();
242
+ }
243
+
244
+ /**
245
+ * Get all islands as an array.
246
+ *
247
+ * @returns Array of all registered islands
248
+ */
249
+ getAllIslands(): DiscoveredIsland[] {
250
+ return Array.from(this._islands.values());
251
+ }
252
+
253
+ /**
254
+ * Get islands from a specific directory.
255
+ *
256
+ * @param directory - The directory to get islands from
257
+ * @returns Array of islands in the directory
258
+ */
259
+ getIslandsInDirectory(directory: IslandDirectory): DiscoveredIsland[] {
260
+ return this.getAllIslands().filter(
261
+ island => island.directory.path === directory.path
262
+ );
263
+ }
264
+
265
+ /**
266
+ * Get islands by namespace.
267
+ *
268
+ * @param namespace - The namespace to filter by (empty string for default)
269
+ * @returns Array of islands in the namespace
270
+ */
271
+ getIslandsByNamespace(namespace: string): DiscoveredIsland[] {
272
+ return this.getAllIslands().filter(
273
+ island => island.namespace === namespace
274
+ );
275
+ }
276
+
277
+ /**
278
+ * Remove an island from the registry.
279
+ *
280
+ * @param qualifiedName - The qualified name of the island to remove
281
+ * @returns True if the island was removed
282
+ */
283
+ unregister(qualifiedName: string): boolean {
284
+ const island = this._islands.get(qualifiedName);
285
+ if (!island) {
286
+ return false;
287
+ }
288
+
289
+ // Remove from main registry
290
+ this._islands.delete(qualifiedName);
291
+
292
+ // Remove from name index
293
+ const byName = this._byName.get(island.name);
294
+ if (byName) {
295
+ const filtered = byName.filter(i =>
296
+ getQualifiedIslandName(i) !== qualifiedName
297
+ );
298
+ if (filtered.length === 0) {
299
+ this._byName.delete(island.name);
300
+ } else {
301
+ this._byName.set(island.name, filtered);
302
+ }
303
+ }
304
+
305
+ return true;
306
+ }
307
+
308
+ /**
309
+ * Create a snapshot of the registry state.
310
+ * Useful for debugging and testing.
311
+ */
312
+ toJSON(): {
313
+ islands: Record<string, DiscoveredIsland>;
314
+ directories: IslandDirectory[];
315
+ collisions: IslandCollision[];
316
+ } {
317
+ return {
318
+ islands: Object.fromEntries(this._islands),
319
+ directories: this._directories,
320
+ collisions: this._collisions,
321
+ };
322
+ }
323
+ }
324
+
325
+ /**
326
+ * Create and initialize an island registry.
327
+ * Convenience function that creates a registry and rebuilds it.
328
+ *
329
+ * @param projectRoot - The root directory of the project
330
+ * @param config - Optional configuration for discovery
331
+ * @returns Initialized island registry
332
+ */
333
+ export async function createIslandRegistry(
334
+ projectRoot: string,
335
+ config: IslandDiscoveryConfig = {}
336
+ ): Promise<IslandRegistry> {
337
+ const registry = new IslandRegistry(projectRoot, config);
338
+ await registry.rebuild();
339
+ return registry;
340
+ }