@rpcbase/server 0.513.0 → 0.514.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/{email-DEw8keax.js → email-DK8uUU4X.js} +5 -2
- package/dist/{email-DEw8keax.js.map → email-DK8uUU4X.js.map} +1 -1
- package/dist/{handler-r-HYO3Oy.js → handler-3gksYOQv.js} +299 -99
- package/dist/{handler-r-HYO3Oy.js.map → handler-3gksYOQv.js.map} +1 -1
- package/dist/{handler-BNrqh1Kb.js → handler-BsauvgjA.js} +58 -19
- package/dist/{handler-BNrqh1Kb.js.map → handler-BsauvgjA.js.map} +1 -1
- package/dist/{handler-Cohj3cz3.js → handler-DY5UEwlw.js} +45 -18
- package/dist/{handler-Cohj3cz3.js.map → handler-DY5UEwlw.js.map} +1 -1
- package/dist/{handler-Bh3a6Br1.js → handler-GZgk5k3c.js} +326 -130
- package/dist/{handler-Bh3a6Br1.js.map → handler-GZgk5k3c.js.map} +1 -1
- package/dist/index.js +293 -155
- package/dist/index.js.map +1 -1
- package/dist/notifications.js +103 -32
- package/dist/notifications.js.map +1 -1
- package/dist/{queryExecutor-DSCpsVI8.js → queryExecutor-Dn62mIDL.js} +47 -33
- package/dist/{queryExecutor-DSCpsVI8.js.map → queryExecutor-Dn62mIDL.js.map} +1 -1
- package/dist/rts/index.js +99 -21
- package/dist/rts/index.js.map +1 -1
- package/dist/{shared-BJomDDWK.js → shared-Dy9x-P7F.js} +10 -7
- package/dist/{shared-BJomDDWK.js.map → shared-Dy9x-P7F.js.map} +1 -1
- package/dist/uploads.js +1 -1
- package/dist/uploads.js.map +1 -1
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"handler-r-HYO3Oy.js","sources":["../src/notifications/api/notifications/shared.ts","../src/notifications/api/notifications/index.ts","../src/notifications/api/notifications/handler.ts"],"sourcesContent":["import type { Ctx } from \"@rpcbase/api\"\n\n\ntype NotificationSessionUser = {\n id?: unknown\n currentTenantId?: unknown\n}\n\nexport const getSessionUser = (ctx: Ctx) => {\n const rawSessionUser = (ctx.req.session as { user?: NotificationSessionUser } | null | undefined)?.user\n const userId = typeof rawSessionUser?.id === \"string\" ? rawSessionUser.id.trim() : \"\"\n const tenantId = typeof rawSessionUser?.currentTenantId === \"string\" ? rawSessionUser.currentTenantId.trim() : \"\"\n\n if (!userId) {\n ctx.res.status(401)\n return null\n }\n\n if (!tenantId) {\n ctx.res.status(400)\n return null\n }\n\n return { userId, tenantId }\n}\n\n","import { z } from \"zod\"\n\n\nexport const ListRoute = \"/api/rb/notifications\"\nexport const CreateRoute = \"/api/rb/notifications/create\"\nexport const MarkReadRoute = \"/api/rb/notifications/:notificationId/read\"\nexport const ArchiveRoute = \"/api/rb/notifications/:notificationId/archive\"\nexport const MarkAllReadRoute = \"/api/rb/notifications/mark-all-read\"\nexport const SettingsRoute = \"/api/rb/notifications/settings\"\nexport const DigestRunRoute = \"/api/rb/notifications/digest/run\"\n\nexport const listRequestSchema = z.object({\n includeArchived: z.boolean().optional(),\n unreadOnly: z.boolean().optional(),\n limit: z.number().int().min(1).max(200).optional(),\n markSeen: z.boolean().optional(),\n})\n\nexport type ListRequestPayload = z.infer<typeof listRequestSchema>\n\nexport const createRequestSchema = z.object({\n topic: z.string().trim().min(1).optional(),\n title: z.string().trim().min(1),\n body: z.string().trim().optional(),\n url: z.string().trim().optional(),\n metadata: z.record(z.string(), z.unknown()).optional(),\n})\n\nexport type CreateRequestPayload = z.infer<typeof createRequestSchema>\n\nexport const notificationSchema = z.object({\n id: z.string(),\n topic: z.string().optional(),\n title: z.string(),\n body: z.string().optional(),\n url: z.string().optional(),\n createdAt: z.string(),\n seenAt: z.string().optional(),\n readAt: z.string().optional(),\n archivedAt: z.string().optional(),\n metadata: z.record(z.string(), z.unknown()).optional(),\n})\n\nexport type NotificationPayload = z.infer<typeof notificationSchema>\n\nexport const listResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n notifications: z.array(notificationSchema).optional(),\n unreadCount: z.number().int().min(0).optional(),\n unseenCount: z.number().int().min(0).optional(),\n})\n\nexport type ListResponsePayload = z.infer<typeof listResponseSchema>\n\nexport const createResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n id: z.string().optional(),\n})\n\nexport type CreateResponsePayload = z.infer<typeof createResponseSchema>\n\nexport const markReadResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type MarkReadResponsePayload = z.infer<typeof markReadResponseSchema>\n\nexport const archiveResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type ArchiveResponsePayload = z.infer<typeof archiveResponseSchema>\n\nexport const markAllReadResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type MarkAllReadResponsePayload = z.infer<typeof markAllReadResponseSchema>\n\nexport const digestFrequencySchema = z.enum([\"off\", \"daily\", \"weekly\"])\n\nexport type DigestFrequency = z.infer<typeof digestFrequencySchema>\n\nexport const topicPreferenceSchema = z.object({\n topic: z.string(),\n inApp: z.boolean(),\n emailDigest: z.boolean(),\n push: z.boolean(),\n})\n\nexport type TopicPreferencePayload = z.infer<typeof topicPreferenceSchema>\n\nexport const settingsSchema = z.object({\n digestFrequency: digestFrequencySchema,\n topicPreferences: z.array(topicPreferenceSchema),\n lastDigestSentAt: z.string().optional(),\n})\n\nexport type SettingsPayload = z.infer<typeof settingsSchema>\n\nexport const settingsResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n settings: settingsSchema.optional(),\n})\n\nexport type SettingsResponsePayload = z.infer<typeof settingsResponseSchema>\n\nexport const updateSettingsRequestSchema = z.object({\n digestFrequency: digestFrequencySchema.optional(),\n topicPreferences: z.array(topicPreferenceSchema).optional(),\n})\n\nexport type UpdateSettingsRequestPayload = z.infer<typeof updateSettingsRequestSchema>\n\nexport const updateSettingsResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n settings: settingsSchema.optional(),\n})\n\nexport type UpdateSettingsResponsePayload = z.infer<typeof updateSettingsResponseSchema>\n\nexport const digestRunRequestSchema = z.object({\n force: z.boolean().optional(),\n})\n\nexport type DigestRunRequestPayload = z.infer<typeof digestRunRequestSchema>\n\nexport const digestRunResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n sent: z.boolean().optional(),\n skippedReason: z.string().optional(),\n})\n\nexport type DigestRunResponsePayload = z.infer<typeof digestRunResponseSchema>\n","import { Api, type ApiHandler } from \"@rpcbase/api\"\nimport { models, type IRBNotification, type IRBNotificationSettings } from \"@rpcbase/db\"\nimport { buildAbilityFromSession, getAccessibleByQuery } from \"@rpcbase/db/acl\"\n\nimport { createNotification } from \"../../createNotification\"\nimport { sendNotificationsDigestForUser } from \"../../digest\"\nimport { getSessionUser } from \"./shared\"\n\nimport * as Notifications from \"./index\"\n\n\ntype NotificationDoc = IRBNotification & { _id: unknown }\ntype SettingsDoc = IRBNotificationSettings & { _id: unknown }\n\nconst toIso = (value: unknown): string | undefined => (value instanceof Date ? value.toISOString() : undefined)\n\nconst buildDisabledTopics = (\n settings: SettingsDoc | null,\n key: \"inApp\" | \"emailDigest\",\n): string[] => {\n const raw = settings?.topicPreferences\n if (!Array.isArray(raw) || raw.length === 0) return []\n\n return raw\n .map((pref) => {\n if (!pref || typeof pref !== \"object\") return null\n const topic = typeof (pref as { topic?: unknown }).topic === \"string\" ? (pref as { topic: string }).topic.trim() : \"\"\n if (!topic) return null\n const enabled = (pref as Record<string, unknown>)[key] === true\n return enabled ? null : topic\n })\n .filter((topic): topic is string => Boolean(topic))\n}\n\nconst listNotifications: ApiHandler<Notifications.ListRequestPayload, Notifications.ListResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.listRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const { userId } = session\n const includeArchived = parsed.data.includeArchived === true\n const unreadOnly = parsed.data.unreadOnly === true\n const limit = parsed.data.limit ?? 50\n const markSeen = parsed.data.markSeen === true\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne({ userId }).lean()) as SettingsDoc | null\n const disabledTopics = buildDisabledTopics(settings, \"inApp\")\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n const queryFilters: Record<string, unknown>[] = [\n { userId },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (!includeArchived) queryFilters.push({ archivedAt: { $exists: false } })\n if (unreadOnly) queryFilters.push({ readAt: { $exists: false } })\n if (disabledTopics.length > 0) queryFilters.push({ topic: { $nin: disabledTopics } })\n const query: Record<string, unknown> = { $and: queryFilters }\n\n const notifications = (await NotificationModel.find(query)\n .sort({ createdAt: -1 })\n .limit(limit)\n .lean()) as NotificationDoc[]\n\n const unseenQueryFilters: Record<string, unknown>[] = [\n { userId },\n { archivedAt: { $exists: false } },\n { seenAt: { $exists: false } },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) unseenQueryFilters.push({ topic: { $nin: disabledTopics } })\n const unseenQuery: Record<string, unknown> = { $and: unseenQueryFilters }\n\n const unreadQueryFilters: Record<string, unknown>[] = [\n { userId },\n { archivedAt: { $exists: false } },\n { readAt: { $exists: false } },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) unreadQueryFilters.push({ topic: { $nin: disabledTopics } })\n const unreadQuery: Record<string, unknown> = { $and: unreadQueryFilters }\n\n const [unreadCount, unseenCount] = await Promise.all([\n NotificationModel.countDocuments(unreadQuery),\n NotificationModel.countDocuments(unseenQuery),\n ])\n\n const now = markSeen ? new Date() : null\n if (now && unseenCount > 0) {\n await NotificationModel.updateMany(unseenQuery, { $set: { seenAt: now } })\n }\n\n return Notifications.listResponseSchema.parse({\n ok: true,\n notifications: notifications.map((n) => ({\n id: String(n._id),\n topic: typeof n.topic === \"string\" ? n.topic : undefined,\n title: typeof n.title === \"string\" ? n.title : \"\",\n body: typeof n.body === \"string\" ? n.body : undefined,\n url: typeof n.url === \"string\" ? n.url : undefined,\n createdAt: toIso(n.createdAt) ?? new Date().toISOString(),\n seenAt: toIso(n.seenAt) ?? (now && !n.archivedAt && !n.seenAt ? now.toISOString() : undefined),\n readAt: toIso(n.readAt),\n archivedAt: toIso(n.archivedAt),\n metadata: typeof n.metadata === \"object\" && n.metadata !== null ? (n.metadata as Record<string, unknown>) : undefined,\n })),\n unreadCount,\n unseenCount: now ? 0 : unseenCount,\n })\n}\n\nconst createNotificationForCurrentUser: ApiHandler<Notifications.CreateRequestPayload, Notifications.CreateResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"create\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.createRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const created = await createNotification(ctx, {\n userId: session.userId,\n topic: parsed.data.topic,\n title: parsed.data.title,\n body: parsed.data.body,\n url: parsed.data.url,\n metadata: parsed.data.metadata,\n })\n\n return Notifications.createResponseSchema.parse({ ok: true, id: created.id })\n}\n\nconst markRead: ApiHandler<unknown, Notifications.MarkReadResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const notificationId = typeof ctx.req.params.notificationId === \"string\" ? ctx.req.params.notificationId.trim() : \"\"\n if (!notificationId) {\n ctx.res.status(400)\n return { ok: false, error: \"missing_notification_id\" }\n }\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n const now = new Date()\n\n try {\n await NotificationModel.updateOne(\n { $and: [{ _id: notificationId }, { archivedAt: { $exists: false } }, getAccessibleByQuery(ability, \"update\", \"RBNotification\")] },\n { $set: { readAt: now, seenAt: now } },\n )\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_notification_id\" }\n }\n\n return { ok: true }\n}\n\nconst markAllRead: ApiHandler<unknown, Notifications.MarkAllReadResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne({ userId: session.userId }).lean()) as SettingsDoc | null\n const disabledTopics = buildDisabledTopics(settings, \"inApp\")\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n const queryFilters: Record<string, unknown>[] = [\n { userId: session.userId },\n { archivedAt: { $exists: false } },\n { readAt: { $exists: false } },\n getAccessibleByQuery(ability, \"update\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) queryFilters.push({ topic: { $nin: disabledTopics } })\n const query: Record<string, unknown> = { $and: queryFilters }\n\n const now = new Date()\n await NotificationModel.updateMany(query, { $set: { readAt: now, seenAt: now } })\n\n return { ok: true }\n}\n\nconst archiveNotification: ApiHandler<unknown, Notifications.ArchiveResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const notificationId = typeof ctx.req.params.notificationId === \"string\" ? ctx.req.params.notificationId.trim() : \"\"\n if (!notificationId) {\n ctx.res.status(400)\n return { ok: false, error: \"missing_notification_id\" }\n }\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n try {\n await NotificationModel.updateOne(\n { $and: [{ _id: notificationId }, { archivedAt: { $exists: false } }, getAccessibleByQuery(ability, \"update\", \"RBNotification\")] },\n { $set: { archivedAt: new Date() } },\n )\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_notification_id\" }\n }\n\n return { ok: true }\n}\n\nconst getSettings: ApiHandler<unknown, Notifications.SettingsResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotificationSettings\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne(\n { $and: [{ userId: session.userId }, getAccessibleByQuery(ability, \"read\", \"RBNotificationSettings\")] },\n ).lean()) as SettingsDoc | null\n\n const digestFrequencyRaw = typeof settings?.digestFrequency === \"string\" ? settings.digestFrequency : \"weekly\"\n const digestFrequency: Notifications.DigestFrequency =\n digestFrequencyRaw === \"off\" || digestFrequencyRaw === \"daily\" || digestFrequencyRaw === \"weekly\"\n ? digestFrequencyRaw\n : \"weekly\"\n\n const topicPreferences = Array.isArray(settings?.topicPreferences)\n ? settings!.topicPreferences.map((pref) => ({\n topic: typeof pref.topic === \"string\" ? pref.topic : \"\",\n inApp: pref.inApp === true,\n emailDigest: pref.emailDigest === true,\n push: pref.push === true,\n })).filter((pref) => pref.topic.length > 0)\n : []\n\n return Notifications.settingsResponseSchema.parse({\n ok: true,\n settings: {\n digestFrequency,\n topicPreferences,\n lastDigestSentAt: toIso(settings?.lastDigestSentAt),\n },\n })\n}\n\nconst updateSettings: ApiHandler<Notifications.UpdateSettingsRequestPayload, Notifications.UpdateSettingsResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotificationSettings\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.updateSettingsRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const nextValues: Record<string, unknown> = {}\n\n if (parsed.data.digestFrequency) {\n nextValues.digestFrequency = parsed.data.digestFrequency\n }\n\n if (parsed.data.topicPreferences) {\n const seen = new Set<string>()\n const next = parsed.data.topicPreferences\n .map((pref) => ({\n topic: pref.topic.trim(),\n inApp: pref.inApp,\n emailDigest: pref.emailDigest,\n push: pref.push,\n }))\n .filter((pref) => pref.topic.length > 0)\n .filter((pref) => {\n if (seen.has(pref.topic)) return false\n seen.add(pref.topic)\n return true\n })\n\n nextValues.topicPreferences = next\n }\n\n const ops: Record<string, unknown> = {\n $setOnInsert: { userId: session.userId },\n }\n\n if (Object.keys(nextValues).length > 0) {\n ops.$set = nextValues\n }\n\n const settings = (await SettingsModel.findOneAndUpdate(\n { $and: [{ userId: session.userId }, getAccessibleByQuery(ability, \"update\", \"RBNotificationSettings\")] },\n ops,\n { upsert: true, new: true, setDefaultsOnInsert: true },\n ).lean()) as SettingsDoc | null\n\n const digestFrequencyRaw = typeof settings?.digestFrequency === \"string\" ? settings.digestFrequency : \"weekly\"\n const digestFrequency: Notifications.DigestFrequency =\n digestFrequencyRaw === \"off\" || digestFrequencyRaw === \"daily\" || digestFrequencyRaw === \"weekly\"\n ? digestFrequencyRaw\n : \"weekly\"\n\n const topicPreferences = Array.isArray(settings?.topicPreferences)\n ? settings!.topicPreferences.map((pref) => ({\n topic: typeof pref.topic === \"string\" ? pref.topic : \"\",\n inApp: pref.inApp === true,\n emailDigest: pref.emailDigest === true,\n push: pref.push === true,\n })).filter((pref) => pref.topic.length > 0)\n : []\n\n return Notifications.updateSettingsResponseSchema.parse({\n ok: true,\n settings: {\n digestFrequency,\n topicPreferences,\n lastDigestSentAt: toIso(settings?.lastDigestSentAt),\n },\n })\n}\n\nconst runDigest: ApiHandler<Notifications.DigestRunRequestPayload, Notifications.DigestRunResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.digestRunRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const result = await sendNotificationsDigestForUser(ctx, {\n userId: session.userId,\n force: parsed.data.force === true,\n })\n\n if (!result.ok) {\n ctx.res.status(500)\n return { ok: false, error: result.error }\n }\n\n return {\n ok: true,\n sent: result.sent,\n ...(result.skippedReason ? { skippedReason: result.skippedReason } : {}),\n }\n}\n\nexport default (api: Api) => {\n api.post(Notifications.ListRoute, listNotifications)\n api.post(Notifications.CreateRoute, createNotificationForCurrentUser)\n api.post(Notifications.MarkReadRoute, markRead)\n api.post(Notifications.MarkAllReadRoute, markAllRead)\n api.post(Notifications.ArchiveRoute, archiveNotification)\n api.get(Notifications.SettingsRoute, getSettings)\n api.put(Notifications.SettingsRoute, updateSettings)\n api.post(Notifications.DigestRunRoute, runDigest)\n}\n"],"names":["z.object","z.boolean","z.number","z.string","z.record","z.unknown","z.array","z.enum","Notifications.listRequestSchema","Notifications.listResponseSchema","Notifications.createRequestSchema","Notifications.createResponseSchema","Notifications.settingsResponseSchema","Notifications.updateSettingsRequestSchema","Notifications.updateSettingsResponseSchema","Notifications.digestRunRequestSchema","Notifications.ListRoute","Notifications.CreateRoute","Notifications.MarkReadRoute","Notifications.MarkAllReadRoute","Notifications.ArchiveRoute","Notifications.SettingsRoute","Notifications.DigestRunRoute"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"handler-3gksYOQv.js","sources":["../src/notifications/api/notifications/shared.ts","../src/notifications/api/notifications/index.ts","../src/notifications/api/notifications/handler.ts"],"sourcesContent":["import type { Ctx } from \"@rpcbase/api\"\n\n\ntype NotificationSessionUser = {\n id?: unknown\n currentTenantId?: unknown\n}\n\nexport const getSessionUser = (ctx: Ctx) => {\n const rawSessionUser = (ctx.req.session as { user?: NotificationSessionUser } | null | undefined)?.user\n const userId = typeof rawSessionUser?.id === \"string\" ? rawSessionUser.id.trim() : \"\"\n const tenantId = typeof rawSessionUser?.currentTenantId === \"string\" ? rawSessionUser.currentTenantId.trim() : \"\"\n\n if (!userId) {\n ctx.res.status(401)\n return null\n }\n\n if (!tenantId) {\n ctx.res.status(400)\n return null\n }\n\n return { userId, tenantId }\n}\n\n","import { z } from \"zod\"\n\n\nexport const ListRoute = \"/api/rb/notifications\"\nexport const CreateRoute = \"/api/rb/notifications/create\"\nexport const MarkReadRoute = \"/api/rb/notifications/:notificationId/read\"\nexport const ArchiveRoute = \"/api/rb/notifications/:notificationId/archive\"\nexport const MarkAllReadRoute = \"/api/rb/notifications/mark-all-read\"\nexport const SettingsRoute = \"/api/rb/notifications/settings\"\nexport const DigestRunRoute = \"/api/rb/notifications/digest/run\"\n\nexport const listRequestSchema = z.object({\n includeArchived: z.boolean().optional(),\n unreadOnly: z.boolean().optional(),\n limit: z.number().int().min(1).max(200).optional(),\n markSeen: z.boolean().optional(),\n})\n\nexport type ListRequestPayload = z.infer<typeof listRequestSchema>\n\nexport const createRequestSchema = z.object({\n topic: z.string().trim().min(1).optional(),\n title: z.string().trim().min(1),\n body: z.string().trim().optional(),\n url: z.string().trim().optional(),\n metadata: z.record(z.string(), z.unknown()).optional(),\n})\n\nexport type CreateRequestPayload = z.infer<typeof createRequestSchema>\n\nexport const notificationSchema = z.object({\n id: z.string(),\n topic: z.string().optional(),\n title: z.string(),\n body: z.string().optional(),\n url: z.string().optional(),\n createdAt: z.string(),\n seenAt: z.string().optional(),\n readAt: z.string().optional(),\n archivedAt: z.string().optional(),\n metadata: z.record(z.string(), z.unknown()).optional(),\n})\n\nexport type NotificationPayload = z.infer<typeof notificationSchema>\n\nexport const listResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n notifications: z.array(notificationSchema).optional(),\n unreadCount: z.number().int().min(0).optional(),\n unseenCount: z.number().int().min(0).optional(),\n})\n\nexport type ListResponsePayload = z.infer<typeof listResponseSchema>\n\nexport const createResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n id: z.string().optional(),\n})\n\nexport type CreateResponsePayload = z.infer<typeof createResponseSchema>\n\nexport const markReadResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type MarkReadResponsePayload = z.infer<typeof markReadResponseSchema>\n\nexport const archiveResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type ArchiveResponsePayload = z.infer<typeof archiveResponseSchema>\n\nexport const markAllReadResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n})\n\nexport type MarkAllReadResponsePayload = z.infer<typeof markAllReadResponseSchema>\n\nexport const digestFrequencySchema = z.enum([\"off\", \"daily\", \"weekly\"])\n\nexport type DigestFrequency = z.infer<typeof digestFrequencySchema>\n\nexport const topicPreferenceSchema = z.object({\n topic: z.string(),\n inApp: z.boolean(),\n emailDigest: z.boolean(),\n push: z.boolean(),\n})\n\nexport type TopicPreferencePayload = z.infer<typeof topicPreferenceSchema>\n\nexport const settingsSchema = z.object({\n digestFrequency: digestFrequencySchema,\n topicPreferences: z.array(topicPreferenceSchema),\n lastDigestSentAt: z.string().optional(),\n})\n\nexport type SettingsPayload = z.infer<typeof settingsSchema>\n\nexport const settingsResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n settings: settingsSchema.optional(),\n})\n\nexport type SettingsResponsePayload = z.infer<typeof settingsResponseSchema>\n\nexport const updateSettingsRequestSchema = z.object({\n digestFrequency: digestFrequencySchema.optional(),\n topicPreferences: z.array(topicPreferenceSchema).optional(),\n})\n\nexport type UpdateSettingsRequestPayload = z.infer<typeof updateSettingsRequestSchema>\n\nexport const updateSettingsResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n settings: settingsSchema.optional(),\n})\n\nexport type UpdateSettingsResponsePayload = z.infer<typeof updateSettingsResponseSchema>\n\nexport const digestRunRequestSchema = z.object({\n force: z.boolean().optional(),\n})\n\nexport type DigestRunRequestPayload = z.infer<typeof digestRunRequestSchema>\n\nexport const digestRunResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n sent: z.boolean().optional(),\n skippedReason: z.string().optional(),\n})\n\nexport type DigestRunResponsePayload = z.infer<typeof digestRunResponseSchema>\n","import { Api, type ApiHandler } from \"@rpcbase/api\"\nimport { models, type IRBNotification, type IRBNotificationSettings } from \"@rpcbase/db\"\nimport { buildAbilityFromSession, getAccessibleByQuery } from \"@rpcbase/db/acl\"\n\nimport { createNotification } from \"../../createNotification\"\nimport { sendNotificationsDigestForUser } from \"../../digest\"\nimport { getSessionUser } from \"./shared\"\n\nimport * as Notifications from \"./index\"\n\n\ntype NotificationDoc = IRBNotification & { _id: unknown }\ntype SettingsDoc = IRBNotificationSettings & { _id: unknown }\n\nconst toIso = (value: unknown): string | undefined => (value instanceof Date ? value.toISOString() : undefined)\n\nconst buildDisabledTopics = (\n settings: SettingsDoc | null,\n key: \"inApp\" | \"emailDigest\",\n): string[] => {\n const raw = settings?.topicPreferences\n if (!Array.isArray(raw) || raw.length === 0) return []\n\n return raw\n .map((pref) => {\n if (!pref || typeof pref !== \"object\") return null\n const topic = typeof (pref as { topic?: unknown }).topic === \"string\" ? (pref as { topic: string }).topic.trim() : \"\"\n if (!topic) return null\n const enabled = (pref as Record<string, unknown>)[key] === true\n return enabled ? null : topic\n })\n .filter((topic): topic is string => Boolean(topic))\n}\n\nconst listNotifications: ApiHandler<Notifications.ListRequestPayload, Notifications.ListResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.listRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const { userId } = session\n const includeArchived = parsed.data.includeArchived === true\n const unreadOnly = parsed.data.unreadOnly === true\n const limit = parsed.data.limit ?? 50\n const markSeen = parsed.data.markSeen === true\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne({ userId }).lean()) as SettingsDoc | null\n const disabledTopics = buildDisabledTopics(settings, \"inApp\")\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n const queryFilters: Record<string, unknown>[] = [\n { userId },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (!includeArchived) queryFilters.push({ archivedAt: { $exists: false } })\n if (unreadOnly) queryFilters.push({ readAt: { $exists: false } })\n if (disabledTopics.length > 0) queryFilters.push({ topic: { $nin: disabledTopics } })\n const query: Record<string, unknown> = { $and: queryFilters }\n\n const notifications = (await NotificationModel.find(query)\n .sort({ createdAt: -1 })\n .limit(limit)\n .lean()) as NotificationDoc[]\n\n const unseenQueryFilters: Record<string, unknown>[] = [\n { userId },\n { archivedAt: { $exists: false } },\n { seenAt: { $exists: false } },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) unseenQueryFilters.push({ topic: { $nin: disabledTopics } })\n const unseenQuery: Record<string, unknown> = { $and: unseenQueryFilters }\n\n const unreadQueryFilters: Record<string, unknown>[] = [\n { userId },\n { archivedAt: { $exists: false } },\n { readAt: { $exists: false } },\n getAccessibleByQuery(ability, \"read\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) unreadQueryFilters.push({ topic: { $nin: disabledTopics } })\n const unreadQuery: Record<string, unknown> = { $and: unreadQueryFilters }\n\n const [unreadCount, unseenCount] = await Promise.all([\n NotificationModel.countDocuments(unreadQuery),\n NotificationModel.countDocuments(unseenQuery),\n ])\n\n const now = markSeen ? new Date() : null\n if (now && unseenCount > 0) {\n await NotificationModel.updateMany(unseenQuery, { $set: { seenAt: now } })\n }\n\n return Notifications.listResponseSchema.parse({\n ok: true,\n notifications: notifications.map((n) => ({\n id: String(n._id),\n topic: typeof n.topic === \"string\" ? n.topic : undefined,\n title: typeof n.title === \"string\" ? n.title : \"\",\n body: typeof n.body === \"string\" ? n.body : undefined,\n url: typeof n.url === \"string\" ? n.url : undefined,\n createdAt: toIso(n.createdAt) ?? new Date().toISOString(),\n seenAt: toIso(n.seenAt) ?? (now && !n.archivedAt && !n.seenAt ? now.toISOString() : undefined),\n readAt: toIso(n.readAt),\n archivedAt: toIso(n.archivedAt),\n metadata: typeof n.metadata === \"object\" && n.metadata !== null ? (n.metadata as Record<string, unknown>) : undefined,\n })),\n unreadCount,\n unseenCount: now ? 0 : unseenCount,\n })\n}\n\nconst createNotificationForCurrentUser: ApiHandler<Notifications.CreateRequestPayload, Notifications.CreateResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"create\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.createRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const created = await createNotification(ctx, {\n userId: session.userId,\n topic: parsed.data.topic,\n title: parsed.data.title,\n body: parsed.data.body,\n url: parsed.data.url,\n metadata: parsed.data.metadata,\n })\n\n return Notifications.createResponseSchema.parse({ ok: true, id: created.id })\n}\n\nconst markRead: ApiHandler<unknown, Notifications.MarkReadResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const notificationId = typeof ctx.req.params.notificationId === \"string\" ? ctx.req.params.notificationId.trim() : \"\"\n if (!notificationId) {\n ctx.res.status(400)\n return { ok: false, error: \"missing_notification_id\" }\n }\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n const now = new Date()\n\n try {\n await NotificationModel.updateOne(\n { $and: [{ _id: notificationId }, { archivedAt: { $exists: false } }, getAccessibleByQuery(ability, \"update\", \"RBNotification\")] },\n { $set: { readAt: now, seenAt: now } },\n )\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_notification_id\" }\n }\n\n return { ok: true }\n}\n\nconst markAllRead: ApiHandler<unknown, Notifications.MarkAllReadResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne({ userId: session.userId }).lean()) as SettingsDoc | null\n const disabledTopics = buildDisabledTopics(settings, \"inApp\")\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n const queryFilters: Record<string, unknown>[] = [\n { userId: session.userId },\n { archivedAt: { $exists: false } },\n { readAt: { $exists: false } },\n getAccessibleByQuery(ability, \"update\", \"RBNotification\"),\n ]\n if (disabledTopics.length > 0) queryFilters.push({ topic: { $nin: disabledTopics } })\n const query: Record<string, unknown> = { $and: queryFilters }\n\n const now = new Date()\n await NotificationModel.updateMany(query, { $set: { readAt: now, seenAt: now } })\n\n return { ok: true }\n}\n\nconst archiveNotification: ApiHandler<unknown, Notifications.ArchiveResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const notificationId = typeof ctx.req.params.notificationId === \"string\" ? ctx.req.params.notificationId.trim() : \"\"\n if (!notificationId) {\n ctx.res.status(400)\n return { ok: false, error: \"missing_notification_id\" }\n }\n\n const NotificationModel = await models.get(\"RBNotification\", ctx)\n\n try {\n await NotificationModel.updateOne(\n { $and: [{ _id: notificationId }, { archivedAt: { $exists: false } }, getAccessibleByQuery(ability, \"update\", \"RBNotification\")] },\n { $set: { archivedAt: new Date() } },\n )\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_notification_id\" }\n }\n\n return { ok: true }\n}\n\nconst getSettings: ApiHandler<unknown, Notifications.SettingsResponsePayload> = async(\n _payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotificationSettings\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const settings = (await SettingsModel.findOne(\n { $and: [{ userId: session.userId }, getAccessibleByQuery(ability, \"read\", \"RBNotificationSettings\")] },\n ).lean()) as SettingsDoc | null\n\n const digestFrequencyRaw = typeof settings?.digestFrequency === \"string\" ? settings.digestFrequency : \"weekly\"\n const digestFrequency: Notifications.DigestFrequency =\n digestFrequencyRaw === \"off\" || digestFrequencyRaw === \"daily\" || digestFrequencyRaw === \"weekly\"\n ? digestFrequencyRaw\n : \"weekly\"\n\n const topicPreferences = Array.isArray(settings?.topicPreferences)\n ? settings!.topicPreferences.map((pref) => ({\n topic: typeof pref.topic === \"string\" ? pref.topic : \"\",\n inApp: pref.inApp === true,\n emailDigest: pref.emailDigest === true,\n push: pref.push === true,\n })).filter((pref) => pref.topic.length > 0)\n : []\n\n return Notifications.settingsResponseSchema.parse({\n ok: true,\n settings: {\n digestFrequency,\n topicPreferences,\n lastDigestSentAt: toIso(settings?.lastDigestSentAt),\n },\n })\n}\n\nconst updateSettings: ApiHandler<Notifications.UpdateSettingsRequestPayload, Notifications.UpdateSettingsResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"update\", \"RBNotificationSettings\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.updateSettingsRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const SettingsModel = await models.get(\"RBNotificationSettings\", ctx)\n const nextValues: Record<string, unknown> = {}\n\n if (parsed.data.digestFrequency) {\n nextValues.digestFrequency = parsed.data.digestFrequency\n }\n\n if (parsed.data.topicPreferences) {\n const seen = new Set<string>()\n const next = parsed.data.topicPreferences\n .map((pref) => ({\n topic: pref.topic.trim(),\n inApp: pref.inApp,\n emailDigest: pref.emailDigest,\n push: pref.push,\n }))\n .filter((pref) => pref.topic.length > 0)\n .filter((pref) => {\n if (seen.has(pref.topic)) return false\n seen.add(pref.topic)\n return true\n })\n\n nextValues.topicPreferences = next\n }\n\n const ops: Record<string, unknown> = {\n $setOnInsert: { userId: session.userId },\n }\n\n if (Object.keys(nextValues).length > 0) {\n ops.$set = nextValues\n }\n\n const settings = (await SettingsModel.findOneAndUpdate(\n { $and: [{ userId: session.userId }, getAccessibleByQuery(ability, \"update\", \"RBNotificationSettings\")] },\n ops,\n { upsert: true, new: true, setDefaultsOnInsert: true },\n ).lean()) as SettingsDoc | null\n\n const digestFrequencyRaw = typeof settings?.digestFrequency === \"string\" ? settings.digestFrequency : \"weekly\"\n const digestFrequency: Notifications.DigestFrequency =\n digestFrequencyRaw === \"off\" || digestFrequencyRaw === \"daily\" || digestFrequencyRaw === \"weekly\"\n ? digestFrequencyRaw\n : \"weekly\"\n\n const topicPreferences = Array.isArray(settings?.topicPreferences)\n ? settings!.topicPreferences.map((pref) => ({\n topic: typeof pref.topic === \"string\" ? pref.topic : \"\",\n inApp: pref.inApp === true,\n emailDigest: pref.emailDigest === true,\n push: pref.push === true,\n })).filter((pref) => pref.topic.length > 0)\n : []\n\n return Notifications.updateSettingsResponseSchema.parse({\n ok: true,\n settings: {\n digestFrequency,\n topicPreferences,\n lastDigestSentAt: toIso(settings?.lastDigestSentAt),\n },\n })\n}\n\nconst runDigest: ApiHandler<Notifications.DigestRunRequestPayload, Notifications.DigestRunResponsePayload> = async(\n payload,\n ctx,\n) => {\n const session = getSessionUser(ctx)\n if (!session) {\n return { ok: false, error: \"unauthorized\" }\n }\n\n const ability = buildAbilityFromSession({ tenantId: session.tenantId, session: ctx.req.session })\n if (!ability.can(\"read\", \"RBNotification\")) {\n ctx.res.status(403)\n return { ok: false, error: \"forbidden\" }\n }\n\n const parsed = Notifications.digestRunRequestSchema.safeParse(payload)\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const result = await sendNotificationsDigestForUser(ctx, {\n userId: session.userId,\n force: parsed.data.force === true,\n })\n\n if (!result.ok) {\n ctx.res.status(500)\n return { ok: false, error: result.error }\n }\n\n return {\n ok: true,\n sent: result.sent,\n ...(result.skippedReason ? { skippedReason: result.skippedReason } : {}),\n }\n}\n\nexport default (api: Api) => {\n api.post(Notifications.ListRoute, listNotifications)\n api.post(Notifications.CreateRoute, createNotificationForCurrentUser)\n api.post(Notifications.MarkReadRoute, markRead)\n api.post(Notifications.MarkAllReadRoute, markAllRead)\n api.post(Notifications.ArchiveRoute, archiveNotification)\n api.get(Notifications.SettingsRoute, getSettings)\n api.put(Notifications.SettingsRoute, updateSettings)\n api.post(Notifications.DigestRunRoute, runDigest)\n}\n"],"names":["getSessionUser","ctx","rawSessionUser","req","session","user","userId","id","trim","tenantId","currentTenantId","res","status","ListRoute","CreateRoute","MarkReadRoute","ArchiveRoute","MarkAllReadRoute","SettingsRoute","DigestRunRoute","listRequestSchema","z","includeArchived","boolean","optional","unreadOnly","limit","int","min","max","markSeen","createRequestSchema","topic","string","title","body","url","metadata","unknown","notificationSchema","createdAt","seenAt","readAt","archivedAt","listResponseSchema","ok","error","notifications","unreadCount","number","unseenCount","createResponseSchema","digestFrequencySchema","topicPreferenceSchema","inApp","emailDigest","push","settingsSchema","digestFrequency","topicPreferences","lastDigestSentAt","settingsResponseSchema","settings","updateSettingsRequestSchema","updateSettingsResponseSchema","digestRunRequestSchema","force","sent","skippedReason","toIso","value","Date","toISOString","undefined","buildDisabledTopics","key","raw","Array","isArray","length","map","pref","enabled","filter","Boolean","listNotifications","payload","ability","buildAbilityFromSession","can","parsed","Notifications","safeParse","success","data","SettingsModel","models","get","findOne","lean","disabledTopics","NotificationModel","queryFilters","getAccessibleByQuery","$exists","$nin","query","$and","find","sort","unseenQueryFilters","unseenQuery","unreadQueryFilters","unreadQuery","Promise","all","countDocuments","now","updateMany","$set","parse","n","String","_id","createNotificationForCurrentUser","created","createNotification","markRead","_payload","notificationId","params","updateOne","markAllRead","archiveNotification","getSettings","digestFrequencyRaw","updateSettings","nextValues","seen","Set","next","has","add","ops","$setOnInsert","Object","keys","findOneAndUpdate","upsert","new","setDefaultsOnInsert","runDigest","result","sendNotificationsDigestForUser","api","post","put"],"mappings":"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"}
|
|
@@ -57,41 +57,80 @@ const changesHandler = async (payload, ctx) => {
|
|
|
57
57
|
const parsed = requestSchema.safeParse(payload ?? {});
|
|
58
58
|
if (!parsed.success) {
|
|
59
59
|
ctx.res.status(400);
|
|
60
|
-
return {
|
|
60
|
+
return {
|
|
61
|
+
ok: false,
|
|
62
|
+
latestSeq: 0,
|
|
63
|
+
changes: []
|
|
64
|
+
};
|
|
61
65
|
}
|
|
62
66
|
const tenantId = getTenantId(ctx);
|
|
63
67
|
if (!tenantId) {
|
|
64
68
|
ctx.res.status(400);
|
|
65
|
-
return {
|
|
69
|
+
return {
|
|
70
|
+
ok: false,
|
|
71
|
+
latestSeq: 0,
|
|
72
|
+
changes: []
|
|
73
|
+
};
|
|
66
74
|
}
|
|
67
75
|
const userId = ensureAuthorized(ctx, tenantId);
|
|
68
76
|
if (!userId) {
|
|
69
77
|
ctx.res.status(401);
|
|
70
|
-
return {
|
|
78
|
+
return {
|
|
79
|
+
ok: false,
|
|
80
|
+
latestSeq: 0,
|
|
81
|
+
changes: []
|
|
82
|
+
};
|
|
71
83
|
}
|
|
72
|
-
const ability = buildAbilityFromSession({
|
|
84
|
+
const ability = buildAbilityFromSession({
|
|
85
|
+
tenantId,
|
|
86
|
+
session: ctx.req.session
|
|
87
|
+
});
|
|
73
88
|
const modelCtx = getModelCtx(ctx, tenantId);
|
|
74
|
-
const [RtsChange, RtsCounter] = await Promise.all([
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
89
|
+
const [RtsChange, RtsCounter] = await Promise.all([models.get("RBRtsChange", modelCtx), models.get("RBRtsCounter", modelCtx)]);
|
|
90
|
+
const counter = await RtsCounter.findOne({
|
|
91
|
+
_id: "rts"
|
|
92
|
+
}, {
|
|
93
|
+
seq: 1
|
|
94
|
+
}).lean();
|
|
79
95
|
const latestSeq = Number(counter?.seq ?? 0) || 0;
|
|
80
|
-
const {
|
|
96
|
+
const {
|
|
97
|
+
sinceSeq,
|
|
98
|
+
limit,
|
|
99
|
+
modelNames
|
|
100
|
+
} = parsed.data;
|
|
81
101
|
const requestedModelNames = Array.isArray(modelNames) && modelNames.length ? modelNames.map((m) => String(m)).filter(Boolean) : null;
|
|
82
|
-
const allowedModelNames = requestedModelNames ? requestedModelNames.filter((m) => ability.can("read", m)) : Array.from(
|
|
83
|
-
new Set(
|
|
84
|
-
(await RtsChange.distinct("modelName")).map((m) => String(m)).filter(Boolean).filter((m) => ability.can("read", m))
|
|
85
|
-
)
|
|
86
|
-
);
|
|
102
|
+
const allowedModelNames = requestedModelNames ? requestedModelNames.filter((m) => ability.can("read", m)) : Array.from(new Set((await RtsChange.distinct("modelName")).map((m) => String(m)).filter(Boolean).filter((m) => ability.can("read", m))));
|
|
87
103
|
let earliestSeq;
|
|
88
104
|
if (allowedModelNames.length) {
|
|
89
|
-
const earliest = await RtsChange.findOne({
|
|
105
|
+
const earliest = await RtsChange.findOne({
|
|
106
|
+
modelName: {
|
|
107
|
+
$in: allowedModelNames
|
|
108
|
+
}
|
|
109
|
+
}, {
|
|
110
|
+
seq: 1
|
|
111
|
+
}).sort({
|
|
112
|
+
seq: 1
|
|
113
|
+
}).lean();
|
|
90
114
|
earliestSeq = earliest?.seq ? Number(earliest.seq) : void 0;
|
|
91
115
|
}
|
|
92
116
|
const needsFullResync = typeof earliestSeq === "number" && sinceSeq < earliestSeq - 1;
|
|
93
|
-
const selector = {
|
|
94
|
-
|
|
117
|
+
const selector = {
|
|
118
|
+
seq: {
|
|
119
|
+
$gt: sinceSeq
|
|
120
|
+
},
|
|
121
|
+
modelName: {
|
|
122
|
+
$in: allowedModelNames
|
|
123
|
+
}
|
|
124
|
+
};
|
|
125
|
+
const changes = await RtsChange.find(selector, {
|
|
126
|
+
_id: 0,
|
|
127
|
+
seq: 1,
|
|
128
|
+
modelName: 1,
|
|
129
|
+
op: 1,
|
|
130
|
+
docId: 1
|
|
131
|
+
}).sort({
|
|
132
|
+
seq: 1
|
|
133
|
+
}).limit(limit).lean();
|
|
95
134
|
return {
|
|
96
135
|
ok: true,
|
|
97
136
|
needsFullResync: needsFullResync || void 0,
|
|
@@ -111,4 +150,4 @@ const handler = (api) => {
|
|
|
111
150
|
export {
|
|
112
151
|
handler as default
|
|
113
152
|
};
|
|
114
|
-
//# sourceMappingURL=handler-
|
|
153
|
+
//# sourceMappingURL=handler-BsauvgjA.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"handler-BNrqh1Kb.js","sources":["../src/rts/api/changes/index.ts","../src/rts/api/changes/handler.ts"],"sourcesContent":["import { z } from \"zod\"\n\n\nexport const Route = \"/api/rb/rts/changes\"\n\nexport const requestSchema = z.object({\n sinceSeq: z.number().int().min(0).default(0),\n limit: z.number().int().min(1).max(5000).default(2000),\n modelNames: z.array(z.string().min(1)).optional(),\n})\n\nexport type RequestPayload = z.infer<typeof requestSchema>\n\nexport const responseSchema = z.object({\n ok: z.boolean(),\n needsFullResync: z.boolean().optional(),\n earliestSeq: z.number().int().min(0).optional(),\n latestSeq: z.number().int().min(0),\n changes: z.array(z.object({\n seq: z.number().int().min(1),\n modelName: z.string().min(1),\n op: z.enum([\"delete\", \"reset_model\"]),\n docId: z.string().optional(),\n })),\n})\n\nexport type ResponsePayload = z.infer<typeof responseSchema>\n\n","import { Api, ApiHandler, Ctx } from \"@rpcbase/api\"\nimport { models, ZRBRtsChangeOp, type IRBRtsChange, type IRBRtsCounter, type LoadModelCtx } from \"@rpcbase/db\"\nimport { buildAbilityFromSession, type AclSubjectType } from \"@rpcbase/db/acl\"\nimport type { Model } from \"mongoose\"\n\nimport * as Changes from \"./index\"\n\n\ntype SessionUser = {\n id?: string\n currentTenantId?: string\n signedInTenants?: string[]\n}\n\ntype RtsCounterDoc = IRBRtsCounter\ntype RtsChangeDoc = IRBRtsChange\n\nconst getTenantId = (ctx: Ctx<SessionUser>): string | null => {\n const raw = ctx.req.query?.[\"rb-tenant-id\"]\n const queryTenantId = Array.isArray(raw) ? raw[0] : raw\n if (typeof queryTenantId === \"string\" && queryTenantId.trim()) return queryTenantId.trim()\n\n const sessionTenantId = ctx.req.session?.user?.currentTenantId\n if (typeof sessionTenantId === \"string\" && sessionTenantId.trim()) return sessionTenantId.trim()\n\n return null\n}\n\nconst ensureAuthorized = (ctx: Ctx<SessionUser>, tenantId: string): string | null => {\n const userId = ctx.req.session?.user?.id\n if (!userId) return null\n\n const signedInTenants = ctx.req.session?.user?.signedInTenants\n const currentTenantId = ctx.req.session?.user?.currentTenantId\n\n const hasTenantAccessFromList = Array.isArray(signedInTenants) && signedInTenants.includes(tenantId)\n\n const normalizedCurrentTenantId = typeof currentTenantId === \"string\" ? currentTenantId.trim() : \"\"\n const hasTenantAccessFromCurrent = Boolean(normalizedCurrentTenantId) && normalizedCurrentTenantId === tenantId\n\n if (!hasTenantAccessFromList && !hasTenantAccessFromCurrent) return null\n return userId\n}\n\nconst getModelCtx = (_ctx: Ctx<SessionUser>, tenantId: string): LoadModelCtx => ({\n req: {\n session: {\n user: {\n currentTenantId: tenantId,\n },\n },\n },\n})\n\nconst isRtsChangeRecord = (value: unknown): value is RtsChangeDoc => {\n if (!value || typeof value !== \"object\") return false\n const obj = value as Partial<RtsChangeDoc>\n const isOp = ZRBRtsChangeOp.safeParse(obj.op).success\n return typeof obj.seq === \"number\" && typeof obj.modelName === \"string\" && isOp\n}\n\nconst changesHandler: ApiHandler<Changes.RequestPayload, Changes.ResponsePayload, SessionUser> = async(\n payload,\n ctx,\n): Promise<Changes.ResponsePayload> => {\n const parsed = Changes.requestSchema.safeParse(payload ?? {})\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const userId = ensureAuthorized(ctx, tenantId)\n if (!userId) {\n ctx.res.status(401)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const ability = buildAbilityFromSession({ tenantId, session: ctx.req.session })\n\n const modelCtx = getModelCtx(ctx, tenantId)\n\n const [RtsChange, RtsCounter] = await Promise.all([\n models.get(\"RBRtsChange\", modelCtx) as Promise<Model<RtsChangeDoc>>,\n models.get(\"RBRtsCounter\", modelCtx) as Promise<Model<RtsCounterDoc>>,\n ])\n\n const counter = await RtsCounter.findOne({ _id: \"rts\" }, { seq: 1 }).lean()\n const latestSeq = Number(counter?.seq ?? 0) || 0\n\n const { sinceSeq, limit, modelNames } = parsed.data\n\n const requestedModelNames = Array.isArray(modelNames) && modelNames.length\n ? modelNames.map((m) => String(m)).filter(Boolean)\n : null\n\n const allowedModelNames = requestedModelNames\n ? requestedModelNames.filter((m) => ability.can(\"read\", m as AclSubjectType))\n : Array.from(\n new Set(\n (await RtsChange.distinct(\"modelName\"))\n .map((m) => String(m))\n .filter(Boolean)\n .filter((m) => ability.can(\"read\", m as AclSubjectType)),\n ),\n )\n\n let earliestSeq: number | undefined\n if (allowedModelNames.length) {\n const earliest = await RtsChange.findOne({ modelName: { $in: allowedModelNames } }, { seq: 1 }).sort({ seq: 1 }).lean()\n earliestSeq = earliest?.seq ? Number(earliest.seq) : undefined\n }\n\n const needsFullResync = typeof earliestSeq === \"number\" && sinceSeq < earliestSeq - 1\n\n const selector: Record<string, unknown> = { seq: { $gt: sinceSeq }, modelName: { $in: allowedModelNames } }\n\n const changes = await RtsChange\n .find(selector, { _id: 0, seq: 1, modelName: 1, op: 1, docId: 1 })\n .sort({ seq: 1 })\n .limit(limit)\n .lean()\n\n return {\n ok: true,\n needsFullResync: needsFullResync || undefined,\n earliestSeq,\n latestSeq,\n changes: Array.isArray(changes)\n ? changes\n .filter(isRtsChangeRecord)\n .filter((c) => ability.can(\"read\", c.modelName as AclSubjectType))\n .map((c) => ({\n seq: Number(c.seq),\n modelName: String(c.modelName),\n op: c.op,\n docId: c.docId ? String(c.docId) : undefined,\n }))\n : [],\n }\n}\n\nexport default (api: Api<SessionUser>) => {\n api.post(Changes.Route, changesHandler)\n}\n"],"names":["z.object","z.number","z.array","z.string","z.boolean","z.enum","Changes.requestSchema","Changes.Route"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"handler-BsauvgjA.js","sources":["../src/rts/api/changes/index.ts","../src/rts/api/changes/handler.ts"],"sourcesContent":["import { z } from \"zod\"\n\n\nexport const Route = \"/api/rb/rts/changes\"\n\nexport const requestSchema = z.object({\n sinceSeq: z.number().int().min(0).default(0),\n limit: z.number().int().min(1).max(5000).default(2000),\n modelNames: z.array(z.string().min(1)).optional(),\n})\n\nexport type RequestPayload = z.infer<typeof requestSchema>\n\nexport const responseSchema = z.object({\n ok: z.boolean(),\n needsFullResync: z.boolean().optional(),\n earliestSeq: z.number().int().min(0).optional(),\n latestSeq: z.number().int().min(0),\n changes: z.array(z.object({\n seq: z.number().int().min(1),\n modelName: z.string().min(1),\n op: z.enum([\"delete\", \"reset_model\"]),\n docId: z.string().optional(),\n })),\n})\n\nexport type ResponsePayload = z.infer<typeof responseSchema>\n\n","import { Api, ApiHandler, Ctx } from \"@rpcbase/api\"\nimport { models, ZRBRtsChangeOp, type IRBRtsChange, type IRBRtsCounter, type LoadModelCtx } from \"@rpcbase/db\"\nimport { buildAbilityFromSession, type AclSubjectType } from \"@rpcbase/db/acl\"\nimport type { Model } from \"mongoose\"\n\nimport * as Changes from \"./index\"\n\n\ntype SessionUser = {\n id?: string\n currentTenantId?: string\n signedInTenants?: string[]\n}\n\ntype RtsCounterDoc = IRBRtsCounter\ntype RtsChangeDoc = IRBRtsChange\n\nconst getTenantId = (ctx: Ctx<SessionUser>): string | null => {\n const raw = ctx.req.query?.[\"rb-tenant-id\"]\n const queryTenantId = Array.isArray(raw) ? raw[0] : raw\n if (typeof queryTenantId === \"string\" && queryTenantId.trim()) return queryTenantId.trim()\n\n const sessionTenantId = ctx.req.session?.user?.currentTenantId\n if (typeof sessionTenantId === \"string\" && sessionTenantId.trim()) return sessionTenantId.trim()\n\n return null\n}\n\nconst ensureAuthorized = (ctx: Ctx<SessionUser>, tenantId: string): string | null => {\n const userId = ctx.req.session?.user?.id\n if (!userId) return null\n\n const signedInTenants = ctx.req.session?.user?.signedInTenants\n const currentTenantId = ctx.req.session?.user?.currentTenantId\n\n const hasTenantAccessFromList = Array.isArray(signedInTenants) && signedInTenants.includes(tenantId)\n\n const normalizedCurrentTenantId = typeof currentTenantId === \"string\" ? currentTenantId.trim() : \"\"\n const hasTenantAccessFromCurrent = Boolean(normalizedCurrentTenantId) && normalizedCurrentTenantId === tenantId\n\n if (!hasTenantAccessFromList && !hasTenantAccessFromCurrent) return null\n return userId\n}\n\nconst getModelCtx = (_ctx: Ctx<SessionUser>, tenantId: string): LoadModelCtx => ({\n req: {\n session: {\n user: {\n currentTenantId: tenantId,\n },\n },\n },\n})\n\nconst isRtsChangeRecord = (value: unknown): value is RtsChangeDoc => {\n if (!value || typeof value !== \"object\") return false\n const obj = value as Partial<RtsChangeDoc>\n const isOp = ZRBRtsChangeOp.safeParse(obj.op).success\n return typeof obj.seq === \"number\" && typeof obj.modelName === \"string\" && isOp\n}\n\nconst changesHandler: ApiHandler<Changes.RequestPayload, Changes.ResponsePayload, SessionUser> = async(\n payload,\n ctx,\n): Promise<Changes.ResponsePayload> => {\n const parsed = Changes.requestSchema.safeParse(payload ?? {})\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const userId = ensureAuthorized(ctx, tenantId)\n if (!userId) {\n ctx.res.status(401)\n return { ok: false, latestSeq: 0, changes: [] }\n }\n\n const ability = buildAbilityFromSession({ tenantId, session: ctx.req.session })\n\n const modelCtx = getModelCtx(ctx, tenantId)\n\n const [RtsChange, RtsCounter] = await Promise.all([\n models.get(\"RBRtsChange\", modelCtx) as Promise<Model<RtsChangeDoc>>,\n models.get(\"RBRtsCounter\", modelCtx) as Promise<Model<RtsCounterDoc>>,\n ])\n\n const counter = await RtsCounter.findOne({ _id: \"rts\" }, { seq: 1 }).lean()\n const latestSeq = Number(counter?.seq ?? 0) || 0\n\n const { sinceSeq, limit, modelNames } = parsed.data\n\n const requestedModelNames = Array.isArray(modelNames) && modelNames.length\n ? modelNames.map((m) => String(m)).filter(Boolean)\n : null\n\n const allowedModelNames = requestedModelNames\n ? requestedModelNames.filter((m) => ability.can(\"read\", m as AclSubjectType))\n : Array.from(\n new Set(\n (await RtsChange.distinct(\"modelName\"))\n .map((m) => String(m))\n .filter(Boolean)\n .filter((m) => ability.can(\"read\", m as AclSubjectType)),\n ),\n )\n\n let earliestSeq: number | undefined\n if (allowedModelNames.length) {\n const earliest = await RtsChange.findOne({ modelName: { $in: allowedModelNames } }, { seq: 1 }).sort({ seq: 1 }).lean()\n earliestSeq = earliest?.seq ? Number(earliest.seq) : undefined\n }\n\n const needsFullResync = typeof earliestSeq === \"number\" && sinceSeq < earliestSeq - 1\n\n const selector: Record<string, unknown> = { seq: { $gt: sinceSeq }, modelName: { $in: allowedModelNames } }\n\n const changes = await RtsChange\n .find(selector, { _id: 0, seq: 1, modelName: 1, op: 1, docId: 1 })\n .sort({ seq: 1 })\n .limit(limit)\n .lean()\n\n return {\n ok: true,\n needsFullResync: needsFullResync || undefined,\n earliestSeq,\n latestSeq,\n changes: Array.isArray(changes)\n ? changes\n .filter(isRtsChangeRecord)\n .filter((c) => ability.can(\"read\", c.modelName as AclSubjectType))\n .map((c) => ({\n seq: Number(c.seq),\n modelName: String(c.modelName),\n op: c.op,\n docId: c.docId ? String(c.docId) : undefined,\n }))\n : [],\n }\n}\n\nexport default (api: Api<SessionUser>) => {\n api.post(Changes.Route, changesHandler)\n}\n"],"names":["Route","requestSchema","z","sinceSeq","number","int","min","default","limit","max","modelNames","string","optional","ok","boolean","needsFullResync","earliestSeq","latestSeq","changes","seq","modelName","op","docId","getTenantId","ctx","raw","req","query","queryTenantId","Array","isArray","trim","sessionTenantId","session","user","currentTenantId","ensureAuthorized","tenantId","userId","id","signedInTenants","hasTenantAccessFromList","includes","normalizedCurrentTenantId","hasTenantAccessFromCurrent","Boolean","getModelCtx","_ctx","isRtsChangeRecord","value","obj","isOp","ZRBRtsChangeOp","safeParse","success","changesHandler","payload","parsed","Changes","res","status","ability","buildAbilityFromSession","modelCtx","RtsChange","RtsCounter","Promise","all","models","get","counter","findOne","_id","lean","Number","data","requestedModelNames","length","map","m","String","filter","allowedModelNames","can","from","Set","distinct","earliest","$in","sort","undefined","selector","$gt","find","c","api","post"],"mappings":"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"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { getTenantFilesystemDb } from "@rpcbase/db";
|
|
2
2
|
import { ObjectId, GridFSBucket } from "mongodb";
|
|
3
|
-
import { g as getTenantId, d as getBucketName, f as getUserId, m as getUploadKeyHash } from "./shared-
|
|
3
|
+
import { g as getTenantId, d as getBucketName, f as getUserId, m as getUploadKeyHash } from "./shared-Dy9x-P7F.js";
|
|
4
4
|
const resolveHeaderString$1 = (value) => {
|
|
5
5
|
if (typeof value !== "string") return null;
|
|
6
6
|
const normalized = value.trim();
|
|
@@ -10,7 +10,10 @@ const deleteFile = async (_payload, ctx) => {
|
|
|
10
10
|
const tenantId = getTenantId(ctx);
|
|
11
11
|
if (!tenantId) {
|
|
12
12
|
ctx.res.status(400);
|
|
13
|
-
return {
|
|
13
|
+
return {
|
|
14
|
+
ok: false,
|
|
15
|
+
error: "tenant_missing"
|
|
16
|
+
};
|
|
14
17
|
}
|
|
15
18
|
const fileIdRaw = String(ctx.req.params?.fileId ?? "").trim();
|
|
16
19
|
let fileObjectId;
|
|
@@ -18,26 +21,41 @@ const deleteFile = async (_payload, ctx) => {
|
|
|
18
21
|
fileObjectId = new ObjectId(fileIdRaw);
|
|
19
22
|
} catch {
|
|
20
23
|
ctx.res.status(400);
|
|
21
|
-
return {
|
|
24
|
+
return {
|
|
25
|
+
ok: false,
|
|
26
|
+
error: "invalid_file_id"
|
|
27
|
+
};
|
|
22
28
|
}
|
|
23
29
|
const fsDb = await getTenantFilesystemDb(tenantId);
|
|
24
30
|
const nativeDb = fsDb.db;
|
|
25
31
|
if (!nativeDb) {
|
|
26
32
|
ctx.res.status(500);
|
|
27
|
-
return {
|
|
33
|
+
return {
|
|
34
|
+
ok: false,
|
|
35
|
+
error: "filesystem_db_unavailable"
|
|
36
|
+
};
|
|
28
37
|
}
|
|
29
38
|
const bucketName = getBucketName();
|
|
30
|
-
const bucket = new GridFSBucket(nativeDb, {
|
|
39
|
+
const bucket = new GridFSBucket(nativeDb, {
|
|
40
|
+
bucketName
|
|
41
|
+
});
|
|
31
42
|
const userId = getUserId(ctx);
|
|
32
43
|
const uploadKeyHash = getUploadKeyHash(ctx);
|
|
33
44
|
if (!userId && !uploadKeyHash) {
|
|
34
45
|
ctx.res.status(401);
|
|
35
|
-
return {
|
|
46
|
+
return {
|
|
47
|
+
ok: false,
|
|
48
|
+
error: "unauthorized"
|
|
49
|
+
};
|
|
36
50
|
}
|
|
37
|
-
const [file] = await bucket.find({
|
|
51
|
+
const [file] = await bucket.find({
|
|
52
|
+
_id: fileObjectId
|
|
53
|
+
}).limit(1).toArray();
|
|
38
54
|
if (!file) {
|
|
39
55
|
ctx.res.status(204);
|
|
40
|
-
return {
|
|
56
|
+
return {
|
|
57
|
+
ok: true
|
|
58
|
+
};
|
|
41
59
|
}
|
|
42
60
|
const metadataUserId = resolveHeaderString$1(file?.metadata?.userId);
|
|
43
61
|
const ownerKeyHash = resolveHeaderString$1(file?.metadata?.ownerKeyHash);
|
|
@@ -45,7 +63,10 @@ const deleteFile = async (_payload, ctx) => {
|
|
|
45
63
|
const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash);
|
|
46
64
|
if (!authorizedByUser && !authorizedByKey) {
|
|
47
65
|
ctx.res.status(401);
|
|
48
|
-
return {
|
|
66
|
+
return {
|
|
67
|
+
ok: false,
|
|
68
|
+
error: "unauthorized"
|
|
69
|
+
};
|
|
49
70
|
}
|
|
50
71
|
try {
|
|
51
72
|
await bucket.delete(fileObjectId);
|
|
@@ -53,11 +74,16 @@ const deleteFile = async (_payload, ctx) => {
|
|
|
53
74
|
const message = error instanceof Error ? error.message : String(error);
|
|
54
75
|
if (!message.includes("FileNotFound")) {
|
|
55
76
|
ctx.res.status(500);
|
|
56
|
-
return {
|
|
77
|
+
return {
|
|
78
|
+
ok: false,
|
|
79
|
+
error: "delete_failed"
|
|
80
|
+
};
|
|
57
81
|
}
|
|
58
82
|
}
|
|
59
83
|
ctx.res.status(204);
|
|
60
|
-
return {
|
|
84
|
+
return {
|
|
85
|
+
ok: true
|
|
86
|
+
};
|
|
61
87
|
};
|
|
62
88
|
const resolveHeaderString = (value) => {
|
|
63
89
|
if (typeof value !== "string") return null;
|
|
@@ -95,8 +121,12 @@ const getFile = async (_payload, ctx) => {
|
|
|
95
121
|
return {};
|
|
96
122
|
}
|
|
97
123
|
const bucketName = getBucketName();
|
|
98
|
-
const bucket = new GridFSBucket(nativeDb, {
|
|
99
|
-
|
|
124
|
+
const bucket = new GridFSBucket(nativeDb, {
|
|
125
|
+
bucketName
|
|
126
|
+
});
|
|
127
|
+
const [file] = await bucket.find({
|
|
128
|
+
_id: fileObjectId
|
|
129
|
+
}).limit(1).toArray();
|
|
100
130
|
if (!file) {
|
|
101
131
|
ctx.res.status(404).end();
|
|
102
132
|
return {};
|
|
@@ -143,10 +173,7 @@ const getFile = async (_payload, ctx) => {
|
|
|
143
173
|
ctx.res.setHeader("ETag", etag);
|
|
144
174
|
ctx.res.setHeader("X-Content-Type-Options", "nosniff");
|
|
145
175
|
if (mimeType === "image/svg+xml") {
|
|
146
|
-
ctx.res.setHeader(
|
|
147
|
-
"Content-Security-Policy",
|
|
148
|
-
"default-src 'none'; style-src 'unsafe-inline'; sandbox; base-uri 'none'; form-action 'none'"
|
|
149
|
-
);
|
|
176
|
+
ctx.res.setHeader("Content-Security-Policy", "default-src 'none'; style-src 'unsafe-inline'; sandbox; base-uri 'none'; form-action 'none'");
|
|
150
177
|
}
|
|
151
178
|
ctx.res.flushHeaders();
|
|
152
179
|
if (ctx.req.method === "HEAD") {
|
|
@@ -173,4 +200,4 @@ const handler = (api) => {
|
|
|
173
200
|
export {
|
|
174
201
|
handler as default
|
|
175
202
|
};
|
|
176
|
-
//# sourceMappingURL=handler-
|
|
203
|
+
//# sourceMappingURL=handler-DY5UEwlw.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"handler-Cohj3cz3.js","sources":["../src/uploads/api/files/handlers/deleteFile.ts","../src/uploads/api/files/handlers/getFile.ts","../src/uploads/api/files/index.ts","../src/uploads/api/files/handler.ts"],"sourcesContent":["import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\ntype DeleteResponsePayload = {\n ok: boolean\n error?: string\n}\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nexport const deleteFile: ApiHandler<Record<string, never>, DeleteResponsePayload, SessionUser> = async (\n _payload,\n ctx,\n): Promise<DeleteResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_file_id\" }\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500)\n return { ok: false, error: \"filesystem_db_unavailable\" }\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n if (!userId && !uploadKeyHash) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(204)\n return { ok: true }\n }\n\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n try {\n await bucket.delete(fileObjectId)\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n if (!message.includes(\"FileNotFound\")) {\n ctx.res.status(500)\n return { ok: false, error: \"delete_failed\" }\n }\n }\n\n ctx.res.status(204)\n return { ok: true }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nconst resolveHeaderBoolean = (value: unknown): boolean | null => {\n if (typeof value === \"boolean\") return value\n if (typeof value !== \"string\") return null\n const normalized = value.trim().toLowerCase()\n if (!normalized) return null\n if (normalized === \"true\") return true\n if (normalized === \"false\") return false\n return null\n}\n\nconst escapeHeaderFilename = (filename: string): string => filename.replace(/[\\\\\"]/g, \"_\")\n\nexport const getFile: ApiHandler<Record<string, never>, Record<string, never>, SessionUser> = async (\n _payload,\n ctx,\n): Promise<Record<string, never>> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400).end()\n return {}\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400).end()\n return {}\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500).end()\n return {}\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(404).end()\n return {}\n }\n\n const isPublic = resolveHeaderBoolean((file as any)?.metadata?.isPublic) ?? false\n if (!isPublic) {\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401).end()\n return {}\n }\n }\n\n const mimeTypeFromMetadata = resolveHeaderString((file as any)?.metadata?.mimeType)\n const mimeType = mimeTypeFromMetadata ?? \"application/octet-stream\"\n const filenameFromDb = resolveHeaderString((file as any)?.filename)\n const filename = filenameFromDb ?? fileIdRaw\n const filenameSafe = escapeHeaderFilename(filename)\n\n const cacheControl = \"private, max-age=0, must-revalidate\"\n const md5 = resolveHeaderString((file as any)?.md5)\n const uploadDate = (file as any)?.uploadDate instanceof Date ? (file as any).uploadDate : null\n const etagValue = md5 ?? `${fileObjectId.toHexString()}-${String((file as any)?.length ?? 0)}-${String(uploadDate?.getTime() ?? 0)}`\n const etag = md5 ? `\"${etagValue}\"` : `W/\"${etagValue}\"`\n const ifNoneMatch = resolveHeaderString((ctx.req.headers as any)?.[\"if-none-match\"])\n if (ifNoneMatch) {\n const candidates = ifNoneMatch.split(\",\").map((value) => value.trim()).filter(Boolean)\n if (candidates.includes(\"*\") || candidates.includes(etag)) {\n ctx.res.status(304)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.end()\n return {}\n }\n }\n\n ctx.res.status(200)\n ctx.res.setHeader(\"Content-Type\", mimeType)\n ctx.res.setHeader(\"Content-Length\", String((file as any)?.length ?? 0))\n ctx.res.setHeader(\"Content-Disposition\", `inline; filename=\"${filenameSafe}\"`)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.setHeader(\"X-Content-Type-Options\", \"nosniff\")\n if (mimeType === \"image/svg+xml\") {\n ctx.res.setHeader(\n \"Content-Security-Policy\",\n \"default-src 'none'; style-src 'unsafe-inline'; sandbox; base-uri 'none'; form-action 'none'\",\n )\n }\n ctx.res.flushHeaders()\n\n if (ctx.req.method === \"HEAD\") {\n ctx.res.end()\n return {}\n }\n\n const stream = bucket.openDownloadStream(fileObjectId)\n\n stream.on(\"error\", () => {\n try {\n ctx.res.destroy()\n } catch {\n //\n }\n })\n\n stream.pipe(ctx.res)\n return {}\n}\n","export const Route = \"/api/rb/files/:fileId\"\n\nexport const GetRoute = Route\nexport const DeleteRoute = Route\n","import { Api } from \"@rpcbase/api\"\n\nimport { deleteFile } from \"./handlers/deleteFile\"\nimport { getFile } from \"./handlers/getFile\"\n\nimport * as Files from \"./index\"\n\n\nexport default (api: Api) => {\n api.get(Files.GetRoute, getFile)\n api.delete(Files.DeleteRoute, deleteFile)\n}\n"],"names":["resolveHeaderString","Files.GetRoute","Files.DeleteRoute"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"handler-DY5UEwlw.js","sources":["../src/uploads/api/files/handlers/deleteFile.ts","../src/uploads/api/files/handlers/getFile.ts","../src/uploads/api/files/index.ts","../src/uploads/api/files/handler.ts"],"sourcesContent":["import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\ntype DeleteResponsePayload = {\n ok: boolean\n error?: string\n}\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nexport const deleteFile: ApiHandler<Record<string, never>, DeleteResponsePayload, SessionUser> = async (\n _payload,\n ctx,\n): Promise<DeleteResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_file_id\" }\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500)\n return { ok: false, error: \"filesystem_db_unavailable\" }\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n if (!userId && !uploadKeyHash) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(204)\n return { ok: true }\n }\n\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n try {\n await bucket.delete(fileObjectId)\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n if (!message.includes(\"FileNotFound\")) {\n ctx.res.status(500)\n return { ok: false, error: \"delete_failed\" }\n }\n }\n\n ctx.res.status(204)\n return { ok: true }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nconst resolveHeaderBoolean = (value: unknown): boolean | null => {\n if (typeof value === \"boolean\") return value\n if (typeof value !== \"string\") return null\n const normalized = value.trim().toLowerCase()\n if (!normalized) return null\n if (normalized === \"true\") return true\n if (normalized === \"false\") return false\n return null\n}\n\nconst escapeHeaderFilename = (filename: string): string => filename.replace(/[\\\\\"]/g, \"_\")\n\nexport const getFile: ApiHandler<Record<string, never>, Record<string, never>, SessionUser> = async (\n _payload,\n ctx,\n): Promise<Record<string, never>> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400).end()\n return {}\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400).end()\n return {}\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500).end()\n return {}\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(404).end()\n return {}\n }\n\n const isPublic = resolveHeaderBoolean((file as any)?.metadata?.isPublic) ?? false\n if (!isPublic) {\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401).end()\n return {}\n }\n }\n\n const mimeTypeFromMetadata = resolveHeaderString((file as any)?.metadata?.mimeType)\n const mimeType = mimeTypeFromMetadata ?? \"application/octet-stream\"\n const filenameFromDb = resolveHeaderString((file as any)?.filename)\n const filename = filenameFromDb ?? fileIdRaw\n const filenameSafe = escapeHeaderFilename(filename)\n\n const cacheControl = \"private, max-age=0, must-revalidate\"\n const md5 = resolveHeaderString((file as any)?.md5)\n const uploadDate = (file as any)?.uploadDate instanceof Date ? (file as any).uploadDate : null\n const etagValue = md5 ?? `${fileObjectId.toHexString()}-${String((file as any)?.length ?? 0)}-${String(uploadDate?.getTime() ?? 0)}`\n const etag = md5 ? `\"${etagValue}\"` : `W/\"${etagValue}\"`\n const ifNoneMatch = resolveHeaderString((ctx.req.headers as any)?.[\"if-none-match\"])\n if (ifNoneMatch) {\n const candidates = ifNoneMatch.split(\",\").map((value) => value.trim()).filter(Boolean)\n if (candidates.includes(\"*\") || candidates.includes(etag)) {\n ctx.res.status(304)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.end()\n return {}\n }\n }\n\n ctx.res.status(200)\n ctx.res.setHeader(\"Content-Type\", mimeType)\n ctx.res.setHeader(\"Content-Length\", String((file as any)?.length ?? 0))\n ctx.res.setHeader(\"Content-Disposition\", `inline; filename=\"${filenameSafe}\"`)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.setHeader(\"X-Content-Type-Options\", \"nosniff\")\n if (mimeType === \"image/svg+xml\") {\n ctx.res.setHeader(\n \"Content-Security-Policy\",\n \"default-src 'none'; style-src 'unsafe-inline'; sandbox; base-uri 'none'; form-action 'none'\",\n )\n }\n ctx.res.flushHeaders()\n\n if (ctx.req.method === \"HEAD\") {\n ctx.res.end()\n return {}\n }\n\n const stream = bucket.openDownloadStream(fileObjectId)\n\n stream.on(\"error\", () => {\n try {\n ctx.res.destroy()\n } catch {\n //\n }\n })\n\n stream.pipe(ctx.res)\n return {}\n}\n","export const Route = \"/api/rb/files/:fileId\"\n\nexport const GetRoute = Route\nexport const DeleteRoute = Route\n","import { Api } from \"@rpcbase/api\"\n\nimport { deleteFile } from \"./handlers/deleteFile\"\nimport { getFile } from \"./handlers/getFile\"\n\nimport * as Files from \"./index\"\n\n\nexport default (api: Api) => {\n api.get(Files.GetRoute, getFile)\n api.delete(Files.DeleteRoute, deleteFile)\n}\n"],"names":["resolveHeaderString","value","normalized","trim","deleteFile","_payload","ctx","tenantId","getTenantId","res","status","ok","error","fileIdRaw","String","req","params","fileId","fileObjectId","ObjectId","fsDb","getTenantFilesystemDb","nativeDb","db","bucketName","getBucketName","bucket","GridFSBucket","userId","getUserId","uploadKeyHash","getUploadKeyHash","file","find","_id","limit","toArray","metadataUserId","metadata","ownerKeyHash","authorizedByUser","Boolean","authorizedByKey","delete","message","Error","includes","resolveHeaderBoolean","toLowerCase","escapeHeaderFilename","filename","replace","getFile","end","isPublic","mimeTypeFromMetadata","mimeType","filenameFromDb","filenameSafe","cacheControl","md5","uploadDate","Date","etagValue","toHexString","length","getTime","etag","ifNoneMatch","headers","candidates","split","map","filter","setHeader","flushHeaders","method","stream","openDownloadStream","on","destroy","pipe","Route","GetRoute","DeleteRoute","api","get","Files"],"mappings":"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"}
|