@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.
- package/dist/beliefDecay.js +229 -1115
- package/dist/beliefDecay.js.map +1 -1
- package/dist/beliefEvidenceLinks.js +53 -834
- package/dist/beliefEvidenceLinks.js.map +1 -1
- package/dist/confidencePropagationDispatch.d.ts +3 -3
- package/dist/confidencePropagationDispatch.js +30 -308
- package/dist/confidencePropagationDispatch.js.map +1 -1
- package/dist/contradictions.js +5 -797
- package/dist/contradictions.js.map +1 -1
- package/dist/edges/contradicts.js +1 -122
- package/dist/edges/contradicts.js.map +1 -1
- package/dist/edges/dependsOn.js +14 -172
- package/dist/edges/dependsOn.js.map +1 -1
- package/dist/edges/elaborates.js +1 -49
- package/dist/edges/elaborates.js.map +1 -1
- package/dist/edges/index.js +14 -277
- package/dist/edges/index.js.map +1 -1
- package/dist/edges/informs.js +1 -62
- package/dist/edges/informs.js.map +1 -1
- package/dist/edges/propagationTypes.d.ts +2 -2
- package/dist/edges/propagationTypes.js.map +1 -1
- package/dist/edges/refutes.js +1 -62
- package/dist/edges/refutes.js.map +1 -1
- package/dist/edges/supports.js +1 -122
- package/dist/edges/supports.js.map +1 -1
- package/dist/edges/utils.d.ts +6 -6
- package/dist/edges/utils.js +1 -130
- package/dist/edges/utils.js.map +1 -1
- package/dist/entityBridge.js +2 -17
- package/dist/entityBridge.js.map +1 -1
- package/dist/entityLifecycle.js +62 -848
- package/dist/entityLifecycle.js.map +1 -1
- package/dist/epistemicAnswers.js +27 -838
- package/dist/epistemicAnswers.js.map +1 -1
- package/dist/epistemicBeliefs.js +186 -2214
- package/dist/epistemicBeliefs.js.map +1 -1
- package/dist/epistemicContractHelpers.js +1 -318
- package/dist/epistemicContractHelpers.js.map +1 -1
- package/dist/epistemicContracts.js +163 -2467
- package/dist/epistemicContracts.js.map +1 -1
- package/dist/epistemicEdges.js +60 -863
- package/dist/epistemicEdges.js.map +1 -1
- package/dist/epistemicEvidence.js +116 -1647
- package/dist/epistemicEvidence.js.map +1 -1
- package/dist/epistemicHelpers.js +3 -2
- package/dist/epistemicHelpers.js.map +1 -1
- package/dist/epistemicLinking.js +2 -785
- package/dist/epistemicLinking.js.map +1 -1
- package/dist/epistemicNodes.js +34 -1427
- package/dist/epistemicNodes.js.map +1 -1
- package/dist/epistemicQuestions.js +88 -1637
- package/dist/epistemicQuestions.js.map +1 -1
- package/dist/epistemicSources.js +28 -1421
- package/dist/epistemicSources.js.map +1 -1
- package/dist/evaluators/index.js +163 -2467
- package/dist/evaluators/index.js.map +1 -1
- package/dist/index.js +486 -3649
- package/dist/index.js.map +1 -1
- package/dist/ontology-matching.js +1 -344
- package/dist/ontology-matching.js.map +1 -1
- package/dist/ontologyApproval.js +1 -13
- package/dist/ontologyApproval.js.map +1 -1
- package/dist/ontologyDefinitions.js +2 -17
- package/dist/ontologyDefinitions.js.map +1 -1
- package/dist/ontologyRegistry.js +2 -17
- package/dist/ontologyRegistry.js.map +1 -1
- package/dist/projectionReconciliation.js +2 -17
- package/dist/projectionReconciliation.js.map +1 -1
- package/dist/questionEvidenceLinks.js +242 -837
- package/dist/questionEvidenceLinks.js.map +1 -1
- package/dist/text-matching.js +1 -244
- package/dist/text-matching.js.map +1 -1
- package/dist/workflowBridge.d.ts +27 -0
- package/dist/workflowBridge.js +303 -0
- package/dist/workflowBridge.js.map +1 -0
- package/dist/workspaceIsolation.js +8 -609
- package/dist/workspaceIsolation.js.map +1 -1
- package/package.json +6 -6
|
@@ -1,937 +1,14 @@
|
|
|
1
1
|
import { v } from 'convex/values';
|
|
2
|
-
import {
|
|
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
|
|
976
|
-
function
|
|
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
|
|
60
|
+
function readStringArray(value) {
|
|
984
61
|
if (!Array.isArray(value)) {
|
|
985
62
|
return [];
|
|
986
63
|
}
|
|
987
|
-
return value.map((entry) =>
|
|
64
|
+
return value.map((entry) => readNonEmptyString(entry)).filter((entry) => Boolean(entry));
|
|
988
65
|
}
|
|
989
|
-
function
|
|
66
|
+
function readMetadata(topic) {
|
|
990
67
|
return topic.metadata && typeof topic.metadata === "object" ? topic.metadata : {};
|
|
991
68
|
}
|
|
992
|
-
function
|
|
69
|
+
function readLegacyProjectId(value) {
|
|
993
70
|
if (!value) {
|
|
994
71
|
return;
|
|
995
72
|
}
|
|
996
|
-
return
|
|
73
|
+
return readNonEmptyString(value[LEGACY_SCOPE_FIELD]);
|
|
997
74
|
}
|
|
998
|
-
function
|
|
75
|
+
function coerceVisibility(value) {
|
|
999
76
|
return value === "private" || value === "team" || value === "firm" || value === "external" || value === "public" ? value : void 0;
|
|
1000
77
|
}
|
|
1001
|
-
function
|
|
78
|
+
function coerceStatus(value) {
|
|
1002
79
|
return value === "active" || value === "archived" || value === "watching" ? value : void 0;
|
|
1003
80
|
}
|
|
1004
|
-
function
|
|
1005
|
-
const explicit =
|
|
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
|
|
89
|
+
return readNonEmptyString(topic.type) || "general";
|
|
1013
90
|
}
|
|
1014
|
-
function
|
|
1015
|
-
const metadata =
|
|
1016
|
-
return topic.type === "theme" || topic.type === "thematic" || topic.type === "deal" || topic.type === "monitoring" ||
|
|
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
|
|
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(
|
|
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(
|
|
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
|
|
1058
|
-
const metadata =
|
|
134
|
+
function materializeTopicProjectOverlay(topic, idMode = "legacy") {
|
|
135
|
+
const metadata = readMetadata(topic);
|
|
1059
136
|
const topicId = String(topic._id);
|
|
1060
|
-
const 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 =
|
|
1064
|
-
const status =
|
|
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:
|
|
1075
|
-
type:
|
|
1076
|
-
description:
|
|
1077
|
-
ownerId:
|
|
1078
|
-
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:
|
|
1081
|
-
workspaceId:
|
|
157
|
+
tenantId: readNonEmptyString(topic.tenantId) || readNonEmptyString(metadata.tenantId),
|
|
158
|
+
workspaceId: readNonEmptyString(topic.workspaceId) || readNonEmptyString(metadata.workspaceId),
|
|
1082
159
|
status,
|
|
1083
|
-
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
|
|
1093
|
-
const topic = await
|
|
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 && !
|
|
174
|
+
if (options.projectLikeOnly !== false && !isProjectLikeTopic(topic)) {
|
|
1098
175
|
return null;
|
|
1099
176
|
}
|
|
1100
|
-
return
|
|
177
|
+
return materializeTopicProjectOverlay(topic, options.idMode);
|
|
1101
178
|
}
|
|
1102
|
-
async function
|
|
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(
|
|
189
|
+
allTopics = (await ctx.runQuery(api.topics.list, {}) ?? []) || [];
|
|
1113
190
|
}
|
|
1114
191
|
return allTopics.filter(
|
|
1115
|
-
(topic) => options.projectLikeOnly === false ||
|
|
1116
|
-
).map((topic) =>
|
|
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
|
|
196
|
+
const topic = await resolveTopicDoc(ctx, scopeId);
|
|
1120
197
|
if (!topic) {
|
|
1121
198
|
return null;
|
|
1122
199
|
}
|
|
1123
|
-
const nextMetadata = { ...
|
|
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 =
|
|
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 =
|
|
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 =
|
|
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(
|
|
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
|
|
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
|
|
312
|
+
function defaultResolvers() {
|
|
1236
313
|
return {
|
|
1237
314
|
async getProject(ctx, projectId) {
|
|
1238
|
-
return await
|
|
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
|
|
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
|
|
333
|
+
var resolverOverrides = {};
|
|
1257
334
|
function resolveGraphPrimitivesAppResolvers(_ctx) {
|
|
1258
335
|
return {
|
|
1259
|
-
...
|
|
1260
|
-
...
|
|
336
|
+
...defaultResolvers(),
|
|
337
|
+
...resolverOverrides
|
|
1261
338
|
};
|
|
1262
339
|
}
|
|
1263
|
-
var
|
|
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[
|
|
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[
|
|
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(
|
|
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(
|
|
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(
|
|
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
|
|
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 (
|
|
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
|
|
622
|
+
const content = `evidence:${text.trim().toLowerCase().replace(/\s+/g, " ").slice(0, 500)}`;
|
|
2137
623
|
let hash = 5381;
|
|
2138
|
-
for (let i = 0; i <
|
|
2139
|
-
hash = (hash << 5) + hash +
|
|
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(
|
|
2159
|
-
if (
|
|
2160
|
-
return
|
|
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
|
|
2249
|
-
|
|
2250
|
-
|
|
2251
|
-
|
|
2252
|
-
|
|
2253
|
-
|
|
2254
|
-
|
|
2255
|
-
|
|
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
|
|
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
|
|
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
|
|
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
|
|
2488
|
-
const
|
|
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
|
|
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
|
|
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" ?
|
|
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
|
|
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
|
|
2638
|
-
|
|
2639
|
-
|
|
2640
|
-
|
|
2641
|
-
|
|
2642
|
-
|
|
2643
|
-
|
|
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
|
|
2668
|
-
|
|
2669
|
-
|
|
2670
|
-
|
|
2671
|
-
|
|
2672
|
-
|
|
2673
|
-
|
|
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
|
|
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
|
|
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
|
|
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
|
|
2942
|
-
const weight = args.evidenceRelation === "supports" ?
|
|
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
|
|
1426
|
+
confidence
|
|
2958
1427
|
}
|
|
2959
1428
|
});
|
|
2960
1429
|
}
|