@lucern/graph-primitives 0.1.0-alpha.4 → 0.3.0-alpha.1

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 (78) hide show
  1. package/dist/beliefDecay.js +229 -1115
  2. package/dist/beliefDecay.js.map +1 -1
  3. package/dist/beliefEvidenceLinks.js +53 -834
  4. package/dist/beliefEvidenceLinks.js.map +1 -1
  5. package/dist/confidencePropagationDispatch.d.ts +3 -3
  6. package/dist/confidencePropagationDispatch.js +30 -308
  7. package/dist/confidencePropagationDispatch.js.map +1 -1
  8. package/dist/contradictions.js +5 -797
  9. package/dist/contradictions.js.map +1 -1
  10. package/dist/edges/contradicts.js +1 -122
  11. package/dist/edges/contradicts.js.map +1 -1
  12. package/dist/edges/dependsOn.js +14 -172
  13. package/dist/edges/dependsOn.js.map +1 -1
  14. package/dist/edges/elaborates.js +1 -49
  15. package/dist/edges/elaborates.js.map +1 -1
  16. package/dist/edges/index.js +14 -277
  17. package/dist/edges/index.js.map +1 -1
  18. package/dist/edges/informs.js +1 -62
  19. package/dist/edges/informs.js.map +1 -1
  20. package/dist/edges/propagationTypes.d.ts +2 -2
  21. package/dist/edges/propagationTypes.js.map +1 -1
  22. package/dist/edges/refutes.js +1 -62
  23. package/dist/edges/refutes.js.map +1 -1
  24. package/dist/edges/supports.js +1 -122
  25. package/dist/edges/supports.js.map +1 -1
  26. package/dist/edges/utils.d.ts +6 -6
  27. package/dist/edges/utils.js +1 -130
  28. package/dist/edges/utils.js.map +1 -1
  29. package/dist/entityBridge.js +2 -17
  30. package/dist/entityBridge.js.map +1 -1
  31. package/dist/entityLifecycle.js +62 -848
  32. package/dist/entityLifecycle.js.map +1 -1
  33. package/dist/epistemicAnswers.js +27 -838
  34. package/dist/epistemicAnswers.js.map +1 -1
  35. package/dist/epistemicBeliefs.js +186 -2214
  36. package/dist/epistemicBeliefs.js.map +1 -1
  37. package/dist/epistemicContractHelpers.js +1 -318
  38. package/dist/epistemicContractHelpers.js.map +1 -1
  39. package/dist/epistemicContracts.js +163 -2467
  40. package/dist/epistemicContracts.js.map +1 -1
  41. package/dist/epistemicEdges.js +60 -863
  42. package/dist/epistemicEdges.js.map +1 -1
  43. package/dist/epistemicEvidence.js +116 -1647
  44. package/dist/epistemicEvidence.js.map +1 -1
  45. package/dist/epistemicHelpers.js +3 -2
  46. package/dist/epistemicHelpers.js.map +1 -1
  47. package/dist/epistemicLinking.js +2 -785
  48. package/dist/epistemicLinking.js.map +1 -1
  49. package/dist/epistemicNodes.js +34 -1427
  50. package/dist/epistemicNodes.js.map +1 -1
  51. package/dist/epistemicQuestions.js +88 -1637
  52. package/dist/epistemicQuestions.js.map +1 -1
  53. package/dist/epistemicSources.js +28 -1421
  54. package/dist/epistemicSources.js.map +1 -1
  55. package/dist/evaluators/index.js +163 -2467
  56. package/dist/evaluators/index.js.map +1 -1
  57. package/dist/index.js +486 -3649
  58. package/dist/index.js.map +1 -1
  59. package/dist/ontology-matching.js +1 -344
  60. package/dist/ontology-matching.js.map +1 -1
  61. package/dist/ontologyApproval.js +1 -13
  62. package/dist/ontologyApproval.js.map +1 -1
  63. package/dist/ontologyDefinitions.js +2 -17
  64. package/dist/ontologyDefinitions.js.map +1 -1
  65. package/dist/ontologyRegistry.js +2 -17
  66. package/dist/ontologyRegistry.js.map +1 -1
  67. package/dist/projectionReconciliation.js +2 -17
  68. package/dist/projectionReconciliation.js.map +1 -1
  69. package/dist/questionEvidenceLinks.js +242 -837
  70. package/dist/questionEvidenceLinks.js.map +1 -1
  71. package/dist/text-matching.js +1 -244
  72. package/dist/text-matching.js.map +1 -1
  73. package/dist/workflowBridge.d.ts +27 -0
  74. package/dist/workflowBridge.js +303 -0
  75. package/dist/workflowBridge.js.map +1 -0
  76. package/dist/workspaceIsolation.js +8 -609
  77. package/dist/workspaceIsolation.js.map +1 -1
  78. package/package.json +6 -6
@@ -1,814 +1,12 @@
1
1
  import { v } from 'convex/values';
2
- import { componentsGeneric, defineTable, queryGeneric, mutationGeneric, anyApi, internalMutationGeneric, internalQueryGeneric } from 'convex/server';
2
+ import { checkScopeAccess, requireProjectAccess } from '@lucern/access-control/access';
3
+ import { permissiveReturn } from '@lucern/contracts/schema-helpers/validators';
4
+ import { componentsGeneric, queryGeneric, mutationGeneric, anyApi, internalMutationGeneric, internalQueryGeneric } from 'convex/server';
5
+ import { isNodeType, getLayerForNodeType } from '@lucern/contracts/schema-helpers/spine/tables/epistemicNodes';
3
6
 
4
7
  // src/epistemicNodes.ts
5
8
  var api = anyApi;
6
9
  componentsGeneric();
7
-
8
- // ../access-control/src/topicProjectOverlay.ts
9
- var LEGACY_SCOPE_FIELD = "graphScopeProjectId";
10
- function readNonEmptyString(value) {
11
- if (typeof value !== "string") {
12
- return;
13
- }
14
- const normalized = value.trim();
15
- return normalized.length > 0 ? normalized : void 0;
16
- }
17
- function readStringArray(value) {
18
- if (!Array.isArray(value)) {
19
- return [];
20
- }
21
- return value.map((entry) => readNonEmptyString(entry)).filter((entry) => Boolean(entry));
22
- }
23
- function readMetadata(topic) {
24
- return topic.metadata && typeof topic.metadata === "object" ? topic.metadata : {};
25
- }
26
- function readLegacyProjectId(value) {
27
- if (!value) {
28
- return;
29
- }
30
- return readNonEmptyString(value[LEGACY_SCOPE_FIELD]);
31
- }
32
- function coerceVisibility(value) {
33
- return value === "private" || value === "team" || value === "firm" || value === "external" || value === "public" ? value : void 0;
34
- }
35
- function coerceStatus(value) {
36
- return value === "active" || value === "archived" || value === "watching" ? value : void 0;
37
- }
38
- function mapProjectType(topic, metadata) {
39
- const explicit = readNonEmptyString(metadata.projectType);
40
- if (explicit) {
41
- return explicit;
42
- }
43
- if (topic.type === "theme") {
44
- return "thematic";
45
- }
46
- return readNonEmptyString(topic.type) || "general";
47
- }
48
- function isProjectLikeTopic(topic) {
49
- const metadata = readMetadata(topic);
50
- return topic.type === "theme" || topic.type === "thematic" || topic.type === "deal" || topic.type === "monitoring" || readLegacyProjectId(topic) !== void 0 || readNonEmptyString(metadata.projectType) !== void 0;
51
- }
52
- async function resolveTopicDoc(ctx, scopeId) {
53
- if (ctx?.db && typeof ctx.db.get === "function") {
54
- try {
55
- const directTopic = await ctx.db.get(scopeId);
56
- if (directTopic) {
57
- return directTopic;
58
- }
59
- } catch {
60
- }
61
- }
62
- if (typeof ctx.runQuery !== "function") {
63
- return null;
64
- }
65
- try {
66
- const topic = await ctx.runQuery(api.topics.get, {
67
- id: String(scopeId)
68
- });
69
- if (topic?.name !== void 0 && topic?.type !== void 0) {
70
- return topic;
71
- }
72
- } catch {
73
- }
74
- try {
75
- const topic = await ctx.runQuery(api.topics.getByLegacyScopeId, {
76
- projectId: String(scopeId)
77
- });
78
- if (topic?.name !== void 0 && topic?.type !== void 0) {
79
- return topic;
80
- }
81
- } catch {
82
- }
83
- return null;
84
- }
85
- function materializeTopicProjectOverlay(topic, idMode = "legacy") {
86
- const metadata = readMetadata(topic);
87
- const topicId = String(topic._id);
88
- const legacyProjectId = readLegacyProjectId(topic) || readLegacyProjectId(metadata) || readNonEmptyString(metadata.legacyProjectId);
89
- const storageProjectId = legacyProjectId || topicId;
90
- const outwardId = idMode === "topic" ? topicId : storageProjectId;
91
- const visibility = coerceVisibility(topic.visibility) || coerceVisibility(metadata.visibility) || "private";
92
- const status = coerceStatus(topic.status) || coerceStatus(metadata.status) || "active";
93
- const createdAt = typeof topic.createdAt === "number" ? topic.createdAt : typeof topic._creationTime === "number" ? topic._creationTime : 0;
94
- const updatedAt = typeof topic.updatedAt === "number" ? topic.updatedAt : typeof metadata.updatedAt === "number" ? metadata.updatedAt : createdAt;
95
- return {
96
- ...metadata,
97
- _id: outwardId,
98
- projectId: outwardId,
99
- topicId,
100
- storageProjectId,
101
- legacyProjectId,
102
- name: readNonEmptyString(topic.name) || "Untitled Theme",
103
- type: mapProjectType(topic, metadata),
104
- description: readNonEmptyString(topic.description),
105
- ownerId: readNonEmptyString(metadata.ownerId) || readNonEmptyString(topic.createdBy) || "system",
106
- sharedWith: readStringArray(metadata.sharedWith),
107
- visibility,
108
- tenantId: readNonEmptyString(topic.tenantId) || readNonEmptyString(metadata.tenantId),
109
- workspaceId: readNonEmptyString(topic.workspaceId) || readNonEmptyString(metadata.workspaceId),
110
- status,
111
- tags: readStringArray(metadata.tags),
112
- chatCount: typeof metadata.chatCount === "number" ? metadata.chatCount : 0,
113
- artifactCount: typeof metadata.artifactCount === "number" ? metadata.artifactCount : 0,
114
- lastActivityAt: typeof metadata.lastActivityAt === "number" ? metadata.lastActivityAt : updatedAt,
115
- _creationTime: typeof topic._creationTime === "number" ? topic._creationTime : createdAt,
116
- createdAt,
117
- updatedAt
118
- };
119
- }
120
- async function resolveTopicProjectOverlay(ctx, scopeId, options = {}) {
121
- const topic = await resolveTopicDoc(ctx, scopeId);
122
- if (!topic) {
123
- return null;
124
- }
125
- if (options.projectLikeOnly !== false && !isProjectLikeTopic(topic)) {
126
- return null;
127
- }
128
- return materializeTopicProjectOverlay(topic, options.idMode);
129
- }
130
- async function listTopicProjectOverlays(ctx, options = {}) {
131
- let allTopics = [];
132
- if (ctx?.db?.query && typeof ctx.db.query === "function") {
133
- try {
134
- allTopics = await ctx.db.query("topics").collect();
135
- } catch {
136
- allTopics = [];
137
- }
138
- }
139
- if (allTopics.length === 0 && typeof ctx.runQuery === "function") {
140
- allTopics = (await ctx.runQuery(api.topics.list, {}) ?? []) || [];
141
- }
142
- return allTopics.filter(
143
- (topic) => options.projectLikeOnly === false || isProjectLikeTopic(topic)
144
- ).map((topic) => materializeTopicProjectOverlay(topic, options.idMode));
145
- }
146
-
147
- // ../access-control/src/projectGrantsBridge.ts
148
- var PROJECT_GRANT_STATUSES = ["active", "revoked", "expired"];
149
- function normalizeString(value) {
150
- if (typeof value !== "string") {
151
- return;
152
- }
153
- const trimmed = value.trim();
154
- return trimmed.length > 0 ? trimmed : void 0;
155
- }
156
- async function resolveGrantScopeIds(ctx, args) {
157
- const topicId = normalizeString(args.topicId);
158
- const projectId = normalizeString(args.projectId);
159
- for (const scopeId of [topicId, projectId]) {
160
- if (!scopeId) {
161
- continue;
162
- }
163
- try {
164
- const overlay = await resolveTopicProjectOverlay(ctx, scopeId, {
165
- idMode: "legacy",
166
- projectLikeOnly: false
167
- });
168
- if (overlay) {
169
- return {
170
- topicId: normalizeString(overlay.topicId) ?? topicId,
171
- projectId: normalizeString(overlay.projectId) ?? projectId ?? scopeId
172
- };
173
- }
174
- } catch {
175
- }
176
- }
177
- return { topicId, projectId };
178
- }
179
- async function normalizeProjectGrantRow(ctx, row) {
180
- const scope = await resolveGrantScopeIds(ctx, {
181
- topicId: row.topicId,
182
- projectId: row.projectId
183
- });
184
- return {
185
- ...row,
186
- ...scope.topicId ? { topicId: scope.topicId } : {},
187
- ...scope.projectId ?? scope.topicId ? { projectId: scope.projectId ?? scope.topicId } : {}
188
- };
189
- }
190
- async function normalizeProjectGrantRows(ctx, rows) {
191
- return await Promise.all(rows.map((row) => normalizeProjectGrantRow(ctx, row)));
192
- }
193
- async function listProjectGrantsByPrincipal(ctx, principalId) {
194
- const rows = await Promise.all(
195
- PROJECT_GRANT_STATUSES.map(
196
- (status) => ctx.db.query("projectGrants").withIndex(
197
- "by_principal_status",
198
- (q) => q.eq("principalId", principalId).eq("status", status)
199
- ).collect()
200
- )
201
- );
202
- return await normalizeProjectGrantRows(ctx, rows.flat());
203
- }
204
- async function listProjectGrantsByGroup(ctx, groupId) {
205
- const rows = await Promise.all(
206
- PROJECT_GRANT_STATUSES.map(
207
- (status) => ctx.db.query("projectGrants").withIndex(
208
- "by_group_status",
209
- (q) => q.eq("groupId", groupId).eq("status", status)
210
- ).collect()
211
- )
212
- );
213
- return await normalizeProjectGrantRows(ctx, rows.flat());
214
- }
215
- function buildScopeMatchers(inputScopeId, resolved) {
216
- return new Set(
217
- [inputScopeId, resolved.topicId, resolved.projectId].map((value) => normalizeString(value)).filter((value) => Boolean(value))
218
- );
219
- }
220
- function matchesResolvedScope(row, scopeIds) {
221
- const rowTopicId = normalizeString(row.topicId);
222
- const rowProjectId = normalizeString(row.projectId);
223
- return rowTopicId !== void 0 && scopeIds.has(rowTopicId) || rowProjectId !== void 0 && scopeIds.has(rowProjectId);
224
- }
225
- async function bridgeListProjectGrantsByTopicAndPrincipal(ctx, topicId, principalId) {
226
- const resolved = await resolveGrantScopeIds(ctx, { topicId });
227
- const scopeIds = buildScopeMatchers(topicId, resolved);
228
- const rows = await listProjectGrantsByPrincipal(ctx, principalId);
229
- return rows.filter((row) => matchesResolvedScope(row, scopeIds));
230
- }
231
- async function bridgeListProjectGrantsByTopicAndGroup(ctx, topicId, groupId) {
232
- const resolved = await resolveGrantScopeIds(ctx, { topicId });
233
- const scopeIds = buildScopeMatchers(topicId, resolved);
234
- const rows = await listProjectGrantsByGroup(ctx, groupId);
235
- return rows.filter((row) => matchesResolvedScope(row, scopeIds));
236
- }
237
- async function bridgeListProjectGrantsByPrincipalStatus(ctx, principalId, status) {
238
- const rows = await listProjectGrantsByPrincipal(ctx, principalId);
239
- return rows.filter((row) => row.status === status);
240
- }
241
- async function bridgeListProjectGrantsByGroupStatus(ctx, groupId, status) {
242
- const rows = await listProjectGrantsByGroup(ctx, groupId);
243
- return rows.filter((row) => row.status === status);
244
- }
245
- async function bridgeInsertProjectGrant(ctx, value) {
246
- const resolved = await resolveGrantScopeIds(ctx, value);
247
- return await ctx.db.insert("projectGrants", {
248
- ...value,
249
- ...resolved.topicId ? { topicId: resolved.topicId } : {},
250
- ...resolved.projectId ?? resolved.topicId ? { projectId: resolved.projectId ?? resolved.topicId } : {}
251
- });
252
- }
253
-
254
- // ../access-control/src/resolvers.ts
255
- async function findUserByClerkId(ctx, clerkId) {
256
- const normalizedClerkId = clerkId.trim();
257
- if (!normalizedClerkId) {
258
- return null;
259
- }
260
- if (typeof ctx.runQuery === "function") {
261
- try {
262
- const bridgedUser = await ctx.runQuery(api.users.getUserByClerkId, {
263
- clerkId: normalizedClerkId
264
- });
265
- if (bridgedUser) {
266
- return bridgedUser;
267
- }
268
- } catch {
269
- }
270
- }
271
- try {
272
- const users = await ctx.db.query("users").collect();
273
- return users.find((user) => String(user.clerkId ?? "") === normalizedClerkId) ?? null;
274
- } catch {
275
- return null;
276
- }
277
- }
278
- async function findUserByPrincipalId(ctx, principalId) {
279
- const normalizedPrincipalId = principalId.trim();
280
- if (!normalizedPrincipalId) {
281
- return null;
282
- }
283
- try {
284
- const users = await ctx.db.query("users").collect();
285
- return users.find(
286
- (user) => String(user.defaultPrincipalId ?? "") === normalizedPrincipalId
287
- ) ?? null;
288
- } catch {
289
- return null;
290
- }
291
- }
292
- async function findAgentByPrincipalId(ctx, principalId) {
293
- const normalizedPrincipalId = principalId.trim();
294
- if (!normalizedPrincipalId) {
295
- return null;
296
- }
297
- if (typeof ctx.runQuery === "function") {
298
- try {
299
- const bridgedAgent = await ctx.runQuery(
300
- api.agents.getAgentByPrincipalId,
301
- {
302
- principalId: normalizedPrincipalId
303
- }
304
- );
305
- if (bridgedAgent) {
306
- return bridgedAgent;
307
- }
308
- } catch {
309
- }
310
- }
311
- try {
312
- const agents = await ctx.db.query("agents").collect();
313
- return agents.find(
314
- (agent) => String(agent.principalId ?? "") === normalizedPrincipalId
315
- ) ?? null;
316
- } catch {
317
- return null;
318
- }
319
- }
320
- function defaultResolvers() {
321
- return {
322
- async getProject(ctx, topicId) {
323
- return await resolveTopicProjectOverlay(ctx, topicId, {
324
- idMode: "legacy",
325
- projectLikeOnly: false
326
- });
327
- },
328
- async listTopics(ctx) {
329
- return await listTopicProjectOverlays(ctx, { idMode: "legacy" });
330
- },
331
- async listTopicsByOwner(ctx, ownerId) {
332
- const topics = await listTopicProjectOverlays(ctx, { idMode: "legacy" });
333
- return topics.filter((topic) => topic.ownerId === ownerId);
334
- },
335
- async listTopicsByVisibility(ctx, visibility) {
336
- const topics = await listTopicProjectOverlays(ctx, { idMode: "legacy" });
337
- return topics.filter((topic) => topic.visibility === visibility);
338
- },
339
- async listProjectGrantsByProjectAndPrincipal(ctx, topicId, principalId) {
340
- return await bridgeListProjectGrantsByTopicAndPrincipal(
341
- ctx,
342
- topicId,
343
- principalId
344
- );
345
- },
346
- async listProjectGrantsByProjectAndGroup(ctx, topicId, groupId) {
347
- return await bridgeListProjectGrantsByTopicAndGroup(ctx, topicId, groupId);
348
- },
349
- async listProjectGrantsByPrincipalStatus(ctx, principalId, status) {
350
- return await bridgeListProjectGrantsByPrincipalStatus(
351
- ctx,
352
- principalId,
353
- status
354
- );
355
- },
356
- async listProjectGrantsByGroupStatus(ctx, groupId, status) {
357
- return await bridgeListProjectGrantsByGroupStatus(ctx, groupId, status);
358
- },
359
- async insertProjectGrant(ctx, value) {
360
- return await bridgeInsertProjectGrant(ctx, value);
361
- },
362
- async getAgentByPrincipalId(ctx, principalId) {
363
- return await findAgentByPrincipalId(ctx, principalId);
364
- },
365
- async getUserByClerkId(ctx, clerkId) {
366
- return await findUserByClerkId(ctx, clerkId);
367
- },
368
- async getUserByPrincipalId(ctx, principalId) {
369
- return await findUserByPrincipalId(ctx, principalId);
370
- }
371
- };
372
- }
373
- var resolverOverrides = {};
374
- function resolveAccessControlAppResolvers(_ctx) {
375
- return {
376
- ...defaultResolvers(),
377
- ...resolverOverrides
378
- };
379
- }
380
-
381
- // ../access-control/src/principalContext.ts
382
- function requireCanonicalResolvedUser(user, clerkId) {
383
- const resolved = user;
384
- if (!resolved) {
385
- throw new Error(
386
- `[AccessControl] Canonical user identity required for ${clerkId}. Sync users.upsertUser before user-bound access checks.`
387
- );
388
- }
389
- const { mcRole, defaultTenantId, defaultWorkspaceId, defaultPrincipalId } = resolved;
390
- if (mcRole !== "platform_admin" && mcRole !== "tenant_admin" && mcRole !== "workspace_admin" && mcRole !== "editor" && mcRole !== "viewer" && mcRole !== "auditor" && mcRole !== "service_agent") {
391
- throw new Error(
392
- `[AccessControl] Canonical MC role required for ${clerkId}. Re-sync Master Control identity before user-bound access checks.`
393
- );
394
- }
395
- if (typeof defaultTenantId !== "string" || defaultTenantId.trim().length === 0) {
396
- throw new Error(
397
- `[AccessControl] Canonical home tenant required for ${clerkId}. Re-sync Master Control identity before user-bound access checks.`
398
- );
399
- }
400
- if (typeof defaultWorkspaceId !== "string" || defaultWorkspaceId.trim().length === 0) {
401
- throw new Error(
402
- `[AccessControl] Canonical home workspace required for ${clerkId}. Re-sync Master Control identity before user-bound access checks.`
403
- );
404
- }
405
- if (typeof defaultPrincipalId !== "string" || defaultPrincipalId.trim().length === 0) {
406
- throw new Error(
407
- `[AccessControl] Canonical federated principal required for ${clerkId}. Re-sync Master Control identity before user-bound access checks.`
408
- );
409
- }
410
- return {
411
- mcRole,
412
- defaultTenantId: defaultTenantId.trim(),
413
- defaultWorkspaceId: defaultWorkspaceId.trim(),
414
- defaultPrincipalId: defaultPrincipalId.trim()
415
- };
416
- }
417
- function isPrincipalIdInput(value) {
418
- return value.startsWith("user:") || value.startsWith("group:") || value.startsWith("service:") || value.startsWith("agent:") || value.startsWith("external_viewer:");
419
- }
420
- async function resolveCanonicalUserRecord(ctx, actorId) {
421
- const normalizedActorId = actorId.trim();
422
- const clerkId = isPrincipalIdInput(normalizedActorId) && normalizedActorId.startsWith("user:") ? normalizedActorId.slice("user:".length) : normalizedActorId;
423
- const resolvers = resolveAccessControlAppResolvers();
424
- const resolvedByClerkId = await resolvers.getUserByClerkId(ctx, clerkId);
425
- if (resolvedByClerkId) {
426
- return {
427
- resolvedUser: resolvedByClerkId,
428
- clerkId,
429
- contextClerkId: clerkId
430
- };
431
- }
432
- const resolvedByPrincipalId = await resolvers.getUserByPrincipalId(
433
- ctx,
434
- normalizedActorId
435
- );
436
- return {
437
- resolvedUser: resolvedByPrincipalId ?? null,
438
- clerkId,
439
- contextClerkId: normalizedActorId.startsWith("user:") && clerkId.length > 0 ? clerkId : normalizedActorId
440
- };
441
- }
442
- function uniqRoles(roles) {
443
- const roleSet = /* @__PURE__ */ new Set();
444
- for (const role of roles) {
445
- if (role === "platform_admin" || role === "tenant_admin" || role === "workspace_admin" || role === "editor" || role === "viewer" || role === "auditor" || role === "service_agent") {
446
- roleSet.add(role);
447
- }
448
- }
449
- return [...roleSet];
450
- }
451
- function normalizeGroupIds(value) {
452
- if (!Array.isArray(value)) {
453
- return [];
454
- }
455
- return [...new Set(
456
- value.filter((entry) => typeof entry === "string").map((entry) => entry.trim()).filter(Boolean)
457
- )];
458
- }
459
- function requireServiceAgentUser(user, actorId) {
460
- const canonicalUser = requireCanonicalResolvedUser(user, actorId);
461
- if (canonicalUser.mcRole !== "service_agent") {
462
- throw new Error(
463
- `[AccessControl] Canonical service_agent identity required for ${actorId}. Sync users.upsertUser before agent-bound access checks.`
464
- );
465
- }
466
- return canonicalUser;
467
- }
468
- function requireCanonicalResolvedAgent(agent, actorId) {
469
- const resolved = agent;
470
- if (!resolved) {
471
- throw new Error(
472
- `[AccessControl] Agent "${actorId}" not found in agents or users table.`
473
- );
474
- }
475
- if (typeof resolved.principalId !== "string" || resolved.principalId.trim().length === 0) {
476
- throw new Error(
477
- `[AccessControl] Canonical agent principalId required for ${actorId}.`
478
- );
479
- }
480
- if (typeof resolved.tenantId !== "string" || resolved.tenantId.trim().length === 0) {
481
- throw new Error(
482
- `[AccessControl] Canonical home tenant required for ${actorId}.`
483
- );
484
- }
485
- if (typeof resolved.workspaceId !== "string" || resolved.workspaceId.trim().length === 0) {
486
- throw new Error(
487
- `[AccessControl] Canonical home workspace required for ${actorId}.`
488
- );
489
- }
490
- return {
491
- principalId: resolved.principalId.trim(),
492
- tenantId: resolved.tenantId.trim(),
493
- workspaceId: resolved.workspaceId.trim(),
494
- roles: uniqRoles(Array.isArray(resolved.roles) ? resolved.roles : []) ?? ["service_agent"],
495
- groupIds: normalizeGroupIds(resolved.groupIds)
496
- };
497
- }
498
- async function resolvePrincipalContext(ctx, actorId) {
499
- if (actorId.startsWith("agent:")) {
500
- const resolvers = resolveAccessControlAppResolvers();
501
- const resolvedAgent = await resolvers.getAgentByPrincipalId(ctx, actorId);
502
- if (resolvedAgent) {
503
- const agent = requireCanonicalResolvedAgent(
504
- resolvedAgent,
505
- actorId
506
- );
507
- return {
508
- principalId: agent.principalId,
509
- principalType: "service",
510
- clerkId: actorId,
511
- tenantId: agent.tenantId,
512
- workspaceId: agent.workspaceId,
513
- roles: agent.roles.length > 0 ? agent.roles : ["service_agent"],
514
- groupIds: agent.groupIds,
515
- isPlatformAdmin: false,
516
- isTenantAdmin: false,
517
- isWorkspaceAdmin: false,
518
- isSystemFallback: false
519
- };
520
- }
521
- const resolvedUser2 = await resolvers.getUserByClerkId(
522
- ctx,
523
- actorId
524
- );
525
- if (!resolvedUser2) {
526
- throw new Error(
527
- `[AccessControl] Agent "${actorId}" not found in agents or users table.`
528
- );
529
- }
530
- const user2 = requireServiceAgentUser(
531
- resolvedUser2,
532
- actorId
533
- );
534
- console.warn(
535
- `[AccessControl] Deprecated legacy service-agent fallback for ${actorId}; migrate this principal into identity.agents.`
536
- );
537
- return {
538
- principalId: user2.defaultPrincipalId,
539
- principalType: "service",
540
- clerkId: actorId,
541
- tenantId: user2.defaultTenantId,
542
- workspaceId: user2.defaultWorkspaceId,
543
- roles: ["service_agent"],
544
- groupIds: normalizeGroupIds(resolvedUser2?.principalGroupIds),
545
- isPlatformAdmin: false,
546
- isTenantAdmin: false,
547
- isWorkspaceAdmin: false,
548
- isSystemFallback: false
549
- };
550
- }
551
- const {
552
- resolvedUser,
553
- contextClerkId
554
- } = await resolveCanonicalUserRecord(ctx, actorId);
555
- const user = requireCanonicalResolvedUser(
556
- resolvedUser,
557
- contextClerkId
558
- );
559
- if (!user.defaultPrincipalId) {
560
- throw new Error(
561
- `[AccessControl] Canonical federated principal required for ${contextClerkId}. Re-sync Master Control identity before user-bound access checks.`
562
- );
563
- }
564
- if (user.mcRole === "service_agent") {
565
- return {
566
- principalId: user.defaultPrincipalId,
567
- principalType: "service",
568
- clerkId: contextClerkId,
569
- tenantId: user.defaultTenantId,
570
- workspaceId: user.defaultWorkspaceId,
571
- roles: ["service_agent"],
572
- groupIds: normalizeGroupIds(resolvedUser?.principalGroupIds),
573
- isPlatformAdmin: false,
574
- isTenantAdmin: false,
575
- isWorkspaceAdmin: false,
576
- isSystemFallback: false
577
- };
578
- }
579
- const principalId = user.defaultPrincipalId;
580
- const effectiveRole = user.mcRole;
581
- const roles = effectiveRole === "platform_admin" ? ["platform_admin", "tenant_admin"] : effectiveRole === "tenant_admin" ? ["tenant_admin"] : [effectiveRole];
582
- const tenantId = user.defaultTenantId;
583
- const workspaceId = user.defaultWorkspaceId;
584
- const isPlatformAdmin = effectiveRole === "platform_admin";
585
- return {
586
- principalId,
587
- principalType: "user",
588
- clerkId: contextClerkId,
589
- tenantId,
590
- workspaceId,
591
- roles: uniqRoles(roles),
592
- groupIds: normalizeGroupIds(resolvedUser?.principalGroupIds),
593
- isPlatformAdmin,
594
- isTenantAdmin: isPlatformAdmin || effectiveRole === "tenant_admin",
595
- isWorkspaceAdmin: isPlatformAdmin || effectiveRole === "tenant_admin" || effectiveRole === "workspace_admin",
596
- isSystemFallback: false
597
- };
598
- }
599
-
600
- // ../access-control/src/access.ts
601
- function isTopicInPrincipalTenant(topic, principalTenantId) {
602
- if (!topic.tenantId) {
603
- return false;
604
- }
605
- if (!principalTenantId) {
606
- return false;
607
- }
608
- return String(topic.tenantId) === String(principalTenantId);
609
- }
610
- function isTopicInPrincipalWorkspace(topic, principalWorkspaceId) {
611
- if (!topic.workspaceId) {
612
- return false;
613
- }
614
- if (!principalWorkspaceId) {
615
- return false;
616
- }
617
- return String(topic.workspaceId) === String(principalWorkspaceId);
618
- }
619
- function isLegacyUnscopedTopic(topic) {
620
- return !topic.tenantId || !topic.workspaceId;
621
- }
622
- function isGrantScopeAlignedToTopic(topic, grant) {
623
- if (topic.tenantId && grant.tenantId && String(topic.tenantId) !== String(grant.tenantId)) {
624
- return false;
625
- }
626
- if (topic.workspaceId && grant.workspaceId && String(topic.workspaceId) !== String(grant.workspaceId)) {
627
- return false;
628
- }
629
- return true;
630
- }
631
- function isGrantSourceAllowedForVisibility(visibility, source) {
632
- if (source !== "external_share") {
633
- return true;
634
- }
635
- return visibility === "external" || visibility === "public";
636
- }
637
- function isGrantActive(grant) {
638
- if (grant.status !== "active") {
639
- return false;
640
- }
641
- if (grant.expiresAt !== void 0 && grant.expiresAt <= Date.now()) {
642
- return false;
643
- }
644
- return true;
645
- }
646
- async function hasPrincipalGrant(ctx, args) {
647
- const grants = await resolveAccessControlAppResolvers().listProjectGrantsByProjectAndPrincipal(
648
- ctx,
649
- args.topic._id,
650
- args.principalId
651
- );
652
- if (grants.some(
653
- (grant) => isGrantActive(grant) && isGrantScopeAlignedToTopic(args.topic, grant) && isGrantSourceAllowedForVisibility(
654
- args.topic.visibility,
655
- grant.source
656
- ) && (!args.principalIsExternal || args.topic.visibility === "public" || grant.source === "external_share")
657
- )) {
658
- return true;
659
- }
660
- return false;
661
- }
662
- async function hasGroupGrant(ctx, args) {
663
- if (args.groupIds.length === 0) {
664
- return false;
665
- }
666
- for (const groupId of args.groupIds) {
667
- const grants = await resolveAccessControlAppResolvers().listProjectGrantsByProjectAndGroup(ctx, args.topic._id, groupId);
668
- if (grants.some(
669
- (grant) => isGrantActive(grant) && isGrantScopeAlignedToTopic(args.topic, grant) && isGrantSourceAllowedForVisibility(
670
- args.topic.visibility,
671
- grant.source
672
- )
673
- )) {
674
- return true;
675
- }
676
- }
677
- return false;
678
- }
679
- function isExternalPrincipal(_ctx, _args) {
680
- return false;
681
- }
682
- async function evaluateTopicAccessDetailed(ctx, args) {
683
- if (args.legacyUserId) {
684
- return {
685
- hasAccess: true,
686
- isAdmin: false,
687
- isOwner: false,
688
- isShared: false,
689
- hasGrant: true,
690
- isFirmVisible: true,
691
- isExternalVisible: false,
692
- isPublicVisible: false,
693
- isTenantScopeMatch: true,
694
- isWorkspaceScopeMatch: true,
695
- isPrincipalExternal: false
696
- };
697
- }
698
- const topic = await resolveAccessControlAppResolvers().getProject(
699
- ctx,
700
- args.topicId
701
- );
702
- if (!topic) {
703
- return {
704
- hasAccess: false,
705
- isAdmin: false,
706
- isOwner: false,
707
- isShared: false,
708
- hasGrant: false,
709
- isFirmVisible: false,
710
- isExternalVisible: false,
711
- isPublicVisible: false,
712
- isTenantScopeMatch: false,
713
- isWorkspaceScopeMatch: false,
714
- isPrincipalExternal: false
715
- };
716
- }
717
- const { principalContext, legacyUserId } = args;
718
- const userIsAdmin = principalContext.isPlatformAdmin;
719
- const isOwner = topic.ownerId === legacyUserId;
720
- const isShared = (topic.sharedWith ?? []).includes(legacyUserId);
721
- const principalIsExternal = await isExternalPrincipal(ctx, {
722
- groupIds: principalContext.groupIds,
723
- topicTenantId: topic.tenantId,
724
- topicWorkspaceId: topic.workspaceId
725
- });
726
- const hasPrincipalGrantResult = await hasPrincipalGrant(ctx, {
727
- topic,
728
- principalId: principalContext.principalId,
729
- principalIsExternal
730
- });
731
- const hasGroupGrantResult = await hasGroupGrant(ctx, {
732
- topic,
733
- groupIds: principalContext.groupIds
734
- });
735
- const hasGrant = isShared || hasPrincipalGrantResult || hasGroupGrantResult;
736
- const legacyUnscoped = isLegacyUnscopedTopic(topic);
737
- const tenantScopeMatch = isTopicInPrincipalTenant(
738
- topic,
739
- principalContext.tenantId
740
- );
741
- const workspaceScopeMatch = isTopicInPrincipalWorkspace(
742
- topic,
743
- principalContext.workspaceId
744
- );
745
- const isPublicVisible = topic.visibility === "public";
746
- const isFirmVisible = topic.visibility === "firm" && !legacyUnscoped && tenantScopeMatch && workspaceScopeMatch && !principalIsExternal;
747
- const hasScopedGrant = hasGrant && (legacyUnscoped || tenantScopeMatch && workspaceScopeMatch);
748
- const isExternalVisible = topic.visibility === "external" && hasScopedGrant;
749
- const hasAccess = userIsAdmin || isOwner || hasScopedGrant || isPublicVisible || isFirmVisible;
750
- return {
751
- hasAccess,
752
- isAdmin: userIsAdmin,
753
- isOwner,
754
- isShared,
755
- hasGrant,
756
- isFirmVisible,
757
- isExternalVisible,
758
- isPublicVisible,
759
- isTenantScopeMatch: tenantScopeMatch,
760
- isWorkspaceScopeMatch: workspaceScopeMatch,
761
- isPrincipalExternal: principalIsExternal
762
- };
763
- }
764
- async function checkTopicAccessDetailed(ctx, topicId, userId) {
765
- const principalContext = await resolvePrincipalContext(ctx, userId);
766
- return evaluateTopicAccessDetailed(ctx, {
767
- topicId,
768
- legacyUserId: userId,
769
- principalContext
770
- });
771
- }
772
- async function checkTopicAccess(ctx, topicId, userId) {
773
- const result = await checkTopicAccessDetailed(ctx, topicId, userId);
774
- return result.hasAccess;
775
- }
776
- async function checkScopeAccess(ctx, scopeId, userId) {
777
- try {
778
- const topic = await ctx.db.get(scopeId);
779
- if (topic && topic.name !== void 0 && topic.type !== void 0) {
780
- return true;
781
- }
782
- } catch {
783
- }
784
- try {
785
- return await checkTopicAccess(ctx, scopeId, userId);
786
- } catch {
787
- return false;
788
- }
789
- }
790
- async function requireTopicAccess(ctx, topicId, userId) {
791
- const hasAccess = await checkTopicAccess(ctx, topicId, userId);
792
- if (!hasAccess) {
793
- throw new Error(
794
- "Access denied: You don't have permission to access this topic"
795
- );
796
- }
797
- }
798
- var requireProjectAccess = requireTopicAccess;
799
- var permissiveReturn = v.optional(v.any());
800
- var looseJsonObject = v.record(v.string(), v.any());
801
- var looseJsonArray = v.array(v.any());
802
- v.union(
803
- v.string(),
804
- v.number(),
805
- v.boolean(),
806
- v.null(),
807
- looseJsonObject,
808
- looseJsonArray
809
- );
810
- var api2 = anyApi;
811
- componentsGeneric();
812
10
  var internal = anyApi;
813
11
  var internalMutation = internalMutationGeneric;
814
12
  var internalQuery = internalQueryGeneric;
@@ -816,7 +14,7 @@ var mutation = mutationGeneric;
816
14
  var query = queryGeneric;
817
15
 
818
16
  // src/graphTypes.ts
819
- function getNodeLayer(nodeType2) {
17
+ function getNodeLayer(nodeType) {
820
18
  const L4_TYPES = ["decision"];
821
19
  const L3_TYPES = ["belief", "question", "theme", "deal"];
822
20
  const L2_TYPES = ["claim", "evidence", "synthesis", "answer"];
@@ -829,25 +27,25 @@ function getNodeLayer(nodeType2) {
829
27
  "value_chain"
830
28
  ];
831
29
  const ORGANIZATIONAL_TYPES = ["topic"];
832
- if (L4_TYPES.includes(nodeType2)) {
30
+ if (L4_TYPES.includes(nodeType)) {
833
31
  return "L4";
834
32
  }
835
- if (L3_TYPES.includes(nodeType2)) {
33
+ if (L3_TYPES.includes(nodeType)) {
836
34
  return "L3";
837
35
  }
838
- if (L2_TYPES.includes(nodeType2)) {
36
+ if (L2_TYPES.includes(nodeType)) {
839
37
  return "L2";
840
38
  }
841
- if (L1_TYPES.includes(nodeType2)) {
39
+ if (L1_TYPES.includes(nodeType)) {
842
40
  return "L1";
843
41
  }
844
- if (ONTOLOGICAL_TYPES.includes(nodeType2)) {
42
+ if (ONTOLOGICAL_TYPES.includes(nodeType)) {
845
43
  return "ontological";
846
44
  }
847
- if (ORGANIZATIONAL_TYPES.includes(nodeType2)) {
45
+ if (ORGANIZATIONAL_TYPES.includes(nodeType)) {
848
46
  return "organizational";
849
47
  }
850
- console.warn(`[GraphTypes] Unknown nodeType "${nodeType2}", defaulting to L2`);
48
+ console.warn(`[GraphTypes] Unknown nodeType "${nodeType}", defaulting to L2`);
851
49
  return "L2";
852
50
  }
853
51
 
@@ -858,11 +56,11 @@ var RESOLVED_PREDICTION_OUTCOMES = [
858
56
  "partial",
859
57
  "expired"
860
58
  ];
861
- function hasResolvedPredictionOutcome(predictionMeta2) {
862
- if (!predictionMeta2 || typeof predictionMeta2 !== "object") {
59
+ function hasResolvedPredictionOutcome(predictionMeta) {
60
+ if (!predictionMeta || typeof predictionMeta !== "object") {
863
61
  return false;
864
62
  }
865
- const outcome = predictionMeta2.outcome;
63
+ const outcome = predictionMeta.outcome;
866
64
  return typeof outcome === "string" && RESOLVED_PREDICTION_OUTCOMES.includes(outcome);
867
65
  }
868
66
 
@@ -1078,17 +276,17 @@ function assertOntologicalNodeSupersedeAllowed(args) {
1078
276
  }
1079
277
  });
1080
278
  }
1081
- var LEGACY_SCOPE_FIELD2 = "graphScopeProjectId";
279
+ var LEGACY_SCOPE_FIELD = "graphScopeProjectId";
1082
280
  function asMappedProjectId(topic) {
1083
281
  if (!topic) {
1084
282
  return;
1085
283
  }
1086
- const directLegacyProjectId = normalizeScopeValue(topic[LEGACY_SCOPE_FIELD2]);
284
+ const directLegacyProjectId = normalizeScopeValue(topic[LEGACY_SCOPE_FIELD]);
1087
285
  if (directLegacyProjectId) {
1088
286
  return directLegacyProjectId;
1089
287
  }
1090
288
  const metadata = topic.metadata || {};
1091
- const candidate = metadata[LEGACY_SCOPE_FIELD2] || metadata.legacyProjectId || metadata.projectId || metadata.scopeProjectId;
289
+ const candidate = metadata[LEGACY_SCOPE_FIELD] || metadata.legacyProjectId || metadata.projectId || metadata.scopeProjectId;
1092
290
  return candidate ? candidate : void 0;
1093
291
  }
1094
292
  function normalizeScopeValue(value) {
@@ -1117,7 +315,7 @@ async function findTopicsByScopeAlias(ctx, scopeId) {
1117
315
  try {
1118
316
  return await ctx.db.query("topics").withIndex(
1119
317
  "by_graph_scope_project",
1120
- (q) => q.eq(LEGACY_SCOPE_FIELD2, scopeId)
318
+ (q) => q.eq(LEGACY_SCOPE_FIELD, scopeId)
1121
319
  ).collect();
1122
320
  } catch {
1123
321
  const topics = await ctx.db.query("topics").collect();
@@ -1133,7 +331,7 @@ async function tryResolveHostTopicById(ctx, topicId) {
1133
331
  return null;
1134
332
  }
1135
333
  try {
1136
- return await ctx.runQuery(api2.topics.get, {
334
+ return await ctx.runQuery(api.topics.get, {
1137
335
  id: topicId
1138
336
  }) ?? null;
1139
337
  } catch {
@@ -1145,7 +343,7 @@ async function tryResolveHostTopicByLegacyScope(ctx, legacyScopeId) {
1145
343
  return null;
1146
344
  }
1147
345
  try {
1148
- return await ctx.runQuery(api2.topics.getByLegacyScopeId, {
346
+ return await ctx.runQuery(api.topics.getByLegacyScopeId, {
1149
347
  projectId: legacyScopeId
1150
348
  }) ?? null;
1151
349
  } catch {
@@ -1264,597 +462,6 @@ var optionalScopeArgs = {
1264
462
  projectId: v.optional(v.string()),
1265
463
  topicId: v.optional(v.string())
1266
464
  };
1267
- v.number();
1268
- v.union(
1269
- v.literal("very_high"),
1270
- // 0.9+
1271
- v.literal("high"),
1272
- // 0.7-0.9
1273
- v.literal("medium"),
1274
- // 0.4-0.7
1275
- v.literal("low"),
1276
- // 0.2-0.4
1277
- v.literal("very_low")
1278
- // 0-0.2
1279
- );
1280
- v.union(
1281
- v.literal(1),
1282
- // Critical
1283
- v.literal(2),
1284
- // High
1285
- v.literal(3),
1286
- // Medium
1287
- v.literal(4),
1288
- // Low
1289
- v.literal(5)
1290
- // Backlog
1291
- );
1292
- v.union(
1293
- v.literal("critical"),
1294
- v.literal("high"),
1295
- v.literal("medium"),
1296
- v.literal("low"),
1297
- v.literal("backlog")
1298
- );
1299
- v.union(
1300
- v.literal("active"),
1301
- v.literal("paused"),
1302
- v.literal("completed"),
1303
- v.literal("archived")
1304
- );
1305
- v.union(
1306
- v.literal("pending"),
1307
- v.literal("processing"),
1308
- v.literal("completed"),
1309
- v.literal("failed")
1310
- );
1311
- v.object({
1312
- crunchbaseId: v.optional(v.string()),
1313
- linkedinUrl: v.optional(v.string()),
1314
- pitchbookId: v.optional(v.string()),
1315
- twitterUrl: v.optional(v.string()),
1316
- domain: v.optional(v.string())
1317
- });
1318
- var sourceType = v.union(
1319
- v.literal("proprietary"),
1320
- // Internal Stack research
1321
- v.literal("primary"),
1322
- // Direct interviews, calls
1323
- v.literal("secondary"),
1324
- // Published sources
1325
- v.literal("ai_generated"),
1326
- // AI-synthesized
1327
- v.literal("user_input"),
1328
- // Manual user entry
1329
- v.literal("inferred")
1330
- // System inference
1331
- );
1332
- v.object({
1333
- sourceType: v.optional(sourceType),
1334
- sourceId: v.optional(v.string()),
1335
- // Reference to source entity
1336
- sourceUrl: v.optional(v.string()),
1337
- sourceDate: v.optional(v.number()),
1338
- sourceName: v.optional(v.string())
1339
- });
1340
- v.object({
1341
- cursor: v.optional(v.string()),
1342
- limit: v.optional(v.number())
1343
- });
1344
- v.object({
1345
- hasMore: v.boolean(),
1346
- nextCursor: v.optional(v.string()),
1347
- totalCount: v.optional(v.number())
1348
- });
1349
- var richTextContent = v.object({
1350
- type: v.literal("doc"),
1351
- content: looseJsonArray
1352
- });
1353
- v.union(v.string(), richTextContent);
1354
- v.object({
1355
- promptTokens: v.optional(v.number()),
1356
- completionTokens: v.optional(v.number()),
1357
- totalTokens: v.optional(v.number())
1358
- });
1359
- v.object({
1360
- fileName: v.optional(v.string()),
1361
- fileSize: v.optional(v.number()),
1362
- mimeType: v.optional(v.string()),
1363
- storageId: v.optional(v.id("_storage")),
1364
- externalUrl: v.optional(v.string())
1365
- });
1366
-
1367
- // ../schema-management/src/spine/tables/epistemicNodes.ts
1368
- var nodeType = v.union(
1369
- // --- L4: Audit Targets (decisions, outcomes) ---
1370
- v.literal("decision"),
1371
- // Investment decision with knowledge horizon snapshot
1372
- // --- L3: Traversal Anchors (epistemic structure) ---
1373
- v.literal("belief"),
1374
- // Structured conviction (immutable formulation)
1375
- v.literal("question"),
1376
- // Unit of uncertainty
1377
- v.literal("theme"),
1378
- // Investment thesis / conviction cluster
1379
- v.literal("deal"),
1380
- // Investment evaluation process
1381
- v.literal("topic"),
1382
- // Hierarchical knowledge container
1383
- // --- L2: Compression Boundary (minimum reasoning unit) ---
1384
- v.literal("claim"),
1385
- // Atomic assertion that can be true/false
1386
- v.literal("evidence"),
1387
- // Interpreted signal linked to beliefs
1388
- v.literal("synthesis"),
1389
- // Primers, deep research
1390
- v.literal("answer"),
1391
- // Immutable answer snapshot for a question
1392
- // --- L1: Terminal Leaves (non-traversable, grounding) ---
1393
- v.literal("atomic_fact"),
1394
- // Raw fact from source (not interpreted)
1395
- v.literal("excerpt"),
1396
- // Direct quote from source document
1397
- v.literal("source"),
1398
- // News, documents, transcripts
1399
- // --- Ontological Entities (things in the world) ---
1400
- v.literal("company"),
1401
- // Organization (subtype: private, corporate, portfolio)
1402
- v.literal("person"),
1403
- // Individual (founder, expert, LP, contact)
1404
- v.literal("investor"),
1405
- // Investment entity (subtype: vc, lp, cvc, pe, family_office, angel)
1406
- v.literal("function"),
1407
- // What a company does (from classifier)
1408
- v.literal("value_chain")
1409
- // Market structure / value flow
1410
- );
1411
- var epistemicLayer = v.union(
1412
- v.literal("L4"),
1413
- // Decisions, outcomes - audit targets
1414
- v.literal("L3"),
1415
- // Beliefs, questions, themes - traversal anchors
1416
- v.literal("L2"),
1417
- // Claims, evidence, synthesis - compression boundary
1418
- v.literal("L1"),
1419
- // Atomic facts, excerpts, sources - terminal leaves
1420
- v.literal("ontological"),
1421
- // Companies, people, etc - not epistemic
1422
- v.literal("organizational")
1423
- // Topics, lenses, worktrees — structural containers
1424
- );
1425
- var nodeStatus = v.union(
1426
- v.literal("active"),
1427
- v.literal("superseded"),
1428
- // Replaced by newer version
1429
- v.literal("archived"),
1430
- v.literal("deleted")
1431
- );
1432
- var sourceType2 = v.union(
1433
- v.literal("human"),
1434
- // User created directly
1435
- v.literal("ai_extracted"),
1436
- // LLM extracted from a source
1437
- v.literal("ai_generated"),
1438
- // LLM synthesized/created
1439
- v.literal("imported"),
1440
- // External system import
1441
- v.literal("system"),
1442
- // System-generated (migrations, classifiers)
1443
- v.literal("verified"),
1444
- // Human-verified source
1445
- v.literal("proprietary")
1446
- // Proprietary/internal data
1447
- );
1448
- var verificationStatus = v.union(
1449
- v.literal("unverified"),
1450
- v.literal("human_verified"),
1451
- v.literal("ai_verified"),
1452
- v.literal("contradicted"),
1453
- v.literal("outdated")
1454
- );
1455
- var syncStatus = v.union(
1456
- v.literal("synced"),
1457
- // Node and edges fully synced to Neo4j
1458
- v.literal("pending_edges"),
1459
- // Node created, edges being created
1460
- v.literal("edge_creation_failed")
1461
- // Edge creation failed, needs retry
1462
- );
1463
- var audienceLabel = v.string();
1464
- var sensitivityTier = v.union(
1465
- v.literal("low"),
1466
- v.literal("medium"),
1467
- v.literal("high"),
1468
- v.literal("restricted")
1469
- );
1470
- var exportClass = v.union(
1471
- v.literal("internal_only"),
1472
- v.literal("client_safe"),
1473
- v.literal("public_safe"),
1474
- v.literal("restricted")
1475
- );
1476
- var anonymizationClass = v.union(
1477
- v.literal("none"),
1478
- v.literal("standard"),
1479
- v.literal("strict")
1480
- );
1481
- var epistemicStatus = v.union(
1482
- v.literal("hypothesis"),
1483
- // Initial conjecture, low evidence
1484
- v.literal("emerging"),
1485
- // Building evidence, gaining traction
1486
- v.literal("established"),
1487
- // Well-evidenced, core to thesis
1488
- v.literal("challenged"),
1489
- // Contradicting evidence appeared
1490
- v.literal("assumption"),
1491
- // Taken as given, not actively tested
1492
- v.literal("deprecated")
1493
- // Superseded or abandoned
1494
- );
1495
- var beliefStatus = v.union(
1496
- v.literal("assumption"),
1497
- v.literal("hypothesis"),
1498
- v.literal("belief"),
1499
- v.literal("fact")
1500
- );
1501
- var reversibility = v.union(
1502
- v.literal("irreversible"),
1503
- // One-way door decision
1504
- v.literal("hard_to_reverse"),
1505
- // Significant cost to undo
1506
- v.literal("reversible"),
1507
- // Can change course with moderate effort
1508
- v.literal("trivial")
1509
- // Easy to adjust
1510
- );
1511
- var predictionOutcome = v.union(
1512
- v.literal("pending"),
1513
- v.literal("confirmed"),
1514
- v.literal("disconfirmed"),
1515
- v.literal("partial"),
1516
- v.literal("expired")
1517
- );
1518
- var predictionMeta = v.object({
1519
- isPrediction: v.boolean(),
1520
- registeredAt: v.number(),
1521
- // When prediction was made
1522
- expectedBy: v.optional(v.number()),
1523
- // When we expect resolution
1524
- outcome: v.optional(predictionOutcome),
1525
- outcomeRecordedAt: v.optional(v.number()),
1526
- outcomeEvidenceId: v.optional(v.string()),
1527
- // globalId of confirming evidence
1528
- confidenceAtPrediction: v.optional(v.number()),
1529
- // 0-1
1530
- actualVsPredicted: v.optional(v.string())
1531
- // Notes on how outcome compared
1532
- });
1533
- var methodology = v.union(
1534
- // Primary Research (high value)
1535
- v.literal("primary_research"),
1536
- // Direct investigation
1537
- v.literal("expert_interview"),
1538
- // Expert call/interview
1539
- v.literal("customer_interview"),
1540
- // Customer research
1541
- v.literal("field_observation"),
1542
- // On-site observation
1543
- v.literal("proprietary_data"),
1544
- // Internal data analysis
1545
- // Secondary Research
1546
- v.literal("desk_research"),
1547
- // Public sources
1548
- v.literal("regulatory_filing"),
1549
- // SEC, regulatory docs
1550
- v.literal("news_article"),
1551
- // News/press
1552
- v.literal("academic_paper"),
1553
- // Academic research
1554
- // AI-Assisted
1555
- v.literal("ai_synthesis"),
1556
- // AI-generated synthesis
1557
- v.literal("ai_extraction")
1558
- // AI-extracted from source
1559
- );
1560
- var informationAsymmetry = v.union(
1561
- v.literal("proprietary"),
1562
- // Only we have this
1563
- v.literal("early"),
1564
- // We're early but others will get it
1565
- v.literal("common")
1566
- // Everyone has access
1567
- );
1568
- var temporalNature = v.union(
1569
- v.literal("factual"),
1570
- // Resolved outcome. Grounded in reality.
1571
- v.literal("forecast"),
1572
- // Prediction. Will resolve. Discounted weight.
1573
- v.literal("unknown")
1574
- // Not yet classified.
1575
- );
1576
- var questionType = v.union(
1577
- v.literal("validation"),
1578
- // Does evidence support this belief?
1579
- v.literal("falsification"),
1580
- // What would prove this belief wrong?
1581
- v.literal("assumption_probe"),
1582
- // Is this unstated assumption true?
1583
- v.literal("prediction_test"),
1584
- // Will this predicted outcome occur?
1585
- v.literal("counterfactual"),
1586
- // What would we expect if X were false?
1587
- v.literal("discovery"),
1588
- // What don't we know yet?
1589
- v.literal("clarification"),
1590
- // What does X actually mean?
1591
- v.literal("comparison"),
1592
- // How does X compare to Y?
1593
- v.literal("causal"),
1594
- // What caused X?
1595
- v.literal("mechanism"),
1596
- // How does X work?
1597
- v.literal("general")
1598
- // Unclassified
1599
- );
1600
- var questionPriority = v.union(
1601
- v.literal("critical"),
1602
- // Blocks decision-making
1603
- v.literal("high"),
1604
- // Important for thesis
1605
- v.literal("medium"),
1606
- // Would be nice to know
1607
- v.literal("low")
1608
- // Background/curiosity
1609
- );
1610
- var answerQuality = v.union(
1611
- v.literal("definitive"),
1612
- // Clear, well-supported
1613
- v.literal("strong"),
1614
- // Good evidence, high confidence
1615
- v.literal("moderate"),
1616
- // Some evidence
1617
- v.literal("weak"),
1618
- // Limited evidence
1619
- v.literal("speculative"),
1620
- // Mostly conjecture
1621
- v.literal("unanswered")
1622
- // No answer yet
1623
- );
1624
- var consensusView = v.union(
1625
- v.literal("aligned"),
1626
- // We agree with market consensus
1627
- v.literal("ahead_of"),
1628
- // We see this before consensus does
1629
- v.literal("contrarian"),
1630
- // We actively disagree with consensus
1631
- v.literal("orthogonal"),
1632
- // We're looking at something consensus isn't discussing
1633
- v.literal("unknown")
1634
- // We don't know what consensus thinks
1635
- );
1636
- var themeConviction = v.union(
1637
- v.literal("high"),
1638
- // Strong conviction, actively deploying
1639
- v.literal("medium"),
1640
- // Building conviction
1641
- v.literal("low"),
1642
- // Exploring, not convicted
1643
- v.literal("negative")
1644
- // Actively avoiding
1645
- );
1646
- var decisionType = v.union(
1647
- v.literal("invest"),
1648
- v.literal("pass"),
1649
- v.literal("follow_on"),
1650
- v.literal("exit"),
1651
- v.literal("deep_dive"),
1652
- v.literal("monitor"),
1653
- v.literal("deprioritize"),
1654
- v.literal("thesis_adopt"),
1655
- v.literal("thesis_revise"),
1656
- v.literal("thesis_abandon")
1657
- );
1658
- var decisionOutcome = v.union(
1659
- v.literal("pending"),
1660
- v.literal("successful"),
1661
- v.literal("unsuccessful"),
1662
- v.literal("mixed"),
1663
- v.literal("unknown")
1664
- );
1665
- var externalIds2 = v.object({
1666
- crunchbase: v.optional(v.string()),
1667
- linkedin: v.optional(v.string()),
1668
- pitchbook: v.optional(v.string()),
1669
- twitter: v.optional(v.string()),
1670
- website: v.optional(v.string())
1671
- });
1672
- defineTable({
1673
- // === IDENTITY ===
1674
- globalId: v.string(),
1675
- // UUID - survives migration to Neo4j
1676
- // === TYPE ===
1677
- nodeType,
1678
- // === EPISTEMIC LAYER ===
1679
- epistemicLayer: v.optional(epistemicLayer),
1680
- // === SUBTYPE (for typed entities) ===
1681
- subtype: v.optional(v.string()),
1682
- // company: private|corporate|portfolio, investor: vc|lp|cvc|pe|family_office|angel
1683
- // === CONTENT ===
1684
- canonicalText: v.string(),
1685
- // The core content (belief statement, company name, etc.)
1686
- contentHash: v.string(),
1687
- // SHA256(nodeType + canonicalText) for deduplication
1688
- // Extended content (for sources/syntheses)
1689
- content: v.optional(v.string()),
1690
- // Full text for documents/articles
1691
- contentType: v.optional(v.string()),
1692
- // "markdown", "html", "pdf", "text"
1693
- // === METADATA ===
1694
- title: v.optional(v.string()),
1695
- // Display title
1696
- tags: v.optional(v.array(v.string())),
1697
- domain: v.optional(v.string()),
1698
- // For companies: website domain
1699
- // Type-specific metadata (flexible object - LEGACY)
1700
- // New code should use the typed fields below when available
1701
- metadata: v.optional(looseJsonObject),
1702
- // === POLICY / ENTITLEMENT ===
1703
- tenantId: v.optional(v.string()),
1704
- workspaceId: v.optional(v.string()),
1705
- ownerPrincipalId: v.optional(v.string()),
1706
- audienceLabel: v.optional(audienceLabel),
1707
- policyTags: v.optional(v.array(v.string())),
1708
- sensitivityTier: v.optional(sensitivityTier),
1709
- exportClass: v.optional(exportClass),
1710
- anonymizationClass: v.optional(anonymizationClass),
1711
- // === PUBLICATION (visibility-based, not copy-based) ===
1712
- // Publication expands who can see a workspace-local node — the node stays
1713
- // in its workspace, like a microservice exposing part of its API surface.
1714
- // Rules-based: pack/tenant-level publicationRules auto-evaluate on
1715
- // confidence changes and node creation. No manual click-by-click.
1716
- publicationStatus: v.optional(
1717
- v.union(
1718
- v.literal("unpublished"),
1719
- // Default: workspace-local only
1720
- v.literal("published"),
1721
- // Visible at tenant scope (rules matched)
1722
- v.literal("suppressed")
1723
- // Manually blocked even if rules match
1724
- )
1725
- ),
1726
- publishedAt: v.optional(v.number()),
1727
- // When publication status last changed to published
1728
- publishedBy: v.optional(v.string()),
1729
- // userId or "system:publication_rules" for auto-publish
1730
- // === TYPED METADATA FIELDS ===
1731
- // --- Belief ---
1732
- // Belief type — validated against schemaEnumConfig category "belief_type"
1733
- // Platform core: hypothesis, belief, principle, invariant, assumption,
1734
- // tenet, prior, preference, goal, forecast
1735
- beliefType: v.optional(v.string()),
1736
- beliefStatus: v.optional(beliefStatus),
1737
- epistemicStatus: v.optional(epistemicStatus),
1738
- reversibility: v.optional(reversibility),
1739
- predictionMeta: v.optional(predictionMeta),
1740
- // Consensus tracking (for non-consensus detection)
1741
- consensusView: v.optional(consensusView),
1742
- consensusConfidence: v.optional(v.number()),
1743
- // 0-1: What we think consensus confidence is
1744
- consensusSource: v.optional(v.string()),
1745
- // Where we got the consensus view (twitter, reports, etc.)
1746
- // --- Evidence ---
1747
- methodology: v.optional(methodology),
1748
- informationAsymmetry: v.optional(informationAsymmetry),
1749
- temporalNature: v.optional(temporalNature),
1750
- // --- Question ---
1751
- questionType: v.optional(questionType),
1752
- questionPriority: v.optional(questionPriority),
1753
- answerQuality: v.optional(answerQuality),
1754
- // --- Theme ---
1755
- themeConviction: v.optional(themeConviction),
1756
- // Market timing (for "early on theme" detection)
1757
- marketAwarenessDate: v.optional(v.number()),
1758
- // When this theme became broadly discussed
1759
- marketAwarenessSource: v.optional(v.string()),
1760
- // How we know (first major report, twitter volume spike, etc.)
1761
- earlySignalIds: v.optional(v.array(v.string())),
1762
- // globalIds of evidence we had before market awareness
1763
- // --- Decision ---
1764
- decisionType: v.optional(decisionType),
1765
- decisionOutcome: v.optional(decisionOutcome),
1766
- // === EXTERNAL IDS (for ontological entities) ===
1767
- externalIds: v.optional(externalIds2),
1768
- // === PROVENANCE ===
1769
- sourceType: sourceType2,
1770
- aiProvider: v.optional(v.string()),
1771
- // "claude", "gemini", "gpt-4", etc.
1772
- extractedFromNodeId: v.optional(v.id("epistemicNodes")),
1773
- // Quick reference to source
1774
- // === EXTRACTION CONTEXT ===
1775
- extractionModel: v.optional(v.string()),
1776
- // "claude-sonnet-4-20250514"
1777
- extractionPromptName: v.optional(v.string()),
1778
- // "lucern/extract-evidence"
1779
- extractionPromptVersion: v.optional(v.number()),
1780
- extractionTemperature: v.optional(v.number()),
1781
- extractionLangfuseTraceId: v.optional(v.string()),
1782
- // === GROUNDING VERIFICATION ===
1783
- groundingVerified: v.optional(v.boolean()),
1784
- groundingConfidence: v.optional(v.number()),
1785
- // 0-1 match quality
1786
- groundingMatchedText: v.optional(v.string()),
1787
- // Actual text from source
1788
- groundingStartOffset: v.optional(v.number()),
1789
- groundingEndOffset: v.optional(v.number()),
1790
- groundingRejectionReason: v.optional(v.string()),
1791
- // === CONFIDENCE & VERIFICATION ===
1792
- confidence: v.optional(v.number()),
1793
- // 0-1 projected probability P(x) = b + a*u
1794
- verificationStatus: v.optional(verificationStatus),
1795
- // === SL OPINION (Subjective Logic — Kernel v2) ===
1796
- // Replaces scalar confidence with rich epistemic state.
1797
- // b + d + u = 1. P(x) = b + a*u is stored in `confidence` for backward compat.
1798
- opinion_b: v.optional(v.number()),
1799
- // Belief: evidence FOR (0-1)
1800
- opinion_d: v.optional(v.number()),
1801
- // Disbelief: evidence AGAINST (0-1)
1802
- opinion_u: v.optional(v.number()),
1803
- // Uncertainty: absence of evidence (0-1)
1804
- opinion_a: v.optional(v.number()),
1805
- // Base rate / prior probability (0-1)
1806
- tupleContradicted: v.optional(v.boolean()),
1807
- // Single-belief tuple-space contradiction flag
1808
- // === LIFECYCLE ===
1809
- status: nodeStatus,
1810
- supersededBy: v.optional(v.id("epistemicNodes")),
1811
- // === OWNERSHIP ===
1812
- topicId: v.optional(v.string()),
1813
- // Canonical scope container (topic-first model)
1814
- projectId: v.optional(v.string()),
1815
- // DEPRECATED: Use belongs_to edges
1816
- createdBy: v.string(),
1817
- // Clerk user ID
1818
- createdAt: v.number(),
1819
- updatedAt: v.number(),
1820
- // === NEO4J SYNC STATUS ===
1821
- syncStatus: v.optional(syncStatus),
1822
- syncError: v.optional(v.string())
1823
- // Error message if sync failed
1824
- }).index("by_globalId", ["globalId"]).index("by_contentHash", ["contentHash"]).index("by_nodeType", ["nodeType"]).index("by_subtype", ["nodeType", "subtype"]).index("by_domain", ["domain"]).index("by_project", ["projectId"]).index("by_project_type", ["projectId", "nodeType"]).index("by_topic", ["topicId"]).index("by_topic_type", ["topicId", "nodeType"]).index("by_tenantId", ["tenantId"]).index("by_workspaceId", ["workspaceId"]).index("by_tenant_workspace", ["tenantId", "workspaceId"]).index("by_audienceLabel", ["audienceLabel"]).index("by_sensitivityTier", ["sensitivityTier"]).index("by_exportClass", ["exportClass"]).index("by_status", ["status"]).index("by_sourceType", ["sourceType"]).index("by_verification", ["verificationStatus"]).index("by_layer", ["epistemicLayer"]).index("by_layer_type", ["epistemicLayer", "nodeType"]).index("by_syncStatus", ["syncStatus"]).index("by_publicationStatus", ["publicationStatus"]).index("by_tenant_publicationStatus", ["tenantId", "publicationStatus"]).index("by_belief_status", ["nodeType", "beliefStatus"]).index("by_epistemic_status", ["nodeType", "epistemicStatus"]).index("by_temporal_nature", ["nodeType", "temporalNature"]).index("by_methodology", ["nodeType", "methodology"]).index("by_reversibility", ["nodeType", "reversibility"]).index("by_questionType", ["nodeType", "questionType"]).index("by_questionPriority", ["nodeType", "questionPriority"]).searchIndex("search_canonicalText", {
1825
- searchField: "canonicalText",
1826
- filterFields: ["nodeType", "projectId", "topicId", "status"]
1827
- });
1828
- function getLayerForNodeType(type) {
1829
- switch (type) {
1830
- case "decision":
1831
- return "L4";
1832
- case "belief":
1833
- case "question":
1834
- case "theme":
1835
- case "deal":
1836
- return "L3";
1837
- case "claim":
1838
- case "evidence":
1839
- case "synthesis":
1840
- case "answer":
1841
- return "L2";
1842
- case "atomic_fact":
1843
- case "excerpt":
1844
- case "source":
1845
- return "L1";
1846
- case "topic":
1847
- return "organizational";
1848
- case "company":
1849
- case "person":
1850
- case "investor":
1851
- case "function":
1852
- case "value_chain":
1853
- return "ontological";
1854
- }
1855
- }
1856
-
1857
- // src/workspaceIsolation.ts
1858
465
  function normalizeScopeValue2(value) {
1859
466
  if (typeof value !== "string") {
1860
467
  return;
@@ -1872,7 +479,7 @@ function throwWorkspaceIsolationError(args) {
1872
479
  throw error;
1873
480
  }
1874
481
  function assertWorkspaceScopedEpistemicNodeScope(args) {
1875
- const layer = getLayerForNodeType(args.nodeType);
482
+ const layer = isNodeType(args.nodeType) ? getLayerForNodeType(args.nodeType) : void 0;
1876
483
  if (layer === "ontological") {
1877
484
  return;
1878
485
  }
@@ -2108,8 +715,8 @@ var getByProjectAndTypeLite = query({
2108
715
  const pageSize = clampNodeLimit(args.limit);
2109
716
  const scanLimit = Math.min(pageSize * 3, MAX_NODE_PAGE_SIZE);
2110
717
  const query2 = ctx.db.query("epistemicNodes").withIndex(
2111
- scope.topicId ? "by_topic_type" : "by_project_type",
2112
- (q) => scope.topicId ? q.eq("topicId", scope.topicId).eq("nodeType", args.nodeType) : q.eq("projectId", scope.projectId).eq("nodeType", args.nodeType)
718
+ "by_topic_type",
719
+ (q) => q.eq("topicId", scope.topicId).eq("nodeType", args.nodeType)
2113
720
  );
2114
721
  const nodes = await query2.order("desc").take(scanLimit);
2115
722
  const statusFiltered = args.status ? nodes.filter((n) => n.status === args.status) : nodes.filter((n) => n.status === "active");
@@ -2315,7 +922,7 @@ var create = mutation({
2315
922
  if (existing && existing.status === "active") {
2316
923
  return { nodeId: existing._id, isDuplicate: true };
2317
924
  }
2318
- const epistemicLayer2 = getNodeLayer(args.nodeType);
925
+ const epistemicLayer = getNodeLayer(args.nodeType);
2319
926
  const resolvedScope = args.topicId || args.projectId ? await resolveTopicProjectScope(ctx, {
2320
927
  topicId: args.topicId,
2321
928
  projectId: args.projectId
@@ -2332,11 +939,11 @@ var create = mutation({
2332
939
  tenantId: resolvedScope.tenantId,
2333
940
  workspaceId: resolvedScope.workspaceId,
2334
941
  nodeType: args.nodeType,
2335
- epistemicLayer: epistemicLayer2
942
+ epistemicLayer
2336
943
  },
2337
944
  mutationName: "epistemicNodes.create"
2338
945
  });
2339
- } else if (epistemicLayer2 !== "ontological") {
946
+ } else if (epistemicLayer !== "ontological") {
2340
947
  throw new Error(
2341
948
  "Workspace-scoped reasoning isolation requires topicId or projectId for non-ontological node creation."
2342
949
  );
@@ -2344,7 +951,7 @@ var create = mutation({
2344
951
  const nodeId = await ctx.db.insert("epistemicNodes", {
2345
952
  globalId: args.globalId,
2346
953
  nodeType: args.nodeType,
2347
- epistemicLayer: epistemicLayer2,
954
+ epistemicLayer,
2348
955
  // Phase 2B: Auto-derived from nodeType
2349
956
  subtype: args.subtype,
2350
957
  canonicalText: args.canonicalText,
@@ -2505,11 +1112,11 @@ var supersede = mutation({
2505
1112
  mutationName: "epistemicNodes.supersede"
2506
1113
  });
2507
1114
  const now = Date.now();
2508
- const epistemicLayer2 = oldNode.epistemicLayer || getNodeLayer(oldNode.nodeType);
1115
+ const epistemicLayer = oldNode.epistemicLayer || getNodeLayer(oldNode.nodeType);
2509
1116
  const newNodeId = await ctx.db.insert("epistemicNodes", {
2510
1117
  globalId: args.newGlobalId,
2511
1118
  nodeType: oldNode.nodeType,
2512
- epistemicLayer: epistemicLayer2,
1119
+ epistemicLayer,
2513
1120
  // Phase 2B: Inherit layer (supersession is same-layer only)
2514
1121
  canonicalText: args.newCanonicalText,
2515
1122
  contentHash: args.newContentHash,
@@ -2742,14 +1349,14 @@ var batchCreate = mutation({
2742
1349
  nodeType: node.nodeType,
2743
1350
  mutationName: "epistemicNodes.batchCreate"
2744
1351
  });
2745
- const epistemicLayer2 = getNodeLayer(node.nodeType);
1352
+ const epistemicLayer = getNodeLayer(node.nodeType);
2746
1353
  const resolvedScope = node.topicId || node.projectId ? await resolveTopicProjectScope(ctx, {
2747
1354
  topicId: node.topicId,
2748
1355
  projectId: node.projectId
2749
1356
  }).catch(() => void 0) : void 0;
2750
1357
  const nodeId = await ctx.db.insert("epistemicNodes", {
2751
1358
  ...node,
2752
- epistemicLayer: epistemicLayer2,
1359
+ epistemicLayer,
2753
1360
  // Phase 2B: Auto-derived from nodeType
2754
1361
  verificationStatus: node.verificationStatus ?? "unverified",
2755
1362
  status: "active",
@@ -2833,11 +1440,11 @@ var createInternal = internalMutation({
2833
1440
  projectId: args.projectId
2834
1441
  }).catch(() => void 0) : void 0;
2835
1442
  const contentHash = args.contentHash || `${args.nodeType}:${args.canonicalText}`.slice(0, 64);
2836
- const epistemicLayer2 = args.epistemicLayer || getNodeLayer(args.nodeType);
1443
+ const epistemicLayer = args.epistemicLayer || getNodeLayer(args.nodeType);
2837
1444
  const nodeId = await ctx.db.insert("epistemicNodes", {
2838
1445
  globalId: args.globalId,
2839
1446
  nodeType: args.nodeType,
2840
- epistemicLayer: epistemicLayer2,
1447
+ epistemicLayer,
2841
1448
  subtype: args.subtype,
2842
1449
  canonicalText: args.canonicalText,
2843
1450
  contentHash,