@rpcbase/server 0.517.0 → 0.519.0
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/{handler-cwV5LCtl.js → handler-3uwH4f67.js} +93 -50
- package/dist/handler-3uwH4f67.js.map +1 -0
- package/dist/index.js +10 -1
- package/dist/index.js.map +1 -1
- package/dist/{queryExecutor-B7lb2FR1.js → queryExecutor-Bg1GGL3j.js} +6 -2
- package/dist/queryExecutor-Bg1GGL3j.js.map +1 -0
- package/dist/rts/index.d.ts.map +1 -1
- package/dist/rts/index.js +4 -1
- package/dist/rts/index.js.map +1 -1
- package/dist/rts/queryExecutor.d.ts +1 -0
- package/dist/rts/queryExecutor.d.ts.map +1 -1
- package/dist/rts/ssrHydration.d.ts.map +1 -1
- package/dist/uploads/api/file-uploads/handlers/completeUpload.d.ts.map +1 -1
- package/dist/uploads/api/file-uploads/postProcessors.d.ts +30 -0
- package/dist/uploads/api/file-uploads/postProcessors.d.ts.map +1 -0
- package/dist/uploads.d.ts +1 -0
- package/dist/uploads.d.ts.map +1 -1
- package/dist/uploads.js +82 -2
- package/dist/uploads.js.map +1 -1
- package/package.json +1 -1
- package/dist/handler-cwV5LCtl.js.map +0 -1
- package/dist/queryExecutor-B7lb2FR1.js.map +0 -1
|
@@ -371,9 +371,13 @@ const runRtsQuery = async ({
|
|
|
371
371
|
signingSecret: getPaginationCursorSigningSecret()
|
|
372
372
|
}
|
|
373
373
|
});
|
|
374
|
+
const totalCount = typeof paginatedResult.totalCount === "number" && Number.isFinite(paginatedResult.totalCount) && paginatedResult.totalCount >= 0 ? Math.floor(paginatedResult.totalCount) : void 0;
|
|
374
375
|
return {
|
|
375
376
|
data: Array.isArray(paginatedResult.nodes) ? paginatedResult.nodes : [],
|
|
376
|
-
pageInfo: paginatedResult.pageInfo
|
|
377
|
+
pageInfo: paginatedResult.pageInfo,
|
|
378
|
+
...totalCount !== void 0 ? {
|
|
379
|
+
totalCount
|
|
380
|
+
} : {}
|
|
377
381
|
};
|
|
378
382
|
}
|
|
379
383
|
const queryPromise = model.find(finalQuery, projection);
|
|
@@ -400,4 +404,4 @@ export {
|
|
|
400
404
|
normalizeRtsQueryOptions as n,
|
|
401
405
|
resolveRtsRequestTenantId as r
|
|
402
406
|
};
|
|
403
|
-
//# sourceMappingURL=queryExecutor-
|
|
407
|
+
//# sourceMappingURL=queryExecutor-Bg1GGL3j.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"queryExecutor-Bg1GGL3j.js","sources":["../src/getDerivedKey.ts","../src/rts/queryExecutor.ts"],"sourcesContent":["import assert from \"assert\"\nimport { hkdfSync } from \"crypto\"\n\n\nexport const getDerivedKey = (\n masterKey: string,\n info: string,\n length: number = 32, // Default to 256-bit keys\n salt: string = \"\",\n): string => {\n assert(masterKey?.length >= 32, \"MASTER_KEY must be 32 chars or longer.\")\n\n return Buffer.from(hkdfSync(\n \"sha256\",\n masterKey,\n Buffer.from(salt),\n Buffer.from(info),\n length,\n )).toString(\"hex\")\n}\n","import type { Request } from \"express\"\nimport type { PaginationPageInfo, PaginationSpec } from \"@rpcbase/api\"\nimport { models, type LoadModelCtx } from \"@rpcbase/db\"\nimport { buildAbility, buildAbilityFromSession, getAccessibleByQuery, getTenantRolesFromSessionUser, type AclSubjectType, type AppAbility } from \"@rpcbase/db/acl\"\nimport type { Model } from \"mongoose\"\n\nimport { getDerivedKey } from \"../getDerivedKey\"\n\n\ntype JsonObject = Record<string, unknown>\n\ntype SessionUser = {\n id?: unknown\n currentTenantId?: unknown\n signedInTenants?: unknown\n}\n\ntype HeaderUserDoc = {\n tenants?: unknown\n tenantRoles?: unknown\n}\n\nexport type RtsPopulateObject = {\n path: string\n model?: string\n select?: string | JsonObject\n match?: JsonObject\n options?: {\n sort?: Record<string, 1 | -1>\n limit?: number\n }\n populate?: RtsPopulateOption\n}\n\nexport type RtsPopulateOption =\n | string\n | RtsPopulateObject\n | Array<string | RtsPopulateObject>\n\nexport type RtsQueryOptions = {\n projection?: JsonObject\n sort?: Record<string, 1 | -1>\n limit?: number\n populate?: RtsPopulateOption\n pagination?: PaginationSpec\n}\n\nexport type RtsQueryResult = {\n data: unknown[]\n pageInfo?: PaginationPageInfo\n totalCount?: number\n}\n\nexport const RTS_TENANT_ID_QUERY_PARAM = \"rb-tenant-id\"\nexport const RTS_USER_ID_HEADER = \"rb-user-id\"\n\nconst QUERY_MAX_LIMIT = 4096\nconst INTERNAL_MODEL_NAMES = new Set([\"RBRtsChange\", \"RBRtsCounter\"])\nlet paginationCursorSigningSecret: string | null = null\n\nconst getPaginationCursorSigningSecret = (): string => {\n if (paginationCursorSigningSecret) return paginationCursorSigningSecret\n const masterKey = process.env.MASTER_KEY?.trim()\n if (!masterKey) {\n throw new Error(\"MASTER_KEY must be defined to derive pagination cursor signing secret\")\n }\n paginationCursorSigningSecret = getDerivedKey(masterKey, \"pagination_cursor_signing\")\n return paginationCursorSigningSecret\n}\n\nconst normalizeTenantId = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nconst normalizeSignedInTenants = (value: unknown): string[] => {\n if (!Array.isArray(value)) return []\n return value\n .map((tenantId) => normalizeTenantId(String(tenantId)))\n .filter((tenantId): tenantId is string => Boolean(tenantId))\n}\n\nconst getTenantIdFromRequest = (req: Request): string | null => {\n const rawQuery = req.query?.[RTS_TENANT_ID_QUERY_PARAM]\n const queryTenantId = Array.isArray(rawQuery) ? rawQuery[0] : rawQuery\n const normalizedFromQuery = normalizeTenantId(queryTenantId)\n if (normalizedFromQuery) return normalizedFromQuery\n\n return normalizeTenantId((req.session?.user as SessionUser | undefined)?.currentTenantId)\n}\n\nexport const resolveRtsRequestTenantId = (req: Request): string | null => {\n return getTenantIdFromRequest(req)\n}\n\nexport const resolveRtsRequestUserId = (req: Request): string | null => {\n const sessionUserId = normalizeTenantId((req.session?.user as SessionUser | undefined)?.id)\n if (sessionUserId) return sessionUserId\n\n const headerValue = req.headers[RTS_USER_ID_HEADER]\n const headerUserId = Array.isArray(headerValue) ? headerValue[0] : headerValue\n return normalizeTenantId(headerUserId)\n}\n\nexport const isRtsRequestAuthorized = (req: Request, tenantId: string): boolean => {\n const sessionUser = req.session?.user as SessionUser | undefined\n if (!sessionUser) return false\n\n const signedInTenants = normalizeSignedInTenants(sessionUser.signedInTenants)\n if (signedInTenants.length > 0) {\n return signedInTenants.includes(tenantId)\n }\n\n const currentTenantId = normalizeTenantId(sessionUser.currentTenantId)\n if (!currentTenantId) return false\n return currentTenantId === tenantId\n}\n\nexport const buildRtsAbilityFromRequest = async (\n req: Request,\n tenantId: string,\n): Promise<{ ability: AppAbility; userId: string | null }> => {\n const sessionUserId = normalizeTenantId((req.session?.user as SessionUser | undefined)?.id)\n if (sessionUserId) {\n const ability = buildAbilityFromSession({ tenantId, session: req.session })\n return { ability, userId: sessionUserId }\n }\n\n const headerValue = req.headers[RTS_USER_ID_HEADER]\n const headerUserIdRaw = Array.isArray(headerValue) ? headerValue[0] : headerValue\n const headerUserId = normalizeTenantId(headerUserIdRaw)\n if (!headerUserId) {\n const ability = buildAbilityFromSession({ tenantId, session: req.session })\n return { ability, userId: null }\n }\n\n const rbCtx: LoadModelCtx = { req: { session: null } }\n const User = await models.getGlobal(\"RBUser\", rbCtx)\n const user = await User.findById(headerUserId, { tenants: 1, tenantRoles: 1 }).lean() as HeaderUserDoc | null\n\n const tenantsRaw = user?.tenants\n const tenants = Array.isArray(tenantsRaw) ? tenantsRaw.map((tenant) => String(tenant)) : []\n if (!tenants.includes(tenantId)) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n\n const roles = getTenantRolesFromSessionUser(user, tenantId)\n return {\n ability: buildAbility({ tenantId, userId: headerUserId, roles: roles.length ? roles : [\"owner\"] }),\n userId: headerUserId,\n }\n}\n\nconst getTenantModel = async (tenantId: string, modelName: string, ability: AppAbility): Promise<Model<any>> => {\n const ctx: LoadModelCtx = {\n req: {\n session: {\n user: {\n currentTenantId: tenantId,\n },\n },\n },\n ability,\n }\n\n return models.get(modelName, ctx)\n}\n\nconst normalizeLimit = (limit?: number): number => {\n if (typeof limit !== \"number\") return QUERY_MAX_LIMIT\n if (!Number.isFinite(limit)) return QUERY_MAX_LIMIT\n return Math.min(QUERY_MAX_LIMIT, Math.abs(limit))\n}\n\nconst normalizeString = (value: unknown): string => {\n return typeof value === \"string\" ? value.trim() : \"\"\n}\n\nconst normalizeObject = (value: unknown): JsonObject | undefined => {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) return undefined\n return value as JsonObject\n}\n\nconst normalizePagination = (value: unknown): PaginationSpec | undefined => {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) return undefined\n return value as PaginationSpec\n}\n\nconst normalizePopulateSelect = (value: unknown): string | JsonObject | undefined => {\n if (typeof value === \"string\") {\n const normalized = value.trim()\n return normalized || undefined\n }\n return normalizeObject(value)\n}\n\nconst normalizePopulateOptions = (value: unknown): RtsPopulateObject[\"options\"] | undefined => {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) return undefined\n const raw = value as { sort?: unknown; limit?: unknown }\n const normalized: RtsPopulateObject[\"options\"] = {}\n\n if (raw.sort && typeof raw.sort === \"object\" && !Array.isArray(raw.sort)) {\n normalized.sort = raw.sort as Record<string, 1 | -1>\n }\n\n if (typeof raw.limit === \"number\" && Number.isFinite(raw.limit)) {\n normalized.limit = Math.max(0, Math.floor(Math.abs(raw.limit)))\n }\n\n if (!normalized.sort && normalized.limit === undefined) return undefined\n return normalized\n}\n\nconst normalizePopulateObject = (value: unknown): RtsPopulateObject | undefined => {\n if (!value || typeof value !== \"object\" || Array.isArray(value)) return undefined\n const raw = value as Record<string, unknown>\n const path = normalizeString(raw.path)\n if (!path) return undefined\n\n const normalized: RtsPopulateObject = { path }\n\n const model = normalizeString(raw.model)\n if (model) normalized.model = model\n\n const select = normalizePopulateSelect(raw.select)\n if (select !== undefined) normalized.select = select\n\n const match = normalizeObject(raw.match)\n if (match) normalized.match = match\n\n const nestedPopulate = normalizeRtsPopulateOption(raw.populate)\n if (nestedPopulate !== undefined) normalized.populate = nestedPopulate\n\n const options = normalizePopulateOptions(raw.options)\n if (options) normalized.options = options\n\n return normalized\n}\n\nconst normalizeRtsPopulateOption = (value: unknown): RtsPopulateOption | undefined => {\n if (typeof value === \"string\") {\n const normalized = value.trim()\n return normalized || undefined\n }\n\n if (Array.isArray(value)) {\n const normalized = value\n .map((entry) => {\n if (typeof entry === \"string\") {\n const path = entry.trim()\n return path || null\n }\n return normalizePopulateObject(entry) ?? null\n })\n .filter((entry): entry is string | RtsPopulateObject => entry !== null)\n\n return normalized.length > 0 ? normalized : undefined\n }\n\n return normalizePopulateObject(value)\n}\n\nconst normalizeModelName = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized || null\n}\n\nconst resolvePopulateRefModelName = (\n model: Model<any>,\n path: string,\n explicitModelName: string | null,\n): string | null => {\n if (explicitModelName) return explicitModelName\n\n const schema = model.schema as any\n const schemaPath = typeof schema.path === \"function\" ? schema.path(path) : null\n const directRef = normalizeModelName(schemaPath?.options?.ref)\n if (directRef) return directRef\n\n const arrayRef = normalizeModelName(schemaPath?.caster?.options?.ref)\n if (arrayRef) return arrayRef\n\n const virtualPath = typeof schema.virtualpath === \"function\" ? schema.virtualpath(path) : null\n const virtualRef = normalizeModelName(virtualPath?.options?.ref)\n if (virtualRef) return virtualRef\n\n return null\n}\n\nconst mergePopulateMatchWithAcl = (\n populateMatch: JsonObject | undefined,\n aclMatch: JsonObject,\n): JsonObject => {\n if (!populateMatch || Object.keys(populateMatch).length === 0) return aclMatch\n return { $and: [populateMatch, aclMatch] }\n}\n\ntype PreparedPopulateObject = {\n path: string\n model?: string\n select?: string | JsonObject\n match?: JsonObject\n options?: {\n sort?: Record<string, 1 | -1>\n limit?: number\n }\n populate?: PreparedPopulateOption\n}\n\ntype PreparedPopulateOption =\n | string\n | PreparedPopulateObject\n | Array<string | PreparedPopulateObject>\n\nconst resolvePopulateSpecForModel = async ({\n tenantId,\n model,\n ability,\n populate,\n allowInternalModels,\n modelCache,\n dependencyModelNames,\n}: {\n tenantId: string\n model: Model<any>\n ability: AppAbility\n populate: RtsPopulateOption | undefined\n allowInternalModels: boolean\n modelCache: Map<string, Model<any>>\n dependencyModelNames: Set<string>\n}): Promise<PreparedPopulateOption | undefined> => {\n if (!populate) return undefined\n\n const getModelCached = async (targetModelName: string): Promise<Model<any>> => {\n const cached = modelCache.get(targetModelName)\n if (cached) return cached\n const loaded = await getTenantModel(tenantId, targetModelName, ability)\n modelCache.set(targetModelName, loaded)\n return loaded\n }\n\n const resolveOne = async (\n entry: string | RtsPopulateObject,\n parentModel: Model<any>,\n ): Promise<string | PreparedPopulateObject | null> => {\n if (typeof entry === \"string\") {\n const path = entry.trim()\n if (!path) return null\n\n const refModelName = resolvePopulateRefModelName(parentModel, path, null)\n if (!refModelName) return path\n if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(refModelName)) {\n throw new Error(\"Model not allowed\")\n }\n if (!ability.can(\"read\", refModelName as AclSubjectType)) {\n throw new Error(\"forbidden\")\n }\n\n dependencyModelNames.add(refModelName)\n\n const aclMatch = getAccessibleByQuery(\n ability,\n \"read\",\n refModelName as Exclude<AclSubjectType, \"all\">,\n )\n return {\n path,\n match: aclMatch as JsonObject,\n }\n }\n\n const path = entry.path.trim()\n if (!path) return null\n\n const explicitModelName = normalizeModelName(entry.model)\n const refModelName = resolvePopulateRefModelName(parentModel, path, explicitModelName)\n let nestedModel = parentModel\n\n const normalizedEntry: PreparedPopulateObject = {\n path,\n }\n\n if (entry.select !== undefined) normalizedEntry.select = entry.select\n if (entry.options !== undefined) normalizedEntry.options = entry.options\n if (explicitModelName) normalizedEntry.model = explicitModelName\n if (entry.match !== undefined) normalizedEntry.match = entry.match\n\n if (refModelName) {\n if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(refModelName)) {\n throw new Error(\"Model not allowed\")\n }\n if (!ability.can(\"read\", refModelName as AclSubjectType)) {\n throw new Error(\"forbidden\")\n }\n\n dependencyModelNames.add(refModelName)\n nestedModel = await getModelCached(refModelName)\n\n const aclMatch = getAccessibleByQuery(\n ability,\n \"read\",\n refModelName as Exclude<AclSubjectType, \"all\">,\n ) as JsonObject\n normalizedEntry.match = mergePopulateMatchWithAcl(\n normalizedEntry.match,\n aclMatch,\n )\n } else if (entry.populate !== undefined) {\n throw new Error(\"Populate path must reference a model when nested populate is used\")\n }\n\n const nestedPopulate = await resolvePopulateSpecForModel({\n tenantId,\n model: nestedModel,\n ability,\n populate: entry.populate,\n allowInternalModels,\n modelCache,\n dependencyModelNames,\n })\n if (nestedPopulate !== undefined) normalizedEntry.populate = nestedPopulate\n\n return normalizedEntry\n }\n\n if (Array.isArray(populate)) {\n const resolved = await Promise.all(populate.map((entry) => resolveOne(entry, model)))\n const filtered = resolved.filter((entry): entry is string | PreparedPopulateObject => entry !== null)\n return filtered.length > 0 ? filtered : undefined\n }\n\n const resolved = await resolveOne(populate, model)\n return resolved ?? undefined\n}\n\nexport const normalizeRtsQueryOptions = (options: RtsQueryOptions | undefined): RtsQueryOptions => {\n if (!options || typeof options !== \"object\") return {}\n const normalized: RtsQueryOptions = {}\n\n if (options.projection && typeof options.projection === \"object\" && !Array.isArray(options.projection)) {\n normalized.projection = options.projection\n }\n\n if (options.sort && typeof options.sort === \"object\" && !Array.isArray(options.sort)) {\n normalized.sort = options.sort\n }\n\n normalized.limit = normalizeLimit(options.limit)\n normalized.populate = normalizeRtsPopulateOption(options.populate)\n normalized.pagination = normalizePagination(options.pagination)\n\n return normalized\n}\n\nexport const resolveRtsQueryDependencyModelNames = async ({\n tenantId,\n ability,\n modelName,\n options,\n allowInternalModels = false,\n}: {\n tenantId: string\n ability: AppAbility\n modelName: string\n options: RtsQueryOptions\n allowInternalModels?: boolean\n}): Promise<string[]> => {\n const model = await getTenantModel(tenantId, modelName, ability)\n const modelCache = new Map<string, Model<any>>()\n modelCache.set(modelName, model)\n\n const dependencyModelNames = new Set<string>()\n await resolvePopulateSpecForModel({\n tenantId,\n model,\n ability,\n populate: options.populate,\n allowInternalModels,\n modelCache,\n dependencyModelNames,\n })\n\n return Array.from(dependencyModelNames)\n}\n\nexport const runRtsQuery = async ({\n tenantId,\n ability,\n modelName,\n query,\n options,\n allowInternalModels = false,\n}: {\n tenantId: string\n ability: AppAbility\n modelName: string\n query: JsonObject\n options: RtsQueryOptions\n allowInternalModels?: boolean\n}): Promise<RtsQueryResult> => {\n if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(modelName)) {\n throw new Error(\"Model not allowed\")\n }\n\n if (!ability.can(\"read\", modelName as AclSubjectType)) {\n throw new Error(\"forbidden\")\n }\n\n const model = await getTenantModel(tenantId, modelName, ability)\n const projection = options.projection ?? undefined\n const sort = options.sort\n const limit = normalizeLimit(options.limit)\n const modelCache = new Map<string, Model<any>>()\n modelCache.set(modelName, model)\n\n const populate = await resolvePopulateSpecForModel({\n tenantId,\n model,\n ability,\n populate: options.populate,\n allowInternalModels,\n modelCache,\n dependencyModelNames: new Set<string>(),\n })\n\n const accessQuery = getAccessibleByQuery(ability, \"read\", modelName as Exclude<AclSubjectType, \"all\">)\n const finalQuery: JsonObject = { $and: [query, accessQuery] }\n\n if (options.pagination) {\n const paginatedQuery = model.find(finalQuery, projection)\n if (populate !== undefined) {\n paginatedQuery.populate(populate as any)\n }\n\n const paginatedResult = await paginatedQuery.paginate(options.pagination, {\n cursor: {\n signingSecret: getPaginationCursorSigningSecret(),\n },\n })\n const totalCount = typeof paginatedResult.totalCount === \"number\"\n && Number.isFinite(paginatedResult.totalCount)\n && paginatedResult.totalCount >= 0\n ? Math.floor(paginatedResult.totalCount)\n : undefined\n\n return {\n data: Array.isArray(paginatedResult.nodes) ? paginatedResult.nodes : [],\n pageInfo: paginatedResult.pageInfo,\n ...(totalCount !== undefined ? { totalCount } : {}),\n }\n }\n\n const queryPromise = model.find(finalQuery, projection)\n if (populate !== undefined) {\n queryPromise.populate(populate as any)\n }\n if (sort && Object.keys(sort).length) {\n queryPromise.sort(sort)\n }\n queryPromise.limit(limit)\n\n const data = await queryPromise\n return { data: Array.isArray(data) ? data : [] }\n}\n"],"names":["getDerivedKey","masterKey","info","length","salt","assert","Buffer","from","hkdfSync","toString","RTS_TENANT_ID_QUERY_PARAM","RTS_USER_ID_HEADER","QUERY_MAX_LIMIT","INTERNAL_MODEL_NAMES","Set","paginationCursorSigningSecret","getPaginationCursorSigningSecret","process","env","MASTER_KEY","trim","Error","normalizeTenantId","value","normalized","normalizeSignedInTenants","Array","isArray","map","tenantId","String","filter","Boolean","getTenantIdFromRequest","req","rawQuery","query","queryTenantId","normalizedFromQuery","session","user","currentTenantId","resolveRtsRequestTenantId","isRtsRequestAuthorized","sessionUser","signedInTenants","includes","buildRtsAbilityFromRequest","sessionUserId","id","ability","buildAbilityFromSession","userId","headerValue","headers","headerUserIdRaw","headerUserId","rbCtx","User","models","getGlobal","findById","tenants","tenantRoles","lean","tenantsRaw","tenant","roles","getTenantRolesFromSessionUser","buildAbility","getTenantModel","modelName","ctx","get","normalizeLimit","limit","Number","isFinite","Math","min","abs","normalizeString","normalizeObject","undefined","normalizePagination","normalizePopulateSelect","normalizePopulateOptions","raw","sort","max","floor","normalizePopulateObject","path","model","select","match","nestedPopulate","normalizeRtsPopulateOption","populate","options","entry","normalizeModelName","resolvePopulateRefModelName","explicitModelName","schema","schemaPath","directRef","ref","arrayRef","caster","virtualPath","virtualpath","virtualRef","mergePopulateMatchWithAcl","populateMatch","aclMatch","Object","keys","$and","resolvePopulateSpecForModel","allowInternalModels","modelCache","dependencyModelNames","getModelCached","targetModelName","cached","loaded","set","resolveOne","parentModel","refModelName","has","can","add","getAccessibleByQuery","nestedModel","normalizedEntry","resolved","Promise","all","filtered","normalizeRtsQueryOptions","projection","pagination","resolveRtsQueryDependencyModelNames","Map","runRtsQuery","accessQuery","finalQuery","paginatedQuery","find","paginatedResult","paginate","cursor","signingSecret","totalCount","data","nodes","pageInfo","queryPromise"],"mappings":"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"}
|
package/dist/rts/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/rts/index.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAmB,MAAM,IAAI,UAAU,EAAE,MAAM,WAAW,CAAA;AAGtE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,SAAS,CAAA;AAG7C,OAAO,EAA6F,KAAK,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAG5I,OAAO,EAAiC,KAAK,SAAS,EAAE,MAAM,IAAI,CAAA;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/rts/index.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAmB,MAAM,IAAI,UAAU,EAAE,MAAM,WAAW,CAAA;AAGtE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,SAAS,CAAA;AAG7C,OAAO,EAA6F,KAAK,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAG5I,OAAO,EAAiC,KAAK,SAAS,EAAE,MAAM,IAAI,CAAA;AAwClE,KAAK,UAAU,GAAG;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,MAAM,EAAE,MAAM,CAAA;IACd,OAAO,EAAE,UAAU,CAAA;CACpB,CAAA;AA0BD,KAAK,SAAS,GAAG,CAAC,MAAM,EAAE,SAAS,KAAK,IAAI,GAAG,CAAC,MAAM,IAAI,CAAC,CAAA;AA6B3D,cAAM,SAAS;IACb,SAAgB,EAAE,EAAE,MAAM,CAAA;IAC1B,SAAgB,QAAQ,EAAE,MAAM,CAAA;IAChC,SAAgB,MAAM,EAAE,MAAM,CAAA;IAE9B,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAW;IAC9B,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAqD;gBAE3D,EACjB,EAAE,EACF,EAAE,EACF,IAAI,GACL,EAAE;QACD,EAAE,EAAE,MAAM,CAAA;QACV,EAAE,EAAE,SAAS,CAAA;QACb,IAAI,EAAE,UAAU,CAAA;KACjB;IAOM,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,GAAG,MAAM,IAAI;IAOlE,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,GAAG,IAAI;IAO7D,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI;IAI5C,KAAK,IAAI,IAAI;IAQb,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,IAAI;CAOvD;AA4lBD,eAAO,MAAM,OAAO,GAAI,4NAQrB;IACD,MAAM,EAAE,UAAU,CAAA;IAClB,IAAI,CAAC,EAAE,MAAM,CAAA;IACb,iBAAiB,CAAC,EAAE,cAAc,CAAA;IAClC,eAAe,CAAC,EAAE,MAAM,CAAA;IACxB,yBAAyB,CAAC,EAAE,MAAM,CAAA;IAClC,kBAAkB,CAAC,EAAE,MAAM,CAAA;IAC3B,mBAAmB,CAAC,EAAE,OAAO,CAAA;CAC9B,KAAG,IAyHH,CAAA;AAED,eAAO,MAAM,kBAAkB,GAAI,SAAS,SAAS,KAAG,IAEvD,CAAA;AAED,eAAO,MAAM,qBAAqB,GAAI,UAAU,MAAM,EAAE,WAAW,MAAM,KAAG,IAE3E,CAAA;AAED,cAAc,UAAU,CAAA"}
|
package/dist/rts/index.js
CHANGED
|
@@ -2,7 +2,7 @@ import { randomUUID } from "node:crypto";
|
|
|
2
2
|
import { models } from "@rpcbase/db";
|
|
3
3
|
import { buildAbilityFromSession, getTenantRolesFromSessionUser, buildAbility } from "@rpcbase/db/acl";
|
|
4
4
|
import { WebSocketServer } from "ws";
|
|
5
|
-
import { R as RTS_TENANT_ID_QUERY_PARAM, c as RTS_USER_ID_HEADER, n as normalizeRtsQueryOptions, d as resolveRtsQueryDependencyModelNames, a as runRtsQuery } from "../queryExecutor-
|
|
5
|
+
import { R as RTS_TENANT_ID_QUERY_PARAM, c as RTS_USER_ID_HEADER, n as normalizeRtsQueryOptions, d as resolveRtsQueryDependencyModelNames, a as runRtsQuery } from "../queryExecutor-Bg1GGL3j.js";
|
|
6
6
|
const routes = Object.entries({
|
|
7
7
|
.../* @__PURE__ */ Object.assign({ "./api/changes/handler.ts": () => import("../handler-BsauvgjA.js") })
|
|
8
8
|
}).reduce((acc, [path, mod]) => {
|
|
@@ -277,6 +277,9 @@ const runAndSendQuery = async ({
|
|
|
277
277
|
data: result.data,
|
|
278
278
|
...result.pageInfo ? {
|
|
279
279
|
pageInfo: result.pageInfo
|
|
280
|
+
} : {},
|
|
281
|
+
...typeof result.totalCount === "number" ? {
|
|
282
|
+
totalCount: result.totalCount
|
|
280
283
|
} : {}
|
|
281
284
|
};
|
|
282
285
|
for (const socketId of targetSocketIds) {
|
package/dist/rts/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sources":["../../src/rts/routes.ts","../../src/rts/index.ts"],"sourcesContent":["export const routes = Object.entries({\n ...import.meta.glob(\"./api/**/handler.ts\"),\n}).reduce<Record<string, unknown>>((acc, [path, mod]) => {\n acc[path.replace(\"./api/\", \"@rpcbase/server/rts/api/\")] = mod\n return acc\n}, {})\n\n","import { randomUUID } from \"node:crypto\"\nimport type { IncomingMessage, Server as HttpServer } from \"node:http\"\nimport type { Socket } from \"node:net\"\n\nimport type { RequestHandler } from \"express\"\nimport type { PaginationPageInfo } from \"@rpcbase/api\"\nimport { models, type LoadModelCtx } from \"@rpcbase/db\"\nimport { buildAbility, buildAbilityFromSession, getTenantRolesFromSessionUser, type AclSubjectType, type AppAbility } from \"@rpcbase/db/acl\"\nimport type { ChangeStream } from \"mongodb\"\nimport type { Model } from \"mongoose\"\nimport { WebSocketServer, type RawData, type WebSocket } from \"ws\"\n\nimport {\n RTS_TENANT_ID_QUERY_PARAM as TENANT_ID_QUERY_PARAM,\n RTS_USER_ID_HEADER as USER_ID_HEADER,\n normalizeRtsQueryOptions,\n resolveRtsQueryDependencyModelNames,\n runRtsQuery,\n type RtsQueryOptions as QueryOptions,\n} from \"./queryExecutor\"\n\n\ntype JsonObject = Record<string, unknown>\n\ntype ClientMessage =\n | { type: \"run-query\"; modelName: string; queryKey: string; query: JsonObject; options?: QueryOptions }\n | {\n type: \"register-query\"\n modelName: string\n queryKey: string\n query: JsonObject\n options?: QueryOptions\n runInitialQuery?: boolean\n }\n | { type: \"remove-query\"; modelName: string; queryKey: string }\n | { type: \"event\"; event: string; payload?: unknown }\n\ntype ServerMessage =\n | {\n type: \"query-payload\"\n modelName: string\n queryKey: string\n data?: unknown\n error?: string\n txnId?: string\n pageInfo?: PaginationPageInfo\n }\n | { type: \"event\"; event: string; payload?: unknown }\n\ntype SocketMeta = {\n tenantId: string\n userId: string\n ability: AppAbility\n}\n\ntype SessionUser = {\n id?: string\n currentTenantId?: string\n signedInTenants?: string[]\n}\n\ntype HeaderUserDoc = {\n tenants?: unknown\n tenantRoles?: unknown\n}\n\ntype UpgradeRequest = IncomingMessage & {\n session?: {\n user?: SessionUser\n }\n}\n\ntype Subscription = {\n query: JsonObject\n options: QueryOptions\n dependencyModelNames: string[]\n socketIds: Set<string>\n}\n\ntype HandlerFn = (socket: RtsSocket) => void | (() => void)\n\nconst QUERY_KEY_MAX_LEN = 4096\n\nconst INTERNAL_MODEL_NAMES = new Set([\"RBRtsChange\", \"RBRtsCounter\"])\n\nconst DEFAULT_MAX_PAYLOAD_BYTES = 1024 * 1024\nconst DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET = 256\nconst DEFAULT_DISPATCH_DEBOUNCE_MS = 25\n\nconst initializedServers = new WeakSet<HttpServer>()\nconst customHandlers: HandlerFn[] = []\n\nconst sockets = new Map<string, WebSocket>()\nconst socketMeta = new Map<string, SocketMeta>()\nconst socketWrappers = new Map<string, RtsSocket>()\nconst socketCleanup = new Map<string, Array<() => void>>()\n\nconst socketSubscriptions = new Map<string, Map<string, Set<string>>>()\nconst subscriptions = new Map<string, Map<string, Map<string, Map<string, Subscription>>>>()\nconst changeStreams = new Map<string, Map<string, ChangeStream>>()\nconst dispatchTimers = new Map<string, ReturnType<typeof setTimeout>>()\nconst upgradeMeta = new WeakMap<IncomingMessage, SocketMeta>()\n\nlet maxPayloadBytes = DEFAULT_MAX_PAYLOAD_BYTES\nlet maxSubscriptionsPerSocket = DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET\nlet dispatchDebounceMs = DEFAULT_DISPATCH_DEBOUNCE_MS\nlet allowInternalModels = false\n\nclass RtsSocket {\n public readonly id: string\n public readonly tenantId: string\n public readonly userId: string\n\n private readonly ws: WebSocket\n private readonly handlers = new Map<string, Set<(payload: unknown) => void>>()\n\n public constructor({\n id,\n ws,\n meta,\n }: {\n id: string\n ws: WebSocket\n meta: SocketMeta\n }) {\n this.id = id\n this.ws = ws\n this.tenantId = meta.tenantId\n this.userId = meta.userId\n }\n\n public on(event: string, handler: (payload: unknown) => void): () => void {\n const set = this.handlers.get(event) ?? new Set()\n set.add(handler)\n this.handlers.set(event, set)\n return () => this.off(event, handler)\n }\n\n public off(event: string, handler: (payload: unknown) => void): void {\n const set = this.handlers.get(event)\n if (!set) return\n set.delete(handler)\n if (!set.size) this.handlers.delete(event)\n }\n\n public emit(event: string, payload?: unknown): void {\n sendWs(this.ws, { type: \"event\", event, payload })\n }\n\n public close(): void {\n try {\n this.ws.close()\n } catch {\n // ignore\n }\n }\n\n public dispatch(event: string, payload: unknown): void {\n const set = this.handlers.get(event)\n if (!set) return\n for (const handler of set) {\n handler(payload)\n }\n }\n}\n\nconst rawToText = (raw: RawData): string => {\n if (typeof raw === \"string\") return raw\n if (raw instanceof ArrayBuffer) return Buffer.from(raw).toString()\n if (Array.isArray(raw)) return Buffer.concat(raw).toString()\n return raw.toString()\n}\n\nconst safeJsonParse = (raw: RawData): unknown => JSON.parse(rawToText(raw))\n\nconst sendWs = (ws: WebSocket, message: ServerMessage): void => {\n if (ws.readyState !== 1) return\n ws.send(JSON.stringify(message))\n}\n\nconst ensureSocketErrorHandler = (socket: Socket, context?: () => Record<string, unknown>): void => {\n if (socket.listenerCount(\"error\") > 0) return\n\n socket.on(\"error\", (err: unknown) => {\n const message = err instanceof Error ? err.message : String(err)\n const name = err instanceof Error ? err.name : typeof err\n const stack = err instanceof Error ? err.stack : undefined\n const code = err && typeof err === \"object\" && \"code\" in err ? (err as any).code : undefined\n\n if (code === \"ECONNRESET\" || (code == null && message.includes(\"ECONNRESET\"))) return\n\n console.warn(\"[rb/rts] socket error\", {\n name,\n code,\n message,\n stack,\n remoteAddress: socket.remoteAddress,\n remotePort: socket.remotePort,\n localAddress: socket.localAddress,\n localPort: socket.localPort,\n ...context?.(),\n })\n })\n}\n\nconst redactErrorMessage = (err: unknown): string => {\n const raw = err instanceof Error ? err.message : \"Unknown error\"\n const trimmedModelList = raw.replace(/\\.\\s+Available models:[\\s\\S]*$/, \"\")\n const maxLen = 256\n if (trimmedModelList.length <= maxLen) return trimmedModelList\n return trimmedModelList.slice(0, maxLen)\n}\n\nconst unauthorized = (socket: Socket, message = \"Unauthorized\"): void => {\n ensureSocketErrorHandler(socket)\n try {\n socket.write(\"HTTP/1.1 401 Unauthorized\\r\\n\\r\\n\")\n socket.write(`Error: ${message}\\r\\n`)\n socket.end()\n } catch {\n socket.destroy()\n }\n}\n\nconst badRequest = (socket: Socket, message = \"Bad Request\"): void => {\n ensureSocketErrorHandler(socket)\n try {\n socket.write(\"HTTP/1.1 400 Bad Request\\r\\n\\r\\n\")\n socket.write(`Error: ${message}\\r\\n`)\n socket.end()\n } catch {\n socket.destroy()\n }\n}\n\nconst runSessionMiddleware = async (sessionMiddleware: RequestHandler, req: UpgradeRequest): Promise<void> => {\n type MiddlewareReq = Parameters<RequestHandler>[0]\n type MiddlewareRes = Parameters<RequestHandler>[1]\n type MiddlewareNext = Parameters<RequestHandler>[2]\n\n await new Promise<void>((resolve, reject) => {\n const next: MiddlewareNext = (err) => {\n if (err) reject(err)\n else resolve()\n }\n\n sessionMiddleware(req as unknown as MiddlewareReq, {} as unknown as MiddlewareRes, next)\n })\n}\n\nconst parseUpgradeMeta = async ({\n req,\n url,\n sessionMiddleware,\n}: {\n req: IncomingMessage\n url: URL\n sessionMiddleware?: RequestHandler\n}): Promise<SocketMeta> => {\n const tenantId = url.searchParams.get(TENANT_ID_QUERY_PARAM)\n if (!tenantId) {\n throw new Error(\"Missing rb-tenant-id query parameter\")\n }\n\n if (sessionMiddleware) {\n const upgradeReq = req as UpgradeRequest\n try {\n await runSessionMiddleware(sessionMiddleware, upgradeReq)\n } catch {\n throw new Error(\"Failed to load session for RTS\")\n }\n\n const sessionUser = upgradeReq.session?.user\n const sessionUserId = sessionUser?.id\n if (!sessionUserId) {\n throw new Error(\"Not signed in (missing session.user.id)\")\n }\n\n const signedInTenants = sessionUser?.signedInTenants\n const currentTenantId = sessionUser?.currentTenantId\n\n if (Array.isArray(signedInTenants) && signedInTenants.length > 0) {\n if (!signedInTenants.includes(tenantId)) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n } else if (currentTenantId) {\n if (currentTenantId !== tenantId) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n } else {\n throw new Error(\"Tenant not authorized for this session\")\n }\n\n const ability = buildAbilityFromSession({ tenantId, session: upgradeReq.session })\n return { tenantId, userId: sessionUserId, ability }\n }\n\n const raw = req.headers[USER_ID_HEADER]\n const headerUserId = Array.isArray(raw) ? raw[0] : raw\n if (!headerUserId) {\n throw new Error(\"Missing rb-user-id header (reverse-proxy) and no session middleware configured\")\n }\n\n const rbCtx: LoadModelCtx = { req: { session: null } }\n const User = await models.getGlobal(\"RBUser\", rbCtx)\n const user = await User.findById(headerUserId, { tenants: 1, tenantRoles: 1 }).lean() as HeaderUserDoc | null\n const tenantsRaw = user?.tenants\n const tenants = Array.isArray(tenantsRaw) ? tenantsRaw.map((t) => String(t)) : []\n if (!tenants.includes(tenantId)) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n\n const roles = getTenantRolesFromSessionUser(user, tenantId)\n const ability = buildAbility({ tenantId, userId: headerUserId, roles: roles.length ? roles : [\"owner\"] })\n\n return { tenantId, userId: headerUserId, ability }\n}\n\nconst getTenantModel = async (tenantId: string, modelName: string): Promise<Model<any>> => {\n const ctx: LoadModelCtx = {\n req: {\n session: {\n user: {\n currentTenantId: tenantId,\n },\n },\n },\n }\n\n return models.getUnsafe(modelName, ctx)\n}\n\nconst makeDispatchKey = (tenantId: string, modelName: string): string => `${tenantId}:${modelName}`\n\nconst clearDispatchTimer = (tenantId: string, modelName: string): void => {\n const key = makeDispatchKey(tenantId, modelName)\n const timer = dispatchTimers.get(key)\n if (!timer) return\n clearTimeout(timer)\n dispatchTimers.delete(key)\n}\n\nconst scheduleDispatchSubscriptionsForModel = (tenantId: string, modelName: string): void => {\n const key = makeDispatchKey(tenantId, modelName)\n if (dispatchTimers.has(key)) return\n\n const delay = Math.max(0, Math.min(1000, Math.floor(dispatchDebounceMs)))\n dispatchTimers.set(key, setTimeout(() => {\n dispatchTimers.delete(key)\n void dispatchSubscriptionsForModel(tenantId, modelName)\n }, delay))\n}\n\nconst runAndSendQuery = async ({\n tenantId,\n targetSocketIds,\n ability,\n modelName,\n queryKey,\n query,\n options,\n}: {\n tenantId: string\n targetSocketIds: string[]\n ability: AppAbility\n modelName: string\n queryKey: string\n query: JsonObject\n options: QueryOptions\n}): Promise<void> => {\n const result = await runRtsQuery({\n tenantId,\n ability,\n modelName,\n query,\n options,\n allowInternalModels,\n })\n const payload: ServerMessage = {\n type: \"query-payload\",\n modelName,\n queryKey,\n data: result.data,\n ...(result.pageInfo ? { pageInfo: result.pageInfo } : {}),\n }\n\n for (const socketId of targetSocketIds) {\n const ws = sockets.get(socketId)\n if (!ws) continue\n sendWs(ws, payload)\n }\n}\n\nconst subscriptionDependsOnModel = (\n changedModelName: string,\n ownerModelName: string,\n subscription: Subscription,\n): boolean => {\n if (ownerModelName === changedModelName) return true\n return subscription.dependencyModelNames.includes(changedModelName)\n}\n\nconst hasAnySubscriptionsDependingOnModel = (\n tenantId: string,\n modelName: string,\n): boolean => {\n const tenantSubs = subscriptions.get(tenantId)\n if (!tenantSubs) return false\n\n for (const userSubs of tenantSubs.values()) {\n for (const [ownerModelName, modelSubs] of userSubs.entries()) {\n for (const subscription of modelSubs.values()) {\n if (subscriptionDependsOnModel(modelName, ownerModelName, subscription)) {\n return true\n }\n }\n }\n }\n\n return false\n}\n\nconst dispatchSubscriptionsForModel = async (tenantId: string, modelName: string): Promise<void> => {\n const tenantSubs = subscriptions.get(tenantId)\n if (!tenantSubs || !tenantSubs.size) return\n\n for (const userSubs of tenantSubs.values()) {\n for (const [ownerModelName, modelSubs] of userSubs.entries()) {\n for (const [queryKey, sub] of modelSubs.entries()) {\n if (!subscriptionDependsOnModel(modelName, ownerModelName, sub)) continue\n\n const targetSocketIds = Array.from(sub.socketIds)\n if (!targetSocketIds.length) continue\n\n const socketId = targetSocketIds[0]\n const meta = socketMeta.get(socketId)\n const ability = meta?.ability\n if (!ability) continue\n\n try {\n await runAndSendQuery({\n tenantId,\n targetSocketIds,\n ability,\n modelName: ownerModelName,\n queryKey,\n query: sub.query,\n options: sub.options,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n const payload: ServerMessage = { type: \"query-payload\", modelName: ownerModelName, queryKey, error }\n for (const socketId of targetSocketIds) {\n const ws = sockets.get(socketId)\n if (!ws) continue\n sendWs(ws, payload)\n }\n }\n }\n }\n }\n}\n\nconst ensureChangeStream = async (tenantId: string, modelName: string): Promise<void> => {\n const tenantStreams = changeStreams.get(tenantId) ?? new Map()\n changeStreams.set(tenantId, tenantStreams)\n if (tenantStreams.has(modelName)) return\n\n const model = await getTenantModel(tenantId, modelName)\n\n const stream = model.watch([], {\n fullDocument: \"updateLookup\",\n })\n\n stream.on(\"change\", () => {\n scheduleDispatchSubscriptionsForModel(tenantId, modelName)\n })\n\n stream.on(\"close\", () => {\n clearDispatchTimer(tenantId, modelName)\n const map = changeStreams.get(tenantId)\n map?.delete(modelName)\n if (map && map.size === 0) changeStreams.delete(tenantId)\n })\n\n stream.on(\"error\", () => {\n try {\n clearDispatchTimer(tenantId, modelName)\n stream.close()\n } catch {\n // ignore\n }\n })\n\n tenantStreams.set(modelName, stream)\n}\n\nconst addSocketSubscription = ({\n socketId,\n tenantId,\n userId,\n modelName,\n queryKey,\n query,\n options,\n dependencyModelNames,\n}: {\n socketId: string\n tenantId: string\n userId: string\n modelName: string\n queryKey: string\n query: JsonObject\n options: QueryOptions\n dependencyModelNames: string[]\n}): void => {\n const tenantSubs = subscriptions.get(tenantId) ?? new Map<string, Map<string, Map<string, Subscription>>>()\n subscriptions.set(tenantId, tenantSubs)\n\n const userSubs = tenantSubs.get(userId) ?? new Map<string, Map<string, Subscription>>()\n tenantSubs.set(userId, userSubs)\n\n const modelSubs = userSubs.get(modelName) ?? new Map<string, Subscription>()\n userSubs.set(modelName, modelSubs)\n\n const existing = modelSubs.get(queryKey)\n if (existing) {\n existing.socketIds.add(socketId)\n existing.dependencyModelNames = Array.from(new Set([...existing.dependencyModelNames, ...dependencyModelNames]))\n } else {\n modelSubs.set(queryKey, {\n query,\n options,\n dependencyModelNames: Array.from(new Set(dependencyModelNames)),\n socketIds: new Set([socketId]),\n })\n }\n\n const byModel = socketSubscriptions.get(socketId) ?? new Map()\n socketSubscriptions.set(socketId, byModel)\n\n const querySet = byModel.get(modelName) ?? new Set()\n byModel.set(modelName, querySet)\n querySet.add(queryKey)\n}\n\nconst removeSocketSubscription = ({\n socketId,\n tenantId,\n userId,\n modelName,\n queryKey,\n}: {\n socketId: string\n tenantId: string\n userId: string\n modelName: string\n queryKey: string\n}): void => {\n const tenantSubs = subscriptions.get(tenantId)\n const userSubs = tenantSubs?.get(userId)\n const modelSubs = userSubs?.get(modelName)\n const sub = modelSubs?.get(queryKey)\n const affectedModelNames = new Set<string>([modelName, ...(sub?.dependencyModelNames ?? [])])\n if (sub) {\n sub.socketIds.delete(socketId)\n if (!sub.socketIds.size) {\n modelSubs?.delete(queryKey)\n }\n }\n\n const byModel = socketSubscriptions.get(socketId)\n const set = byModel?.get(modelName)\n if (set) {\n set.delete(queryKey)\n if (!set.size) {\n byModel?.delete(modelName)\n }\n }\n\n if (modelSubs && modelSubs.size === 0) {\n userSubs?.delete(modelName)\n }\n\n if (userSubs && userSubs.size === 0) {\n tenantSubs?.delete(userId)\n }\n\n for (const affectedModelName of affectedModelNames) {\n if (hasAnySubscriptionsDependingOnModel(tenantId, affectedModelName)) continue\n\n const tenantStreams = changeStreams.get(tenantId)\n const stream = tenantStreams?.get(affectedModelName)\n if (stream) {\n try {\n stream.close()\n } catch {\n // ignore\n }\n clearDispatchTimer(tenantId, affectedModelName)\n tenantStreams?.delete(affectedModelName)\n if (tenantStreams && tenantStreams.size === 0) changeStreams.delete(tenantId)\n }\n }\n\n if (tenantSubs && tenantSubs.size === 0) subscriptions.delete(tenantId)\n if (byModel && byModel.size === 0) socketSubscriptions.delete(socketId)\n}\n\nconst cleanupSocket = (socketId: string): void => {\n const meta = socketMeta.get(socketId)\n if (meta) {\n const byModel = socketSubscriptions.get(socketId)\n if (byModel) {\n for (const [modelName, keys] of byModel.entries()) {\n for (const queryKey of keys.values()) {\n removeSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName,\n queryKey,\n })\n }\n }\n }\n }\n\n socketSubscriptions.delete(socketId)\n\n const cleanupFns = socketCleanup.get(socketId) ?? []\n socketCleanup.delete(socketId)\n for (const fn of cleanupFns) {\n try {\n fn()\n } catch {\n // ignore\n }\n }\n\n sockets.delete(socketId)\n socketMeta.delete(socketId)\n socketWrappers.delete(socketId)\n}\n\nconst handleClientMessage = async ({\n socketId,\n meta,\n message,\n}: {\n socketId: string\n meta: SocketMeta\n message: ClientMessage\n}): Promise<void> => {\n const ws = sockets.get(socketId)\n if (!ws) return\n\n if (message.type === \"event\") {\n const wrapper = socketWrappers.get(socketId)\n wrapper?.dispatch(message.event, message.payload)\n return\n }\n\n if (!message.modelName || typeof message.modelName !== \"string\") return\n if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(message.modelName)) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey ?? \"\", error: \"Model not allowed\" })\n return\n }\n if (!message.queryKey || typeof message.queryKey !== \"string\") return\n if (message.queryKey.length > QUERY_KEY_MAX_LEN) return\n\n if (message.type === \"remove-query\") {\n removeSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName: message.modelName,\n queryKey: message.queryKey,\n })\n return\n }\n\n if (!message.query || typeof message.query !== \"object\") return\n\n const options = normalizeRtsQueryOptions(message.options)\n const ability = meta.ability\n\n if (!ability.can(\"read\", message.modelName as AclSubjectType)) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error: \"forbidden\" })\n return\n }\n\n if (message.type === \"register-query\") {\n const existing = socketSubscriptions.get(socketId)?.get(message.modelName)?.has(message.queryKey) ?? false\n if (!existing) {\n let count = 0\n const byModel = socketSubscriptions.get(socketId)\n if (byModel) {\n for (const set of byModel.values()) count += set.size\n }\n\n if (count >= maxSubscriptionsPerSocket) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error: \"Too many subscriptions\" })\n return\n }\n }\n\n let dependencyModelNames: string[] = []\n if (options.populate !== undefined) {\n try {\n dependencyModelNames = await resolveRtsQueryDependencyModelNames({\n tenantId: meta.tenantId,\n ability,\n modelName: message.modelName,\n options,\n allowInternalModels,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n return\n }\n }\n\n addSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName: message.modelName,\n queryKey: message.queryKey,\n query: message.query,\n options,\n dependencyModelNames,\n })\n\n try {\n const modelNamesToWatch = new Set<string>([message.modelName, ...dependencyModelNames])\n for (const modelName of modelNamesToWatch) {\n await ensureChangeStream(meta.tenantId, modelName)\n }\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n return\n }\n\n if (message.runInitialQuery === false) {\n return\n }\n }\n\n try {\n await runAndSendQuery({\n tenantId: meta.tenantId,\n targetSocketIds: [socketId],\n ability,\n modelName: message.modelName,\n queryKey: message.queryKey,\n query: message.query,\n options,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n }\n}\n\nexport const initRts = ({\n server,\n path = \"/rts\",\n sessionMiddleware,\n maxPayloadBytes: maxPayloadBytesArg,\n maxSubscriptionsPerSocket: maxSubscriptionsPerSocketArg,\n dispatchDebounceMs: dispatchDebounceMsArg,\n allowInternalModels: allowInternalModelsArg,\n}: {\n server: HttpServer\n path?: string\n sessionMiddleware?: RequestHandler\n maxPayloadBytes?: number\n maxSubscriptionsPerSocket?: number\n dispatchDebounceMs?: number\n allowInternalModels?: boolean\n}): void => {\n if (initializedServers.has(server)) return\n initializedServers.add(server)\n\n if (typeof maxPayloadBytesArg === \"number\" && Number.isFinite(maxPayloadBytesArg) && maxPayloadBytesArg > 0) {\n maxPayloadBytes = Math.floor(maxPayloadBytesArg)\n }\n\n if (\n typeof maxSubscriptionsPerSocketArg === \"number\"\n && Number.isFinite(maxSubscriptionsPerSocketArg)\n && maxSubscriptionsPerSocketArg > 0\n ) {\n maxSubscriptionsPerSocket = Math.floor(maxSubscriptionsPerSocketArg)\n }\n\n if (typeof dispatchDebounceMsArg === \"number\" && Number.isFinite(dispatchDebounceMsArg) && dispatchDebounceMsArg >= 0) {\n dispatchDebounceMs = Math.floor(dispatchDebounceMsArg)\n }\n\n allowInternalModels = Boolean(allowInternalModelsArg)\n\n const wss = new WebSocketServer({ noServer: true, maxPayload: maxPayloadBytes })\n\n server.on(\"upgrade\", (req: IncomingMessage, socket: Socket, head: Buffer) => {\n ensureSocketErrorHandler(socket, () => ({\n upgradeHost: req.headers.host ?? \"\",\n upgradeUrl: req.url ?? \"\",\n userAgent: typeof req.headers[\"user-agent\"] === \"string\" ? req.headers[\"user-agent\"] : \"\",\n }))\n upgradeMeta.delete(req)\n\n let url: URL\n try {\n url = new URL(req.url ?? \"\", `http://${req.headers.host ?? \"localhost\"}`)\n } catch {\n badRequest(socket, \"Invalid URL\")\n return\n }\n\n if (url.pathname !== path) return\n\n void (async() => {\n try {\n const meta = await parseUpgradeMeta({ req, url, sessionMiddleware })\n upgradeMeta.set(req, meta)\n\n wss.handleUpgrade(req, socket, head, (ws: WebSocket) => {\n wss.emit(\"connection\", ws, req)\n })\n } catch (err) {\n const message = err instanceof Error ? err.message : \"RTS upgrade failed\"\n if (message.startsWith(\"Missing rb-tenant-id\")) {\n badRequest(socket, message)\n return\n }\n unauthorized(socket, message)\n return\n }\n })().catch(() => {\n badRequest(socket, \"RTS upgrade failed\")\n })\n })\n\n wss.on(\"connection\", (ws: WebSocket, req: IncomingMessage) => {\n const meta = upgradeMeta.get(req)\n upgradeMeta.delete(req)\n if (!meta) {\n try {\n ws.close()\n } catch {\n // ignore\n }\n return\n }\n\n const socketId = randomUUID()\n sockets.set(socketId, ws)\n socketMeta.set(socketId, meta)\n\n const wrapper = new RtsSocket({ id: socketId, ws, meta })\n socketWrappers.set(socketId, wrapper)\n\n const cleanupFns: Array<() => void> = []\n for (const handler of customHandlers) {\n try {\n const cleanup = handler(wrapper)\n if (typeof cleanup === \"function\") cleanupFns.push(cleanup)\n } catch {\n // ignore\n }\n }\n if (cleanupFns.length) socketCleanup.set(socketId, cleanupFns)\n\n ws.on(\"message\", (raw: RawData) => {\n let parsed: unknown\n try {\n parsed = safeJsonParse(raw)\n } catch {\n return\n }\n\n if (!parsed || typeof parsed !== \"object\") return\n const message = parsed as { type?: unknown }\n if (\n message.type !== \"event\"\n && message.type !== \"run-query\"\n && message.type !== \"register-query\"\n && message.type !== \"remove-query\"\n ) return\n void handleClientMessage({ socketId, meta, message: message as ClientMessage })\n })\n\n ws.on(\"close\", () => {\n cleanupSocket(socketId)\n })\n\n ws.on(\"error\", () => {\n cleanupSocket(socketId)\n })\n })\n}\n\nexport const registerRtsHandler = (handler: HandlerFn): void => {\n customHandlers.push(handler)\n}\n\nexport const notifyRtsModelChanged = (tenantId: string, modelName: string): void => {\n scheduleDispatchSubscriptionsForModel(tenantId, modelName)\n}\n\nexport * from \"./routes\"\n"],"names":["routes","Object","entries","import","reduce","acc","path","mod","replace","QUERY_KEY_MAX_LEN","INTERNAL_MODEL_NAMES","Set","DEFAULT_MAX_PAYLOAD_BYTES","DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET","DEFAULT_DISPATCH_DEBOUNCE_MS","initializedServers","WeakSet","customHandlers","sockets","Map","socketMeta","socketWrappers","socketCleanup","socketSubscriptions","subscriptions","changeStreams","dispatchTimers","upgradeMeta","WeakMap","maxPayloadBytes","maxSubscriptionsPerSocket","dispatchDebounceMs","allowInternalModels","RtsSocket","id","tenantId","userId","ws","handlers","constructor","meta","on","event","handler","set","get","add","off","delete","size","emit","payload","sendWs","type","close","dispatch","rawToText","raw","ArrayBuffer","Buffer","from","toString","Array","isArray","concat","safeJsonParse","JSON","parse","message","readyState","send","stringify","ensureSocketErrorHandler","socket","context","listenerCount","err","Error","String","name","stack","undefined","code","includes","console","warn","remoteAddress","remotePort","localAddress","localPort","redactErrorMessage","trimmedModelList","maxLen","length","slice","unauthorized","write","end","destroy","badRequest","runSessionMiddleware","sessionMiddleware","req","Promise","resolve","reject","next","parseUpgradeMeta","url","searchParams","TENANT_ID_QUERY_PARAM","upgradeReq","sessionUser","session","user","sessionUserId","signedInTenants","currentTenantId","ability","buildAbilityFromSession","headers","USER_ID_HEADER","headerUserId","rbCtx","User","models","getGlobal","findById","tenants","tenantRoles","lean","tenantsRaw","map","t","roles","getTenantRolesFromSessionUser","buildAbility","getTenantModel","modelName","ctx","getUnsafe","makeDispatchKey","clearDispatchTimer","key","timer","clearTimeout","scheduleDispatchSubscriptionsForModel","has","delay","Math","max","min","floor","setTimeout","dispatchSubscriptionsForModel","runAndSendQuery","targetSocketIds","queryKey","query","options","result","runRtsQuery","data","pageInfo","socketId","subscriptionDependsOnModel","changedModelName","ownerModelName","subscription","dependencyModelNames","hasAnySubscriptionsDependingOnModel","tenantSubs","userSubs","values","modelSubs","sub","socketIds","error","ensureChangeStream","tenantStreams","model","stream","watch","fullDocument","addSocketSubscription","existing","byModel","querySet","removeSocketSubscription","affectedModelNames","affectedModelName","cleanupSocket","keys","cleanupFns","fn","handleClientMessage","wrapper","normalizeRtsQueryOptions","can","count","populate","resolveRtsQueryDependencyModelNames","modelNamesToWatch","runInitialQuery","initRts","server","maxPayloadBytesArg","maxSubscriptionsPerSocketArg","dispatchDebounceMsArg","allowInternalModelsArg","Number","isFinite","Boolean","wss","WebSocketServer","noServer","maxPayload","head","upgradeHost","host","upgradeUrl","userAgent","URL","pathname","handleUpgrade","startsWith","catch","randomUUID","cleanup","push","parsed","registerRtsHandler","notifyRtsModelChanged"],"mappings":";;;;;AAAO,MAAMA,SAASC,OAAOC,QAAQ;AAAA,EACnC,GAAGC,uBAAAA,OAAAA,EAAAA,4BAAAA,MAAAA,OAAAA,wBAAAA,EAAAA,CAAAA;AACL,CAAC,EAAEC,OAAgC,CAACC,KAAK,CAACC,MAAMC,GAAG,MAAM;AACvDF,MAAIC,KAAKE,QAAQ,UAAU,0BAA0B,CAAC,IAAID;AAC1D,SAAOF;AACT,GAAG,CAAA,CAAE;AC4EL,MAAMI,oBAAoB;AAE1B,MAAMC,uBAAuB,oBAAIC,IAAI,CAAC,eAAe,cAAc,CAAC;AAEpE,MAAMC,4BAA4B,OAAO;AACzC,MAAMC,uCAAuC;AAC7C,MAAMC,+BAA+B;AAErC,MAAMC,yCAAyBC,QAAAA;AAC/B,MAAMC,iBAA8B,CAAA;AAEpC,MAAMC,8BAAcC,IAAAA;AACpB,MAAMC,iCAAiBD,IAAAA;AACvB,MAAME,qCAAqBF,IAAAA;AAC3B,MAAMG,oCAAoBH,IAAAA;AAE1B,MAAMI,0CAA0BJ,IAAAA;AAChC,MAAMK,oCAAoBL,IAAAA;AAC1B,MAAMM,oCAAoBN,IAAAA;AAC1B,MAAMO,qCAAqBP,IAAAA;AAC3B,MAAMQ,kCAAkBC,QAAAA;AAExB,IAAIC,kBAAkBjB;AACtB,IAAIkB,4BAA4BjB;AAChC,IAAIkB,qBAAqBjB;AACzB,IAAIkB,sBAAsB;AAE1B,MAAMC,UAAU;AAAA,EACEC;AAAAA,EACAC;AAAAA,EACAC;AAAAA,EAECC;AAAAA,EACAC,+BAAenB,IAAAA;AAAAA,EAEzBoB,YAAY;AAAA,IACjBL;AAAAA,IACAG;AAAAA,IACAG;AAAAA,EAAAA,GAKC;AACD,SAAKN,KAAKA;AACV,SAAKG,KAAKA;AACV,SAAKF,WAAWK,KAAKL;AACrB,SAAKC,SAASI,KAAKJ;AAAAA,EACrB;AAAA,EAEOK,GAAGC,OAAeC,SAAiD;AACxE,UAAMC,MAAM,KAAKN,SAASO,IAAIH,KAAK,yBAAS/B,IAAAA;AAC5CiC,QAAIE,IAAIH,OAAO;AACf,SAAKL,SAASM,IAAIF,OAAOE,GAAG;AAC5B,WAAO,MAAM,KAAKG,IAAIL,OAAOC,OAAO;AAAA,EACtC;AAAA,EAEOI,IAAIL,OAAeC,SAA2C;AACnE,UAAMC,MAAM,KAAKN,SAASO,IAAIH,KAAK;AACnC,QAAI,CAACE,IAAK;AACVA,QAAII,OAAOL,OAAO;AAClB,QAAI,CAACC,IAAIK,KAAM,MAAKX,SAASU,OAAON,KAAK;AAAA,EAC3C;AAAA,EAEOQ,KAAKR,OAAeS,SAAyB;AAClDC,WAAO,KAAKf,IAAI;AAAA,MAAEgB,MAAM;AAAA,MAASX;AAAAA,MAAOS;AAAAA,IAAAA,CAAS;AAAA,EACnD;AAAA,EAEOG,QAAc;AACnB,QAAI;AACF,WAAKjB,GAAGiB,MAAAA;AAAAA,IACV,QAAQ;AAAA,IACN;AAAA,EAEJ;AAAA,EAEOC,SAASb,OAAeS,SAAwB;AACrD,UAAMP,MAAM,KAAKN,SAASO,IAAIH,KAAK;AACnC,QAAI,CAACE,IAAK;AACV,eAAWD,WAAWC,KAAK;AACzBD,cAAQQ,OAAO;AAAA,IACjB;AAAA,EACF;AACF;AAEA,MAAMK,YAAYA,CAACC,QAAyB;AAC1C,MAAI,OAAOA,QAAQ,SAAU,QAAOA;AACpC,MAAIA,eAAeC,YAAa,QAAOC,OAAOC,KAAKH,GAAG,EAAEI,SAAAA;AACxD,MAAIC,MAAMC,QAAQN,GAAG,UAAUE,OAAOK,OAAOP,GAAG,EAAEI,SAAAA;AAClD,SAAOJ,IAAII,SAAAA;AACb;AAEA,MAAMI,gBAAgBA,CAACR,QAA0BS,KAAKC,MAAMX,UAAUC,GAAG,CAAC;AAE1E,MAAML,SAASA,CAACf,IAAe+B,YAAiC;AAC9D,MAAI/B,GAAGgC,eAAe,EAAG;AACzBhC,KAAGiC,KAAKJ,KAAKK,UAAUH,OAAO,CAAC;AACjC;AAEA,MAAMI,2BAA2BA,CAACC,QAAgBC,YAAkD;AAClG,MAAID,OAAOE,cAAc,OAAO,IAAI,EAAG;AAEvCF,SAAOhC,GAAG,SAAS,CAACmC,QAAiB;AACnC,UAAMR,UAAUQ,eAAeC,QAAQD,IAAIR,UAAUU,OAAOF,GAAG;AAC/D,UAAMG,OAAOH,eAAeC,QAAQD,IAAIG,OAAO,OAAOH;AACtD,UAAMI,QAAQJ,eAAeC,QAAQD,IAAII,QAAQC;AACjD,UAAMC,OAAON,OAAO,OAAOA,QAAQ,YAAY,UAAUA,MAAOA,IAAYM,OAAOD;AAEnF,QAAIC,SAAS,gBAAiBA,QAAQ,QAAQd,QAAQe,SAAS,YAAY,EAAI;AAE/EC,YAAQC,KAAK,yBAAyB;AAAA,MACpCN;AAAAA,MACAG;AAAAA,MACAd;AAAAA,MACAY;AAAAA,MACAM,eAAeb,OAAOa;AAAAA,MACtBC,YAAYd,OAAOc;AAAAA,MACnBC,cAAcf,OAAOe;AAAAA,MACrBC,WAAWhB,OAAOgB;AAAAA,MAClB,GAAGf,UAAAA;AAAAA,IAAU,CACd;AAAA,EACH,CAAC;AACH;AAEA,MAAMgB,qBAAqBA,CAACd,QAAyB;AACnD,QAAMnB,MAAMmB,eAAeC,QAAQD,IAAIR,UAAU;AACjD,QAAMuB,mBAAmBlC,IAAIjD,QAAQ,kCAAkC,EAAE;AACzE,QAAMoF,SAAS;AACf,MAAID,iBAAiBE,UAAUD,OAAQ,QAAOD;AAC9C,SAAOA,iBAAiBG,MAAM,GAAGF,MAAM;AACzC;AAEA,MAAMG,eAAeA,CAACtB,QAAgBL,UAAU,mBAAyB;AACvEI,2BAAyBC,MAAM;AAC/B,MAAI;AACFA,WAAOuB,MAAM,mCAAmC;AAChDvB,WAAOuB,MAAM,UAAU5B,OAAO;AAAA,CAAM;AACpCK,WAAOwB,IAAAA;AAAAA,EACT,QAAQ;AACNxB,WAAOyB,QAAAA;AAAAA,EACT;AACF;AAEA,MAAMC,aAAaA,CAAC1B,QAAgBL,UAAU,kBAAwB;AACpEI,2BAAyBC,MAAM;AAC/B,MAAI;AACFA,WAAOuB,MAAM,kCAAkC;AAC/CvB,WAAOuB,MAAM,UAAU5B,OAAO;AAAA,CAAM;AACpCK,WAAOwB,IAAAA;AAAAA,EACT,QAAQ;AACNxB,WAAOyB,QAAAA;AAAAA,EACT;AACF;AAEA,MAAME,uBAAuB,OAAOC,mBAAmCC,QAAuC;AAK5G,QAAM,IAAIC,QAAc,CAACC,SAASC,WAAW;AAC3C,UAAMC,OAAwB9B,CAAAA,QAAQ;AACpC,UAAIA,YAAYA,GAAG;AAAA,UACd4B,SAAAA;AAAAA,IACP;AAEAH,sBAAkBC,KAAiC,CAAA,GAAgCI,IAAI;AAAA,EACzF,CAAC;AACH;AAEA,MAAMC,mBAAmB,OAAO;AAAA,EAC9BL;AAAAA,EACAM;AAAAA,EACAP;AAKF,MAA2B;AACzB,QAAMlE,WAAWyE,IAAIC,aAAahE,IAAIiE,yBAAqB;AAC3D,MAAI,CAAC3E,UAAU;AACb,UAAM,IAAI0C,MAAM,sCAAsC;AAAA,EACxD;AAEA,MAAIwB,mBAAmB;AACrB,UAAMU,aAAaT;AACnB,QAAI;AACF,YAAMF,qBAAqBC,mBAAmBU,UAAU;AAAA,IAC1D,QAAQ;AACN,YAAM,IAAIlC,MAAM,gCAAgC;AAAA,IAClD;AAEA,UAAMmC,cAAcD,WAAWE,SAASC;AACxC,UAAMC,gBAAgBH,aAAa9E;AACnC,QAAI,CAACiF,eAAe;AAClB,YAAM,IAAItC,MAAM,yCAAyC;AAAA,IAC3D;AAEA,UAAMuC,kBAAkBJ,aAAaI;AACrC,UAAMC,kBAAkBL,aAAaK;AAErC,QAAIvD,MAAMC,QAAQqD,eAAe,KAAKA,gBAAgBvB,SAAS,GAAG;AAChE,UAAI,CAACuB,gBAAgBjC,SAAShD,QAAQ,GAAG;AACvC,cAAM,IAAI0C,MAAM,wCAAwC;AAAA,MAC1D;AAAA,IACF,WAAWwC,iBAAiB;AAC1B,UAAIA,oBAAoBlF,UAAU;AAChC,cAAM,IAAI0C,MAAM,wCAAwC;AAAA,MAC1D;AAAA,IACF,OAAO;AACL,YAAM,IAAIA,MAAM,wCAAwC;AAAA,IAC1D;AAEA,UAAMyC,WAAUC,wBAAwB;AAAA,MAAEpF;AAAAA,MAAU8E,SAASF,WAAWE;AAAAA,IAAAA,CAAS;AACjF,WAAO;AAAA,MAAE9E;AAAAA,MAAUC,QAAQ+E;AAAAA,MAAeG,SAAAA;AAAAA,IAAAA;AAAAA,EAC5C;AAEA,QAAM7D,MAAM6C,IAAIkB,QAAQC,kBAAc;AACtC,QAAMC,eAAe5D,MAAMC,QAAQN,GAAG,IAAIA,IAAI,CAAC,IAAIA;AACnD,MAAI,CAACiE,cAAc;AACjB,UAAM,IAAI7C,MAAM,gFAAgF;AAAA,EAClG;AAEA,QAAM8C,QAAsB;AAAA,IAAErB,KAAK;AAAA,MAAEW,SAAS;AAAA,IAAA;AAAA,EAAK;AACnD,QAAMW,OAAO,MAAMC,OAAOC,UAAU,UAAUH,KAAK;AACnD,QAAMT,OAAO,MAAMU,KAAKG,SAASL,cAAc;AAAA,IAAEM,SAAS;AAAA,IAAGC,aAAa;AAAA,EAAA,CAAG,EAAEC,KAAAA;AAC/E,QAAMC,aAAajB,MAAMc;AACzB,QAAMA,UAAUlE,MAAMC,QAAQoE,UAAU,IAAIA,WAAWC,IAAKC,CAAAA,MAAMvD,OAAOuD,CAAC,CAAC,IAAI,CAAA;AAC/E,MAAI,CAACL,QAAQ7C,SAAShD,QAAQ,GAAG;AAC/B,UAAM,IAAI0C,MAAM,wCAAwC;AAAA,EAC1D;AAEA,QAAMyD,QAAQC,8BAA8BrB,MAAM/E,QAAQ;AAC1D,QAAMmF,UAAUkB,aAAa;AAAA,IAAErG;AAAAA,IAAUC,QAAQsF;AAAAA,IAAcY,OAAOA,MAAMzC,SAASyC,QAAQ,CAAC,OAAO;AAAA,EAAA,CAAG;AAExG,SAAO;AAAA,IAAEnG;AAAAA,IAAUC,QAAQsF;AAAAA,IAAcJ;AAAAA,EAAAA;AAC3C;AAEA,MAAMmB,iBAAiB,OAAOtG,UAAkBuG,cAA2C;AACzF,QAAMC,MAAoB;AAAA,IACxBrC,KAAK;AAAA,MACHW,SAAS;AAAA,QACPC,MAAM;AAAA,UACJG,iBAAiBlF;AAAAA,QAAAA;AAAAA,MACnB;AAAA,IACF;AAAA,EACF;AAGF,SAAO0F,OAAOe,UAAUF,WAAWC,GAAG;AACxC;AAEA,MAAME,kBAAkBA,CAAC1G,UAAkBuG,cAA8B,GAAGvG,QAAQ,IAAIuG,SAAS;AAEjG,MAAMI,qBAAqBA,CAAC3G,UAAkBuG,cAA4B;AACxE,QAAMK,MAAMF,gBAAgB1G,UAAUuG,SAAS;AAC/C,QAAMM,QAAQtH,eAAemB,IAAIkG,GAAG;AACpC,MAAI,CAACC,MAAO;AACZC,eAAaD,KAAK;AAClBtH,iBAAesB,OAAO+F,GAAG;AAC3B;AAEA,MAAMG,wCAAwCA,CAAC/G,UAAkBuG,cAA4B;AAC3F,QAAMK,MAAMF,gBAAgB1G,UAAUuG,SAAS;AAC/C,MAAIhH,eAAeyH,IAAIJ,GAAG,EAAG;AAE7B,QAAMK,QAAQC,KAAKC,IAAI,GAAGD,KAAKE,IAAI,KAAMF,KAAKG,MAAMzH,kBAAkB,CAAC,CAAC;AACxEL,iBAAekB,IAAImG,KAAKU,WAAW,MAAM;AACvC/H,mBAAesB,OAAO+F,GAAG;AACzB,SAAKW,8BAA8BvH,UAAUuG,SAAS;AAAA,EACxD,GAAGU,KAAK,CAAC;AACX;AAEA,MAAMO,kBAAkB,OAAO;AAAA,EAC7BxH;AAAAA,EACAyH;AAAAA,EACAtC;AAAAA,EACAoB;AAAAA,EACAmB;AAAAA,EACAC;AAAAA,EACAC;AASF,MAAqB;AACnB,QAAMC,SAAS,MAAMC,YAAY;AAAA,IAC/B9H;AAAAA,IACAmF;AAAAA,IACAoB;AAAAA,IACAoB;AAAAA,IACAC;AAAAA,IACA/H;AAAAA,EAAAA,CACD;AACD,QAAMmB,UAAyB;AAAA,IAC7BE,MAAM;AAAA,IACNqF;AAAAA,IACAmB;AAAAA,IACAK,MAAMF,OAAOE;AAAAA,IACb,GAAIF,OAAOG,WAAW;AAAA,MAAEA,UAAUH,OAAOG;AAAAA,IAAAA,IAAa,CAAA;AAAA,EAAC;AAGzD,aAAWC,YAAYR,iBAAiB;AACtC,UAAMvH,KAAKnB,QAAQ2B,IAAIuH,QAAQ;AAC/B,QAAI,CAAC/H,GAAI;AACTe,WAAOf,IAAIc,OAAO;AAAA,EACpB;AACF;AAEA,MAAMkH,6BAA6BA,CACjCC,kBACAC,gBACAC,iBACY;AACZ,MAAID,mBAAmBD,iBAAkB,QAAO;AAChD,SAAOE,aAAaC,qBAAqBtF,SAASmF,gBAAgB;AACpE;AAEA,MAAMI,sCAAsCA,CAC1CvI,UACAuG,cACY;AACZ,QAAMiC,aAAanJ,cAAcqB,IAAIV,QAAQ;AAC7C,MAAI,CAACwI,WAAY,QAAO;AAExB,aAAWC,YAAYD,WAAWE,UAAU;AAC1C,eAAW,CAACN,gBAAgBO,SAAS,KAAKF,SAAS1K,WAAW;AAC5D,iBAAWsK,gBAAgBM,UAAUD,UAAU;AAC7C,YAAIR,2BAA2B3B,WAAW6B,gBAAgBC,YAAY,GAAG;AACvE,iBAAO;AAAA,QACT;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAEA,SAAO;AACT;AAEA,MAAMd,gCAAgC,OAAOvH,UAAkBuG,cAAqC;AAClG,QAAMiC,aAAanJ,cAAcqB,IAAIV,QAAQ;AAC7C,MAAI,CAACwI,cAAc,CAACA,WAAW1H,KAAM;AAErC,aAAW2H,YAAYD,WAAWE,UAAU;AAC1C,eAAW,CAACN,gBAAgBO,SAAS,KAAKF,SAAS1K,WAAW;AAC5D,iBAAW,CAAC2J,UAAUkB,GAAG,KAAKD,UAAU5K,WAAW;AACjD,YAAI,CAACmK,2BAA2B3B,WAAW6B,gBAAgBQ,GAAG,EAAG;AAEjE,cAAMnB,kBAAkB9F,MAAMF,KAAKmH,IAAIC,SAAS;AAChD,YAAI,CAACpB,gBAAgB/D,OAAQ;AAE7B,cAAMuE,WAAWR,gBAAgB,CAAC;AAClC,cAAMpH,OAAOpB,WAAWyB,IAAIuH,QAAQ;AACpC,cAAM9C,UAAU9E,MAAM8E;AACtB,YAAI,CAACA,QAAS;AAEd,YAAI;AACF,gBAAMqC,gBAAgB;AAAA,YACpBxH;AAAAA,YACAyH;AAAAA,YACAtC;AAAAA,YACAoB,WAAW6B;AAAAA,YACXV;AAAAA,YACAC,OAAOiB,IAAIjB;AAAAA,YACXC,SAASgB,IAAIhB;AAAAA,UAAAA,CACd;AAAA,QACH,SAASnF,KAAK;AACZ,gBAAMqG,QAAQvF,mBAAmBd,GAAG;AACpC,gBAAMzB,UAAyB;AAAA,YAAEE,MAAM;AAAA,YAAiBqF,WAAW6B;AAAAA,YAAgBV;AAAAA,YAAUoB;AAAAA,UAAAA;AAC7F,qBAAWb,aAAYR,iBAAiB;AACtC,kBAAMvH,KAAKnB,QAAQ2B,IAAIuH,SAAQ;AAC/B,gBAAI,CAAC/H,GAAI;AACTe,mBAAOf,IAAIc,OAAO;AAAA,UACpB;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;AAEA,MAAM+H,qBAAqB,OAAO/I,UAAkBuG,cAAqC;AACvF,QAAMyC,gBAAgB1J,cAAcoB,IAAIV,QAAQ,yBAAShB,IAAAA;AACzDM,gBAAcmB,IAAIT,UAAUgJ,aAAa;AACzC,MAAIA,cAAchC,IAAIT,SAAS,EAAG;AAElC,QAAM0C,QAAQ,MAAM3C,eAAetG,UAAUuG,SAAS;AAEtD,QAAM2C,SAASD,MAAME,MAAM,IAAI;AAAA,IAC7BC,cAAc;AAAA,EAAA,CACf;AAEDF,SAAO5I,GAAG,UAAU,MAAM;AACxByG,0CAAsC/G,UAAUuG,SAAS;AAAA,EAC3D,CAAC;AAED2C,SAAO5I,GAAG,SAAS,MAAM;AACvBqG,uBAAmB3G,UAAUuG,SAAS;AACtC,UAAMN,MAAM3G,cAAcoB,IAAIV,QAAQ;AACtCiG,SAAKpF,OAAO0F,SAAS;AACrB,QAAIN,OAAOA,IAAInF,SAAS,EAAGxB,eAAcuB,OAAOb,QAAQ;AAAA,EAC1D,CAAC;AAEDkJ,SAAO5I,GAAG,SAAS,MAAM;AACvB,QAAI;AACFqG,yBAAmB3G,UAAUuG,SAAS;AACtC2C,aAAO/H,MAAAA;AAAAA,IACT,QAAQ;AAAA,IACN;AAAA,EAEJ,CAAC;AAED6H,gBAAcvI,IAAI8F,WAAW2C,MAAM;AACrC;AAEA,MAAMG,wBAAwBA,CAAC;AAAA,EAC7BpB;AAAAA,EACAjI;AAAAA,EACAC;AAAAA,EACAsG;AAAAA,EACAmB;AAAAA,EACAC;AAAAA,EACAC;AAAAA,EACAU;AAUF,MAAY;AACV,QAAME,aAAanJ,cAAcqB,IAAIV,QAAQ,yBAAShB,IAAAA;AACtDK,gBAAcoB,IAAIT,UAAUwI,UAAU;AAEtC,QAAMC,WAAWD,WAAW9H,IAAIT,MAAM,yBAASjB,IAAAA;AAC/CwJ,aAAW/H,IAAIR,QAAQwI,QAAQ;AAE/B,QAAME,YAAYF,SAAS/H,IAAI6F,SAAS,yBAASvH,IAAAA;AACjDyJ,WAAShI,IAAI8F,WAAWoC,SAAS;AAEjC,QAAMW,WAAWX,UAAUjI,IAAIgH,QAAQ;AACvC,MAAI4B,UAAU;AACZA,aAAST,UAAUlI,IAAIsH,QAAQ;AAC/BqB,aAAShB,uBAAuB3G,MAAMF,KAAK,oBAAIjD,IAAI,CAAC,GAAG8K,SAAShB,sBAAsB,GAAGA,oBAAoB,CAAC,CAAC;AAAA,EACjH,OAAO;AACLK,cAAUlI,IAAIiH,UAAU;AAAA,MACtBC;AAAAA,MACAC;AAAAA,MACAU,sBAAsB3G,MAAMF,KAAK,IAAIjD,IAAI8J,oBAAoB,CAAC;AAAA,MAC9DO,WAAW,oBAAIrK,IAAI,CAACyJ,QAAQ,CAAC;AAAA,IAAA,CAC9B;AAAA,EACH;AAEA,QAAMsB,UAAUnK,oBAAoBsB,IAAIuH,QAAQ,yBAASjJ,IAAAA;AACzDI,sBAAoBqB,IAAIwH,UAAUsB,OAAO;AAEzC,QAAMC,WAAWD,QAAQ7I,IAAI6F,SAAS,yBAAS/H,IAAAA;AAC/C+K,UAAQ9I,IAAI8F,WAAWiD,QAAQ;AAC/BA,WAAS7I,IAAI+G,QAAQ;AACvB;AAEA,MAAM+B,2BAA2BA,CAAC;AAAA,EAChCxB;AAAAA,EACAjI;AAAAA,EACAC;AAAAA,EACAsG;AAAAA,EACAmB;AAOF,MAAY;AACV,QAAMc,aAAanJ,cAAcqB,IAAIV,QAAQ;AAC7C,QAAMyI,WAAWD,YAAY9H,IAAIT,MAAM;AACvC,QAAM0I,YAAYF,UAAU/H,IAAI6F,SAAS;AACzC,QAAMqC,MAAMD,WAAWjI,IAAIgH,QAAQ;AACnC,QAAMgC,qBAAqB,oBAAIlL,IAAY,CAAC+H,WAAW,GAAIqC,KAAKN,wBAAwB,CAAA,CAAG,CAAC;AAC5F,MAAIM,KAAK;AACPA,QAAIC,UAAUhI,OAAOoH,QAAQ;AAC7B,QAAI,CAACW,IAAIC,UAAU/H,MAAM;AACvB6H,iBAAW9H,OAAO6G,QAAQ;AAAA,IAC5B;AAAA,EACF;AAEA,QAAM6B,UAAUnK,oBAAoBsB,IAAIuH,QAAQ;AAChD,QAAMxH,MAAM8I,SAAS7I,IAAI6F,SAAS;AAClC,MAAI9F,KAAK;AACPA,QAAII,OAAO6G,QAAQ;AACnB,QAAI,CAACjH,IAAIK,MAAM;AACbyI,eAAS1I,OAAO0F,SAAS;AAAA,IAC3B;AAAA,EACF;AAEA,MAAIoC,aAAaA,UAAU7H,SAAS,GAAG;AACrC2H,cAAU5H,OAAO0F,SAAS;AAAA,EAC5B;AAEA,MAAIkC,YAAYA,SAAS3H,SAAS,GAAG;AACnC0H,gBAAY3H,OAAOZ,MAAM;AAAA,EAC3B;AAEA,aAAW0J,qBAAqBD,oBAAoB;AAClD,QAAInB,oCAAoCvI,UAAU2J,iBAAiB,EAAG;AAEtE,UAAMX,gBAAgB1J,cAAcoB,IAAIV,QAAQ;AAChD,UAAMkJ,SAASF,eAAetI,IAAIiJ,iBAAiB;AACnD,QAAIT,QAAQ;AACV,UAAI;AACFA,eAAO/H,MAAAA;AAAAA,MACT,QAAQ;AAAA,MACN;AAEFwF,yBAAmB3G,UAAU2J,iBAAiB;AAC9CX,qBAAenI,OAAO8I,iBAAiB;AACvC,UAAIX,iBAAiBA,cAAclI,SAAS,EAAGxB,eAAcuB,OAAOb,QAAQ;AAAA,IAC9E;AAAA,EACF;AAEA,MAAIwI,cAAcA,WAAW1H,SAAS,EAAGzB,eAAcwB,OAAOb,QAAQ;AACtE,MAAIuJ,WAAWA,QAAQzI,SAAS,EAAG1B,qBAAoByB,OAAOoH,QAAQ;AACxE;AAEA,MAAM2B,gBAAgBA,CAAC3B,aAA2B;AAChD,QAAM5H,OAAOpB,WAAWyB,IAAIuH,QAAQ;AACpC,MAAI5H,MAAM;AACR,UAAMkJ,UAAUnK,oBAAoBsB,IAAIuH,QAAQ;AAChD,QAAIsB,SAAS;AACX,iBAAW,CAAChD,WAAWsD,IAAI,KAAKN,QAAQxL,WAAW;AACjD,mBAAW2J,YAAYmC,KAAKnB,UAAU;AACpCe,mCAAyB;AAAA,YACvBxB;AAAAA,YACAjI,UAAUK,KAAKL;AAAAA,YACfC,QAAQI,KAAKJ;AAAAA,YACbsG;AAAAA,YACAmB;AAAAA,UAAAA,CACD;AAAA,QACH;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAEAtI,sBAAoByB,OAAOoH,QAAQ;AAEnC,QAAM6B,aAAa3K,cAAcuB,IAAIuH,QAAQ,KAAK,CAAA;AAClD9I,gBAAc0B,OAAOoH,QAAQ;AAC7B,aAAW8B,MAAMD,YAAY;AAC3B,QAAI;AACFC,SAAAA;AAAAA,IACF,QAAQ;AAAA,IACN;AAAA,EAEJ;AAEAhL,UAAQ8B,OAAOoH,QAAQ;AACvBhJ,aAAW4B,OAAOoH,QAAQ;AAC1B/I,iBAAe2B,OAAOoH,QAAQ;AAChC;AAEA,MAAM+B,sBAAsB,OAAO;AAAA,EACjC/B;AAAAA,EACA5H;AAAAA,EACA4B;AAKF,MAAqB;AACnB,QAAM/B,KAAKnB,QAAQ2B,IAAIuH,QAAQ;AAC/B,MAAI,CAAC/H,GAAI;AAET,MAAI+B,QAAQf,SAAS,SAAS;AAC5B,UAAM+I,UAAU/K,eAAewB,IAAIuH,QAAQ;AAC3CgC,aAAS7I,SAASa,QAAQ1B,OAAO0B,QAAQjB,OAAO;AAChD;AAAA,EACF;AAEA,MAAI,CAACiB,QAAQsE,aAAa,OAAOtE,QAAQsE,cAAc,SAAU;AACjE,MAAI,CAAC1G,uBAAuBtB,qBAAqByI,IAAI/E,QAAQsE,SAAS,GAAG;AACvEtF,WAAOf,IAAI;AAAA,MAAEgB,MAAM;AAAA,MAAiBqF,WAAWtE,QAAQsE;AAAAA,MAAWmB,UAAUzF,QAAQyF,YAAY;AAAA,MAAIoB,OAAO;AAAA,IAAA,CAAqB;AAChI;AAAA,EACF;AACA,MAAI,CAAC7G,QAAQyF,YAAY,OAAOzF,QAAQyF,aAAa,SAAU;AAC/D,MAAIzF,QAAQyF,SAAShE,SAASpF,kBAAmB;AAEjD,MAAI2D,QAAQf,SAAS,gBAAgB;AACnCuI,6BAAyB;AAAA,MACvBxB;AAAAA,MACAjI,UAAUK,KAAKL;AAAAA,MACfC,QAAQI,KAAKJ;AAAAA,MACbsG,WAAWtE,QAAQsE;AAAAA,MACnBmB,UAAUzF,QAAQyF;AAAAA,IAAAA,CACnB;AACD;AAAA,EACF;AAEA,MAAI,CAACzF,QAAQ0F,SAAS,OAAO1F,QAAQ0F,UAAU,SAAU;AAEzD,QAAMC,UAAUsC,yBAAyBjI,QAAQ2F,OAAO;AACxD,QAAMzC,UAAU9E,KAAK8E;AAErB,MAAI,CAACA,QAAQgF,IAAI,QAAQlI,QAAQsE,SAA2B,GAAG;AAC7DtF,WAAOf,IAAI;AAAA,MAAEgB,MAAM;AAAA,MAAiBqF,WAAWtE,QAAQsE;AAAAA,MAAWmB,UAAUzF,QAAQyF;AAAAA,MAAUoB,OAAO;AAAA,IAAA,CAAa;AAClH;AAAA,EACF;AAEA,MAAI7G,QAAQf,SAAS,kBAAkB;AACrC,UAAMoI,WAAWlK,oBAAoBsB,IAAIuH,QAAQ,GAAGvH,IAAIuB,QAAQsE,SAAS,GAAGS,IAAI/E,QAAQyF,QAAQ,KAAK;AACrG,QAAI,CAAC4B,UAAU;AACb,UAAIc,QAAQ;AACZ,YAAMb,UAAUnK,oBAAoBsB,IAAIuH,QAAQ;AAChD,UAAIsB,SAAS;AACX,mBAAW9I,OAAO8I,QAAQb,OAAAA,YAAmBjI,IAAIK;AAAAA,MACnD;AAEA,UAAIsJ,SAASzK,2BAA2B;AACtCsB,eAAOf,IAAI;AAAA,UAAEgB,MAAM;AAAA,UAAiBqF,WAAWtE,QAAQsE;AAAAA,UAAWmB,UAAUzF,QAAQyF;AAAAA,UAAUoB,OAAO;AAAA,QAAA,CAA0B;AAC/H;AAAA,MACF;AAAA,IACF;AAEA,QAAIR,uBAAiC,CAAA;AACrC,QAAIV,QAAQyC,aAAavH,QAAW;AAClC,UAAI;AACFwF,+BAAuB,MAAMgC,oCAAoC;AAAA,UAC/DtK,UAAUK,KAAKL;AAAAA,UACfmF;AAAAA,UACAoB,WAAWtE,QAAQsE;AAAAA,UACnBqB;AAAAA,UACA/H;AAAAA,QAAAA,CACD;AAAA,MACH,SAAS4C,KAAK;AACZ,cAAMqG,QAAQvF,mBAAmBd,GAAG;AACpCxB,eAAOf,IAAI;AAAA,UAAEgB,MAAM;AAAA,UAAiBqF,WAAWtE,QAAQsE;AAAAA,UAAWmB,UAAUzF,QAAQyF;AAAAA,UAAUoB;AAAAA,QAAAA,CAAO;AACrG;AAAA,MACF;AAAA,IACF;AAEAO,0BAAsB;AAAA,MACpBpB;AAAAA,MACAjI,UAAUK,KAAKL;AAAAA,MACfC,QAAQI,KAAKJ;AAAAA,MACbsG,WAAWtE,QAAQsE;AAAAA,MACnBmB,UAAUzF,QAAQyF;AAAAA,MAClBC,OAAO1F,QAAQ0F;AAAAA,MACfC;AAAAA,MACAU;AAAAA,IAAAA,CACD;AAED,QAAI;AACF,YAAMiC,wCAAwB/L,IAAY,CAACyD,QAAQsE,WAAW,GAAG+B,oBAAoB,CAAC;AACtF,iBAAW/B,aAAagE,mBAAmB;AACzC,cAAMxB,mBAAmB1I,KAAKL,UAAUuG,SAAS;AAAA,MACnD;AAAA,IACF,SAAS9D,KAAK;AACZ,YAAMqG,QAAQvF,mBAAmBd,GAAG;AACpCxB,aAAOf,IAAI;AAAA,QAAEgB,MAAM;AAAA,QAAiBqF,WAAWtE,QAAQsE;AAAAA,QAAWmB,UAAUzF,QAAQyF;AAAAA,QAAUoB;AAAAA,MAAAA,CAAO;AACrG;AAAA,IACF;AAEA,QAAI7G,QAAQuI,oBAAoB,OAAO;AACrC;AAAA,IACF;AAAA,EACF;AAEA,MAAI;AACF,UAAMhD,gBAAgB;AAAA,MACpBxH,UAAUK,KAAKL;AAAAA,MACfyH,iBAAiB,CAACQ,QAAQ;AAAA,MAC1B9C;AAAAA,MACAoB,WAAWtE,QAAQsE;AAAAA,MACnBmB,UAAUzF,QAAQyF;AAAAA,MAClBC,OAAO1F,QAAQ0F;AAAAA,MACfC;AAAAA,IAAAA,CACD;AAAA,EACH,SAASnF,KAAK;AACZ,UAAMqG,QAAQvF,mBAAmBd,GAAG;AACpCxB,WAAOf,IAAI;AAAA,MAAEgB,MAAM;AAAA,MAAiBqF,WAAWtE,QAAQsE;AAAAA,MAAWmB,UAAUzF,QAAQyF;AAAAA,MAAUoB;AAAAA,IAAAA,CAAO;AAAA,EACvG;AACF;AAEO,MAAM2B,UAAUA,CAAC;AAAA,EACtBC;AAAAA,EACAvM,OAAO;AAAA,EACP+F;AAAAA,EACAxE,iBAAiBiL;AAAAA,EACjBhL,2BAA2BiL;AAAAA,EAC3BhL,oBAAoBiL;AAAAA,EACpBhL,qBAAqBiL;AASvB,MAAY;AACV,MAAIlM,mBAAmBoI,IAAI0D,MAAM,EAAG;AACpC9L,qBAAmB+B,IAAI+J,MAAM;AAE7B,MAAI,OAAOC,uBAAuB,YAAYI,OAAOC,SAASL,kBAAkB,KAAKA,qBAAqB,GAAG;AAC3GjL,sBAAkBwH,KAAKG,MAAMsD,kBAAkB;AAAA,EACjD;AAEA,MACE,OAAOC,iCAAiC,YACrCG,OAAOC,SAASJ,4BAA4B,KAC5CA,+BAA+B,GAClC;AACAjL,gCAA4BuH,KAAKG,MAAMuD,4BAA4B;AAAA,EACrE;AAEA,MAAI,OAAOC,0BAA0B,YAAYE,OAAOC,SAASH,qBAAqB,KAAKA,yBAAyB,GAAG;AACrHjL,yBAAqBsH,KAAKG,MAAMwD,qBAAqB;AAAA,EACvD;AAEAhL,wBAAsBoL,QAAQH,sBAAsB;AAEpD,QAAMI,MAAM,IAAIC,gBAAgB;AAAA,IAAEC,UAAU;AAAA,IAAMC,YAAY3L;AAAAA,EAAAA,CAAiB;AAE/EgL,SAAOpK,GAAG,WAAW,CAAC6D,KAAsB7B,QAAgBgJ,SAAiB;AAC3EjJ,6BAAyBC,QAAQ,OAAO;AAAA,MACtCiJ,aAAapH,IAAIkB,QAAQmG,QAAQ;AAAA,MACjCC,YAAYtH,IAAIM,OAAO;AAAA,MACvBiH,WAAW,OAAOvH,IAAIkB,QAAQ,YAAY,MAAM,WAAWlB,IAAIkB,QAAQ,YAAY,IAAI;AAAA,IAAA,EACvF;AACF7F,gBAAYqB,OAAOsD,GAAG;AAEtB,QAAIM;AACJ,QAAI;AACFA,YAAM,IAAIkH,IAAIxH,IAAIM,OAAO,IAAI,UAAUN,IAAIkB,QAAQmG,QAAQ,WAAW,EAAE;AAAA,IAC1E,QAAQ;AACNxH,iBAAW1B,QAAQ,aAAa;AAChC;AAAA,IACF;AAEA,QAAImC,IAAImH,aAAazN,KAAM;AAE3B,UAAM,YAAW;AACf,UAAI;AACF,cAAMkC,OAAO,MAAMmE,iBAAiB;AAAA,UAAEL;AAAAA,UAAKM;AAAAA,UAAKP;AAAAA,QAAAA,CAAmB;AACnE1E,oBAAYiB,IAAI0D,KAAK9D,IAAI;AAEzB6K,YAAIW,cAAc1H,KAAK7B,QAAQgJ,MAAM,CAACpL,OAAkB;AACtDgL,cAAInK,KAAK,cAAcb,IAAIiE,GAAG;AAAA,QAChC,CAAC;AAAA,MACH,SAAS1B,KAAK;AACZ,cAAMR,UAAUQ,eAAeC,QAAQD,IAAIR,UAAU;AACrD,YAAIA,QAAQ6J,WAAW,sBAAsB,GAAG;AAC9C9H,qBAAW1B,QAAQL,OAAO;AAC1B;AAAA,QACF;AACA2B,qBAAatB,QAAQL,OAAO;AAC5B;AAAA,MACF;AAAA,IACF,GAAA,EAAK8J,MAAM,MAAM;AACf/H,iBAAW1B,QAAQ,oBAAoB;AAAA,IACzC,CAAC;AAAA,EACH,CAAC;AAED4I,MAAI5K,GAAG,cAAc,CAACJ,IAAeiE,QAAyB;AAC5D,UAAM9D,OAAOb,YAAYkB,IAAIyD,GAAG;AAChC3E,gBAAYqB,OAAOsD,GAAG;AACtB,QAAI,CAAC9D,MAAM;AACT,UAAI;AACFH,WAAGiB,MAAAA;AAAAA,MACL,QAAQ;AAAA,MACN;AAEF;AAAA,IACF;AAEA,UAAM8G,WAAW+D,WAAAA;AACjBjN,YAAQ0B,IAAIwH,UAAU/H,EAAE;AACxBjB,eAAWwB,IAAIwH,UAAU5H,IAAI;AAE7B,UAAM4J,UAAU,IAAInK,UAAU;AAAA,MAAEC,IAAIkI;AAAAA,MAAU/H;AAAAA,MAAIG;AAAAA,IAAAA,CAAM;AACxDnB,mBAAeuB,IAAIwH,UAAUgC,OAAO;AAEpC,UAAMH,aAAgC,CAAA;AACtC,eAAWtJ,WAAW1B,gBAAgB;AACpC,UAAI;AACF,cAAMmN,UAAUzL,QAAQyJ,OAAO;AAC/B,YAAI,OAAOgC,YAAY,WAAYnC,YAAWoC,KAAKD,OAAO;AAAA,MAC5D,QAAQ;AAAA,MACN;AAAA,IAEJ;AACA,QAAInC,WAAWpG,OAAQvE,eAAcsB,IAAIwH,UAAU6B,UAAU;AAE7D5J,OAAGI,GAAG,WAAW,CAACgB,QAAiB;AACjC,UAAI6K;AACJ,UAAI;AACFA,iBAASrK,cAAcR,GAAG;AAAA,MAC5B,QAAQ;AACN;AAAA,MACF;AAEA,UAAI,CAAC6K,UAAU,OAAOA,WAAW,SAAU;AAC3C,YAAMlK,UAAUkK;AAChB,UACElK,QAAQf,SAAS,WACde,QAAQf,SAAS,eACjBe,QAAQf,SAAS,oBACjBe,QAAQf,SAAS,eACpB;AACF,WAAK8I,oBAAoB;AAAA,QAAE/B;AAAAA,QAAU5H;AAAAA,QAAM4B;AAAAA,MAAAA,CAAmC;AAAA,IAChF,CAAC;AAED/B,OAAGI,GAAG,SAAS,MAAM;AACnBsJ,oBAAc3B,QAAQ;AAAA,IACxB,CAAC;AAED/H,OAAGI,GAAG,SAAS,MAAM;AACnBsJ,oBAAc3B,QAAQ;AAAA,IACxB,CAAC;AAAA,EACH,CAAC;AACH;AAEO,MAAMmE,qBAAqBA,CAAC5L,YAA6B;AAC9D1B,iBAAeoN,KAAK1L,OAAO;AAC7B;AAEO,MAAM6L,wBAAwBA,CAACrM,UAAkBuG,cAA4B;AAClFQ,wCAAsC/G,UAAUuG,SAAS;AAC3D;"}
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../../src/rts/routes.ts","../../src/rts/index.ts"],"sourcesContent":["export const routes = Object.entries({\n ...import.meta.glob(\"./api/**/handler.ts\"),\n}).reduce<Record<string, unknown>>((acc, [path, mod]) => {\n acc[path.replace(\"./api/\", \"@rpcbase/server/rts/api/\")] = mod\n return acc\n}, {})\n\n","import { randomUUID } from \"node:crypto\"\nimport type { IncomingMessage, Server as HttpServer } from \"node:http\"\nimport type { Socket } from \"node:net\"\n\nimport type { RequestHandler } from \"express\"\nimport type { PaginationPageInfo } from \"@rpcbase/api\"\nimport { models, type LoadModelCtx } from \"@rpcbase/db\"\nimport { buildAbility, buildAbilityFromSession, getTenantRolesFromSessionUser, type AclSubjectType, type AppAbility } from \"@rpcbase/db/acl\"\nimport type { ChangeStream } from \"mongodb\"\nimport type { Model } from \"mongoose\"\nimport { WebSocketServer, type RawData, type WebSocket } from \"ws\"\n\nimport {\n RTS_TENANT_ID_QUERY_PARAM as TENANT_ID_QUERY_PARAM,\n RTS_USER_ID_HEADER as USER_ID_HEADER,\n normalizeRtsQueryOptions,\n resolveRtsQueryDependencyModelNames,\n runRtsQuery,\n type RtsQueryOptions as QueryOptions,\n} from \"./queryExecutor\"\n\n\ntype JsonObject = Record<string, unknown>\n\ntype ClientMessage =\n | { type: \"run-query\"; modelName: string; queryKey: string; query: JsonObject; options?: QueryOptions }\n | {\n type: \"register-query\"\n modelName: string\n queryKey: string\n query: JsonObject\n options?: QueryOptions\n runInitialQuery?: boolean\n }\n | { type: \"remove-query\"; modelName: string; queryKey: string }\n | { type: \"event\"; event: string; payload?: unknown }\n\ntype ServerMessage =\n | {\n type: \"query-payload\"\n modelName: string\n queryKey: string\n data?: unknown\n error?: string\n txnId?: string\n pageInfo?: PaginationPageInfo\n totalCount?: number\n }\n | { type: \"event\"; event: string; payload?: unknown }\n\ntype SocketMeta = {\n tenantId: string\n userId: string\n ability: AppAbility\n}\n\ntype SessionUser = {\n id?: string\n currentTenantId?: string\n signedInTenants?: string[]\n}\n\ntype HeaderUserDoc = {\n tenants?: unknown\n tenantRoles?: unknown\n}\n\ntype UpgradeRequest = IncomingMessage & {\n session?: {\n user?: SessionUser\n }\n}\n\ntype Subscription = {\n query: JsonObject\n options: QueryOptions\n dependencyModelNames: string[]\n socketIds: Set<string>\n}\n\ntype HandlerFn = (socket: RtsSocket) => void | (() => void)\n\nconst QUERY_KEY_MAX_LEN = 4096\n\nconst INTERNAL_MODEL_NAMES = new Set([\"RBRtsChange\", \"RBRtsCounter\"])\n\nconst DEFAULT_MAX_PAYLOAD_BYTES = 1024 * 1024\nconst DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET = 256\nconst DEFAULT_DISPATCH_DEBOUNCE_MS = 25\n\nconst initializedServers = new WeakSet<HttpServer>()\nconst customHandlers: HandlerFn[] = []\n\nconst sockets = new Map<string, WebSocket>()\nconst socketMeta = new Map<string, SocketMeta>()\nconst socketWrappers = new Map<string, RtsSocket>()\nconst socketCleanup = new Map<string, Array<() => void>>()\n\nconst socketSubscriptions = new Map<string, Map<string, Set<string>>>()\nconst subscriptions = new Map<string, Map<string, Map<string, Map<string, Subscription>>>>()\nconst changeStreams = new Map<string, Map<string, ChangeStream>>()\nconst dispatchTimers = new Map<string, ReturnType<typeof setTimeout>>()\nconst upgradeMeta = new WeakMap<IncomingMessage, SocketMeta>()\n\nlet maxPayloadBytes = DEFAULT_MAX_PAYLOAD_BYTES\nlet maxSubscriptionsPerSocket = DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET\nlet dispatchDebounceMs = DEFAULT_DISPATCH_DEBOUNCE_MS\nlet allowInternalModels = false\n\nclass RtsSocket {\n public readonly id: string\n public readonly tenantId: string\n public readonly userId: string\n\n private readonly ws: WebSocket\n private readonly handlers = new Map<string, Set<(payload: unknown) => void>>()\n\n public constructor({\n id,\n ws,\n meta,\n }: {\n id: string\n ws: WebSocket\n meta: SocketMeta\n }) {\n this.id = id\n this.ws = ws\n this.tenantId = meta.tenantId\n this.userId = meta.userId\n }\n\n public on(event: string, handler: (payload: unknown) => void): () => void {\n const set = this.handlers.get(event) ?? new Set()\n set.add(handler)\n this.handlers.set(event, set)\n return () => this.off(event, handler)\n }\n\n public off(event: string, handler: (payload: unknown) => void): void {\n const set = this.handlers.get(event)\n if (!set) return\n set.delete(handler)\n if (!set.size) this.handlers.delete(event)\n }\n\n public emit(event: string, payload?: unknown): void {\n sendWs(this.ws, { type: \"event\", event, payload })\n }\n\n public close(): void {\n try {\n this.ws.close()\n } catch {\n // ignore\n }\n }\n\n public dispatch(event: string, payload: unknown): void {\n const set = this.handlers.get(event)\n if (!set) return\n for (const handler of set) {\n handler(payload)\n }\n }\n}\n\nconst rawToText = (raw: RawData): string => {\n if (typeof raw === \"string\") return raw\n if (raw instanceof ArrayBuffer) return Buffer.from(raw).toString()\n if (Array.isArray(raw)) return Buffer.concat(raw).toString()\n return raw.toString()\n}\n\nconst safeJsonParse = (raw: RawData): unknown => JSON.parse(rawToText(raw))\n\nconst sendWs = (ws: WebSocket, message: ServerMessage): void => {\n if (ws.readyState !== 1) return\n ws.send(JSON.stringify(message))\n}\n\nconst ensureSocketErrorHandler = (socket: Socket, context?: () => Record<string, unknown>): void => {\n if (socket.listenerCount(\"error\") > 0) return\n\n socket.on(\"error\", (err: unknown) => {\n const message = err instanceof Error ? err.message : String(err)\n const name = err instanceof Error ? err.name : typeof err\n const stack = err instanceof Error ? err.stack : undefined\n const code = err && typeof err === \"object\" && \"code\" in err ? (err as any).code : undefined\n\n if (code === \"ECONNRESET\" || (code == null && message.includes(\"ECONNRESET\"))) return\n\n console.warn(\"[rb/rts] socket error\", {\n name,\n code,\n message,\n stack,\n remoteAddress: socket.remoteAddress,\n remotePort: socket.remotePort,\n localAddress: socket.localAddress,\n localPort: socket.localPort,\n ...context?.(),\n })\n })\n}\n\nconst redactErrorMessage = (err: unknown): string => {\n const raw = err instanceof Error ? err.message : \"Unknown error\"\n const trimmedModelList = raw.replace(/\\.\\s+Available models:[\\s\\S]*$/, \"\")\n const maxLen = 256\n if (trimmedModelList.length <= maxLen) return trimmedModelList\n return trimmedModelList.slice(0, maxLen)\n}\n\nconst unauthorized = (socket: Socket, message = \"Unauthorized\"): void => {\n ensureSocketErrorHandler(socket)\n try {\n socket.write(\"HTTP/1.1 401 Unauthorized\\r\\n\\r\\n\")\n socket.write(`Error: ${message}\\r\\n`)\n socket.end()\n } catch {\n socket.destroy()\n }\n}\n\nconst badRequest = (socket: Socket, message = \"Bad Request\"): void => {\n ensureSocketErrorHandler(socket)\n try {\n socket.write(\"HTTP/1.1 400 Bad Request\\r\\n\\r\\n\")\n socket.write(`Error: ${message}\\r\\n`)\n socket.end()\n } catch {\n socket.destroy()\n }\n}\n\nconst runSessionMiddleware = async (sessionMiddleware: RequestHandler, req: UpgradeRequest): Promise<void> => {\n type MiddlewareReq = Parameters<RequestHandler>[0]\n type MiddlewareRes = Parameters<RequestHandler>[1]\n type MiddlewareNext = Parameters<RequestHandler>[2]\n\n await new Promise<void>((resolve, reject) => {\n const next: MiddlewareNext = (err) => {\n if (err) reject(err)\n else resolve()\n }\n\n sessionMiddleware(req as unknown as MiddlewareReq, {} as unknown as MiddlewareRes, next)\n })\n}\n\nconst parseUpgradeMeta = async ({\n req,\n url,\n sessionMiddleware,\n}: {\n req: IncomingMessage\n url: URL\n sessionMiddleware?: RequestHandler\n}): Promise<SocketMeta> => {\n const tenantId = url.searchParams.get(TENANT_ID_QUERY_PARAM)\n if (!tenantId) {\n throw new Error(\"Missing rb-tenant-id query parameter\")\n }\n\n if (sessionMiddleware) {\n const upgradeReq = req as UpgradeRequest\n try {\n await runSessionMiddleware(sessionMiddleware, upgradeReq)\n } catch {\n throw new Error(\"Failed to load session for RTS\")\n }\n\n const sessionUser = upgradeReq.session?.user\n const sessionUserId = sessionUser?.id\n if (!sessionUserId) {\n throw new Error(\"Not signed in (missing session.user.id)\")\n }\n\n const signedInTenants = sessionUser?.signedInTenants\n const currentTenantId = sessionUser?.currentTenantId\n\n if (Array.isArray(signedInTenants) && signedInTenants.length > 0) {\n if (!signedInTenants.includes(tenantId)) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n } else if (currentTenantId) {\n if (currentTenantId !== tenantId) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n } else {\n throw new Error(\"Tenant not authorized for this session\")\n }\n\n const ability = buildAbilityFromSession({ tenantId, session: upgradeReq.session })\n return { tenantId, userId: sessionUserId, ability }\n }\n\n const raw = req.headers[USER_ID_HEADER]\n const headerUserId = Array.isArray(raw) ? raw[0] : raw\n if (!headerUserId) {\n throw new Error(\"Missing rb-user-id header (reverse-proxy) and no session middleware configured\")\n }\n\n const rbCtx: LoadModelCtx = { req: { session: null } }\n const User = await models.getGlobal(\"RBUser\", rbCtx)\n const user = await User.findById(headerUserId, { tenants: 1, tenantRoles: 1 }).lean() as HeaderUserDoc | null\n const tenantsRaw = user?.tenants\n const tenants = Array.isArray(tenantsRaw) ? tenantsRaw.map((t) => String(t)) : []\n if (!tenants.includes(tenantId)) {\n throw new Error(\"Tenant not authorized for this session\")\n }\n\n const roles = getTenantRolesFromSessionUser(user, tenantId)\n const ability = buildAbility({ tenantId, userId: headerUserId, roles: roles.length ? roles : [\"owner\"] })\n\n return { tenantId, userId: headerUserId, ability }\n}\n\nconst getTenantModel = async (tenantId: string, modelName: string): Promise<Model<any>> => {\n const ctx: LoadModelCtx = {\n req: {\n session: {\n user: {\n currentTenantId: tenantId,\n },\n },\n },\n }\n\n return models.getUnsafe(modelName, ctx)\n}\n\nconst makeDispatchKey = (tenantId: string, modelName: string): string => `${tenantId}:${modelName}`\n\nconst clearDispatchTimer = (tenantId: string, modelName: string): void => {\n const key = makeDispatchKey(tenantId, modelName)\n const timer = dispatchTimers.get(key)\n if (!timer) return\n clearTimeout(timer)\n dispatchTimers.delete(key)\n}\n\nconst scheduleDispatchSubscriptionsForModel = (tenantId: string, modelName: string): void => {\n const key = makeDispatchKey(tenantId, modelName)\n if (dispatchTimers.has(key)) return\n\n const delay = Math.max(0, Math.min(1000, Math.floor(dispatchDebounceMs)))\n dispatchTimers.set(key, setTimeout(() => {\n dispatchTimers.delete(key)\n void dispatchSubscriptionsForModel(tenantId, modelName)\n }, delay))\n}\n\nconst runAndSendQuery = async ({\n tenantId,\n targetSocketIds,\n ability,\n modelName,\n queryKey,\n query,\n options,\n}: {\n tenantId: string\n targetSocketIds: string[]\n ability: AppAbility\n modelName: string\n queryKey: string\n query: JsonObject\n options: QueryOptions\n}): Promise<void> => {\n const result = await runRtsQuery({\n tenantId,\n ability,\n modelName,\n query,\n options,\n allowInternalModels,\n })\n const payload: ServerMessage = {\n type: \"query-payload\",\n modelName,\n queryKey,\n data: result.data,\n ...(result.pageInfo ? { pageInfo: result.pageInfo } : {}),\n ...(typeof result.totalCount === \"number\" ? { totalCount: result.totalCount } : {}),\n }\n\n for (const socketId of targetSocketIds) {\n const ws = sockets.get(socketId)\n if (!ws) continue\n sendWs(ws, payload)\n }\n}\n\nconst subscriptionDependsOnModel = (\n changedModelName: string,\n ownerModelName: string,\n subscription: Subscription,\n): boolean => {\n if (ownerModelName === changedModelName) return true\n return subscription.dependencyModelNames.includes(changedModelName)\n}\n\nconst hasAnySubscriptionsDependingOnModel = (\n tenantId: string,\n modelName: string,\n): boolean => {\n const tenantSubs = subscriptions.get(tenantId)\n if (!tenantSubs) return false\n\n for (const userSubs of tenantSubs.values()) {\n for (const [ownerModelName, modelSubs] of userSubs.entries()) {\n for (const subscription of modelSubs.values()) {\n if (subscriptionDependsOnModel(modelName, ownerModelName, subscription)) {\n return true\n }\n }\n }\n }\n\n return false\n}\n\nconst dispatchSubscriptionsForModel = async (tenantId: string, modelName: string): Promise<void> => {\n const tenantSubs = subscriptions.get(tenantId)\n if (!tenantSubs || !tenantSubs.size) return\n\n for (const userSubs of tenantSubs.values()) {\n for (const [ownerModelName, modelSubs] of userSubs.entries()) {\n for (const [queryKey, sub] of modelSubs.entries()) {\n if (!subscriptionDependsOnModel(modelName, ownerModelName, sub)) continue\n\n const targetSocketIds = Array.from(sub.socketIds)\n if (!targetSocketIds.length) continue\n\n const socketId = targetSocketIds[0]\n const meta = socketMeta.get(socketId)\n const ability = meta?.ability\n if (!ability) continue\n\n try {\n await runAndSendQuery({\n tenantId,\n targetSocketIds,\n ability,\n modelName: ownerModelName,\n queryKey,\n query: sub.query,\n options: sub.options,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n const payload: ServerMessage = { type: \"query-payload\", modelName: ownerModelName, queryKey, error }\n for (const socketId of targetSocketIds) {\n const ws = sockets.get(socketId)\n if (!ws) continue\n sendWs(ws, payload)\n }\n }\n }\n }\n }\n}\n\nconst ensureChangeStream = async (tenantId: string, modelName: string): Promise<void> => {\n const tenantStreams = changeStreams.get(tenantId) ?? new Map()\n changeStreams.set(tenantId, tenantStreams)\n if (tenantStreams.has(modelName)) return\n\n const model = await getTenantModel(tenantId, modelName)\n\n const stream = model.watch([], {\n fullDocument: \"updateLookup\",\n })\n\n stream.on(\"change\", () => {\n scheduleDispatchSubscriptionsForModel(tenantId, modelName)\n })\n\n stream.on(\"close\", () => {\n clearDispatchTimer(tenantId, modelName)\n const map = changeStreams.get(tenantId)\n map?.delete(modelName)\n if (map && map.size === 0) changeStreams.delete(tenantId)\n })\n\n stream.on(\"error\", () => {\n try {\n clearDispatchTimer(tenantId, modelName)\n stream.close()\n } catch {\n // ignore\n }\n })\n\n tenantStreams.set(modelName, stream)\n}\n\nconst addSocketSubscription = ({\n socketId,\n tenantId,\n userId,\n modelName,\n queryKey,\n query,\n options,\n dependencyModelNames,\n}: {\n socketId: string\n tenantId: string\n userId: string\n modelName: string\n queryKey: string\n query: JsonObject\n options: QueryOptions\n dependencyModelNames: string[]\n}): void => {\n const tenantSubs = subscriptions.get(tenantId) ?? new Map<string, Map<string, Map<string, Subscription>>>()\n subscriptions.set(tenantId, tenantSubs)\n\n const userSubs = tenantSubs.get(userId) ?? new Map<string, Map<string, Subscription>>()\n tenantSubs.set(userId, userSubs)\n\n const modelSubs = userSubs.get(modelName) ?? new Map<string, Subscription>()\n userSubs.set(modelName, modelSubs)\n\n const existing = modelSubs.get(queryKey)\n if (existing) {\n existing.socketIds.add(socketId)\n existing.dependencyModelNames = Array.from(new Set([...existing.dependencyModelNames, ...dependencyModelNames]))\n } else {\n modelSubs.set(queryKey, {\n query,\n options,\n dependencyModelNames: Array.from(new Set(dependencyModelNames)),\n socketIds: new Set([socketId]),\n })\n }\n\n const byModel = socketSubscriptions.get(socketId) ?? new Map()\n socketSubscriptions.set(socketId, byModel)\n\n const querySet = byModel.get(modelName) ?? new Set()\n byModel.set(modelName, querySet)\n querySet.add(queryKey)\n}\n\nconst removeSocketSubscription = ({\n socketId,\n tenantId,\n userId,\n modelName,\n queryKey,\n}: {\n socketId: string\n tenantId: string\n userId: string\n modelName: string\n queryKey: string\n}): void => {\n const tenantSubs = subscriptions.get(tenantId)\n const userSubs = tenantSubs?.get(userId)\n const modelSubs = userSubs?.get(modelName)\n const sub = modelSubs?.get(queryKey)\n const affectedModelNames = new Set<string>([modelName, ...(sub?.dependencyModelNames ?? [])])\n if (sub) {\n sub.socketIds.delete(socketId)\n if (!sub.socketIds.size) {\n modelSubs?.delete(queryKey)\n }\n }\n\n const byModel = socketSubscriptions.get(socketId)\n const set = byModel?.get(modelName)\n if (set) {\n set.delete(queryKey)\n if (!set.size) {\n byModel?.delete(modelName)\n }\n }\n\n if (modelSubs && modelSubs.size === 0) {\n userSubs?.delete(modelName)\n }\n\n if (userSubs && userSubs.size === 0) {\n tenantSubs?.delete(userId)\n }\n\n for (const affectedModelName of affectedModelNames) {\n if (hasAnySubscriptionsDependingOnModel(tenantId, affectedModelName)) continue\n\n const tenantStreams = changeStreams.get(tenantId)\n const stream = tenantStreams?.get(affectedModelName)\n if (stream) {\n try {\n stream.close()\n } catch {\n // ignore\n }\n clearDispatchTimer(tenantId, affectedModelName)\n tenantStreams?.delete(affectedModelName)\n if (tenantStreams && tenantStreams.size === 0) changeStreams.delete(tenantId)\n }\n }\n\n if (tenantSubs && tenantSubs.size === 0) subscriptions.delete(tenantId)\n if (byModel && byModel.size === 0) socketSubscriptions.delete(socketId)\n}\n\nconst cleanupSocket = (socketId: string): void => {\n const meta = socketMeta.get(socketId)\n if (meta) {\n const byModel = socketSubscriptions.get(socketId)\n if (byModel) {\n for (const [modelName, keys] of byModel.entries()) {\n for (const queryKey of keys.values()) {\n removeSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName,\n queryKey,\n })\n }\n }\n }\n }\n\n socketSubscriptions.delete(socketId)\n\n const cleanupFns = socketCleanup.get(socketId) ?? []\n socketCleanup.delete(socketId)\n for (const fn of cleanupFns) {\n try {\n fn()\n } catch {\n // ignore\n }\n }\n\n sockets.delete(socketId)\n socketMeta.delete(socketId)\n socketWrappers.delete(socketId)\n}\n\nconst handleClientMessage = async ({\n socketId,\n meta,\n message,\n}: {\n socketId: string\n meta: SocketMeta\n message: ClientMessage\n}): Promise<void> => {\n const ws = sockets.get(socketId)\n if (!ws) return\n\n if (message.type === \"event\") {\n const wrapper = socketWrappers.get(socketId)\n wrapper?.dispatch(message.event, message.payload)\n return\n }\n\n if (!message.modelName || typeof message.modelName !== \"string\") return\n if (!allowInternalModels && INTERNAL_MODEL_NAMES.has(message.modelName)) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey ?? \"\", error: \"Model not allowed\" })\n return\n }\n if (!message.queryKey || typeof message.queryKey !== \"string\") return\n if (message.queryKey.length > QUERY_KEY_MAX_LEN) return\n\n if (message.type === \"remove-query\") {\n removeSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName: message.modelName,\n queryKey: message.queryKey,\n })\n return\n }\n\n if (!message.query || typeof message.query !== \"object\") return\n\n const options = normalizeRtsQueryOptions(message.options)\n const ability = meta.ability\n\n if (!ability.can(\"read\", message.modelName as AclSubjectType)) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error: \"forbidden\" })\n return\n }\n\n if (message.type === \"register-query\") {\n const existing = socketSubscriptions.get(socketId)?.get(message.modelName)?.has(message.queryKey) ?? false\n if (!existing) {\n let count = 0\n const byModel = socketSubscriptions.get(socketId)\n if (byModel) {\n for (const set of byModel.values()) count += set.size\n }\n\n if (count >= maxSubscriptionsPerSocket) {\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error: \"Too many subscriptions\" })\n return\n }\n }\n\n let dependencyModelNames: string[] = []\n if (options.populate !== undefined) {\n try {\n dependencyModelNames = await resolveRtsQueryDependencyModelNames({\n tenantId: meta.tenantId,\n ability,\n modelName: message.modelName,\n options,\n allowInternalModels,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n return\n }\n }\n\n addSocketSubscription({\n socketId,\n tenantId: meta.tenantId,\n userId: meta.userId,\n modelName: message.modelName,\n queryKey: message.queryKey,\n query: message.query,\n options,\n dependencyModelNames,\n })\n\n try {\n const modelNamesToWatch = new Set<string>([message.modelName, ...dependencyModelNames])\n for (const modelName of modelNamesToWatch) {\n await ensureChangeStream(meta.tenantId, modelName)\n }\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n return\n }\n\n if (message.runInitialQuery === false) {\n return\n }\n }\n\n try {\n await runAndSendQuery({\n tenantId: meta.tenantId,\n targetSocketIds: [socketId],\n ability,\n modelName: message.modelName,\n queryKey: message.queryKey,\n query: message.query,\n options,\n })\n } catch (err) {\n const error = redactErrorMessage(err)\n sendWs(ws, { type: \"query-payload\", modelName: message.modelName, queryKey: message.queryKey, error })\n }\n}\n\nexport const initRts = ({\n server,\n path = \"/rts\",\n sessionMiddleware,\n maxPayloadBytes: maxPayloadBytesArg,\n maxSubscriptionsPerSocket: maxSubscriptionsPerSocketArg,\n dispatchDebounceMs: dispatchDebounceMsArg,\n allowInternalModels: allowInternalModelsArg,\n}: {\n server: HttpServer\n path?: string\n sessionMiddleware?: RequestHandler\n maxPayloadBytes?: number\n maxSubscriptionsPerSocket?: number\n dispatchDebounceMs?: number\n allowInternalModels?: boolean\n}): void => {\n if (initializedServers.has(server)) return\n initializedServers.add(server)\n\n if (typeof maxPayloadBytesArg === \"number\" && Number.isFinite(maxPayloadBytesArg) && maxPayloadBytesArg > 0) {\n maxPayloadBytes = Math.floor(maxPayloadBytesArg)\n }\n\n if (\n typeof maxSubscriptionsPerSocketArg === \"number\"\n && Number.isFinite(maxSubscriptionsPerSocketArg)\n && maxSubscriptionsPerSocketArg > 0\n ) {\n maxSubscriptionsPerSocket = Math.floor(maxSubscriptionsPerSocketArg)\n }\n\n if (typeof dispatchDebounceMsArg === \"number\" && Number.isFinite(dispatchDebounceMsArg) && dispatchDebounceMsArg >= 0) {\n dispatchDebounceMs = Math.floor(dispatchDebounceMsArg)\n }\n\n allowInternalModels = Boolean(allowInternalModelsArg)\n\n const wss = new WebSocketServer({ noServer: true, maxPayload: maxPayloadBytes })\n\n server.on(\"upgrade\", (req: IncomingMessage, socket: Socket, head: Buffer) => {\n ensureSocketErrorHandler(socket, () => ({\n upgradeHost: req.headers.host ?? \"\",\n upgradeUrl: req.url ?? \"\",\n userAgent: typeof req.headers[\"user-agent\"] === \"string\" ? req.headers[\"user-agent\"] : \"\",\n }))\n upgradeMeta.delete(req)\n\n let url: URL\n try {\n url = new URL(req.url ?? \"\", `http://${req.headers.host ?? \"localhost\"}`)\n } catch {\n badRequest(socket, \"Invalid URL\")\n return\n }\n\n if (url.pathname !== path) return\n\n void (async() => {\n try {\n const meta = await parseUpgradeMeta({ req, url, sessionMiddleware })\n upgradeMeta.set(req, meta)\n\n wss.handleUpgrade(req, socket, head, (ws: WebSocket) => {\n wss.emit(\"connection\", ws, req)\n })\n } catch (err) {\n const message = err instanceof Error ? err.message : \"RTS upgrade failed\"\n if (message.startsWith(\"Missing rb-tenant-id\")) {\n badRequest(socket, message)\n return\n }\n unauthorized(socket, message)\n return\n }\n })().catch(() => {\n badRequest(socket, \"RTS upgrade failed\")\n })\n })\n\n wss.on(\"connection\", (ws: WebSocket, req: IncomingMessage) => {\n const meta = upgradeMeta.get(req)\n upgradeMeta.delete(req)\n if (!meta) {\n try {\n ws.close()\n } catch {\n // ignore\n }\n return\n }\n\n const socketId = randomUUID()\n sockets.set(socketId, ws)\n socketMeta.set(socketId, meta)\n\n const wrapper = new RtsSocket({ id: socketId, ws, meta })\n socketWrappers.set(socketId, wrapper)\n\n const cleanupFns: Array<() => void> = []\n for (const handler of customHandlers) {\n try {\n const cleanup = handler(wrapper)\n if (typeof cleanup === \"function\") cleanupFns.push(cleanup)\n } catch {\n // ignore\n }\n }\n if (cleanupFns.length) socketCleanup.set(socketId, cleanupFns)\n\n ws.on(\"message\", (raw: RawData) => {\n let parsed: unknown\n try {\n parsed = safeJsonParse(raw)\n } catch {\n return\n }\n\n if (!parsed || typeof parsed !== \"object\") return\n const message = parsed as { type?: unknown }\n if (\n message.type !== \"event\"\n && message.type !== \"run-query\"\n && message.type !== \"register-query\"\n && message.type !== \"remove-query\"\n ) return\n void handleClientMessage({ socketId, meta, message: message as ClientMessage })\n })\n\n ws.on(\"close\", () => {\n cleanupSocket(socketId)\n })\n\n ws.on(\"error\", () => {\n cleanupSocket(socketId)\n })\n })\n}\n\nexport const registerRtsHandler = (handler: HandlerFn): void => {\n customHandlers.push(handler)\n}\n\nexport const notifyRtsModelChanged = (tenantId: string, modelName: string): void => {\n scheduleDispatchSubscriptionsForModel(tenantId, modelName)\n}\n\nexport * from \"./routes\"\n"],"names":["routes","Object","entries","import","reduce","acc","path","mod","replace","QUERY_KEY_MAX_LEN","INTERNAL_MODEL_NAMES","Set","DEFAULT_MAX_PAYLOAD_BYTES","DEFAULT_MAX_SUBSCRIPTIONS_PER_SOCKET","DEFAULT_DISPATCH_DEBOUNCE_MS","initializedServers","WeakSet","customHandlers","sockets","Map","socketMeta","socketWrappers","socketCleanup","socketSubscriptions","subscriptions","changeStreams","dispatchTimers","upgradeMeta","WeakMap","maxPayloadBytes","maxSubscriptionsPerSocket","dispatchDebounceMs","allowInternalModels","RtsSocket","id","tenantId","userId","ws","handlers","constructor","meta","on","event","handler","set","get","add","off","delete","size","emit","payload","sendWs","type","close","dispatch","rawToText","raw","ArrayBuffer","Buffer","from","toString","Array","isArray","concat","safeJsonParse","JSON","parse","message","readyState","send","stringify","ensureSocketErrorHandler","socket","context","listenerCount","err","Error","String","name","stack","undefined","code","includes","console","warn","remoteAddress","remotePort","localAddress","localPort","redactErrorMessage","trimmedModelList","maxLen","length","slice","unauthorized","write","end","destroy","badRequest","runSessionMiddleware","sessionMiddleware","req","Promise","resolve","reject","next","parseUpgradeMeta","url","searchParams","TENANT_ID_QUERY_PARAM","upgradeReq","sessionUser","session","user","sessionUserId","signedInTenants","currentTenantId","ability","buildAbilityFromSession","headers","USER_ID_HEADER","headerUserId","rbCtx","User","models","getGlobal","findById","tenants","tenantRoles","lean","tenantsRaw","map","t","roles","getTenantRolesFromSessionUser","buildAbility","getTenantModel","modelName","ctx","getUnsafe","makeDispatchKey","clearDispatchTimer","key","timer","clearTimeout","scheduleDispatchSubscriptionsForModel","has","delay","Math","max","min","floor","setTimeout","dispatchSubscriptionsForModel","runAndSendQuery","targetSocketIds","queryKey","query","options","result","runRtsQuery","data","pageInfo","totalCount","socketId","subscriptionDependsOnModel","changedModelName","ownerModelName","subscription","dependencyModelNames","hasAnySubscriptionsDependingOnModel","tenantSubs","userSubs","values","modelSubs","sub","socketIds","error","ensureChangeStream","tenantStreams","model","stream","watch","fullDocument","addSocketSubscription","existing","byModel","querySet","removeSocketSubscription","affectedModelNames","affectedModelName","cleanupSocket","keys","cleanupFns","fn","handleClientMessage","wrapper","normalizeRtsQueryOptions","can","count","populate","resolveRtsQueryDependencyModelNames","modelNamesToWatch","runInitialQuery","initRts","server","maxPayloadBytesArg","maxSubscriptionsPerSocketArg","dispatchDebounceMsArg","allowInternalModelsArg","Number","isFinite","Boolean","wss","WebSocketServer","noServer","maxPayload","head","upgradeHost","host","upgradeUrl","userAgent","URL","pathname","handleUpgrade","startsWith","catch","randomUUID","cleanup","push","parsed","registerRtsHandler","notifyRtsModelChanged"],"mappings":"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"}
|
|
@@ -24,6 +24,7 @@ export type RtsQueryOptions = {
|
|
|
24
24
|
export type RtsQueryResult = {
|
|
25
25
|
data: unknown[];
|
|
26
26
|
pageInfo?: PaginationPageInfo;
|
|
27
|
+
totalCount?: number;
|
|
27
28
|
};
|
|
28
29
|
export declare const RTS_TENANT_ID_QUERY_PARAM = "rb-tenant-id";
|
|
29
30
|
export declare const RTS_USER_ID_HEADER = "rb-user-id";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"queryExecutor.d.ts","sourceRoot":"","sources":["../../src/rts/queryExecutor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,SAAS,CAAA;AACtC,OAAO,KAAK,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,cAAc,CAAA;AAEtE,OAAO,EAAmH,KAAK,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAMlK,KAAK,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;AAazC,MAAM,MAAM,iBAAiB,GAAG;IAC9B,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,MAAM,CAAC,EAAE,MAAM,GAAG,UAAU,CAAA;IAC5B,KAAK,CAAC,EAAE,UAAU,CAAA;IAClB,OAAO,CAAC,EAAE;QACR,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAA;QAC7B,KAAK,CAAC,EAAE,MAAM,CAAA;KACf,CAAA;IACD,QAAQ,CAAC,EAAE,iBAAiB,CAAA;CAC7B,CAAA;AAED,MAAM,MAAM,iBAAiB,GACzB,MAAM,GACN,iBAAiB,GACjB,KAAK,CAAC,MAAM,GAAG,iBAAiB,CAAC,CAAA;AAErC,MAAM,MAAM,eAAe,GAAG;IAC5B,UAAU,CAAC,EAAE,UAAU,CAAA;IACvB,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAA;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,QAAQ,CAAC,EAAE,iBAAiB,CAAA;IAC5B,UAAU,CAAC,EAAE,cAAc,CAAA;CAC5B,CAAA;AAED,MAAM,MAAM,cAAc,GAAG;IAC3B,IAAI,EAAE,OAAO,EAAE,CAAA;IACf,QAAQ,CAAC,EAAE,kBAAkB,CAAA;
|
|
1
|
+
{"version":3,"file":"queryExecutor.d.ts","sourceRoot":"","sources":["../../src/rts/queryExecutor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,SAAS,CAAA;AACtC,OAAO,KAAK,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,cAAc,CAAA;AAEtE,OAAO,EAAmH,KAAK,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAMlK,KAAK,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;AAazC,MAAM,MAAM,iBAAiB,GAAG;IAC9B,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,MAAM,CAAC,EAAE,MAAM,GAAG,UAAU,CAAA;IAC5B,KAAK,CAAC,EAAE,UAAU,CAAA;IAClB,OAAO,CAAC,EAAE;QACR,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAA;QAC7B,KAAK,CAAC,EAAE,MAAM,CAAA;KACf,CAAA;IACD,QAAQ,CAAC,EAAE,iBAAiB,CAAA;CAC7B,CAAA;AAED,MAAM,MAAM,iBAAiB,GACzB,MAAM,GACN,iBAAiB,GACjB,KAAK,CAAC,MAAM,GAAG,iBAAiB,CAAC,CAAA;AAErC,MAAM,MAAM,eAAe,GAAG;IAC5B,UAAU,CAAC,EAAE,UAAU,CAAA;IACvB,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAA;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,QAAQ,CAAC,EAAE,iBAAiB,CAAA;IAC5B,UAAU,CAAC,EAAE,cAAc,CAAA;CAC5B,CAAA;AAED,MAAM,MAAM,cAAc,GAAG;IAC3B,IAAI,EAAE,OAAO,EAAE,CAAA;IACf,QAAQ,CAAC,EAAE,kBAAkB,CAAA;IAC7B,UAAU,CAAC,EAAE,MAAM,CAAA;CACpB,CAAA;AAED,eAAO,MAAM,yBAAyB,iBAAiB,CAAA;AACvD,eAAO,MAAM,kBAAkB,eAAe,CAAA;AAsC9C,eAAO,MAAM,yBAAyB,GAAI,KAAK,OAAO,KAAG,MAAM,GAAG,IAEjE,CAAA;AAED,eAAO,MAAM,uBAAuB,GAAI,KAAK,OAAO,KAAG,MAAM,GAAG,IAO/D,CAAA;AAED,eAAO,MAAM,sBAAsB,GAAI,KAAK,OAAO,EAAE,UAAU,MAAM,KAAG,OAYvE,CAAA;AAED,eAAO,MAAM,0BAA0B,GACrC,KAAK,OAAO,EACZ,UAAU,MAAM,KACf,OAAO,CAAC;IAAE,OAAO,EAAE,UAAU,CAAC;IAAC,MAAM,EAAE,MAAM,GAAG,IAAI,CAAA;CAAE,CA8BxD,CAAA;AA6RD,eAAO,MAAM,wBAAwB,GAAI,SAAS,eAAe,GAAG,SAAS,KAAG,eAiB/E,CAAA;AAED,eAAO,MAAM,mCAAmC,GAAU,iEAMvD;IACD,QAAQ,EAAE,MAAM,CAAA;IAChB,OAAO,EAAE,UAAU,CAAA;IACnB,SAAS,EAAE,MAAM,CAAA;IACjB,OAAO,EAAE,eAAe,CAAA;IACxB,mBAAmB,CAAC,EAAE,OAAO,CAAA;CAC9B,KAAG,OAAO,CAAC,MAAM,EAAE,CAiBnB,CAAA;AAED,eAAO,MAAM,WAAW,GAAU,wEAO/B;IACD,QAAQ,EAAE,MAAM,CAAA;IAChB,OAAO,EAAE,UAAU,CAAA;IACnB,SAAS,EAAE,MAAM,CAAA;IACjB,KAAK,EAAE,UAAU,CAAA;IACjB,OAAO,EAAE,eAAe,CAAA;IACxB,mBAAmB,CAAC,EAAE,OAAO,CAAA;CAC9B,KAAG,OAAO,CAAC,cAAc,CAgEzB,CAAA"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ssrHydration.d.ts","sourceRoot":"","sources":["../../src/rts/ssrHydration.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,SAAS,CAAA;AACtC,OAAO,EAIL,KAAK,aAAa,EAClB,KAAK,6BAA6B,EACnC,MAAM,iBAAiB,CAAA;AA6BxB,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,aAAa,CAAA;IACtB,gBAAgB,EAAE,MAAM,OAAO,CAAA;IAC/B,OAAO,EAAE,MAAM,OAAO,CAAC,6BAA6B,GAAG,IAAI,CAAC,CAAA;CAC7D,CAAA;AAED,eAAO,MAAM,qBAAqB,GAChC,KAAK,OAAO,EACZ,OAAO;IACL,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,eAAe,CAAC,EAAE,MAAM,CAAA;IACxB,kBAAkB,CAAC,EAAE,MAAM,CAAA;CAC5B,KACA,
|
|
1
|
+
{"version":3,"file":"ssrHydration.d.ts","sourceRoot":"","sources":["../../src/rts/ssrHydration.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,SAAS,CAAA;AACtC,OAAO,EAIL,KAAK,aAAa,EAClB,KAAK,6BAA6B,EACnC,MAAM,iBAAiB,CAAA;AA6BxB,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,aAAa,CAAA;IACtB,gBAAgB,EAAE,MAAM,OAAO,CAAA;IAC/B,OAAO,EAAE,MAAM,OAAO,CAAC,6BAA6B,GAAG,IAAI,CAAC,CAAA;CAC7D,CAAA;AAED,eAAO,MAAM,qBAAqB,GAChC,KAAK,OAAO,EACZ,OAAO;IACL,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,eAAe,CAAC,EAAE,MAAM,CAAA;IACxB,kBAAkB,CAAC,EAAE,MAAM,CAAA;CAC5B,KACA,eAmHF,CAAA;AAED,eAAO,MAAM,wBAAwB,GACnC,MAAM,6BAA6B,GAAG,IAAI,GAAG,SAAS,KACrD,MAIF,CAAA"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"completeUpload.d.ts","sourceRoot":"","sources":["../../../../../src/uploads/api/file-uploads/handlers/completeUpload.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AAKzC,OAAO,KAAK,OAAO,MAAM,UAAU,CAAA;
|
|
1
|
+
{"version":3,"file":"completeUpload.d.ts","sourceRoot":"","sources":["../../../../../src/uploads/api/file-uploads/handlers/completeUpload.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AAKzC,OAAO,KAAK,OAAO,MAAM,UAAU,CAAA;AAGnC,OAAO,EACL,KAAK,WAAW,EASjB,MAAM,WAAW,CAAA;AAiDlB,eAAO,MAAM,cAAc,EAAE,UAAU,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,uBAAuB,EAAE,WAAW,CAoT1G,CAAA"}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
export type UploadPostProcessorContext = {
|
|
2
|
+
tenantId: string;
|
|
3
|
+
uploadId: string;
|
|
4
|
+
fileId: string;
|
|
5
|
+
filename: string;
|
|
6
|
+
mimeType: string;
|
|
7
|
+
clientMimeType: string;
|
|
8
|
+
totalSize: number;
|
|
9
|
+
isPublic?: boolean;
|
|
10
|
+
userId?: string;
|
|
11
|
+
ownerKeyHash?: string;
|
|
12
|
+
inlineProcessors: string[];
|
|
13
|
+
metadata: Record<string, unknown>;
|
|
14
|
+
};
|
|
15
|
+
export type UploadPostProcessor = {
|
|
16
|
+
id: string;
|
|
17
|
+
version?: number;
|
|
18
|
+
match?: (ctx: UploadPostProcessorContext) => boolean;
|
|
19
|
+
process: (ctx: UploadPostProcessorContext) => Promise<unknown> | unknown;
|
|
20
|
+
};
|
|
21
|
+
export declare const registerUploadPostProcessor: (processor: UploadPostProcessor) => void;
|
|
22
|
+
export declare const registerUploadProcessor: (processor: UploadPostProcessor) => void;
|
|
23
|
+
export declare const unregisterUploadPostProcessor: (id: string) => void;
|
|
24
|
+
export declare const unregisterUploadProcessor: (id: string) => void;
|
|
25
|
+
export declare const clearUploadPostProcessors: () => void;
|
|
26
|
+
export declare const clearUploadProcessors: () => void;
|
|
27
|
+
export declare const getUploadPostProcessors: () => UploadPostProcessor[];
|
|
28
|
+
export declare const getUploadProcessors: () => UploadPostProcessor[];
|
|
29
|
+
export declare const runUploadPostProcessors: (ctx: UploadPostProcessorContext) => Promise<void>;
|
|
30
|
+
//# sourceMappingURL=postProcessors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postProcessors.d.ts","sourceRoot":"","sources":["../../../../src/uploads/api/file-uploads/postProcessors.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,0BAA0B,GAAG;IACvC,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,cAAc,EAAE,MAAM,CAAA;IACtB,SAAS,EAAE,MAAM,CAAA;IACjB,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,gBAAgB,EAAE,MAAM,EAAE,CAAA;IAC1B,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAClC,CAAA;AAED,MAAM,MAAM,mBAAmB,GAAG;IAChC,EAAE,EAAE,MAAM,CAAA;IACV,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,KAAK,CAAC,EAAE,CAAC,GAAG,EAAE,0BAA0B,KAAK,OAAO,CAAA;IACpD,OAAO,EAAE,CAAC,GAAG,EAAE,0BAA0B,KAAK,OAAO,CAAC,OAAO,CAAC,GAAG,OAAO,CAAA;CACzE,CAAA;AAYD,eAAO,MAAM,2BAA2B,GAAI,WAAW,mBAAmB,KAAG,IAa5E,CAAA;AAED,eAAO,MAAM,uBAAuB,cAfmB,mBAAmB,KAAG,IAeX,CAAA;AAElE,eAAO,MAAM,6BAA6B,GAAI,IAAI,MAAM,KAAG,IAI1D,CAAA;AAED,eAAO,MAAM,yBAAyB,OANY,MAAM,KAAG,IAMW,CAAA;AAEtE,eAAO,MAAM,yBAAyB,QAAO,IAI5C,CAAA;AAED,eAAO,MAAM,qBAAqB,QANW,IAMiB,CAAA;AAE9D,eAAO,MAAM,uBAAuB,QAAO,mBAAmB,EAAmC,CAAA;AAEjG,eAAO,MAAM,mBAAmB,QAFW,mBAAmB,EAEJ,CAAA;AAE1D,eAAO,MAAM,uBAAuB,GAAU,KAAK,0BAA0B,KAAG,OAAO,CAAC,IAAI,CAoC3F,CAAA"}
|
package/dist/uploads.d.ts
CHANGED
package/dist/uploads.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"uploads.d.ts","sourceRoot":"","sources":["../src/uploads.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAA"}
|
|
1
|
+
{"version":3,"file":"uploads.d.ts","sourceRoot":"","sources":["../src/uploads.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAA;AAChC,cAAc,2CAA2C,CAAA"}
|
package/dist/uploads.js
CHANGED
|
@@ -1,10 +1,90 @@
|
|
|
1
1
|
const routes = Object.entries({
|
|
2
|
-
.../* @__PURE__ */ Object.assign({ "./api/file-uploads/handler.ts": () => import("./handler-
|
|
2
|
+
.../* @__PURE__ */ Object.assign({ "./api/file-uploads/handler.ts": () => import("./handler-3uwH4f67.js"), "./api/files/handler.ts": () => import("./handler-DnSJAQ_B.js") })
|
|
3
3
|
}).reduce((acc, [path, mod]) => {
|
|
4
4
|
acc[path.replace("./api/", "@rpcbase/server/uploads/api/")] = mod;
|
|
5
5
|
return acc;
|
|
6
6
|
}, {});
|
|
7
|
+
const processorsById = /* @__PURE__ */ Object.create(null);
|
|
8
|
+
const normalizeProcessorId = (value) => typeof value === "string" ? value.trim() : "";
|
|
9
|
+
const normalizeProcessorVersion = (value) => {
|
|
10
|
+
if (typeof value !== "number") return void 0;
|
|
11
|
+
if (!Number.isInteger(value) || value < 1) return void 0;
|
|
12
|
+
return value;
|
|
13
|
+
};
|
|
14
|
+
const registerUploadPostProcessor = (processor) => {
|
|
15
|
+
const normalizedId = normalizeProcessorId(processor.id);
|
|
16
|
+
if (!normalizedId) {
|
|
17
|
+
throw new Error("Upload post processor id is required.");
|
|
18
|
+
}
|
|
19
|
+
const normalizedVersion = normalizeProcessorVersion(processor.version);
|
|
20
|
+
processorsById[normalizedId] = {
|
|
21
|
+
...processor,
|
|
22
|
+
id: normalizedId,
|
|
23
|
+
...normalizedVersion ? {
|
|
24
|
+
version: normalizedVersion
|
|
25
|
+
} : {}
|
|
26
|
+
};
|
|
27
|
+
};
|
|
28
|
+
const registerUploadProcessor = registerUploadPostProcessor;
|
|
29
|
+
const unregisterUploadPostProcessor = (id) => {
|
|
30
|
+
const normalizedId = normalizeProcessorId(id);
|
|
31
|
+
if (!normalizedId) return;
|
|
32
|
+
delete processorsById[normalizedId];
|
|
33
|
+
};
|
|
34
|
+
const unregisterUploadProcessor = unregisterUploadPostProcessor;
|
|
35
|
+
const clearUploadPostProcessors = () => {
|
|
36
|
+
for (const id of Object.keys(processorsById)) {
|
|
37
|
+
delete processorsById[id];
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
const clearUploadProcessors = clearUploadPostProcessors;
|
|
41
|
+
const getUploadPostProcessors = () => Object.values(processorsById);
|
|
42
|
+
const getUploadProcessors = getUploadPostProcessors;
|
|
43
|
+
const runUploadPostProcessors = async (ctx) => {
|
|
44
|
+
const processors = getUploadPostProcessors();
|
|
45
|
+
if (!processors.length) return;
|
|
46
|
+
for (const processor of processors) {
|
|
47
|
+
if (processor.match) {
|
|
48
|
+
try {
|
|
49
|
+
if (!processor.match(ctx)) continue;
|
|
50
|
+
} catch (error) {
|
|
51
|
+
console.error("Upload post processor failed", {
|
|
52
|
+
processorId: processor.id,
|
|
53
|
+
processorVersion: processor.version,
|
|
54
|
+
tenantId: ctx.tenantId,
|
|
55
|
+
uploadId: ctx.uploadId,
|
|
56
|
+
fileId: ctx.fileId,
|
|
57
|
+
stage: "match",
|
|
58
|
+
error
|
|
59
|
+
});
|
|
60
|
+
continue;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
try {
|
|
64
|
+
await processor.process(ctx);
|
|
65
|
+
} catch (error) {
|
|
66
|
+
console.error("Upload post processor failed", {
|
|
67
|
+
processorId: processor.id,
|
|
68
|
+
processorVersion: processor.version,
|
|
69
|
+
tenantId: ctx.tenantId,
|
|
70
|
+
uploadId: ctx.uploadId,
|
|
71
|
+
fileId: ctx.fileId,
|
|
72
|
+
stage: "process",
|
|
73
|
+
error
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
};
|
|
7
78
|
export {
|
|
8
|
-
|
|
79
|
+
clearUploadPostProcessors,
|
|
80
|
+
clearUploadProcessors,
|
|
81
|
+
getUploadPostProcessors,
|
|
82
|
+
getUploadProcessors,
|
|
83
|
+
registerUploadPostProcessor,
|
|
84
|
+
registerUploadProcessor,
|
|
85
|
+
routes,
|
|
86
|
+
runUploadPostProcessors,
|
|
87
|
+
unregisterUploadPostProcessor,
|
|
88
|
+
unregisterUploadProcessor
|
|
9
89
|
};
|
|
10
90
|
//# sourceMappingURL=uploads.js.map
|
package/dist/uploads.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"uploads.js","sources":["../src/uploads/routes.ts"],"sourcesContent":["export const routes = Object.entries({\n ...import.meta.glob(\"./api/**/handler.ts\"),\n}).reduce<Record<string, unknown>>((acc, [path, mod]) => {\n acc[path.replace(\"./api/\", \"@rpcbase/server/uploads/api/\")] = mod\n return acc\n}, {})\n\n"],"names":["routes","Object","entries","import","reduce","acc","path","mod","replace"],"mappings":"AAAO,MAAMA,SAASC,OAAOC,QAAQ;AAAA,EACnC,GAAGC,uBAAAA,OAAAA,EAAAA,iCAAAA,MAAAA,OAAAA,uBAAAA,GAAAA,0BAAAA,MAAAA,OAAAA,uBAAAA,EAAAA,CAAAA;AACL,CAAC,EAAEC,OAAgC,CAACC,KAAK,CAACC,MAAMC,GAAG,MAAM;AACvDF,MAAIC,KAAKE,QAAQ,UAAU,8BAA8B,CAAC,IAAID;AAC9D,SAAOF;AACT,GAAG,CAAA,CAAE;"}
|
|
1
|
+
{"version":3,"file":"uploads.js","sources":["../src/uploads/routes.ts","../src/uploads/api/file-uploads/postProcessors.ts"],"sourcesContent":["export const routes = Object.entries({\n ...import.meta.glob(\"./api/**/handler.ts\"),\n}).reduce<Record<string, unknown>>((acc, [path, mod]) => {\n acc[path.replace(\"./api/\", \"@rpcbase/server/uploads/api/\")] = mod\n return acc\n}, {})\n\n","export type UploadPostProcessorContext = {\n tenantId: string\n uploadId: string\n fileId: string\n filename: string\n mimeType: string\n clientMimeType: string\n totalSize: number\n isPublic?: boolean\n userId?: string\n ownerKeyHash?: string\n inlineProcessors: string[]\n metadata: Record<string, unknown>\n}\n\nexport type UploadPostProcessor = {\n id: string\n version?: number\n match?: (ctx: UploadPostProcessorContext) => boolean\n process: (ctx: UploadPostProcessorContext) => Promise<unknown> | unknown\n}\n\nconst processorsById: Record<string, UploadPostProcessor> = Object.create(null)\n\nconst normalizeProcessorId = (value: unknown): string => (typeof value === \"string\" ? value.trim() : \"\")\n\nconst normalizeProcessorVersion = (value: unknown): number | undefined => {\n if (typeof value !== \"number\") return undefined\n if (!Number.isInteger(value) || value < 1) return undefined\n return value\n}\n\nexport const registerUploadPostProcessor = (processor: UploadPostProcessor): void => {\n const normalizedId = normalizeProcessorId(processor.id)\n if (!normalizedId) {\n throw new Error(\"Upload post processor id is required.\")\n }\n\n const normalizedVersion = normalizeProcessorVersion(processor.version)\n\n processorsById[normalizedId] = {\n ...processor,\n id: normalizedId,\n ...(normalizedVersion ? { version: normalizedVersion } : {}),\n }\n}\n\nexport const registerUploadProcessor = registerUploadPostProcessor\n\nexport const unregisterUploadPostProcessor = (id: string): void => {\n const normalizedId = normalizeProcessorId(id)\n if (!normalizedId) return\n delete processorsById[normalizedId]\n}\n\nexport const unregisterUploadProcessor = unregisterUploadPostProcessor\n\nexport const clearUploadPostProcessors = (): void => {\n for (const id of Object.keys(processorsById)) {\n delete processorsById[id]\n }\n}\n\nexport const clearUploadProcessors = clearUploadPostProcessors\n\nexport const getUploadPostProcessors = (): UploadPostProcessor[] => Object.values(processorsById)\n\nexport const getUploadProcessors = getUploadPostProcessors\n\nexport const runUploadPostProcessors = async (ctx: UploadPostProcessorContext): Promise<void> => {\n const processors = getUploadPostProcessors()\n if (!processors.length) return\n\n for (const processor of processors) {\n if (processor.match) {\n try {\n if (!processor.match(ctx)) continue\n } catch (error) {\n console.error(\"Upload post processor failed\", {\n processorId: processor.id,\n processorVersion: processor.version,\n tenantId: ctx.tenantId,\n uploadId: ctx.uploadId,\n fileId: ctx.fileId,\n stage: \"match\",\n error,\n })\n continue\n }\n }\n\n try {\n await processor.process(ctx)\n } catch (error) {\n console.error(\"Upload post processor failed\", {\n processorId: processor.id,\n processorVersion: processor.version,\n tenantId: ctx.tenantId,\n uploadId: ctx.uploadId,\n fileId: ctx.fileId,\n stage: \"process\",\n error,\n })\n }\n }\n}\n"],"names":["routes","Object","entries","import","reduce","acc","path","mod","replace","processorsById","create","normalizeProcessorId","value","trim","normalizeProcessorVersion","undefined","Number","isInteger","registerUploadPostProcessor","processor","normalizedId","id","Error","normalizedVersion","version","registerUploadProcessor","unregisterUploadPostProcessor","unregisterUploadProcessor","clearUploadPostProcessors","keys","clearUploadProcessors","getUploadPostProcessors","values","getUploadProcessors","runUploadPostProcessors","ctx","processors","length","match","error","console","processorId","processorVersion","tenantId","uploadId","fileId","stage","process"],"mappings":"AAAO,MAAMA,SAASC,OAAOC,QAAQ;AAAA,EACnC,GAAGC,uBAAAA,OAAAA,EAAAA,iCAAAA,MAAAA,OAAAA,uBAAAA,GAAAA,0BAAAA,MAAAA,OAAAA,uBAAAA,EAAAA,CAAAA;AACL,CAAC,EAAEC,OAAgC,CAACC,KAAK,CAACC,MAAMC,GAAG,MAAM;AACvDF,MAAIC,KAAKE,QAAQ,UAAU,8BAA8B,CAAC,IAAID;AAC9D,SAAOF;AACT,GAAG,CAAA,CAAE;ACiBL,MAAMI,iBAAsDR,uBAAOS,OAAO,IAAI;AAE9E,MAAMC,uBAAuBA,CAACC,UAA4B,OAAOA,UAAU,WAAWA,MAAMC,SAAS;AAErG,MAAMC,4BAA4BA,CAACF,UAAuC;AACxE,MAAI,OAAOA,UAAU,SAAU,QAAOG;AACtC,MAAI,CAACC,OAAOC,UAAUL,KAAK,KAAKA,QAAQ,EAAG,QAAOG;AAClD,SAAOH;AACT;AAEO,MAAMM,8BAA8BA,CAACC,cAAyC;AACnF,QAAMC,eAAeT,qBAAqBQ,UAAUE,EAAE;AACtD,MAAI,CAACD,cAAc;AACjB,UAAM,IAAIE,MAAM,uCAAuC;AAAA,EACzD;AAEA,QAAMC,oBAAoBT,0BAA0BK,UAAUK,OAAO;AAErEf,iBAAeW,YAAY,IAAI;AAAA,IAC7B,GAAGD;AAAAA,IACHE,IAAID;AAAAA,IACJ,GAAIG,oBAAoB;AAAA,MAAEC,SAASD;AAAAA,IAAAA,IAAsB,CAAA;AAAA,EAAC;AAE9D;AAEO,MAAME,0BAA0BP;AAEhC,MAAMQ,gCAAgCA,CAACL,OAAqB;AACjE,QAAMD,eAAeT,qBAAqBU,EAAE;AAC5C,MAAI,CAACD,aAAc;AACnB,SAAOX,eAAeW,YAAY;AACpC;AAEO,MAAMO,4BAA4BD;AAElC,MAAME,4BAA4BA,MAAY;AACnD,aAAWP,MAAMpB,OAAO4B,KAAKpB,cAAc,GAAG;AAC5C,WAAOA,eAAeY,EAAE;AAAA,EAC1B;AACF;AAEO,MAAMS,wBAAwBF;AAE9B,MAAMG,0BAA0BA,MAA6B9B,OAAO+B,OAAOvB,cAAc;AAEzF,MAAMwB,sBAAsBF;AAE5B,MAAMG,0BAA0B,OAAOC,QAAmD;AAC/F,QAAMC,aAAaL,wBAAAA;AACnB,MAAI,CAACK,WAAWC,OAAQ;AAExB,aAAWlB,aAAaiB,YAAY;AAClC,QAAIjB,UAAUmB,OAAO;AACnB,UAAI;AACF,YAAI,CAACnB,UAAUmB,MAAMH,GAAG,EAAG;AAAA,MAC7B,SAASI,OAAO;AACdC,gBAAQD,MAAM,gCAAgC;AAAA,UAC5CE,aAAatB,UAAUE;AAAAA,UACvBqB,kBAAkBvB,UAAUK;AAAAA,UAC5BmB,UAAUR,IAAIQ;AAAAA,UACdC,UAAUT,IAAIS;AAAAA,UACdC,QAAQV,IAAIU;AAAAA,UACZC,OAAO;AAAA,UACPP;AAAAA,QAAAA,CACD;AACD;AAAA,MACF;AAAA,IACF;AAEA,QAAI;AACF,YAAMpB,UAAU4B,QAAQZ,GAAG;AAAA,IAC7B,SAASI,OAAO;AACdC,cAAQD,MAAM,gCAAgC;AAAA,QAC5CE,aAAatB,UAAUE;AAAAA,QACvBqB,kBAAkBvB,UAAUK;AAAAA,QAC5BmB,UAAUR,IAAIQ;AAAAA,QACdC,UAAUT,IAAIS;AAAAA,QACdC,QAAQV,IAAIU;AAAAA,QACZC,OAAO;AAAA,QACPP;AAAAA,MAAAA,CACD;AAAA,IACH;AAAA,EACF;AACF;"}
|