holosphere 1.1.20 → 2.0.0-alpha1

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 (147) hide show
  1. package/.env.example +36 -0
  2. package/.eslintrc.json +16 -0
  3. package/.prettierrc.json +7 -0
  4. package/LICENSE +162 -38
  5. package/README.md +483 -367
  6. package/bin/holosphere-activitypub.js +158 -0
  7. package/cleanup-test-data.js +204 -0
  8. package/examples/demo.html +1333 -0
  9. package/examples/example-bot.js +197 -0
  10. package/package.json +47 -87
  11. package/scripts/check-bundle-size.js +54 -0
  12. package/scripts/check-quest-ids.js +77 -0
  13. package/scripts/import-holons.js +578 -0
  14. package/scripts/publish-to-relay.js +101 -0
  15. package/scripts/read-example.js +186 -0
  16. package/scripts/relay-diagnostic.js +59 -0
  17. package/scripts/relay-example.js +179 -0
  18. package/scripts/resync-to-relay.js +245 -0
  19. package/scripts/revert-import.js +196 -0
  20. package/scripts/test-hybrid-mode.js +108 -0
  21. package/scripts/test-local-storage.js +63 -0
  22. package/scripts/test-nostr-direct.js +55 -0
  23. package/scripts/test-read-data.js +45 -0
  24. package/scripts/test-write-read.js +63 -0
  25. package/scripts/verify-import.js +95 -0
  26. package/scripts/verify-relay-data.js +139 -0
  27. package/src/ai/aggregation.js +319 -0
  28. package/src/ai/breakdown.js +511 -0
  29. package/src/ai/classifier.js +217 -0
  30. package/src/ai/council.js +228 -0
  31. package/src/ai/embeddings.js +279 -0
  32. package/src/ai/federation-ai.js +324 -0
  33. package/src/ai/h3-ai.js +955 -0
  34. package/src/ai/index.js +112 -0
  35. package/src/ai/json-ops.js +225 -0
  36. package/src/ai/llm-service.js +205 -0
  37. package/src/ai/nl-query.js +223 -0
  38. package/src/ai/relationships.js +353 -0
  39. package/src/ai/schema-extractor.js +218 -0
  40. package/src/ai/spatial.js +293 -0
  41. package/src/ai/tts.js +194 -0
  42. package/src/content/social-protocols.js +168 -0
  43. package/src/core/holosphere.js +273 -0
  44. package/src/crypto/secp256k1.js +259 -0
  45. package/src/federation/discovery.js +334 -0
  46. package/src/federation/hologram.js +1042 -0
  47. package/src/federation/registry.js +386 -0
  48. package/src/hierarchical/upcast.js +110 -0
  49. package/src/index.js +2669 -0
  50. package/src/schema/validator.js +91 -0
  51. package/src/spatial/h3-operations.js +110 -0
  52. package/src/storage/backend-factory.js +125 -0
  53. package/src/storage/backend-interface.js +142 -0
  54. package/src/storage/backends/activitypub/server.js +653 -0
  55. package/src/storage/backends/activitypub-backend.js +272 -0
  56. package/src/storage/backends/gundb-backend.js +233 -0
  57. package/src/storage/backends/nostr-backend.js +136 -0
  58. package/src/storage/filesystem-storage-browser.js +41 -0
  59. package/src/storage/filesystem-storage.js +138 -0
  60. package/src/storage/global-tables.js +81 -0
  61. package/src/storage/gun-async.js +281 -0
  62. package/src/storage/gun-wrapper.js +221 -0
  63. package/src/storage/indexeddb-storage.js +122 -0
  64. package/src/storage/key-storage-simple.js +76 -0
  65. package/src/storage/key-storage.js +136 -0
  66. package/src/storage/memory-storage.js +59 -0
  67. package/src/storage/migration.js +338 -0
  68. package/src/storage/nostr-async.js +811 -0
  69. package/src/storage/nostr-client.js +939 -0
  70. package/src/storage/nostr-wrapper.js +211 -0
  71. package/src/storage/outbox-queue.js +208 -0
  72. package/src/storage/persistent-storage.js +109 -0
  73. package/src/storage/sync-service.js +164 -0
  74. package/src/subscriptions/manager.js +142 -0
  75. package/test-ai-real-api.js +202 -0
  76. package/tests/unit/ai/aggregation.test.js +295 -0
  77. package/tests/unit/ai/breakdown.test.js +446 -0
  78. package/tests/unit/ai/classifier.test.js +294 -0
  79. package/tests/unit/ai/council.test.js +262 -0
  80. package/tests/unit/ai/embeddings.test.js +384 -0
  81. package/tests/unit/ai/federation-ai.test.js +344 -0
  82. package/tests/unit/ai/h3-ai.test.js +458 -0
  83. package/tests/unit/ai/index.test.js +304 -0
  84. package/tests/unit/ai/json-ops.test.js +307 -0
  85. package/tests/unit/ai/llm-service.test.js +390 -0
  86. package/tests/unit/ai/nl-query.test.js +383 -0
  87. package/tests/unit/ai/relationships.test.js +311 -0
  88. package/tests/unit/ai/schema-extractor.test.js +384 -0
  89. package/tests/unit/ai/spatial.test.js +279 -0
  90. package/tests/unit/ai/tts.test.js +279 -0
  91. package/tests/unit/content.test.js +332 -0
  92. package/tests/unit/contract/core.test.js +88 -0
  93. package/tests/unit/contract/crypto.test.js +198 -0
  94. package/tests/unit/contract/data.test.js +223 -0
  95. package/tests/unit/contract/federation.test.js +181 -0
  96. package/tests/unit/contract/hierarchical.test.js +113 -0
  97. package/tests/unit/contract/schema.test.js +114 -0
  98. package/tests/unit/contract/social.test.js +217 -0
  99. package/tests/unit/contract/spatial.test.js +110 -0
  100. package/tests/unit/contract/subscriptions.test.js +128 -0
  101. package/tests/unit/contract/utils.test.js +159 -0
  102. package/tests/unit/core.test.js +152 -0
  103. package/tests/unit/crypto.test.js +328 -0
  104. package/tests/unit/federation.test.js +234 -0
  105. package/tests/unit/gun-async.test.js +252 -0
  106. package/tests/unit/hierarchical.test.js +399 -0
  107. package/tests/unit/integration/scenario-01-geographic-storage.test.js +74 -0
  108. package/tests/unit/integration/scenario-02-federation.test.js +76 -0
  109. package/tests/unit/integration/scenario-03-subscriptions.test.js +102 -0
  110. package/tests/unit/integration/scenario-04-validation.test.js +129 -0
  111. package/tests/unit/integration/scenario-05-hierarchy.test.js +125 -0
  112. package/tests/unit/integration/scenario-06-social.test.js +135 -0
  113. package/tests/unit/integration/scenario-07-persistence.test.js +130 -0
  114. package/tests/unit/integration/scenario-08-authorization.test.js +161 -0
  115. package/tests/unit/integration/scenario-09-cross-dimensional.test.js +139 -0
  116. package/tests/unit/integration/scenario-10-cross-holosphere-capabilities.test.js +357 -0
  117. package/tests/unit/integration/scenario-11-cross-holosphere-federation.test.js +410 -0
  118. package/tests/unit/integration/scenario-12-capability-federated-read.test.js +719 -0
  119. package/tests/unit/performance/benchmark.test.js +85 -0
  120. package/tests/unit/schema.test.js +213 -0
  121. package/tests/unit/spatial.test.js +158 -0
  122. package/tests/unit/storage.test.js +195 -0
  123. package/tests/unit/subscriptions.test.js +328 -0
  124. package/tests/unit/test-data-permanence-debug.js +197 -0
  125. package/tests/unit/test-data-permanence.js +340 -0
  126. package/tests/unit/test-key-persistence-fixed.js +148 -0
  127. package/tests/unit/test-key-persistence.js +172 -0
  128. package/tests/unit/test-relay-permanence.js +376 -0
  129. package/tests/unit/test-second-node.js +95 -0
  130. package/tests/unit/test-simple-write.js +89 -0
  131. package/vite.config.js +49 -0
  132. package/vitest.config.js +20 -0
  133. package/FEDERATION.md +0 -213
  134. package/compute.js +0 -298
  135. package/content.js +0 -980
  136. package/federation.js +0 -1234
  137. package/global.js +0 -736
  138. package/hexlib.js +0 -335
  139. package/hologram.js +0 -183
  140. package/holosphere-bundle.esm.js +0 -33256
  141. package/holosphere-bundle.js +0 -33287
  142. package/holosphere-bundle.min.js +0 -39
  143. package/holosphere.d.ts +0 -601
  144. package/holosphere.js +0 -719
  145. package/node.js +0 -246
  146. package/schema.js +0 -139
  147. package/utils.js +0 -302
@@ -0,0 +1,458 @@
1
+ import { describe, it, expect, beforeEach, vi } from 'vitest';
2
+ import { H3AI } from '../../../src/ai/h3-ai.js';
3
+
4
+ // Mock h3-js
5
+ vi.mock('h3-js', () => ({
6
+ isValidCell: vi.fn().mockReturnValue(true),
7
+ getResolution: vi.fn().mockReturnValue(7),
8
+ cellToParent: vi.fn().mockReturnValue('parent_cell'),
9
+ cellToChildren: vi.fn().mockReturnValue(['child1', 'child2', 'child3']),
10
+ gridDisk: vi.fn().mockReturnValue(['center', 'neighbor1', 'neighbor2']),
11
+ cellToLatLng: vi.fn().mockReturnValue([41.9028, 12.4964]),
12
+ cellToBoundary: vi.fn().mockReturnValue([[0, 0], [0, 1], [1, 1], [1, 0]]),
13
+ cellArea: vi.fn().mockReturnValue(5.16)
14
+ }));
15
+
16
+ describe('Unit: H3AI', () => {
17
+ let h3ai;
18
+ let mockLLM;
19
+ let mockHolosphere;
20
+
21
+ beforeEach(async () => {
22
+ vi.clearAllMocks();
23
+
24
+ mockLLM = {
25
+ getJSON: vi.fn()
26
+ };
27
+
28
+ mockHolosphere = {
29
+ getAll: vi.fn().mockResolvedValue([])
30
+ };
31
+
32
+ h3ai = new H3AI(mockLLM, mockHolosphere);
33
+ });
34
+
35
+ describe('Constructor', () => {
36
+ it('should initialize with LLM service', () => {
37
+ const h = new H3AI(mockLLM);
38
+ expect(h.llm).toBe(mockLLM);
39
+ expect(h.holosphere).toBeNull();
40
+ });
41
+
42
+ it('should accept optional HoloSphere instance', () => {
43
+ expect(h3ai.holosphere).toBe(mockHolosphere);
44
+ });
45
+ });
46
+
47
+ describe('setHoloSphere', () => {
48
+ it('should set HoloSphere instance', () => {
49
+ const h = new H3AI(mockLLM);
50
+ h.setHoloSphere(mockHolosphere);
51
+ expect(h.holosphere).toBe(mockHolosphere);
52
+ });
53
+ });
54
+
55
+ describe('suggestResolution', () => {
56
+ it('should suggest optimal H3 resolution', async () => {
57
+ mockLLM.getJSON.mockResolvedValue({
58
+ recommendedResolution: 8,
59
+ reasoning: 'Project scope covers neighborhood area',
60
+ alternativeResolutions: [
61
+ { resolution: 7, useCase: 'Broader community' },
62
+ { resolution: 9, useCase: 'More focused blocks' }
63
+ ],
64
+ geographicScope: '~1 km area',
65
+ scaleSuggestions: {
66
+ expansion: { resolution: 6, reason: 'Regional collaboration' },
67
+ contraction: { resolution: 9, reason: 'Local focus' }
68
+ }
69
+ });
70
+
71
+ const item = { title: 'Neighborhood Garden Project', scope: 'local' };
72
+ const result = await h3ai.suggestResolution(item);
73
+
74
+ expect(result.recommendedResolution).toBe(8);
75
+ expect(result.reasoning).toBeDefined();
76
+ expect(result.alternativeResolutions).toHaveLength(2);
77
+ });
78
+
79
+ it('should include current resolution in prompt when provided', async () => {
80
+ mockLLM.getJSON.mockResolvedValue({ recommendedResolution: 7 });
81
+
82
+ await h3ai.suggestResolution({ title: 'Project' }, { currentResolution: 5 });
83
+
84
+ const call = mockLLM.getJSON.mock.calls[0];
85
+ expect(call[0]).toContain('Current resolution: 5');
86
+ });
87
+
88
+ it('should include context when provided', async () => {
89
+ mockLLM.getJSON.mockResolvedValue({ recommendedResolution: 7 });
90
+
91
+ await h3ai.suggestResolution({ title: 'Project' }, { context: { similarProjects: 10 } });
92
+
93
+ const call = mockLLM.getJSON.mock.calls[0];
94
+ expect(call[0]).toContain('similarProjects');
95
+ });
96
+ });
97
+
98
+ describe('analyzeDistribution', () => {
99
+ it('should analyze data distribution across H3 cells', async () => {
100
+ mockHolosphere.getAll
101
+ .mockResolvedValueOnce([{ id: 1 }, { id: 2 }]) // parent
102
+ .mockResolvedValueOnce([{ id: 3 }]) // child1
103
+ .mockResolvedValueOnce([]) // child2
104
+ .mockResolvedValueOnce([{ id: 4 }]); // child3
105
+
106
+ mockLLM.getJSON.mockResolvedValue({
107
+ distribution: { pattern: 'clustered', density: 'medium', coverage: 0.6 },
108
+ hotspots: [{ holonId: 'child1', reason: 'Most activity' }],
109
+ gaps: [{ description: 'Child2 area', suggestedAction: 'Outreach' }],
110
+ clusters: [{ theme: 'Active areas', holons: ['child1', 'child3'] }],
111
+ recommendations: ['Expand to child2'],
112
+ summary: 'Moderate distribution with gaps'
113
+ });
114
+
115
+ const result = await h3ai.analyzeDistribution('holon1', 'projects');
116
+
117
+ expect(result.holonId).toBe('holon1');
118
+ expect(result.parentDataCount).toBe(2);
119
+ expect(result.analysis.distribution.pattern).toBe('clustered');
120
+ });
121
+
122
+ it('should throw error if HoloSphere not available', async () => {
123
+ const h = new H3AI(mockLLM);
124
+
125
+ await expect(h.analyzeDistribution('holon', 'lens'))
126
+ .rejects.toThrow('HoloSphere instance required');
127
+ });
128
+
129
+ it('should skip children analysis when includeChildren is false', async () => {
130
+ mockHolosphere.getAll.mockResolvedValue([{ id: 1 }]);
131
+ mockLLM.getJSON.mockResolvedValue({ distribution: {} });
132
+
133
+ await h3ai.analyzeDistribution('holon1', 'lens', { includeChildren: false });
134
+
135
+ expect(mockHolosphere.getAll).toHaveBeenCalledTimes(1);
136
+ });
137
+ });
138
+
139
+ describe('findNeighborRelevance', () => {
140
+ it('should find relevant data from neighboring cells', async () => {
141
+ mockHolosphere.getAll
142
+ .mockResolvedValueOnce([{ id: 1, title: 'Center item' }])
143
+ .mockResolvedValueOnce([{ id: 2, title: 'Neighbor item' }])
144
+ .mockResolvedValueOnce([]);
145
+
146
+ mockLLM.getJSON.mockResolvedValue({
147
+ relevantNeighbors: [
148
+ {
149
+ holonId: 'neighbor1',
150
+ relevanceScore: 0.8,
151
+ relevantItems: [{ id: 2, reason: 'Similar topic' }],
152
+ collaborationPotential: 'High potential for collaboration'
153
+ }
154
+ ],
155
+ crossBoundaryOpportunities: [
156
+ { description: 'Joint project', involvedHolons: ['center', 'neighbor1'] }
157
+ ],
158
+ geographicPatterns: ['Activity cluster'],
159
+ summary: 'Strong neighbor relevance'
160
+ });
161
+
162
+ const result = await h3ai.findNeighborRelevance('center', 'projects');
163
+
164
+ expect(result.holonId).toBe('center');
165
+ expect(result.analysis.relevantNeighbors).toHaveLength(1);
166
+ });
167
+
168
+ it('should throw error if HoloSphere not available', async () => {
169
+ const h = new H3AI(mockLLM);
170
+
171
+ await expect(h.findNeighborRelevance('holon', 'lens'))
172
+ .rejects.toThrow('HoloSphere instance required');
173
+ });
174
+
175
+ it('should filter by relevance threshold', async () => {
176
+ mockHolosphere.getAll.mockResolvedValue([{ id: 1 }]);
177
+
178
+ mockLLM.getJSON.mockResolvedValue({
179
+ relevantNeighbors: [
180
+ { holonId: 'n1', relevanceScore: 0.9 },
181
+ { holonId: 'n2', relevanceScore: 0.3 }
182
+ ]
183
+ });
184
+
185
+ const result = await h3ai.findNeighborRelevance('center', 'lens', { relevanceThreshold: 0.5 });
186
+
187
+ expect(result.analysis.relevantNeighbors).toHaveLength(1);
188
+ });
189
+
190
+ it('should return empty if no neighbor data', async () => {
191
+ mockHolosphere.getAll.mockResolvedValue([{ id: 1 }]);
192
+
193
+ // Center has data, neighbors don't
194
+ mockHolosphere.getAll
195
+ .mockResolvedValueOnce([{ id: 1 }])
196
+ .mockResolvedValue([]);
197
+
198
+ const result = await h3ai.findNeighborRelevance('center', 'lens');
199
+
200
+ expect(result.neighbors).toEqual([]);
201
+ });
202
+ });
203
+
204
+ describe('suggestGeographicScope', () => {
205
+ it('should suggest expansion or contraction', async () => {
206
+ mockHolosphere.getAll
207
+ .mockResolvedValueOnce([{ id: 1 }]) // parent
208
+ .mockResolvedValueOnce([{ id: 2 }]); // child
209
+
210
+ mockLLM.getJSON.mockResolvedValue({
211
+ currentScopeAssessment: { appropriate: false, reasoning: 'Too narrow' },
212
+ expansionRecommendation: {
213
+ recommended: true,
214
+ targetResolution: 6,
215
+ reasoning: 'Project has regional impact',
216
+ benefits: ['More visibility'],
217
+ risks: ['Diluted focus']
218
+ },
219
+ contractionRecommendation: {
220
+ recommended: false,
221
+ targetResolution: 8,
222
+ reasoning: 'Already focused',
223
+ benefits: [],
224
+ risks: []
225
+ },
226
+ optimalAction: 'expand',
227
+ summary: 'Consider expanding scope'
228
+ });
229
+
230
+ const result = await h3ai.suggestGeographicScope(
231
+ { title: 'Regional Project' },
232
+ 'current_cell',
233
+ 'projects'
234
+ );
235
+
236
+ expect(result.optimalAction).toBe('expand');
237
+ expect(result.expansionRecommendation.recommended).toBe(true);
238
+ });
239
+
240
+ it('should throw error if HoloSphere not available', async () => {
241
+ const h = new H3AI(mockLLM);
242
+
243
+ await expect(h.suggestGeographicScope({}, 'cell', 'lens'))
244
+ .rejects.toThrow('HoloSphere instance required');
245
+ });
246
+ });
247
+
248
+ describe('analyzeCoverage', () => {
249
+ it('should analyze coverage gaps in region', async () => {
250
+ mockHolosphere.getAll
251
+ .mockResolvedValueOnce([{ id: 1 }]) // child1
252
+ .mockResolvedValueOnce([]) // child2
253
+ .mockResolvedValueOnce([{ id: 2 }]); // child3
254
+
255
+ mockLLM.getJSON.mockResolvedValue({
256
+ coverageScore: 0.67,
257
+ coverageQuality: 'moderate',
258
+ patterns: { type: 'scattered', description: 'Patchy coverage' },
259
+ gaps: { count: 1, significance: 'medium', likelyReasons: ['New area'] },
260
+ recommendations: [{ action: 'Outreach', priority: 'high' }],
261
+ summary: 'Moderate coverage with gaps'
262
+ });
263
+
264
+ const result = await h3ai.analyzeCoverage('parent_cell', 'projects');
265
+
266
+ expect(result.coverage.ratio).toBeCloseTo(0.67, 1);
267
+ expect(result.analysis.coverageQuality).toBe('moderate');
268
+ });
269
+
270
+ it('should throw error if HoloSphere not available', async () => {
271
+ const h = new H3AI(mockLLM);
272
+
273
+ await expect(h.analyzeCoverage('cell', 'lens'))
274
+ .rejects.toThrow('HoloSphere instance required');
275
+ });
276
+ });
277
+
278
+ describe('crossResolutionInsights', () => {
279
+ it('should find patterns across H3 resolutions', async () => {
280
+ mockHolosphere.getAll.mockResolvedValue([{ id: 1 }]);
281
+
282
+ mockLLM.getJSON.mockResolvedValue({
283
+ scalePatterns: [
284
+ { pattern: 'Local clustering', visibleAt: [8, 9], significance: 'Community focus' }
285
+ ],
286
+ themeEvolution: {
287
+ localThemes: ['Gardening'],
288
+ regionalThemes: ['Food security'],
289
+ consistentThemes: ['Sustainability']
290
+ },
291
+ optimalResolutions: {
292
+ forCollaboration: 7,
293
+ forResources: 8,
294
+ forCommunity: 9,
295
+ reasoning: 'Different scales for different needs'
296
+ },
297
+ insights: ['Multi-scale approach recommended'],
298
+ recommendations: ['Focus on resolution 8 for main activities'],
299
+ summary: 'Rich multi-scale patterns'
300
+ });
301
+
302
+ const result = await h3ai.crossResolutionInsights('cell', 'projects');
303
+
304
+ expect(result.scalePatterns).toBeDefined();
305
+ expect(result.optimalResolutions).toBeDefined();
306
+ });
307
+
308
+ it('should throw error if HoloSphere not available', async () => {
309
+ const h = new H3AI(mockLLM);
310
+
311
+ await expect(h.crossResolutionInsights('cell', 'lens'))
312
+ .rejects.toThrow('HoloSphere instance required');
313
+ });
314
+ });
315
+
316
+ describe('suggestMigration', () => {
317
+ it('should suggest item migration between cells', async () => {
318
+ mockHolosphere.getAll
319
+ .mockResolvedValueOnce([{ id: 'nearby', title: 'Similar project' }])
320
+ .mockResolvedValueOnce([{ id: 'parent', title: 'Parent data' }]);
321
+
322
+ mockLLM.getJSON.mockResolvedValue({
323
+ currentFit: { score: 0.5, reasoning: 'Marginal fit' },
324
+ migrationRecommended: true,
325
+ suggestedDestinations: [
326
+ {
327
+ holonId: 'neighbor1',
328
+ fitScore: 0.9,
329
+ reasoning: 'Better thematic match',
330
+ benefits: ['More visibility'],
331
+ drawbacks: ['Smaller community']
332
+ }
333
+ ],
334
+ stayReasons: ['Established presence'],
335
+ moveReasons: ['Better fit elsewhere'],
336
+ recommendation: 'move',
337
+ summary: 'Consider moving to neighbor1'
338
+ });
339
+
340
+ const result = await h3ai.suggestMigration(
341
+ { id: 'item1', title: 'Project' },
342
+ 'current_cell',
343
+ 'projects'
344
+ );
345
+
346
+ expect(result.migrationRecommended).toBe(true);
347
+ expect(result.recommendation).toBe('move');
348
+ });
349
+
350
+ it('should throw error if HoloSphere not available', async () => {
351
+ const h = new H3AI(mockLLM);
352
+
353
+ await expect(h.suggestMigration({}, 'cell', 'lens'))
354
+ .rejects.toThrow('HoloSphere instance required');
355
+ });
356
+ });
357
+
358
+ describe('generateGeographicReport', () => {
359
+ it('should generate comprehensive geographic report', async () => {
360
+ mockHolosphere.getAll
361
+ .mockResolvedValueOnce([{ id: 1 }]) // projects
362
+ .mockResolvedValueOnce([{ id: 2 }]) // quests
363
+ .mockResolvedValueOnce([]); // events
364
+
365
+ mockLLM.getJSON.mockResolvedValue({
366
+ title: 'Region Report',
367
+ executiveSummary: 'Active region with 2 items',
368
+ activityOverview: { totalItems: 2, categorySummaries: { projects: '1 project' } },
369
+ highlights: [{ title: 'Growth', description: 'Increasing activity', category: 'projects' }],
370
+ geographicPatterns: ['Central clustering'],
371
+ strengths: ['Active community'],
372
+ challenges: ['Limited resources'],
373
+ opportunities: ['Expansion potential'],
374
+ recommendations: [{ priority: 'high', action: 'Recruit', rationale: 'Grow team' }],
375
+ metrics: { activityLevel: 'medium', diversity: 'low', growth: 'growing' }
376
+ });
377
+
378
+ const result = await h3ai.generateGeographicReport('cell');
379
+
380
+ expect(result.holonId).toBe('cell');
381
+ expect(result.geoContext).toBeDefined();
382
+ expect(result.report.title).toBe('Region Report');
383
+ });
384
+
385
+ it('should throw error if HoloSphere not available', async () => {
386
+ const h = new H3AI(mockLLM);
387
+
388
+ await expect(h.generateGeographicReport('cell'))
389
+ .rejects.toThrow('HoloSphere instance required');
390
+ });
391
+ });
392
+
393
+ describe('findGeographicClusters', () => {
394
+ it('should find thematic clusters in spatial data', async () => {
395
+ mockHolosphere.getAll
396
+ .mockResolvedValueOnce([{ id: 1, theme: 'A' }])
397
+ .mockResolvedValueOnce([{ id: 2, theme: 'A' }])
398
+ .mockResolvedValueOnce([{ id: 3, theme: 'B' }]);
399
+
400
+ mockLLM.getJSON.mockResolvedValue({
401
+ clusters: [
402
+ { name: 'Theme A cluster', theme: 'A', cells: ['child1', 'child2'], strength: 0.9 }
403
+ ],
404
+ isolatedCells: [{ cellId: 'child3', uniqueAspect: 'Different theme' }],
405
+ interClusterConnections: [],
406
+ spatialPatterns: ['Eastern concentration'],
407
+ recommendations: ['Connect isolated cell']
408
+ });
409
+
410
+ const result = await h3ai.findGeographicClusters('parent', 'lens');
411
+
412
+ expect(result.analysis.clusters).toHaveLength(1);
413
+ expect(result.analysis.isolatedCells).toHaveLength(1);
414
+ });
415
+
416
+ it('should throw error if HoloSphere not available', async () => {
417
+ const h = new H3AI(mockLLM);
418
+
419
+ await expect(h.findGeographicClusters('cell', 'lens'))
420
+ .rejects.toThrow('HoloSphere instance required');
421
+ });
422
+
423
+ it('should return message if not enough data', async () => {
424
+ // Only one child has data (cellToChildren returns 3, we need less than 2 with data)
425
+ mockHolosphere.getAll
426
+ .mockResolvedValueOnce([{ id: 1 }]) // child1 has data
427
+ .mockResolvedValueOnce([]) // child2 empty
428
+ .mockResolvedValueOnce([]); // child3 empty
429
+
430
+ const result = await h3ai.findGeographicClusters('parent', 'lens');
431
+
432
+ expect(result.message).toBe('Not enough data for clustering');
433
+ });
434
+ });
435
+
436
+ describe('analyzeGeographicImpact', () => {
437
+ it('should analyze geographic impact of item', async () => {
438
+ mockLLM.getJSON.mockResolvedValue({
439
+ directImpact: { areaKm2: 5.16, description: 'Immediate neighborhood', affectedPopulation: '~500' },
440
+ indirectImpact: { estimatedReach: 20, mechanisms: ['Word of mouth'], neighboringAreas: ['Area B'] },
441
+ potentialReach: { ifExpanded: { maxAreaKm2: 50, optimalResolution: 5, limitingFactors: ['Resources'] } },
442
+ geographicFactors: { enablers: ['Good transport'], barriers: ['Physical barriers'] },
443
+ impactScore: { local: 0.8, regional: 0.4, network: 0.6 },
444
+ recommendations: ['Expand to adjacent areas'],
445
+ summary: 'Strong local impact with expansion potential'
446
+ });
447
+
448
+ const result = await h3ai.analyzeGeographicImpact(
449
+ { id: 'item1', title: 'Community Project' },
450
+ 'cell',
451
+ 'projects'
452
+ );
453
+
454
+ expect(result.directImpact.areaKm2).toBe(5.16);
455
+ expect(result.impactScore.local).toBe(0.8);
456
+ });
457
+ });
458
+ });