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.
- package/dist/cjs/holosphere.cjs +2 -0
- package/dist/cjs/holosphere.cjs.map +1 -0
- package/dist/esm/holosphere.js +56 -0
- package/dist/esm/holosphere.js.map +1 -0
- package/dist/index-CDfIuXew.js +15974 -0
- package/dist/index-CDfIuXew.js.map +1 -0
- package/dist/index-ifOgtDvd.cjs +3 -0
- package/dist/index-ifOgtDvd.cjs.map +1 -0
- package/dist/indexeddb-storage-CMW4qRQS.js +96 -0
- package/dist/indexeddb-storage-CMW4qRQS.js.map +1 -0
- package/dist/indexeddb-storage-DLZOgetM.cjs +2 -0
- package/dist/indexeddb-storage-DLZOgetM.cjs.map +1 -0
- package/dist/memory-storage-DQzcAZlf.js +47 -0
- package/dist/memory-storage-DQzcAZlf.js.map +1 -0
- package/dist/memory-storage-DmePEP2q.cjs +2 -0
- package/dist/memory-storage-DmePEP2q.cjs.map +1 -0
- package/dist/secp256k1-CP0ZkpAx.cjs +13 -0
- package/dist/secp256k1-CP0ZkpAx.cjs.map +1 -0
- package/dist/secp256k1-vOXp40Fx.js +2281 -0
- package/dist/secp256k1-vOXp40Fx.js.map +1 -0
- package/docs/FOSDEM_PROPOSAL.md +388 -0
- package/docs/LOCALFIRST.md +266 -0
- package/docs/contracts/api-interface.md +793 -0
- package/docs/data-model.md +476 -0
- package/docs/gun-async-usage.md +338 -0
- package/docs/plan.md +349 -0
- package/docs/quickstart.md +674 -0
- package/docs/research.md +362 -0
- package/docs/spec.md +244 -0
- package/docs/storage-backends.md +326 -0
- package/docs/tasks.md +947 -0
- package/package.json +1 -1
- package/tests/unit/ai/aggregation.test.js +0 -295
- package/tests/unit/ai/breakdown.test.js +0 -446
- package/tests/unit/ai/classifier.test.js +0 -294
- package/tests/unit/ai/council.test.js +0 -262
- package/tests/unit/ai/embeddings.test.js +0 -384
- package/tests/unit/ai/federation-ai.test.js +0 -344
- package/tests/unit/ai/h3-ai.test.js +0 -458
- package/tests/unit/ai/index.test.js +0 -304
- package/tests/unit/ai/json-ops.test.js +0 -307
- package/tests/unit/ai/llm-service.test.js +0 -390
- package/tests/unit/ai/nl-query.test.js +0 -383
- package/tests/unit/ai/relationships.test.js +0 -311
- package/tests/unit/ai/schema-extractor.test.js +0 -384
- package/tests/unit/ai/spatial.test.js +0 -279
- package/tests/unit/ai/tts.test.js +0 -279
- package/tests/unit/content.test.js +0 -332
- package/tests/unit/contract/core.test.js +0 -88
- package/tests/unit/contract/crypto.test.js +0 -198
- package/tests/unit/contract/data.test.js +0 -223
- package/tests/unit/contract/federation.test.js +0 -181
- package/tests/unit/contract/hierarchical.test.js +0 -113
- package/tests/unit/contract/schema.test.js +0 -114
- package/tests/unit/contract/social.test.js +0 -217
- package/tests/unit/contract/spatial.test.js +0 -110
- package/tests/unit/contract/subscriptions.test.js +0 -128
- package/tests/unit/contract/utils.test.js +0 -159
- package/tests/unit/core.test.js +0 -152
- package/tests/unit/crypto.test.js +0 -328
- package/tests/unit/federation.test.js +0 -234
- package/tests/unit/gun-async.test.js +0 -252
- package/tests/unit/hierarchical.test.js +0 -399
- package/tests/unit/integration/scenario-01-geographic-storage.test.js +0 -74
- package/tests/unit/integration/scenario-02-federation.test.js +0 -76
- package/tests/unit/integration/scenario-03-subscriptions.test.js +0 -102
- package/tests/unit/integration/scenario-04-validation.test.js +0 -129
- package/tests/unit/integration/scenario-05-hierarchy.test.js +0 -125
- package/tests/unit/integration/scenario-06-social.test.js +0 -135
- package/tests/unit/integration/scenario-07-persistence.test.js +0 -130
- package/tests/unit/integration/scenario-08-authorization.test.js +0 -161
- package/tests/unit/integration/scenario-09-cross-dimensional.test.js +0 -139
- package/tests/unit/integration/scenario-10-cross-holosphere-capabilities.test.js +0 -357
- package/tests/unit/integration/scenario-11-cross-holosphere-federation.test.js +0 -410
- package/tests/unit/integration/scenario-12-capability-federated-read.test.js +0 -719
- package/tests/unit/performance/benchmark.test.js +0 -85
- package/tests/unit/schema.test.js +0 -213
- package/tests/unit/spatial.test.js +0 -158
- package/tests/unit/storage.test.js +0 -195
- package/tests/unit/subscriptions.test.js +0 -328
- package/tests/unit/test-data-permanence-debug.js +0 -197
- package/tests/unit/test-data-permanence.js +0 -340
- package/tests/unit/test-key-persistence-fixed.js +0 -148
- package/tests/unit/test-key-persistence.js +0 -172
- package/tests/unit/test-relay-permanence.js +0 -376
- package/tests/unit/test-second-node.js +0 -95
- 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
|
-
});
|