holosphere 2.0.0-alpha0 → 2.0.0-alpha2

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 (88) hide show
  1. package/LICENSE +162 -38
  2. package/dist/cjs/holosphere.cjs +2 -0
  3. package/dist/cjs/holosphere.cjs.map +1 -0
  4. package/dist/esm/holosphere.js +56 -0
  5. package/dist/esm/holosphere.js.map +1 -0
  6. package/dist/index-CDfIuXew.js +15974 -0
  7. package/dist/index-CDfIuXew.js.map +1 -0
  8. package/dist/index-ifOgtDvd.cjs +3 -0
  9. package/dist/index-ifOgtDvd.cjs.map +1 -0
  10. package/dist/indexeddb-storage-CMW4qRQS.js +96 -0
  11. package/dist/indexeddb-storage-CMW4qRQS.js.map +1 -0
  12. package/dist/indexeddb-storage-DLZOgetM.cjs +2 -0
  13. package/dist/indexeddb-storage-DLZOgetM.cjs.map +1 -0
  14. package/dist/memory-storage-DQzcAZlf.js +47 -0
  15. package/dist/memory-storage-DQzcAZlf.js.map +1 -0
  16. package/dist/memory-storage-DmePEP2q.cjs +2 -0
  17. package/dist/memory-storage-DmePEP2q.cjs.map +1 -0
  18. package/dist/secp256k1-CP0ZkpAx.cjs +13 -0
  19. package/dist/secp256k1-CP0ZkpAx.cjs.map +1 -0
  20. package/dist/secp256k1-vOXp40Fx.js +2281 -0
  21. package/dist/secp256k1-vOXp40Fx.js.map +1 -0
  22. package/docs/FOSDEM_PROPOSAL.md +388 -0
  23. package/docs/LOCALFIRST.md +266 -0
  24. package/docs/contracts/api-interface.md +793 -0
  25. package/docs/data-model.md +476 -0
  26. package/docs/gun-async-usage.md +338 -0
  27. package/docs/plan.md +349 -0
  28. package/docs/quickstart.md +674 -0
  29. package/docs/research.md +362 -0
  30. package/docs/spec.md +244 -0
  31. package/docs/storage-backends.md +326 -0
  32. package/docs/tasks.md +947 -0
  33. package/package.json +1 -1
  34. package/tests/unit/ai/aggregation.test.js +0 -295
  35. package/tests/unit/ai/breakdown.test.js +0 -446
  36. package/tests/unit/ai/classifier.test.js +0 -294
  37. package/tests/unit/ai/council.test.js +0 -262
  38. package/tests/unit/ai/embeddings.test.js +0 -384
  39. package/tests/unit/ai/federation-ai.test.js +0 -344
  40. package/tests/unit/ai/h3-ai.test.js +0 -458
  41. package/tests/unit/ai/index.test.js +0 -304
  42. package/tests/unit/ai/json-ops.test.js +0 -307
  43. package/tests/unit/ai/llm-service.test.js +0 -390
  44. package/tests/unit/ai/nl-query.test.js +0 -383
  45. package/tests/unit/ai/relationships.test.js +0 -311
  46. package/tests/unit/ai/schema-extractor.test.js +0 -384
  47. package/tests/unit/ai/spatial.test.js +0 -279
  48. package/tests/unit/ai/tts.test.js +0 -279
  49. package/tests/unit/content.test.js +0 -332
  50. package/tests/unit/contract/core.test.js +0 -88
  51. package/tests/unit/contract/crypto.test.js +0 -198
  52. package/tests/unit/contract/data.test.js +0 -223
  53. package/tests/unit/contract/federation.test.js +0 -181
  54. package/tests/unit/contract/hierarchical.test.js +0 -113
  55. package/tests/unit/contract/schema.test.js +0 -114
  56. package/tests/unit/contract/social.test.js +0 -217
  57. package/tests/unit/contract/spatial.test.js +0 -110
  58. package/tests/unit/contract/subscriptions.test.js +0 -128
  59. package/tests/unit/contract/utils.test.js +0 -159
  60. package/tests/unit/core.test.js +0 -152
  61. package/tests/unit/crypto.test.js +0 -328
  62. package/tests/unit/federation.test.js +0 -234
  63. package/tests/unit/gun-async.test.js +0 -252
  64. package/tests/unit/hierarchical.test.js +0 -399
  65. package/tests/unit/integration/scenario-01-geographic-storage.test.js +0 -74
  66. package/tests/unit/integration/scenario-02-federation.test.js +0 -76
  67. package/tests/unit/integration/scenario-03-subscriptions.test.js +0 -102
  68. package/tests/unit/integration/scenario-04-validation.test.js +0 -129
  69. package/tests/unit/integration/scenario-05-hierarchy.test.js +0 -125
  70. package/tests/unit/integration/scenario-06-social.test.js +0 -135
  71. package/tests/unit/integration/scenario-07-persistence.test.js +0 -130
  72. package/tests/unit/integration/scenario-08-authorization.test.js +0 -161
  73. package/tests/unit/integration/scenario-09-cross-dimensional.test.js +0 -139
  74. package/tests/unit/integration/scenario-10-cross-holosphere-capabilities.test.js +0 -357
  75. package/tests/unit/integration/scenario-11-cross-holosphere-federation.test.js +0 -410
  76. package/tests/unit/integration/scenario-12-capability-federated-read.test.js +0 -719
  77. package/tests/unit/performance/benchmark.test.js +0 -85
  78. package/tests/unit/schema.test.js +0 -213
  79. package/tests/unit/spatial.test.js +0 -158
  80. package/tests/unit/storage.test.js +0 -195
  81. package/tests/unit/subscriptions.test.js +0 -328
  82. package/tests/unit/test-data-permanence-debug.js +0 -197
  83. package/tests/unit/test-data-permanence.js +0 -340
  84. package/tests/unit/test-key-persistence-fixed.js +0 -148
  85. package/tests/unit/test-key-persistence.js +0 -172
  86. package/tests/unit/test-relay-permanence.js +0 -376
  87. package/tests/unit/test-second-node.js +0 -95
  88. package/tests/unit/test-simple-write.js +0 -89
@@ -1,458 +0,0 @@
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
- });