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