@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,937 +1,14 @@
1
1
  import { v } from 'convex/values';
2
- import { componentsGeneric, defineTable, mutationGeneric, anyApi, queryGeneric, internalQueryGeneric, internalMutationGeneric } from 'convex/server';
2
+ import { requireProjectAccess, checkScopeAccess, checkProjectAccess } from '@lucern/access-control/access';
3
+ import { canAudienceClassAccess, normalizeAudienceKey, classFromAudienceKey } from '@lucern/access-control/audience';
4
+ import { listAudienceRegistryRows } from '@lucern/access-control/audienceRegistry';
5
+ import { permissiveReturn } from '@lucern/contracts/schema-helpers/validators';
6
+ import { componentsGeneric, mutationGeneric, anyApi, queryGeneric, internalQueryGeneric, internalMutationGeneric } from 'convex/server';
7
+ import { isNodeType, getLayerForNodeType } from '@lucern/contracts/schema-helpers/spine/tables/epistemicNodes';
3
8
 
4
9
  // src/epistemicEvidence.ts
5
10
  var api = anyApi;
6
11
  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 checkProjectAccess = checkTopicAccess;
799
- var requireProjectAccess = requireTopicAccess;
800
-
801
- // ../access-control/src/audience.ts
802
- var AUDIENCE_CLASS_RANK = {
803
- public: 0,
804
- restricted_external: 1,
805
- internal: 2
806
- };
807
- function normalizeKey(key) {
808
- return (key ?? "").trim().toLowerCase().replace(/[^a-z0-9:_-]+/g, "_").replace(/_+/g, "_").replace(/^_+|_+$/g, "");
809
- }
810
- function normalizeAudienceKey(key) {
811
- return normalizeKey(key);
812
- }
813
- function classFromAudienceKey(audienceKey, fallback = "internal") {
814
- const key = normalizeKey(audienceKey);
815
- if (!key) {
816
- return fallback;
817
- }
818
- if (key === "internal") {
819
- return "internal";
820
- }
821
- if (key === "public") {
822
- return "public";
823
- }
824
- if (key === "lp" || key === "external" || key === "client" || key === "partner" || key === "portfolio" || key === "network" || key === "restricted_external") {
825
- return "restricted_external";
826
- }
827
- return fallback;
828
- }
829
- function canAudienceClassAccess(viewerClass, resourceClass) {
830
- return AUDIENCE_CLASS_RANK[viewerClass] >= AUDIENCE_CLASS_RANK[resourceClass];
831
- }
832
-
833
- // ../access-control/src/audienceRegistry.ts
834
- var DEFAULT_AUDIENCES = [
835
- {
836
- audienceKey: "internal",
837
- audienceLabel: "Internal",
838
- audienceClass: "internal"
839
- },
840
- {
841
- audienceKey: "lp",
842
- audienceLabel: "Limited Partners",
843
- audienceClass: "restricted_external"
844
- },
845
- {
846
- audienceKey: "public",
847
- audienceLabel: "Public",
848
- audienceClass: "public"
849
- }
850
- ];
851
- var AUDIENCE_CLASS_PRIORITY = {
852
- internal: 0,
853
- restricted_external: 1,
854
- public: 2
855
- };
856
- function normalizeRegistryRow(row) {
857
- return {
858
- audienceKey: normalizeAudienceKey(row.audienceKey),
859
- audienceLabel: row.audienceLabel,
860
- audienceClass: row.audienceClass,
861
- workspaceId: row.workspaceId
862
- };
863
- }
864
- function dedupeRegistryRows(rows) {
865
- const byKey = /* @__PURE__ */ new Map();
866
- for (const row of rows) {
867
- const key = normalizeAudienceKey(row.audienceKey);
868
- if (!key) {
869
- continue;
870
- }
871
- const existing = byKey.get(key);
872
- const isWorkspaceScoped = row.workspaceId !== void 0;
873
- const existingWorkspaceScoped = existing?.workspaceId !== void 0;
874
- if (!existing || isWorkspaceScoped && !existingWorkspaceScoped) {
875
- byKey.set(key, {
876
- ...row,
877
- audienceKey: key
878
- });
879
- }
880
- }
881
- const normalized = [...byKey.values()];
882
- normalized.sort((a, b) => {
883
- const classDelta = AUDIENCE_CLASS_PRIORITY[a.audienceClass] - AUDIENCE_CLASS_PRIORITY[b.audienceClass];
884
- if (classDelta !== 0) {
885
- return classDelta;
886
- }
887
- return a.audienceKey.localeCompare(b.audienceKey);
888
- });
889
- return normalized;
890
- }
891
- async function queryRegistryRows(ctx, args) {
892
- if (!args.tenantId) {
893
- return [...DEFAULT_AUDIENCES];
894
- }
895
- const rows = await ctx.db.query("platformAudiences").withIndex("by_tenantId", (q) => q.eq("tenantId", args.tenantId)).collect();
896
- const workspaceIdString = args.workspaceId ? String(args.workspaceId) : null;
897
- const tenantScoped = rows.filter((row) => row.status === "active");
898
- const applicable = tenantScoped.filter((row) => {
899
- if (!row.workspaceId) {
900
- return true;
901
- }
902
- if (!workspaceIdString) {
903
- return false;
904
- }
905
- return String(row.workspaceId) === workspaceIdString;
906
- });
907
- return dedupeRegistryRows([
908
- ...DEFAULT_AUDIENCES,
909
- ...applicable.map(
910
- (row) => normalizeRegistryRow({
911
- audienceKey: row.audienceKey,
912
- audienceLabel: row.audienceLabel,
913
- audienceClass: row.audienceClass,
914
- workspaceId: row.workspaceId
915
- })
916
- )
917
- ]);
918
- }
919
- async function listAudienceRegistryRows(ctx, args) {
920
- return queryRegistryRows(ctx, args);
921
- }
922
- var permissiveReturn = v.optional(v.any());
923
- var looseJsonObject = v.record(v.string(), v.any());
924
- var looseJsonArray = v.array(v.any());
925
- v.union(
926
- v.string(),
927
- v.number(),
928
- v.boolean(),
929
- v.null(),
930
- looseJsonObject,
931
- looseJsonArray
932
- );
933
- var api2 = anyApi;
934
- componentsGeneric();
935
12
  var internal = anyApi;
936
13
  var internalMutation = internalMutationGeneric;
937
14
  var internalQuery = internalQueryGeneric;
@@ -972,48 +49,48 @@ function generateGlobalId() {
972
49
  }
973
50
 
974
51
  // src/topicProjectOverlay.ts
975
- var LEGACY_SCOPE_FIELD2 = "graphScopeProjectId";
976
- function readNonEmptyString2(value) {
52
+ var LEGACY_SCOPE_FIELD = "graphScopeProjectId";
53
+ function readNonEmptyString(value) {
977
54
  if (typeof value !== "string") {
978
55
  return;
979
56
  }
980
57
  const normalized = value.trim();
981
58
  return normalized.length > 0 ? normalized : void 0;
982
59
  }
983
- function readStringArray2(value) {
60
+ function readStringArray(value) {
984
61
  if (!Array.isArray(value)) {
985
62
  return [];
986
63
  }
987
- return value.map((entry) => readNonEmptyString2(entry)).filter((entry) => Boolean(entry));
64
+ return value.map((entry) => readNonEmptyString(entry)).filter((entry) => Boolean(entry));
988
65
  }
989
- function readMetadata2(topic) {
66
+ function readMetadata(topic) {
990
67
  return topic.metadata && typeof topic.metadata === "object" ? topic.metadata : {};
991
68
  }
992
- function readLegacyProjectId2(value) {
69
+ function readLegacyProjectId(value) {
993
70
  if (!value) {
994
71
  return;
995
72
  }
996
- return readNonEmptyString2(value[LEGACY_SCOPE_FIELD2]);
73
+ return readNonEmptyString(value[LEGACY_SCOPE_FIELD]);
997
74
  }
998
- function coerceVisibility2(value) {
75
+ function coerceVisibility(value) {
999
76
  return value === "private" || value === "team" || value === "firm" || value === "external" || value === "public" ? value : void 0;
1000
77
  }
1001
- function coerceStatus2(value) {
78
+ function coerceStatus(value) {
1002
79
  return value === "active" || value === "archived" || value === "watching" ? value : void 0;
1003
80
  }
1004
- function mapProjectType2(topic, metadata) {
1005
- const explicit = readNonEmptyString2(metadata.projectType);
81
+ function mapProjectType(topic, metadata) {
82
+ const explicit = readNonEmptyString(metadata.projectType);
1006
83
  if (explicit) {
1007
84
  return explicit;
1008
85
  }
1009
86
  if (topic.type === "theme") {
1010
87
  return "thematic";
1011
88
  }
1012
- return readNonEmptyString2(topic.type) || "general";
89
+ return readNonEmptyString(topic.type) || "general";
1013
90
  }
1014
- function isProjectLikeTopic2(topic) {
1015
- const metadata = readMetadata2(topic);
1016
- return topic.type === "theme" || topic.type === "thematic" || topic.type === "deal" || topic.type === "monitoring" || readLegacyProjectId2(topic) !== void 0 || readNonEmptyString2(metadata.projectType) !== void 0;
91
+ function isProjectLikeTopic(topic) {
92
+ const metadata = readMetadata(topic);
93
+ return topic.type === "theme" || topic.type === "thematic" || topic.type === "deal" || topic.type === "monitoring" || readLegacyProjectId(topic) !== void 0 || readNonEmptyString(metadata.projectType) !== void 0;
1017
94
  }
1018
95
  function isMissingLucernChildComponentError(error) {
1019
96
  const message = error instanceof Error ? error.message : String(error);
@@ -1021,7 +98,7 @@ function isMissingLucernChildComponentError(error) {
1021
98
  'Child component ComponentName(Identifier("lucern")) not found'
1022
99
  ) || message.includes("Child component") && message.includes("lucern") && message.includes("not found");
1023
100
  }
1024
- async function resolveTopicDoc2(ctx, scopeId) {
101
+ async function resolveTopicDoc(ctx, scopeId) {
1025
102
  if (ctx?.db && typeof ctx.db.get === "function") {
1026
103
  try {
1027
104
  const directTopic = await ctx.db.get(scopeId);
@@ -1035,7 +112,7 @@ async function resolveTopicDoc2(ctx, scopeId) {
1035
112
  return null;
1036
113
  }
1037
114
  try {
1038
- const topic = await ctx.runQuery(api2.topics.get, {
115
+ const topic = await ctx.runQuery(api.topics.get, {
1039
116
  id: String(scopeId)
1040
117
  });
1041
118
  if (topic?.name !== void 0 && topic?.type !== void 0) {
@@ -1044,7 +121,7 @@ async function resolveTopicDoc2(ctx, scopeId) {
1044
121
  } catch {
1045
122
  }
1046
123
  try {
1047
- const topic = await ctx.runQuery(api2.topics.getByLegacyScopeId, {
124
+ const topic = await ctx.runQuery(api.topics.getByLegacyScopeId, {
1048
125
  projectId: String(scopeId)
1049
126
  });
1050
127
  if (topic?.name !== void 0 && topic?.type !== void 0) {
@@ -1054,14 +131,14 @@ async function resolveTopicDoc2(ctx, scopeId) {
1054
131
  }
1055
132
  return null;
1056
133
  }
1057
- function materializeTopicProjectOverlay2(topic, idMode = "legacy") {
1058
- const metadata = readMetadata2(topic);
134
+ function materializeTopicProjectOverlay(topic, idMode = "legacy") {
135
+ const metadata = readMetadata(topic);
1059
136
  const topicId = String(topic._id);
1060
- const legacyProjectId = readLegacyProjectId2(topic) || readLegacyProjectId2(metadata) || readNonEmptyString2(metadata.legacyProjectId);
137
+ const legacyProjectId = readLegacyProjectId(topic) || readLegacyProjectId(metadata) || readNonEmptyString(metadata.legacyProjectId);
1061
138
  const storageProjectId = legacyProjectId || topicId;
1062
139
  const outwardId = idMode === "topic" ? topicId : storageProjectId;
1063
- const visibility = coerceVisibility2(topic.visibility) || coerceVisibility2(metadata.visibility) || "private";
1064
- const status = coerceStatus2(topic.status) || coerceStatus2(metadata.status) || "active";
140
+ const visibility = coerceVisibility(topic.visibility) || coerceVisibility(metadata.visibility) || "private";
141
+ const status = coerceStatus(topic.status) || coerceStatus(metadata.status) || "active";
1065
142
  const createdAt = typeof topic.createdAt === "number" ? topic.createdAt : typeof topic._creationTime === "number" ? topic._creationTime : 0;
1066
143
  const updatedAt = typeof topic.updatedAt === "number" ? topic.updatedAt : typeof metadata.updatedAt === "number" ? metadata.updatedAt : createdAt;
1067
144
  return {
@@ -1071,16 +148,16 @@ function materializeTopicProjectOverlay2(topic, idMode = "legacy") {
1071
148
  topicId,
1072
149
  storageProjectId,
1073
150
  legacyProjectId,
1074
- name: readNonEmptyString2(topic.name) || "Untitled Theme",
1075
- type: mapProjectType2(topic, metadata),
1076
- description: readNonEmptyString2(topic.description),
1077
- ownerId: readNonEmptyString2(metadata.ownerId) || readNonEmptyString2(topic.createdBy) || "system",
1078
- sharedWith: readStringArray2(metadata.sharedWith),
151
+ name: readNonEmptyString(topic.name) || "Untitled Theme",
152
+ type: mapProjectType(topic, metadata),
153
+ description: readNonEmptyString(topic.description),
154
+ ownerId: readNonEmptyString(metadata.ownerId) || readNonEmptyString(topic.createdBy) || "system",
155
+ sharedWith: readStringArray(metadata.sharedWith),
1079
156
  visibility,
1080
- tenantId: readNonEmptyString2(topic.tenantId) || readNonEmptyString2(metadata.tenantId),
1081
- workspaceId: readNonEmptyString2(topic.workspaceId) || readNonEmptyString2(metadata.workspaceId),
157
+ tenantId: readNonEmptyString(topic.tenantId) || readNonEmptyString(metadata.tenantId),
158
+ workspaceId: readNonEmptyString(topic.workspaceId) || readNonEmptyString(metadata.workspaceId),
1082
159
  status,
1083
- tags: readStringArray2(metadata.tags),
160
+ tags: readStringArray(metadata.tags),
1084
161
  chatCount: typeof metadata.chatCount === "number" ? metadata.chatCount : 0,
1085
162
  artifactCount: typeof metadata.artifactCount === "number" ? metadata.artifactCount : 0,
1086
163
  lastActivityAt: typeof metadata.lastActivityAt === "number" ? metadata.lastActivityAt : updatedAt,
@@ -1089,17 +166,17 @@ function materializeTopicProjectOverlay2(topic, idMode = "legacy") {
1089
166
  updatedAt
1090
167
  };
1091
168
  }
1092
- async function resolveTopicProjectOverlay2(ctx, scopeId, options = {}) {
1093
- const topic = await resolveTopicDoc2(ctx, scopeId);
169
+ async function resolveTopicProjectOverlay(ctx, scopeId, options = {}) {
170
+ const topic = await resolveTopicDoc(ctx, scopeId);
1094
171
  if (!topic) {
1095
172
  return null;
1096
173
  }
1097
- if (options.projectLikeOnly !== false && !isProjectLikeTopic2(topic)) {
174
+ if (options.projectLikeOnly !== false && !isProjectLikeTopic(topic)) {
1098
175
  return null;
1099
176
  }
1100
- return materializeTopicProjectOverlay2(topic, options.idMode);
177
+ return materializeTopicProjectOverlay(topic, options.idMode);
1101
178
  }
1102
- async function listTopicProjectOverlays2(ctx, options = {}) {
179
+ async function listTopicProjectOverlays(ctx, options = {}) {
1103
180
  let allTopics = [];
1104
181
  if (ctx?.db?.query && typeof ctx.db.query === "function") {
1105
182
  try {
@@ -1109,18 +186,18 @@ async function listTopicProjectOverlays2(ctx, options = {}) {
1109
186
  }
1110
187
  }
1111
188
  if (allTopics.length === 0 && typeof ctx.runQuery === "function") {
1112
- allTopics = (await ctx.runQuery(api2.topics.list, {}) ?? []) || [];
189
+ allTopics = (await ctx.runQuery(api.topics.list, {}) ?? []) || [];
1113
190
  }
1114
191
  return allTopics.filter(
1115
- (topic) => options.projectLikeOnly === false || isProjectLikeTopic2(topic)
1116
- ).map((topic) => materializeTopicProjectOverlay2(topic, options.idMode));
192
+ (topic) => options.projectLikeOnly === false || isProjectLikeTopic(topic)
193
+ ).map((topic) => materializeTopicProjectOverlay(topic, options.idMode));
1117
194
  }
1118
195
  async function patchTopicProjectOverlay(ctx, scopeId, value) {
1119
- const topic = await resolveTopicDoc2(ctx, scopeId);
196
+ const topic = await resolveTopicDoc(ctx, scopeId);
1120
197
  if (!topic) {
1121
198
  return null;
1122
199
  }
1123
- const nextMetadata = { ...readMetadata2(topic) };
200
+ const nextMetadata = { ...readMetadata(topic) };
1124
201
  const patch = {};
1125
202
  const topicUpdateArgs = {
1126
203
  id: String(topic._id)
@@ -1145,7 +222,7 @@ async function patchTopicProjectOverlay(ctx, scopeId, value) {
1145
222
  `patchTopicProjectOverlay cannot mutate ${key} via component-owned topics`
1146
223
  );
1147
224
  case "status": {
1148
- const status = coerceStatus2(rawValue);
225
+ const status = coerceStatus(rawValue);
1149
226
  if (status) {
1150
227
  patch.status = status;
1151
228
  topicUpdateArgs.status = status;
@@ -1153,7 +230,7 @@ async function patchTopicProjectOverlay(ctx, scopeId, value) {
1153
230
  break;
1154
231
  }
1155
232
  case "visibility": {
1156
- const visibility = coerceVisibility2(rawValue);
233
+ const visibility = coerceVisibility(rawValue);
1157
234
  if (visibility) {
1158
235
  patch.visibility = visibility;
1159
236
  topicUpdateArgs.visibility = visibility;
@@ -1161,7 +238,7 @@ async function patchTopicProjectOverlay(ctx, scopeId, value) {
1161
238
  break;
1162
239
  }
1163
240
  case "type": {
1164
- const projectType = readNonEmptyString2(rawValue);
241
+ const projectType = readNonEmptyString(rawValue);
1165
242
  if (projectType) {
1166
243
  nextMetadata.projectType = projectType;
1167
244
  } else {
@@ -1185,7 +262,7 @@ async function patchTopicProjectOverlay(ctx, scopeId, value) {
1185
262
  topicUpdateArgs.metadata = nextMetadata;
1186
263
  if (typeof ctx.runMutation === "function") {
1187
264
  try {
1188
- await ctx.runMutation(api2.topics.update, topicUpdateArgs);
265
+ await ctx.runMutation(api.topics.update, topicUpdateArgs);
1189
266
  } catch (error) {
1190
267
  if (!isMissingLucernChildComponentError(error) || !ctx?.db || typeof ctx.db.patch !== "function") {
1191
268
  throw error;
@@ -1199,7 +276,7 @@ async function patchTopicProjectOverlay(ctx, scopeId, value) {
1199
276
  "Cannot patch topic without component adapter (ctx.runMutation unavailable)"
1200
277
  );
1201
278
  }
1202
- return materializeTopicProjectOverlay2(
279
+ return materializeTopicProjectOverlay(
1203
280
  {
1204
281
  ...topic,
1205
282
  ...patch,
@@ -1232,10 +309,10 @@ async function patchProjectWithTolerance(ctx, projectId, value) {
1232
309
  });
1233
310
  }
1234
311
  }
1235
- function defaultResolvers2() {
312
+ function defaultResolvers() {
1236
313
  return {
1237
314
  async getProject(ctx, projectId) {
1238
- return await resolveTopicProjectOverlay2(ctx, projectId, {
315
+ return await resolveTopicProjectOverlay(ctx, projectId, {
1239
316
  idMode: "legacy",
1240
317
  projectLikeOnly: false
1241
318
  });
@@ -1244,7 +321,7 @@ function defaultResolvers2() {
1244
321
  await patchProjectWithTolerance(ctx, projectId, value);
1245
322
  },
1246
323
  async listTopics(ctx) {
1247
- return await listTopicProjectOverlays2(ctx, {
324
+ return await listTopicProjectOverlays(ctx, {
1248
325
  idMode: "legacy"
1249
326
  });
1250
327
  },
@@ -1253,24 +330,24 @@ function defaultResolvers2() {
1253
330
  }
1254
331
  };
1255
332
  }
1256
- var resolverOverrides2 = {};
333
+ var resolverOverrides = {};
1257
334
  function resolveGraphPrimitivesAppResolvers(_ctx) {
1258
335
  return {
1259
- ...defaultResolvers2(),
1260
- ...resolverOverrides2
336
+ ...defaultResolvers(),
337
+ ...resolverOverrides
1261
338
  };
1262
339
  }
1263
- var LEGACY_SCOPE_FIELD3 = "graphScopeProjectId";
340
+ var LEGACY_SCOPE_FIELD2 = "graphScopeProjectId";
1264
341
  function asMappedProjectId(topic) {
1265
342
  if (!topic) {
1266
343
  return;
1267
344
  }
1268
- const directLegacyProjectId = normalizeScopeValue(topic[LEGACY_SCOPE_FIELD3]);
345
+ const directLegacyProjectId = normalizeScopeValue(topic[LEGACY_SCOPE_FIELD2]);
1269
346
  if (directLegacyProjectId) {
1270
347
  return directLegacyProjectId;
1271
348
  }
1272
349
  const metadata = topic.metadata || {};
1273
- const candidate = metadata[LEGACY_SCOPE_FIELD3] || metadata.legacyProjectId || metadata.projectId || metadata.scopeProjectId;
350
+ const candidate = metadata[LEGACY_SCOPE_FIELD2] || metadata.legacyProjectId || metadata.projectId || metadata.scopeProjectId;
1274
351
  return candidate ? candidate : void 0;
1275
352
  }
1276
353
  function normalizeScopeValue(value) {
@@ -1299,7 +376,7 @@ async function findTopicsByScopeAlias(ctx, scopeId) {
1299
376
  try {
1300
377
  return await ctx.db.query("topics").withIndex(
1301
378
  "by_graph_scope_project",
1302
- (q) => q.eq(LEGACY_SCOPE_FIELD3, scopeId)
379
+ (q) => q.eq(LEGACY_SCOPE_FIELD2, scopeId)
1303
380
  ).collect();
1304
381
  } catch {
1305
382
  const topics = await ctx.db.query("topics").collect();
@@ -1315,7 +392,7 @@ async function tryResolveHostTopicById(ctx, topicId) {
1315
392
  return null;
1316
393
  }
1317
394
  try {
1318
- return await ctx.runQuery(api2.topics.get, {
395
+ return await ctx.runQuery(api.topics.get, {
1319
396
  id: topicId
1320
397
  }) ?? null;
1321
398
  } catch {
@@ -1327,7 +404,7 @@ async function tryResolveHostTopicByLegacyScope(ctx, legacyScopeId) {
1327
404
  return null;
1328
405
  }
1329
406
  try {
1330
- return await ctx.runQuery(api2.topics.getByLegacyScopeId, {
407
+ return await ctx.runQuery(api.topics.getByLegacyScopeId, {
1331
408
  projectId: legacyScopeId
1332
409
  }) ?? null;
1333
410
  } catch {
@@ -1446,597 +523,6 @@ var optionalScopeArgs = {
1446
523
  projectId: v.optional(v.string()),
1447
524
  topicId: v.optional(v.string())
1448
525
  };
1449
- v.number();
1450
- v.union(
1451
- v.literal("very_high"),
1452
- // 0.9+
1453
- v.literal("high"),
1454
- // 0.7-0.9
1455
- v.literal("medium"),
1456
- // 0.4-0.7
1457
- v.literal("low"),
1458
- // 0.2-0.4
1459
- v.literal("very_low")
1460
- // 0-0.2
1461
- );
1462
- v.union(
1463
- v.literal(1),
1464
- // Critical
1465
- v.literal(2),
1466
- // High
1467
- v.literal(3),
1468
- // Medium
1469
- v.literal(4),
1470
- // Low
1471
- v.literal(5)
1472
- // Backlog
1473
- );
1474
- v.union(
1475
- v.literal("critical"),
1476
- v.literal("high"),
1477
- v.literal("medium"),
1478
- v.literal("low"),
1479
- v.literal("backlog")
1480
- );
1481
- v.union(
1482
- v.literal("active"),
1483
- v.literal("paused"),
1484
- v.literal("completed"),
1485
- v.literal("archived")
1486
- );
1487
- v.union(
1488
- v.literal("pending"),
1489
- v.literal("processing"),
1490
- v.literal("completed"),
1491
- v.literal("failed")
1492
- );
1493
- v.object({
1494
- crunchbaseId: v.optional(v.string()),
1495
- linkedinUrl: v.optional(v.string()),
1496
- pitchbookId: v.optional(v.string()),
1497
- twitterUrl: v.optional(v.string()),
1498
- domain: v.optional(v.string())
1499
- });
1500
- var sourceType = v.union(
1501
- v.literal("proprietary"),
1502
- // Internal Stack research
1503
- v.literal("primary"),
1504
- // Direct interviews, calls
1505
- v.literal("secondary"),
1506
- // Published sources
1507
- v.literal("ai_generated"),
1508
- // AI-synthesized
1509
- v.literal("user_input"),
1510
- // Manual user entry
1511
- v.literal("inferred")
1512
- // System inference
1513
- );
1514
- v.object({
1515
- sourceType: v.optional(sourceType),
1516
- sourceId: v.optional(v.string()),
1517
- // Reference to source entity
1518
- sourceUrl: v.optional(v.string()),
1519
- sourceDate: v.optional(v.number()),
1520
- sourceName: v.optional(v.string())
1521
- });
1522
- v.object({
1523
- cursor: v.optional(v.string()),
1524
- limit: v.optional(v.number())
1525
- });
1526
- v.object({
1527
- hasMore: v.boolean(),
1528
- nextCursor: v.optional(v.string()),
1529
- totalCount: v.optional(v.number())
1530
- });
1531
- var richTextContent = v.object({
1532
- type: v.literal("doc"),
1533
- content: looseJsonArray
1534
- });
1535
- v.union(v.string(), richTextContent);
1536
- v.object({
1537
- promptTokens: v.optional(v.number()),
1538
- completionTokens: v.optional(v.number()),
1539
- totalTokens: v.optional(v.number())
1540
- });
1541
- v.object({
1542
- fileName: v.optional(v.string()),
1543
- fileSize: v.optional(v.number()),
1544
- mimeType: v.optional(v.string()),
1545
- storageId: v.optional(v.id("_storage")),
1546
- externalUrl: v.optional(v.string())
1547
- });
1548
-
1549
- // ../schema-management/src/spine/tables/epistemicNodes.ts
1550
- var nodeType = v.union(
1551
- // --- L4: Audit Targets (decisions, outcomes) ---
1552
- v.literal("decision"),
1553
- // Investment decision with knowledge horizon snapshot
1554
- // --- L3: Traversal Anchors (epistemic structure) ---
1555
- v.literal("belief"),
1556
- // Structured conviction (immutable formulation)
1557
- v.literal("question"),
1558
- // Unit of uncertainty
1559
- v.literal("theme"),
1560
- // Investment thesis / conviction cluster
1561
- v.literal("deal"),
1562
- // Investment evaluation process
1563
- v.literal("topic"),
1564
- // Hierarchical knowledge container
1565
- // --- L2: Compression Boundary (minimum reasoning unit) ---
1566
- v.literal("claim"),
1567
- // Atomic assertion that can be true/false
1568
- v.literal("evidence"),
1569
- // Interpreted signal linked to beliefs
1570
- v.literal("synthesis"),
1571
- // Primers, deep research
1572
- v.literal("answer"),
1573
- // Immutable answer snapshot for a question
1574
- // --- L1: Terminal Leaves (non-traversable, grounding) ---
1575
- v.literal("atomic_fact"),
1576
- // Raw fact from source (not interpreted)
1577
- v.literal("excerpt"),
1578
- // Direct quote from source document
1579
- v.literal("source"),
1580
- // News, documents, transcripts
1581
- // --- Ontological Entities (things in the world) ---
1582
- v.literal("company"),
1583
- // Organization (subtype: private, corporate, portfolio)
1584
- v.literal("person"),
1585
- // Individual (founder, expert, LP, contact)
1586
- v.literal("investor"),
1587
- // Investment entity (subtype: vc, lp, cvc, pe, family_office, angel)
1588
- v.literal("function"),
1589
- // What a company does (from classifier)
1590
- v.literal("value_chain")
1591
- // Market structure / value flow
1592
- );
1593
- var epistemicLayer = v.union(
1594
- v.literal("L4"),
1595
- // Decisions, outcomes - audit targets
1596
- v.literal("L3"),
1597
- // Beliefs, questions, themes - traversal anchors
1598
- v.literal("L2"),
1599
- // Claims, evidence, synthesis - compression boundary
1600
- v.literal("L1"),
1601
- // Atomic facts, excerpts, sources - terminal leaves
1602
- v.literal("ontological"),
1603
- // Companies, people, etc - not epistemic
1604
- v.literal("organizational")
1605
- // Topics, lenses, worktrees — structural containers
1606
- );
1607
- var nodeStatus = v.union(
1608
- v.literal("active"),
1609
- v.literal("superseded"),
1610
- // Replaced by newer version
1611
- v.literal("archived"),
1612
- v.literal("deleted")
1613
- );
1614
- var sourceType2 = v.union(
1615
- v.literal("human"),
1616
- // User created directly
1617
- v.literal("ai_extracted"),
1618
- // LLM extracted from a source
1619
- v.literal("ai_generated"),
1620
- // LLM synthesized/created
1621
- v.literal("imported"),
1622
- // External system import
1623
- v.literal("system"),
1624
- // System-generated (migrations, classifiers)
1625
- v.literal("verified"),
1626
- // Human-verified source
1627
- v.literal("proprietary")
1628
- // Proprietary/internal data
1629
- );
1630
- var verificationStatus = v.union(
1631
- v.literal("unverified"),
1632
- v.literal("human_verified"),
1633
- v.literal("ai_verified"),
1634
- v.literal("contradicted"),
1635
- v.literal("outdated")
1636
- );
1637
- var syncStatus = v.union(
1638
- v.literal("synced"),
1639
- // Node and edges fully synced to Neo4j
1640
- v.literal("pending_edges"),
1641
- // Node created, edges being created
1642
- v.literal("edge_creation_failed")
1643
- // Edge creation failed, needs retry
1644
- );
1645
- var audienceLabel = v.string();
1646
- var sensitivityTier = v.union(
1647
- v.literal("low"),
1648
- v.literal("medium"),
1649
- v.literal("high"),
1650
- v.literal("restricted")
1651
- );
1652
- var exportClass = v.union(
1653
- v.literal("internal_only"),
1654
- v.literal("client_safe"),
1655
- v.literal("public_safe"),
1656
- v.literal("restricted")
1657
- );
1658
- var anonymizationClass = v.union(
1659
- v.literal("none"),
1660
- v.literal("standard"),
1661
- v.literal("strict")
1662
- );
1663
- var epistemicStatus = v.union(
1664
- v.literal("hypothesis"),
1665
- // Initial conjecture, low evidence
1666
- v.literal("emerging"),
1667
- // Building evidence, gaining traction
1668
- v.literal("established"),
1669
- // Well-evidenced, core to thesis
1670
- v.literal("challenged"),
1671
- // Contradicting evidence appeared
1672
- v.literal("assumption"),
1673
- // Taken as given, not actively tested
1674
- v.literal("deprecated")
1675
- // Superseded or abandoned
1676
- );
1677
- var beliefStatus = v.union(
1678
- v.literal("assumption"),
1679
- v.literal("hypothesis"),
1680
- v.literal("belief"),
1681
- v.literal("fact")
1682
- );
1683
- var reversibility = v.union(
1684
- v.literal("irreversible"),
1685
- // One-way door decision
1686
- v.literal("hard_to_reverse"),
1687
- // Significant cost to undo
1688
- v.literal("reversible"),
1689
- // Can change course with moderate effort
1690
- v.literal("trivial")
1691
- // Easy to adjust
1692
- );
1693
- var predictionOutcome = v.union(
1694
- v.literal("pending"),
1695
- v.literal("confirmed"),
1696
- v.literal("disconfirmed"),
1697
- v.literal("partial"),
1698
- v.literal("expired")
1699
- );
1700
- var predictionMeta = v.object({
1701
- isPrediction: v.boolean(),
1702
- registeredAt: v.number(),
1703
- // When prediction was made
1704
- expectedBy: v.optional(v.number()),
1705
- // When we expect resolution
1706
- outcome: v.optional(predictionOutcome),
1707
- outcomeRecordedAt: v.optional(v.number()),
1708
- outcomeEvidenceId: v.optional(v.string()),
1709
- // globalId of confirming evidence
1710
- confidenceAtPrediction: v.optional(v.number()),
1711
- // 0-1
1712
- actualVsPredicted: v.optional(v.string())
1713
- // Notes on how outcome compared
1714
- });
1715
- var methodology = v.union(
1716
- // Primary Research (high value)
1717
- v.literal("primary_research"),
1718
- // Direct investigation
1719
- v.literal("expert_interview"),
1720
- // Expert call/interview
1721
- v.literal("customer_interview"),
1722
- // Customer research
1723
- v.literal("field_observation"),
1724
- // On-site observation
1725
- v.literal("proprietary_data"),
1726
- // Internal data analysis
1727
- // Secondary Research
1728
- v.literal("desk_research"),
1729
- // Public sources
1730
- v.literal("regulatory_filing"),
1731
- // SEC, regulatory docs
1732
- v.literal("news_article"),
1733
- // News/press
1734
- v.literal("academic_paper"),
1735
- // Academic research
1736
- // AI-Assisted
1737
- v.literal("ai_synthesis"),
1738
- // AI-generated synthesis
1739
- v.literal("ai_extraction")
1740
- // AI-extracted from source
1741
- );
1742
- var informationAsymmetry = v.union(
1743
- v.literal("proprietary"),
1744
- // Only we have this
1745
- v.literal("early"),
1746
- // We're early but others will get it
1747
- v.literal("common")
1748
- // Everyone has access
1749
- );
1750
- var temporalNature = v.union(
1751
- v.literal("factual"),
1752
- // Resolved outcome. Grounded in reality.
1753
- v.literal("forecast"),
1754
- // Prediction. Will resolve. Discounted weight.
1755
- v.literal("unknown")
1756
- // Not yet classified.
1757
- );
1758
- var questionType = v.union(
1759
- v.literal("validation"),
1760
- // Does evidence support this belief?
1761
- v.literal("falsification"),
1762
- // What would prove this belief wrong?
1763
- v.literal("assumption_probe"),
1764
- // Is this unstated assumption true?
1765
- v.literal("prediction_test"),
1766
- // Will this predicted outcome occur?
1767
- v.literal("counterfactual"),
1768
- // What would we expect if X were false?
1769
- v.literal("discovery"),
1770
- // What don't we know yet?
1771
- v.literal("clarification"),
1772
- // What does X actually mean?
1773
- v.literal("comparison"),
1774
- // How does X compare to Y?
1775
- v.literal("causal"),
1776
- // What caused X?
1777
- v.literal("mechanism"),
1778
- // How does X work?
1779
- v.literal("general")
1780
- // Unclassified
1781
- );
1782
- var questionPriority = v.union(
1783
- v.literal("critical"),
1784
- // Blocks decision-making
1785
- v.literal("high"),
1786
- // Important for thesis
1787
- v.literal("medium"),
1788
- // Would be nice to know
1789
- v.literal("low")
1790
- // Background/curiosity
1791
- );
1792
- var answerQuality = v.union(
1793
- v.literal("definitive"),
1794
- // Clear, well-supported
1795
- v.literal("strong"),
1796
- // Good evidence, high confidence
1797
- v.literal("moderate"),
1798
- // Some evidence
1799
- v.literal("weak"),
1800
- // Limited evidence
1801
- v.literal("speculative"),
1802
- // Mostly conjecture
1803
- v.literal("unanswered")
1804
- // No answer yet
1805
- );
1806
- var consensusView = v.union(
1807
- v.literal("aligned"),
1808
- // We agree with market consensus
1809
- v.literal("ahead_of"),
1810
- // We see this before consensus does
1811
- v.literal("contrarian"),
1812
- // We actively disagree with consensus
1813
- v.literal("orthogonal"),
1814
- // We're looking at something consensus isn't discussing
1815
- v.literal("unknown")
1816
- // We don't know what consensus thinks
1817
- );
1818
- var themeConviction = v.union(
1819
- v.literal("high"),
1820
- // Strong conviction, actively deploying
1821
- v.literal("medium"),
1822
- // Building conviction
1823
- v.literal("low"),
1824
- // Exploring, not convicted
1825
- v.literal("negative")
1826
- // Actively avoiding
1827
- );
1828
- var decisionType = v.union(
1829
- v.literal("invest"),
1830
- v.literal("pass"),
1831
- v.literal("follow_on"),
1832
- v.literal("exit"),
1833
- v.literal("deep_dive"),
1834
- v.literal("monitor"),
1835
- v.literal("deprioritize"),
1836
- v.literal("thesis_adopt"),
1837
- v.literal("thesis_revise"),
1838
- v.literal("thesis_abandon")
1839
- );
1840
- var decisionOutcome = v.union(
1841
- v.literal("pending"),
1842
- v.literal("successful"),
1843
- v.literal("unsuccessful"),
1844
- v.literal("mixed"),
1845
- v.literal("unknown")
1846
- );
1847
- var externalIds2 = v.object({
1848
- crunchbase: v.optional(v.string()),
1849
- linkedin: v.optional(v.string()),
1850
- pitchbook: v.optional(v.string()),
1851
- twitter: v.optional(v.string()),
1852
- website: v.optional(v.string())
1853
- });
1854
- defineTable({
1855
- // === IDENTITY ===
1856
- globalId: v.string(),
1857
- // UUID - survives migration to Neo4j
1858
- // === TYPE ===
1859
- nodeType,
1860
- // === EPISTEMIC LAYER ===
1861
- epistemicLayer: v.optional(epistemicLayer),
1862
- // === SUBTYPE (for typed entities) ===
1863
- subtype: v.optional(v.string()),
1864
- // company: private|corporate|portfolio, investor: vc|lp|cvc|pe|family_office|angel
1865
- // === CONTENT ===
1866
- canonicalText: v.string(),
1867
- // The core content (belief statement, company name, etc.)
1868
- contentHash: v.string(),
1869
- // SHA256(nodeType + canonicalText) for deduplication
1870
- // Extended content (for sources/syntheses)
1871
- content: v.optional(v.string()),
1872
- // Full text for documents/articles
1873
- contentType: v.optional(v.string()),
1874
- // "markdown", "html", "pdf", "text"
1875
- // === METADATA ===
1876
- title: v.optional(v.string()),
1877
- // Display title
1878
- tags: v.optional(v.array(v.string())),
1879
- domain: v.optional(v.string()),
1880
- // For companies: website domain
1881
- // Type-specific metadata (flexible object - LEGACY)
1882
- // New code should use the typed fields below when available
1883
- metadata: v.optional(looseJsonObject),
1884
- // === POLICY / ENTITLEMENT ===
1885
- tenantId: v.optional(v.string()),
1886
- workspaceId: v.optional(v.string()),
1887
- ownerPrincipalId: v.optional(v.string()),
1888
- audienceLabel: v.optional(audienceLabel),
1889
- policyTags: v.optional(v.array(v.string())),
1890
- sensitivityTier: v.optional(sensitivityTier),
1891
- exportClass: v.optional(exportClass),
1892
- anonymizationClass: v.optional(anonymizationClass),
1893
- // === PUBLICATION (visibility-based, not copy-based) ===
1894
- // Publication expands who can see a workspace-local node — the node stays
1895
- // in its workspace, like a microservice exposing part of its API surface.
1896
- // Rules-based: pack/tenant-level publicationRules auto-evaluate on
1897
- // confidence changes and node creation. No manual click-by-click.
1898
- publicationStatus: v.optional(
1899
- v.union(
1900
- v.literal("unpublished"),
1901
- // Default: workspace-local only
1902
- v.literal("published"),
1903
- // Visible at tenant scope (rules matched)
1904
- v.literal("suppressed")
1905
- // Manually blocked even if rules match
1906
- )
1907
- ),
1908
- publishedAt: v.optional(v.number()),
1909
- // When publication status last changed to published
1910
- publishedBy: v.optional(v.string()),
1911
- // userId or "system:publication_rules" for auto-publish
1912
- // === TYPED METADATA FIELDS ===
1913
- // --- Belief ---
1914
- // Belief type — validated against schemaEnumConfig category "belief_type"
1915
- // Platform core: hypothesis, belief, principle, invariant, assumption,
1916
- // tenet, prior, preference, goal, forecast
1917
- beliefType: v.optional(v.string()),
1918
- beliefStatus: v.optional(beliefStatus),
1919
- epistemicStatus: v.optional(epistemicStatus),
1920
- reversibility: v.optional(reversibility),
1921
- predictionMeta: v.optional(predictionMeta),
1922
- // Consensus tracking (for non-consensus detection)
1923
- consensusView: v.optional(consensusView),
1924
- consensusConfidence: v.optional(v.number()),
1925
- // 0-1: What we think consensus confidence is
1926
- consensusSource: v.optional(v.string()),
1927
- // Where we got the consensus view (twitter, reports, etc.)
1928
- // --- Evidence ---
1929
- methodology: v.optional(methodology),
1930
- informationAsymmetry: v.optional(informationAsymmetry),
1931
- temporalNature: v.optional(temporalNature),
1932
- // --- Question ---
1933
- questionType: v.optional(questionType),
1934
- questionPriority: v.optional(questionPriority),
1935
- answerQuality: v.optional(answerQuality),
1936
- // --- Theme ---
1937
- themeConviction: v.optional(themeConviction),
1938
- // Market timing (for "early on theme" detection)
1939
- marketAwarenessDate: v.optional(v.number()),
1940
- // When this theme became broadly discussed
1941
- marketAwarenessSource: v.optional(v.string()),
1942
- // How we know (first major report, twitter volume spike, etc.)
1943
- earlySignalIds: v.optional(v.array(v.string())),
1944
- // globalIds of evidence we had before market awareness
1945
- // --- Decision ---
1946
- decisionType: v.optional(decisionType),
1947
- decisionOutcome: v.optional(decisionOutcome),
1948
- // === EXTERNAL IDS (for ontological entities) ===
1949
- externalIds: v.optional(externalIds2),
1950
- // === PROVENANCE ===
1951
- sourceType: sourceType2,
1952
- aiProvider: v.optional(v.string()),
1953
- // "claude", "gemini", "gpt-4", etc.
1954
- extractedFromNodeId: v.optional(v.id("epistemicNodes")),
1955
- // Quick reference to source
1956
- // === EXTRACTION CONTEXT ===
1957
- extractionModel: v.optional(v.string()),
1958
- // "claude-sonnet-4-20250514"
1959
- extractionPromptName: v.optional(v.string()),
1960
- // "lucern/extract-evidence"
1961
- extractionPromptVersion: v.optional(v.number()),
1962
- extractionTemperature: v.optional(v.number()),
1963
- extractionLangfuseTraceId: v.optional(v.string()),
1964
- // === GROUNDING VERIFICATION ===
1965
- groundingVerified: v.optional(v.boolean()),
1966
- groundingConfidence: v.optional(v.number()),
1967
- // 0-1 match quality
1968
- groundingMatchedText: v.optional(v.string()),
1969
- // Actual text from source
1970
- groundingStartOffset: v.optional(v.number()),
1971
- groundingEndOffset: v.optional(v.number()),
1972
- groundingRejectionReason: v.optional(v.string()),
1973
- // === CONFIDENCE & VERIFICATION ===
1974
- confidence: v.optional(v.number()),
1975
- // 0-1 projected probability P(x) = b + a*u
1976
- verificationStatus: v.optional(verificationStatus),
1977
- // === SL OPINION (Subjective Logic — Kernel v2) ===
1978
- // Replaces scalar confidence with rich epistemic state.
1979
- // b + d + u = 1. P(x) = b + a*u is stored in `confidence` for backward compat.
1980
- opinion_b: v.optional(v.number()),
1981
- // Belief: evidence FOR (0-1)
1982
- opinion_d: v.optional(v.number()),
1983
- // Disbelief: evidence AGAINST (0-1)
1984
- opinion_u: v.optional(v.number()),
1985
- // Uncertainty: absence of evidence (0-1)
1986
- opinion_a: v.optional(v.number()),
1987
- // Base rate / prior probability (0-1)
1988
- tupleContradicted: v.optional(v.boolean()),
1989
- // Single-belief tuple-space contradiction flag
1990
- // === LIFECYCLE ===
1991
- status: nodeStatus,
1992
- supersededBy: v.optional(v.id("epistemicNodes")),
1993
- // === OWNERSHIP ===
1994
- topicId: v.optional(v.string()),
1995
- // Canonical scope container (topic-first model)
1996
- projectId: v.optional(v.string()),
1997
- // DEPRECATED: Use belongs_to edges
1998
- createdBy: v.string(),
1999
- // Clerk user ID
2000
- createdAt: v.number(),
2001
- updatedAt: v.number(),
2002
- // === NEO4J SYNC STATUS ===
2003
- syncStatus: v.optional(syncStatus),
2004
- syncError: v.optional(v.string())
2005
- // Error message if sync failed
2006
- }).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", {
2007
- searchField: "canonicalText",
2008
- filterFields: ["nodeType", "projectId", "topicId", "status"]
2009
- });
2010
- function getLayerForNodeType(type) {
2011
- switch (type) {
2012
- case "decision":
2013
- return "L4";
2014
- case "belief":
2015
- case "question":
2016
- case "theme":
2017
- case "deal":
2018
- return "L3";
2019
- case "claim":
2020
- case "evidence":
2021
- case "synthesis":
2022
- case "answer":
2023
- return "L2";
2024
- case "atomic_fact":
2025
- case "excerpt":
2026
- case "source":
2027
- return "L1";
2028
- case "topic":
2029
- return "organizational";
2030
- case "company":
2031
- case "person":
2032
- case "investor":
2033
- case "function":
2034
- case "value_chain":
2035
- return "ontological";
2036
- }
2037
- }
2038
-
2039
- // src/workspaceIsolation.ts
2040
526
  function normalizeScopeValue2(value) {
2041
527
  if (typeof value !== "string") {
2042
528
  return;
@@ -2054,7 +540,7 @@ function throwWorkspaceIsolationError(args) {
2054
540
  throw error;
2055
541
  }
2056
542
  function assertWorkspaceScopedEpistemicNodeScope(args) {
2057
- const layer = getLayerForNodeType(args.nodeType);
543
+ const layer = isNodeType(args.nodeType) ? getLayerForNodeType(args.nodeType) : void 0;
2058
544
  if (layer === "ontological") {
2059
545
  return;
2060
546
  }
@@ -2082,11 +568,11 @@ function nodeMatchesWorkspaceReasoningScope(node, scope) {
2082
568
  const scopeWorkspaceId = normalizeScopeValue2(scope.workspaceId);
2083
569
  const nodeTenantId = normalizeScopeValue2(node.tenantId);
2084
570
  const nodeWorkspaceId = normalizeScopeValue2(node.workspaceId);
2085
- const epistemicLayer2 = typeof node.epistemicLayer === "string" ? node.epistemicLayer : void 0;
571
+ const epistemicLayer = typeof node.epistemicLayer === "string" ? node.epistemicLayer : void 0;
2086
572
  if (scopeTenantId && nodeTenantId && scopeTenantId !== nodeTenantId) {
2087
573
  return false;
2088
574
  }
2089
- if (epistemicLayer2 === "ontological" && nodeWorkspaceId === void 0) {
575
+ if (epistemicLayer === "ontological" && nodeWorkspaceId === void 0) {
2090
576
  return true;
2091
577
  }
2092
578
  if (!scopeWorkspaceId && node.publicationStatus === "published") {
@@ -2133,10 +619,10 @@ function assertTenantPackWorkspaceMutationAllowed(args) {
2133
619
 
2134
620
  // src/epistemicEvidence.ts
2135
621
  function generateContentHash(text) {
2136
- const content2 = `evidence:${text.trim().toLowerCase().replace(/\s+/g, " ").slice(0, 500)}`;
622
+ const content = `evidence:${text.trim().toLowerCase().replace(/\s+/g, " ").slice(0, 500)}`;
2137
623
  let hash = 5381;
2138
- for (let i = 0; i < content2.length; i++) {
2139
- hash = (hash << 5) + hash + content2.charCodeAt(i);
624
+ for (let i = 0; i < content.length; i++) {
625
+ hash = (hash << 5) + hash + content.charCodeAt(i);
2140
626
  hash &= hash;
2141
627
  }
2142
628
  return Math.abs(hash).toString(16).padStart(8, "0");
@@ -2155,9 +641,9 @@ function normalizeKind(kind) {
2155
641
  ];
2156
642
  return validKinds.find((k) => kind.toLowerCase() === k) || "observation";
2157
643
  }
2158
- function normalizeSourceType(sourceType3) {
2159
- if (sourceType3 === "proprietary" || sourceType3 === "verified") {
2160
- return sourceType3;
644
+ function normalizeSourceType(sourceType) {
645
+ if (sourceType === "proprietary" || sourceType === "verified") {
646
+ return sourceType;
2161
647
  }
2162
648
  return "ai_generated";
2163
649
  }
@@ -2245,23 +731,14 @@ async function resolveEvidenceScopeOrNull(ctx, args) {
2245
731
  }
2246
732
  async function getEvidenceNodesForScope(ctx, scope, args) {
2247
733
  const scanLimit = typeof args?.scanLimit === "number" ? args.scanLimit : void 0;
2248
- const [topicNodes, projectNodes] = await Promise.all([
2249
- scope.topicId ? scanLimit ? ctx.db.query("epistemicNodes").withIndex(
2250
- "by_topic_type",
2251
- (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
2252
- ).order("desc").take(scanLimit) : ctx.db.query("epistemicNodes").withIndex(
2253
- "by_topic_type",
2254
- (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
2255
- ).collect() : Promise.resolve([]),
2256
- scope.projectId ? scanLimit ? ctx.db.query("epistemicNodes").withIndex(
2257
- "by_project_type",
2258
- (q) => q.eq("projectId", scope.projectId).eq("nodeType", "evidence")
2259
- ).order("desc").take(scanLimit) : ctx.db.query("epistemicNodes").withIndex(
2260
- "by_project_type",
2261
- (q) => q.eq("projectId", scope.projectId).eq("nodeType", "evidence")
2262
- ).collect() : Promise.resolve([])
2263
- ]);
2264
- return dedupeEvidenceNodes([...topicNodes, ...projectNodes]).filter(
734
+ const topicNodes = await (scanLimit ? ctx.db.query("epistemicNodes").withIndex(
735
+ "by_topic_type",
736
+ (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
737
+ ).order("desc").take(scanLimit) : ctx.db.query("epistemicNodes").withIndex(
738
+ "by_topic_type",
739
+ (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
740
+ ).collect());
741
+ return dedupeEvidenceNodes(topicNodes).filter(
2265
742
  (node) => evidenceMatchesScope(node, scope)
2266
743
  );
2267
744
  }
@@ -2312,6 +789,7 @@ var create = mutation({
2312
789
  sourceUrl: v.optional(v.string()),
2313
790
  sourceQuestionId: v.optional(v.string()),
2314
791
  userId: v.string(),
792
+ rationale: v.string(),
2315
793
  // Classification fields (from AI tools)
2316
794
  methodology: v.optional(v.string()),
2317
795
  informationAsymmetry: v.optional(v.string()),
@@ -2342,7 +820,7 @@ var create = mutation({
2342
820
  const globalId = generateGlobalId();
2343
821
  const contentHash = generateContentHash(args.text);
2344
822
  const kind = normalizeKind(args.kind);
2345
- const sourceType3 = normalizeSourceType(args.sourceType);
823
+ const sourceType = normalizeSourceType(args.sourceType);
2346
824
  const additionalMetadata = args.metadata && typeof args.metadata === "object" ? args.metadata : {};
2347
825
  const nodeId = await ctx.db.insert("epistemicNodes", {
2348
826
  globalId,
@@ -2359,7 +837,7 @@ var create = mutation({
2359
837
  status: "active",
2360
838
  epistemicLayer: "L2",
2361
839
  // L2: Compression Boundary (Evidence/Claims)
2362
- sourceType: sourceType3,
840
+ sourceType,
2363
841
  createdAt: now,
2364
842
  updatedAt: now,
2365
843
  createdBy: args.userId,
@@ -2369,6 +847,7 @@ var create = mutation({
2369
847
  externalSourceType: args.externalSourceType,
2370
848
  sourceUrl: args.sourceUrl,
2371
849
  sourceQuestionId: args.sourceQuestionId,
850
+ rationale: args.rationale,
2372
851
  linkedBeliefNodeId: args.linkedBeliefNodeId,
2373
852
  evidenceRelation: args.evidenceRelation,
2374
853
  confidence: args.confidence,
@@ -2422,10 +901,11 @@ var create = mutation({
2422
901
  changedBy: args.userId,
2423
902
  isAgent: false,
2424
903
  projectId: scope.projectId,
904
+ rationale: args.rationale,
2425
905
  newState: {
2426
906
  text: args.text.slice(0, 200),
2427
907
  kind,
2428
- sourceType: sourceType3,
908
+ sourceType,
2429
909
  linkedBeliefNodeId: args.linkedBeliefNodeId,
2430
910
  evidenceRelation: args.evidenceRelation
2431
911
  }
@@ -2484,8 +964,8 @@ var createAndLink = mutation({
2484
964
  const globalId = generateGlobalId();
2485
965
  const contentHash = generateContentHash(args.text);
2486
966
  const kind = normalizeKind(args.kind);
2487
- const sourceType3 = normalizeSourceType(args.sourceType);
2488
- const confidence2 = args.confidence ?? 0.7;
967
+ const sourceType = normalizeSourceType(args.sourceType);
968
+ const confidence = args.confidence ?? 0.7;
2489
969
  const nodeId = await ctx.db.insert("epistemicNodes", {
2490
970
  globalId,
2491
971
  topicId: scope.topicId,
@@ -2497,7 +977,7 @@ var createAndLink = mutation({
2497
977
  contentHash,
2498
978
  status: "active",
2499
979
  epistemicLayer: "L2",
2500
- sourceType: sourceType3,
980
+ sourceType,
2501
981
  createdAt: now,
2502
982
  updatedAt: now,
2503
983
  createdBy: args.userId,
@@ -2506,7 +986,7 @@ var createAndLink = mutation({
2506
986
  tags: args.tags || [],
2507
987
  linkedBeliefNodeId: args.beliefNodeId,
2508
988
  evidenceRelation: args.relation,
2509
- confidence: confidence2
989
+ confidence
2510
990
  }
2511
991
  });
2512
992
  await ctx.scheduler.runAfter(0, internal.neo4jSync.syncNodeToNeo4j, {
@@ -2517,7 +997,7 @@ var createAndLink = mutation({
2517
997
  if (!beliefNode) {
2518
998
  throw new Error("Belief node not found for edge creation");
2519
999
  }
2520
- const weight = args.relation === "supports" ? confidence2 : -confidence2;
1000
+ const weight = args.relation === "supports" ? confidence : -confidence;
2521
1001
  const edgeGlobalId = crypto.randomUUID();
2522
1002
  await ctx.scheduler.runAfter(5e3, internal.neo4jEdgeAPI.createEdge, {
2523
1003
  globalId: edgeGlobalId,
@@ -2533,7 +1013,7 @@ var createAndLink = mutation({
2533
1013
  toLayer: "L3",
2534
1014
  metadata: {
2535
1015
  relation: args.relation,
2536
- confidence: confidence2
1016
+ confidence
2537
1017
  }
2538
1018
  });
2539
1019
  await markProjectGraphDirty(ctx, scope.projectId, String(scope.topicId));
@@ -2634,20 +1114,13 @@ var getByProject = query({
2634
1114
  return [];
2635
1115
  }
2636
1116
  }
2637
- const [topicNodes, projectNodes] = await Promise.all([
2638
- scope.topicId ? ctx.db.query("epistemicNodes").withIndex(
2639
- "by_topic_type",
2640
- (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
2641
- ).order("desc").take(scanLimit) : Promise.resolve([]),
2642
- scope.projectId ? ctx.db.query("epistemicNodes").withIndex(
2643
- "by_project_type",
2644
- (q) => q.eq("projectId", scope.projectId).eq("nodeType", "evidence")
2645
- ).order("desc").take(scanLimit) : Promise.resolve([])
2646
- ]);
2647
- const scopedNodes = dedupeEvidenceNodes([
2648
- ...topicNodes,
2649
- ...projectNodes
2650
- ]).filter((node) => evidenceMatchesScope(node, scope));
1117
+ const topicNodes = await ctx.db.query("epistemicNodes").withIndex(
1118
+ "by_topic_type",
1119
+ (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
1120
+ ).order("desc").take(scanLimit);
1121
+ const scopedNodes = dedupeEvidenceNodes(topicNodes).filter(
1122
+ (node) => evidenceMatchesScope(node, scope)
1123
+ );
2651
1124
  const filteredNodes = args.status ? scopedNodes.filter((node) => node.status === args.status) : scopedNodes;
2652
1125
  return filteredNodes.map(flattenEvidenceNode).slice(0, pageSize);
2653
1126
  }
@@ -2664,20 +1137,13 @@ var getByTopic = query({
2664
1137
  const pageSize = clampEvidenceLimit(args.limit);
2665
1138
  const scanLimit = Math.min(pageSize * 3, MAX_EVIDENCE_PAGE_SIZE);
2666
1139
  const scope = await resolveTopicProjectScope(ctx, { topicId: args.topicId });
2667
- const [topicNodes, projectNodes] = await Promise.all([
2668
- ctx.db.query("epistemicNodes").withIndex(
2669
- "by_topic_type",
2670
- (q) => q.eq("topicId", args.topicId).eq("nodeType", "evidence")
2671
- ).order("desc").take(scanLimit),
2672
- scope.projectId ? ctx.db.query("epistemicNodes").withIndex(
2673
- "by_project_type",
2674
- (q) => q.eq("projectId", scope.projectId).eq("nodeType", "evidence")
2675
- ).order("desc").take(scanLimit) : Promise.resolve([])
2676
- ]);
2677
- const scopedNodes = dedupeEvidenceNodes([
2678
- ...topicNodes,
2679
- ...projectNodes
2680
- ]).filter((node) => evidenceMatchesScope(node, scope));
1140
+ const topicNodes = await ctx.db.query("epistemicNodes").withIndex(
1141
+ "by_topic_type",
1142
+ (q) => q.eq("topicId", scope.topicId).eq("nodeType", "evidence")
1143
+ ).order("desc").take(scanLimit);
1144
+ const scopedNodes = dedupeEvidenceNodes(topicNodes).filter(
1145
+ (node) => evidenceMatchesScope(node, scope)
1146
+ );
2681
1147
  const filteredNodes = args.status ? scopedNodes.filter((node) => node.status === args.status) : scopedNodes;
2682
1148
  return filteredNodes.map(flattenEvidenceNode).slice(0, pageSize);
2683
1149
  }
@@ -2842,6 +1308,7 @@ var internalCreate = internalMutation({
2842
1308
  sourceUrl: v.optional(v.string()),
2843
1309
  sourceQuestionId: v.optional(v.string()),
2844
1310
  userId: v.string(),
1311
+ rationale: v.string(),
2845
1312
  linkedBeliefNodeId: v.optional(v.id("epistemicNodes")),
2846
1313
  evidenceRelation: v.optional(v.string()),
2847
1314
  confidence: v.optional(v.number()),
@@ -2879,7 +1346,7 @@ var internalCreate = internalMutation({
2879
1346
  const globalId = generateGlobalId();
2880
1347
  const contentHash = generateContentHash(args.text);
2881
1348
  const kind = normalizeKind(args.kind);
2882
- const sourceType3 = normalizeSourceType(args.sourceType);
1349
+ const sourceType = normalizeSourceType(args.sourceType);
2883
1350
  const additionalMetadata = args.metadata && typeof args.metadata === "object" ? args.metadata : {};
2884
1351
  const nodeId = await ctx.db.insert("epistemicNodes", {
2885
1352
  globalId,
@@ -2895,7 +1362,7 @@ var internalCreate = internalMutation({
2895
1362
  ...typeof args.contentType === "string" && args.contentType.trim().length > 0 ? { contentType: args.contentType.trim() } : {},
2896
1363
  status: "active",
2897
1364
  epistemicLayer: "L2",
2898
- sourceType: sourceType3,
1365
+ sourceType,
2899
1366
  createdAt: now,
2900
1367
  updatedAt: now,
2901
1368
  createdBy: args.userId,
@@ -2905,6 +1372,7 @@ var internalCreate = internalMutation({
2905
1372
  externalSourceType: args.externalSourceType,
2906
1373
  sourceUrl: args.sourceUrl,
2907
1374
  sourceQuestionId: args.sourceQuestionId,
1375
+ rationale: args.rationale,
2908
1376
  linkedBeliefNodeId: args.linkedBeliefNodeId,
2909
1377
  evidenceRelation: args.evidenceRelation,
2910
1378
  confidence: args.confidence,
@@ -2919,10 +1387,11 @@ var internalCreate = internalMutation({
2919
1387
  changedBy: args.userId,
2920
1388
  isAgent: false,
2921
1389
  projectId: scope.projectId,
1390
+ rationale: args.rationale,
2922
1391
  newState: {
2923
1392
  text: args.text.slice(0, 200),
2924
1393
  kind,
2925
- sourceType: sourceType3,
1394
+ sourceType,
2926
1395
  externalSourceType: args.externalSourceType,
2927
1396
  sourceUrl: args.sourceUrl,
2928
1397
  linkedBeliefNodeId: args.linkedBeliefNodeId,
@@ -2938,8 +1407,8 @@ var internalCreate = internalMutation({
2938
1407
  if (args.linkedBeliefNodeId && args.evidenceRelation) {
2939
1408
  const beliefNode = await ctx.db.get(args.linkedBeliefNodeId);
2940
1409
  if (beliefNode) {
2941
- const confidence2 = args.confidence ?? 0.7;
2942
- const weight = args.evidenceRelation === "supports" ? confidence2 : -confidence2;
1410
+ const confidence = args.confidence ?? 0.7;
1411
+ const weight = args.evidenceRelation === "supports" ? confidence : -confidence;
2943
1412
  await ctx.scheduler.runAfter(5e3, internal.neo4jEdgeAPI.createEdge, {
2944
1413
  globalId: crypto.randomUUID(),
2945
1414
  fromGlobalId: globalId,
@@ -2954,7 +1423,7 @@ var internalCreate = internalMutation({
2954
1423
  toLayer: "L3",
2955
1424
  metadata: {
2956
1425
  relation: args.evidenceRelation,
2957
- confidence: confidence2
1426
+ confidence
2958
1427
  }
2959
1428
  });
2960
1429
  }