@atproto/ozone 0.2.4 → 0.2.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (78) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/dist/api/label/queryLabels.d.ts.map +1 -1
  3. package/dist/api/label/queryLabels.js +13 -21
  4. package/dist/api/label/queryLabels.js.map +1 -1
  5. package/dist/assignment/index.d.ts.map +1 -1
  6. package/dist/assignment/index.js +9 -12
  7. package/dist/assignment/index.js.map +1 -1
  8. package/dist/daemon/event-pusher.d.ts +7 -1
  9. package/dist/daemon/event-pusher.d.ts.map +1 -1
  10. package/dist/db/index.d.ts +4 -5
  11. package/dist/db/index.d.ts.map +1 -1
  12. package/dist/db/index.js +2 -1
  13. package/dist/db/index.js.map +1 -1
  14. package/dist/db/migrations/20241220T144630860Z-stats-materialized-views.d.ts +1 -2
  15. package/dist/db/migrations/20241220T144630860Z-stats-materialized-views.d.ts.map +1 -1
  16. package/dist/db/migrations/20241220T144630860Z-stats-materialized-views.js.map +1 -1
  17. package/dist/db/migrations/20250718T150931000Z-update-appeal-reason-stats.d.ts +1 -2
  18. package/dist/db/migrations/20250718T150931000Z-update-appeal-reason-stats.d.ts.map +1 -1
  19. package/dist/db/migrations/20250718T150931000Z-update-appeal-reason-stats.js.map +1 -1
  20. package/dist/db/migrations/provider.d.ts +2 -1
  21. package/dist/db/migrations/provider.d.ts.map +1 -1
  22. package/dist/db/migrations/provider.js.map +1 -1
  23. package/dist/db/pagination.d.ts +4 -3
  24. package/dist/db/pagination.d.ts.map +1 -1
  25. package/dist/db/pagination.js +4 -4
  26. package/dist/db/pagination.js.map +1 -1
  27. package/dist/db/types.d.ts +1 -1
  28. package/dist/db/types.d.ts.map +1 -1
  29. package/dist/db/types.js.map +1 -1
  30. package/dist/mod-service/index.d.ts.map +1 -1
  31. package/dist/mod-service/index.js +28 -52
  32. package/dist/mod-service/index.js.map +1 -1
  33. package/dist/mod-service/report.d.ts.map +1 -1
  34. package/dist/mod-service/report.js.map +1 -1
  35. package/dist/mod-service/status.d.ts +23 -128
  36. package/dist/mod-service/status.d.ts.map +1 -1
  37. package/dist/mod-service/views.js +7 -11
  38. package/dist/mod-service/views.js.map +1 -1
  39. package/dist/queue/service.js +1 -3
  40. package/dist/queue/service.js.map +1 -1
  41. package/dist/report/activity.d.ts +12 -1
  42. package/dist/report/activity.d.ts.map +1 -1
  43. package/dist/report/stats.d.ts.map +1 -1
  44. package/dist/report/stats.js.map +1 -1
  45. package/dist/scheduled-action/service.d.ts.map +1 -1
  46. package/dist/scheduled-action/service.js +16 -20
  47. package/dist/scheduled-action/service.js.map +1 -1
  48. package/dist/set/service.d.ts +10 -1
  49. package/dist/set/service.d.ts.map +1 -1
  50. package/dist/set/service.js +5 -2
  51. package/dist/set/service.js.map +1 -1
  52. package/dist/team/index.d.ts.map +1 -1
  53. package/dist/team/index.js +5 -4
  54. package/dist/team/index.js.map +1 -1
  55. package/dist/verification/issuer.d.ts +13 -3
  56. package/dist/verification/issuer.d.ts.map +1 -1
  57. package/dist/verification/service.d.ts +13 -1
  58. package/dist/verification/service.d.ts.map +1 -1
  59. package/dist/verification/service.js +1 -1
  60. package/dist/verification/service.js.map +1 -1
  61. package/package.json +7 -7
  62. package/src/api/label/queryLabels.ts +11 -14
  63. package/src/assignment/index.ts +15 -18
  64. package/src/db/index.ts +1 -1
  65. package/src/db/migrations/20241220T144630860Z-stats-materialized-views.ts +1 -2
  66. package/src/db/migrations/20250718T150931000Z-update-appeal-reason-stats.ts +1 -2
  67. package/src/db/migrations/provider.ts +2 -1
  68. package/src/db/pagination.ts +18 -18
  69. package/src/db/types.ts +3 -1
  70. package/src/mod-service/index.ts +78 -71
  71. package/src/mod-service/report.ts +5 -3
  72. package/src/mod-service/views.ts +16 -16
  73. package/src/queue/service.ts +5 -5
  74. package/src/report/stats.ts +5 -3
  75. package/src/scheduled-action/service.ts +22 -20
  76. package/src/set/service.ts +17 -14
  77. package/src/team/index.ts +6 -5
  78. package/src/verification/service.ts +2 -2
package/CHANGELOG.md CHANGED
@@ -1,5 +1,11 @@
1
1
  # @atproto/ozone
2
2
 
3
+ ## 0.2.5
4
+
5
+ ### Patch Changes
6
+
7
+ - [#5086](https://github.com/bluesky-social/atproto/pull/5086) [`20665c1`](https://github.com/bluesky-social/atproto/commit/20665c18322effe648f73d70fc1a8dcc7e312992) Thanks [@devinivy](https://github.com/devinivy)! - Upgrade kysely from 0.22 to 0.29
8
+
3
9
  ## 0.2.4
4
10
 
5
11
  ### Patch Changes
@@ -1 +1 @@
1
- {"version":3,"file":"queryLabels.d.ts","sourceRoot":"","sources":["../../../src/api/label/queryLabels.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAA;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,wBAAwB,CAAA;AAE/C,MAAM,CAAC,OAAO,WAAW,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,UAAU,QAsDvD"}
1
+ {"version":3,"file":"queryLabels.d.ts","sourceRoot":"","sources":["../../../src/api/label/queryLabels.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAA;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,wBAAwB,CAAA;AAE/C,MAAM,CAAC,OAAO,WAAW,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,UAAU,QAoDvD"}
@@ -1,4 +1,3 @@
1
- import { sql } from 'kysely';
2
1
  import { InvalidRequestError } from '@atproto/xrpc-server';
3
2
  export default function (server, ctx) {
4
3
  server.com.atproto.label.queryLabels(async ({ params }) => {
@@ -6,27 +5,20 @@ export default function (server, ctx) {
6
5
  let builder = ctx.db.db.selectFrom('label').selectAll().limit(limit);
7
6
  // if includes '*', then we don't need a where clause
8
7
  if (!uriPatterns.includes('*')) {
9
- builder = builder.where((qb) => {
10
- // starter where clause that is always false so that we can chain `orWhere`s
11
- qb = qb.where(sql `1 = 0`);
12
- for (const pattern of uriPatterns) {
13
- // if no '*', then we're looking for an exact match
14
- if (!pattern.includes('*')) {
15
- qb = qb.orWhere('uri', '=', pattern);
16
- }
17
- else {
18
- if (pattern.indexOf('*') < pattern.length - 1) {
19
- throw new InvalidRequestError(`invalid pattern: ${pattern}`);
20
- }
21
- const searchPattern = pattern
22
- .slice(0, -1)
23
- .replaceAll('%', '') // sanitize search pattern
24
- .replaceAll('_', '\\_'); // escape any underscores
25
- qb = qb.orWhere('uri', 'like', `${searchPattern}%`);
26
- }
8
+ builder = builder.where((eb) => eb.or(uriPatterns.map((pattern) => {
9
+ // if no '*', then we're looking for an exact match
10
+ if (!pattern.includes('*')) {
11
+ return eb('uri', '=', pattern);
27
12
  }
28
- return qb;
29
- });
13
+ if (pattern.indexOf('*') < pattern.length - 1) {
14
+ throw new InvalidRequestError(`invalid pattern: ${pattern}`);
15
+ }
16
+ const searchPattern = pattern
17
+ .slice(0, -1)
18
+ .replaceAll('%', '') // sanitize search pattern
19
+ .replaceAll('_', '\\_'); // escape any underscores
20
+ return eb('uri', 'like', `${searchPattern}%`);
21
+ })));
30
22
  }
31
23
  if (sources && sources.length > 0) {
32
24
  builder = builder.where('src', 'in', sources);
@@ -1 +1 @@
1
- {"version":3,"file":"queryLabels.js","sourceRoot":"","sources":["../../../src/api/label/queryLabels.ts"],"names":[],"mappings":"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","sourcesContent":["import { sql } from 'kysely'\nimport { InvalidRequestError } from '@atproto/xrpc-server'\nimport { AppContext } from '../../context.js'\nimport { Server } from '../../lexicon/index.js'\n\nexport default function (server: Server, ctx: AppContext) {\n server.com.atproto.label.queryLabels(async ({ params }) => {\n const { uriPatterns, sources, limit, cursor } = params\n let builder = ctx.db.db.selectFrom('label').selectAll().limit(limit)\n // if includes '*', then we don't need a where clause\n if (!uriPatterns.includes('*')) {\n builder = builder.where((qb) => {\n // starter where clause that is always false so that we can chain `orWhere`s\n qb = qb.where(sql`1 = 0`)\n for (const pattern of uriPatterns) {\n // if no '*', then we're looking for an exact match\n if (!pattern.includes('*')) {\n qb = qb.orWhere('uri', '=', pattern)\n } else {\n if (pattern.indexOf('*') < pattern.length - 1) {\n throw new InvalidRequestError(`invalid pattern: ${pattern}`)\n }\n const searchPattern = pattern\n .slice(0, -1)\n .replaceAll('%', '') // sanitize search pattern\n .replaceAll('_', '\\\\_') // escape any underscores\n qb = qb.orWhere('uri', 'like', `${searchPattern}%`)\n }\n }\n return qb\n })\n }\n if (sources && sources.length > 0) {\n builder = builder.where('src', 'in', sources)\n }\n if (cursor) {\n const cursorId = parseInt(cursor, 10)\n if (isNaN(cursorId)) {\n throw new InvalidRequestError('invalid cursor')\n }\n builder = builder.where('id', '>', cursorId)\n }\n\n const res = await builder.execute()\n\n const modSrvc = ctx.modService(ctx.db)\n const labels = await Promise.all(\n res.map((l) => modSrvc.views.formatLabelAndEnsureSig(l)),\n )\n const resCursor = res.at(-1)?.id.toString(10)\n\n return {\n encoding: 'application/json',\n body: {\n cursor: resCursor,\n labels,\n },\n }\n })\n}\n"]}
1
+ {"version":3,"file":"queryLabels.js","sourceRoot":"","sources":["../../../src/api/label/queryLabels.ts"],"names":[],"mappings":"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","sourcesContent":["import { InvalidRequestError } from '@atproto/xrpc-server'\nimport { AppContext } from '../../context.js'\nimport { Server } from '../../lexicon/index.js'\n\nexport default function (server: Server, ctx: AppContext) {\n server.com.atproto.label.queryLabels(async ({ params }) => {\n const { uriPatterns, sources, limit, cursor } = params\n let builder = ctx.db.db.selectFrom('label').selectAll().limit(limit)\n // if includes '*', then we don't need a where clause\n if (!uriPatterns.includes('*')) {\n builder = builder.where((eb) =>\n eb.or(\n uriPatterns.map((pattern) => {\n // if no '*', then we're looking for an exact match\n if (!pattern.includes('*')) {\n return eb('uri', '=', pattern)\n }\n if (pattern.indexOf('*') < pattern.length - 1) {\n throw new InvalidRequestError(`invalid pattern: ${pattern}`)\n }\n const searchPattern = pattern\n .slice(0, -1)\n .replaceAll('%', '') // sanitize search pattern\n .replaceAll('_', '\\\\_') // escape any underscores\n return eb('uri', 'like', `${searchPattern}%`)\n }),\n ),\n )\n }\n if (sources && sources.length > 0) {\n builder = builder.where('src', 'in', sources)\n }\n if (cursor) {\n const cursorId = parseInt(cursor, 10)\n if (isNaN(cursorId)) {\n throw new InvalidRequestError('invalid cursor')\n }\n builder = builder.where('id', '>', cursorId)\n }\n\n const res = await builder.execute()\n\n const modSrvc = ctx.modService(ctx.db)\n const labels = await Promise.all(\n res.map((l) => modSrvc.views.formatLabelAndEnsureSig(l)),\n )\n const resCursor = res.at(-1)?.id.toString(10)\n\n return {\n encoding: 'application/json',\n body: {\n cursor: resCursor,\n labels,\n },\n }\n })\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/assignment/index.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/assignment/index.ts"],"names":[],"mappings":"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"}
@@ -37,7 +37,7 @@ export class AssignmentService {
37
37
  .where('queueId', 'is not', null);
38
38
  if (onlyActive) {
39
39
  const now = new Date().toISOString();
40
- query = query.where((qb) => qb.where('endAt', 'is', null).orWhere('endAt', '>', now));
40
+ query = query.where((eb) => eb.or([eb('endAt', 'is', null), eb('endAt', '>', now)]));
41
41
  }
42
42
  if (queueIds?.length) {
43
43
  query = query.where('queueId', 'in', queueIds);
@@ -84,7 +84,7 @@ export class AssignmentService {
84
84
  .where('reportId', 'is not', null);
85
85
  if (onlyActive) {
86
86
  const now = new Date().toISOString();
87
- query = query.where((qb) => qb.where('endAt', '>', now).orWhere('endAt', 'is', null));
87
+ query = query.where((eb) => eb.or([eb('endAt', '>', now), eb('endAt', 'is', null)]));
88
88
  }
89
89
  if (reportIds?.length) {
90
90
  query = query.where('reportId', 'in', reportIds);
@@ -134,9 +134,7 @@ export class AssignmentService {
134
134
  .where('did', '=', did)
135
135
  .where('queueId', '=', queueId)
136
136
  .where('reportId', 'is', null)
137
- .where((qb) => qb
138
- .where('endAt', 'is', null)
139
- .orWhere('endAt', '>', now.toISOString()))
137
+ .where((eb) => eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]))
140
138
  .executeTakeFirst();
141
139
  if (existing) {
142
140
  if (existing.endAt === null) {
@@ -195,7 +193,7 @@ export class AssignmentService {
195
193
  .where('did', '=', did)
196
194
  .where('queueId', '=', queueId)
197
195
  .where('reportId', 'is', null)
198
- .where((qb) => qb.where('endAt', 'is', null).orWhere('endAt', '>', now.toISOString()))
196
+ .where((eb) => eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]))
199
197
  .executeTakeFirst();
200
198
  if (!existing) {
201
199
  throw new InvalidRequestError('No active assignment found', 'InvalidAssignment');
@@ -285,9 +283,7 @@ export class AssignmentService {
285
283
  .selectFrom('moderator_assignment')
286
284
  .selectAll()
287
285
  .where('reportId', '=', reportId)
288
- .where((qb) => qb
289
- .where('endAt', '>', now.toISOString())
290
- .orWhere('endAt', 'is', null))
286
+ .where((eb) => eb.or([eb('endAt', '>', now.toISOString()), eb('endAt', 'is', null)]))
291
287
  .executeTakeFirst();
292
288
  if (existing) {
293
289
  if (existing.did !== did) {
@@ -353,9 +349,10 @@ export class AssignmentService {
353
349
  .selectFrom('moderator_assignment')
354
350
  .selectAll()
355
351
  .where('reportId', '=', reportId)
356
- .where((qb) => qb
357
- .where('endAt', '>', now.toISOString())
358
- .orWhere('endAt', 'is', null))
352
+ .where((eb) => eb.or([
353
+ eb('endAt', '>', now.toISOString()),
354
+ eb('endAt', 'is', null),
355
+ ]))
359
356
  .executeTakeFirst();
360
357
  if (!existing) {
361
358
  throw new InvalidRequestError('Report is not assigned', 'InvalidAssignment');
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/assignment/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Selectable } from 'kysely'\nimport { ToolsOzoneQueueDefs } from '@atproto/api'\nimport { InvalidRequestError } from '@atproto/xrpc-server'\nimport { Database } from '../db/index.js'\nimport { EndAtIdKeyset, paginate } from '../db/pagination.js'\nimport { ModeratorAssignment } from '../db/schema/moderator_assignment.js'\nimport { ReportQueue } from '../db/schema/report_queue.js'\nimport type * as ToolsOzoneReportDefs from '../lexicon/types/tools/ozone/report/defs.js'\nimport type { Member as TeamMember } from '../lexicon/types/tools/ozone/team/defs.js'\nimport { QueueService, QueueServiceCreator } from '../queue/service.js'\nimport { createReportActivity } from '../report/activity.js'\nimport { TeamService, TeamServiceCreator } from '../team/index.js'\n\nexport interface AssignmentServiceOpts {\n queueDurationMs: number\n reportDurationMs: number\n}\n\n// Queue\nexport interface GetQueueAssignmentsInput {\n onlyActive?: boolean\n queueIds?: number[]\n dids?: string[]\n limit?: number\n cursor?: string\n}\nexport interface AssignQueueInput {\n did: string\n queueId: number\n}\nexport interface UnassignQueueInput {\n did: string\n queueId: number\n}\n\n// Report\nexport interface GetReportAssignmentsInput {\n onlyActive?: boolean\n reportIds?: number[]\n queueIds?: number[]\n dids?: string[]\n limit?: number\n cursor?: string\n}\nexport interface AssignReportInput {\n did: string\n reportId: number\n queueId?: number | null\n isPermanent?: boolean\n createdBy?: string\n}\nexport interface UnassignReportInput {\n reportId: number\n createdBy?: string\n}\n\ntype AssignmentRowWithQueue = Selectable<ModeratorAssignment> & {\n queueName: string | null\n queueSubjectTypes: string[] | null\n queueCollection: string | null\n queueDescription: string | null\n queueReportTypes: string[] | null\n queueCreatedBy: string | null\n queueCreatedAt: string | null\n queueUpdatedAt: string | null\n queueEnabled: boolean | null\n queueDeletedAt: string | null\n}\n\nexport type AssignmentServiceCreator = (db: Database) => AssignmentService\n\nexport class AssignmentService {\n constructor(\n public db: Database,\n public opts: AssignmentServiceOpts,\n private queueService: QueueService,\n private teamService: TeamService,\n ) {}\n\n static creator(\n opts: AssignmentServiceOpts,\n queueServiceCreator: QueueServiceCreator,\n teamServiceCreator: TeamServiceCreator,\n ): AssignmentServiceCreator {\n return (db: Database) =>\n new AssignmentService(\n db,\n opts,\n queueServiceCreator(db),\n teamServiceCreator(db),\n )\n }\n\n private async fetchMemberViews(\n dids: string[],\n ): Promise<Map<string, TeamMember>> {\n return this.teamService.viewByDids(dids)\n }\n\n async getQueueAssignments(input: GetQueueAssignmentsInput): Promise<{\n assignments: ToolsOzoneQueueDefs.AssignmentView[]\n cursor?: string\n }> {\n const { onlyActive, queueIds, dids, limit, cursor } = input\n const { ref } = this.db.db.dynamic\n\n let query = this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('reportId', 'is', null)\n .where('queueId', 'is not', null)\n\n if (onlyActive) {\n const now = new Date().toISOString()\n query = query.where((qb) =>\n qb.where('endAt', 'is', null).orWhere('endAt', '>', now),\n )\n }\n\n if (queueIds?.length) {\n query = query.where('queueId', 'in', queueIds)\n }\n\n if (dids?.length) {\n query = query.where('did', 'in', dids)\n }\n\n // use endAt to take advantage of indexes\n // Qualify column refs to avoid ambiguity with the report_queue join\n const keyset = new EndAtIdKeyset(\n ref('moderator_assignment.endAt'),\n ref('moderator_assignment.id'),\n )\n const paginatedQuery = paginate(query, {\n limit,\n cursor,\n keyset,\n direction: 'desc',\n tryIndex: true,\n })\n\n const results = await paginatedQuery.execute()\n const memberViews = await this.fetchMemberViews(results.map((r) => r.did))\n\n return {\n assignments: results.map((row) =>\n this.viewQueueAssignment(row, memberViews.get(row.did)),\n ),\n cursor: keyset.packFromResult(results),\n }\n }\n\n async getReportAssignments(input: GetReportAssignmentsInput): Promise<{\n assignments: ToolsOzoneReportDefs.AssignmentView[]\n cursor?: string\n }> {\n const { onlyActive, reportIds, queueIds, dids, limit, cursor } = input\n const { ref } = this.db.db.dynamic\n\n let query = this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('reportId', 'is not', null)\n\n if (onlyActive) {\n const now = new Date().toISOString()\n query = query.where((qb) =>\n qb.where('endAt', '>', now).orWhere('endAt', 'is', null),\n )\n }\n\n if (reportIds?.length) {\n query = query.where('reportId', 'in', reportIds)\n }\n\n if (queueIds?.length) {\n query = query.where('queueId', 'in', queueIds)\n }\n\n if (dids?.length) {\n query = query.where('did', 'in', dids)\n }\n\n // Qualify column refs to avoid ambiguity with the report_queue join\n const keyset = new EndAtIdKeyset(\n ref('moderator_assignment.endAt'),\n ref('moderator_assignment.id'),\n )\n const paginatedQuery = paginate(query, {\n limit,\n cursor,\n keyset,\n direction: 'desc',\n tryIndex: true,\n })\n\n const results = await paginatedQuery.execute()\n const memberViews = await this.fetchMemberViews(results.map((r) => r.did))\n\n return {\n assignments: results.map((row) =>\n this.viewReportAssignment(row, memberViews.get(row.did)),\n ),\n cursor: keyset.packFromResult(results),\n }\n }\n\n async assignQueue(\n input: AssignQueueInput,\n ): Promise<ToolsOzoneQueueDefs.AssignmentView> {\n const { did, queueId } = input\n const now = new Date()\n\n // Check queue since we aren't using foreign keys\n const queue = await this.db.db\n .selectFrom('report_queue')\n .selectAll()\n .where('id', '=', queueId)\n .where('enabled', '=', true)\n .where('deletedAt', 'is', null)\n .executeTakeFirst()\n if (!queue) {\n throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')\n }\n\n // Queue assignments are permanent (endAt = null). Upgrade any existing\n // active expiry-based row to permanent; otherwise insert a new permanent row.\n const result = await this.db.transaction(async (dbTxn) => {\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('did', '=', did)\n .where('queueId', '=', queueId)\n .where('reportId', 'is', null)\n .where((qb) =>\n qb\n .where('endAt', 'is', null)\n .orWhere('endAt', '>', now.toISOString()),\n )\n .executeTakeFirst()\n if (existing) {\n if (existing.endAt === null) {\n return existing\n }\n const updated = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ endAt: null })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n return updated\n }\n const created = await dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n queueId,\n startAt: now.toISOString(),\n endAt: null,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n return created\n })\n\n if (result.queueId === null || result.reportId !== null) {\n throw new Error('Failed to assign moderator to queue')\n }\n\n const row = await this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('moderator_assignment.id', '=', result.id)\n .executeTakeFirstOrThrow()\n\n const memberViews = await this.fetchMemberViews([result.did])\n return this.viewQueueAssignment(row, memberViews.get(result.did))\n }\n\n async unassignQueue(input: UnassignQueueInput): Promise<void> {\n const { did, queueId } = input\n const now = new Date()\n\n const existing = await this.db.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('did', '=', did)\n .where('queueId', '=', queueId)\n .where('reportId', 'is', null)\n .where((qb) =>\n qb.where('endAt', 'is', null).orWhere('endAt', '>', now.toISOString()),\n )\n .executeTakeFirst()\n\n if (!existing) {\n throw new InvalidRequestError(\n 'No active assignment found',\n 'InvalidAssignment',\n )\n }\n\n await this.db.db\n .updateTable('moderator_assignment')\n .set({ endAt: now.toISOString() })\n .where('id', '=', existing.id)\n .execute()\n }\n\n async assignReport(\n input: AssignReportInput,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const { did, reportId, queueId, isPermanent = false } = input\n const now = new Date()\n\n // Check report and queue since we aren't using foreign keys\n await this.checkReport(reportId)\n if (queueId !== undefined && queueId !== null) {\n await this.checkQueue(queueId)\n }\n\n // Make assignment\n const result = await this.db.transaction(async (dbTxn) => {\n if (isPermanent) {\n // Check for an existing permanent assignment (endAt IS NULL)\n const permanentExisting = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where('endAt', 'is', null)\n .executeTakeFirst()\n\n let result: Selectable<ModeratorAssignment>\n\n if (permanentExisting) {\n if (permanentExisting.did !== did) {\n throw new InvalidRequestError(\n 'Report already assigned',\n 'AlreadyAssigned',\n )\n }\n // Same user — update queueId if provided\n result = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ queueId: queueId ?? permanentExisting.queueId ?? null })\n .where('id', '=', permanentExisting.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n } else {\n // Upgrade an existing active (non-permanent) assignment to permanent\n const activeExisting = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where('endAt', '>', now.toISOString())\n .executeTakeFirst()\n\n if (activeExisting) {\n result = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({\n did,\n endAt: null,\n queueId: queueId ?? activeExisting.queueId ?? null,\n })\n .where('id', '=', activeExisting.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n } else {\n // Create new permanent assignment\n result = await dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n reportId,\n queueId: queueId,\n startAt: now.toISOString(),\n endAt: null,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n }\n }\n\n // Sync denormalized assignment fields on report table\n await dbTxn.db\n .updateTable('report')\n .set({ assignedTo: did, assignedAt: now.toISOString() })\n .where('id', '=', reportId)\n .execute()\n\n return result\n }\n\n // Non-permanent: find any active or permanent assignment\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where((qb) =>\n qb\n .where('endAt', '>', now.toISOString())\n .orWhere('endAt', 'is', null),\n )\n .executeTakeFirst()\n\n if (existing) {\n if (existing.did !== did) {\n throw new InvalidRequestError(\n 'Report already assigned',\n 'AlreadyAssigned',\n )\n }\n // Refresh the expiry unless the assignment is already permanent\n const newEndAt =\n existing.endAt === null\n ? null\n : new Date(now.getTime() + this.opts.reportDurationMs).toISOString()\n return dbTxn.db\n .updateTable('moderator_assignment')\n .set({\n endAt: newEndAt,\n queueId: queueId ?? existing.queueId ?? null,\n })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n }\n\n const endAt = new Date(\n now.getTime() + this.opts.reportDurationMs,\n ).toISOString()\n return dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n reportId,\n queueId: queueId,\n startAt: now.toISOString(),\n endAt,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n })\n\n // Log an assignmentActivity ONLY for permanent assignments. Swallow AlreadyInTargetState\n // so that re-assignments (e.g. refreshing expiry) don't throw.\n if (input.isPermanent) {\n try {\n await createReportActivity(this.db, {\n reportId,\n activityType: 'assignmentActivity',\n isAutomated: false,\n createdBy: input.createdBy ?? did,\n meta: { assignedTo: did },\n })\n } catch (err) {\n if (\n err instanceof InvalidRequestError &&\n err.customErrorName === 'AlreadyInTargetState'\n ) {\n // no-op — report already assigned, no state change to record\n } else {\n throw err\n }\n }\n }\n\n return this.hydrateReportAssignment(result.id)\n }\n\n async unassignReport(\n input: UnassignReportInput,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const { reportId, createdBy } = input\n const now = new Date()\n\n await this.checkReport(reportId)\n\n const { result, reportStatus } = await this.db.transaction(\n async (dbTxn) => {\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where((qb) =>\n qb\n .where('endAt', '>', now.toISOString())\n .orWhere('endAt', 'is', null),\n )\n .executeTakeFirst()\n\n if (!existing) {\n throw new InvalidRequestError(\n 'Report is not assigned',\n 'InvalidAssignment',\n )\n }\n\n const updated = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ endAt: now.toISOString() })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n\n // Capture status before any update so we can decide on the next status.\n const reportRow = await dbTxn.db\n .selectFrom('report')\n .select('status')\n .where('id', '=', reportId)\n .forUpdate()\n .executeTakeFirstOrThrow()\n\n // If the report had moved to 'assigned' and the assignment was not tied\n // to a queue, send it back to 'open' so it isn't stuck in 'assigned'\n // after the moderator releases it. The 'queued' transition (when there\n // is a queueId) is handled below via createReportActivity.\n const updateSet: Record<string, string | null> = {\n assignedTo: null,\n assignedAt: null,\n }\n if (reportRow.status === 'assigned' && existing.queueId === null) {\n updateSet.status = 'open'\n updateSet.updatedAt = now.toISOString()\n }\n await dbTxn.db\n .updateTable('report')\n .set(updateSet)\n .where('id', '=', reportId)\n .execute()\n\n return { result: updated, reportStatus: reportRow.status }\n },\n )\n\n // If unassigning from a queued report (status moved to 'assigned' on\n // permanent assign) before any other status change, send it back to\n // 'queued' so other moderators can pick it up.\n if (reportStatus === 'assigned' && result.queueId !== null) {\n try {\n await createReportActivity(this.db, {\n reportId,\n activityType: 'queueActivity',\n isAutomated: false,\n createdBy: createdBy ?? result.did,\n })\n } catch (err) {\n if (\n err instanceof InvalidRequestError &&\n (err.customErrorName === 'AlreadyInTargetState' ||\n err.customErrorName === 'InvalidStateTransition')\n ) {\n // no-op — status changed concurrently; leave it alone\n } else {\n throw err\n }\n }\n }\n\n return this.hydrateReportAssignment(result.id)\n }\n\n private async checkReport(reportId: number): Promise<void> {\n const report = await this.db.db\n .selectFrom('report')\n .selectAll()\n .where('id', '=', reportId)\n .executeTakeFirst()\n if (!report) {\n throw new InvalidRequestError('Invalid report', 'InvalidAssignment')\n }\n }\n\n private async checkQueue(queueId: number): Promise<void> {\n const queue = await this.db.db\n .selectFrom('report_queue')\n .selectAll()\n .where('id', '=', queueId)\n .where('enabled', '=', true)\n .where('deletedAt', 'is', null)\n .executeTakeFirst()\n if (!queue) {\n throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')\n }\n }\n\n private async hydrateReportAssignment(\n assignmentId: number,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const row = await this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('moderator_assignment.id', '=', assignmentId)\n .executeTakeFirstOrThrow()\n\n const memberViews = await this.fetchMemberViews([row.did])\n return this.viewReportAssignment(row, memberViews.get(row.did))\n }\n\n queueFromJoined(\n row: AssignmentRowWithQueue,\n ): Selectable<ReportQueue> | undefined {\n if (row.queueId === null || row.queueName === null) {\n return undefined\n }\n\n return {\n id: row.queueId,\n name: row.queueName,\n subjectTypes: row.queueSubjectTypes ?? [],\n collection: row.queueCollection,\n reportTypes: row.queueReportTypes ?? [],\n description: row.queueDescription ?? null,\n createdBy: row.queueCreatedBy ?? '',\n createdAt: row.queueCreatedAt ?? '',\n updatedAt: row.queueUpdatedAt ?? '',\n enabled: row.queueEnabled ?? false,\n deletedAt: row.queueDeletedAt,\n }\n }\n\n viewQueueAssignment(\n row: AssignmentRowWithQueue,\n member?: TeamMember,\n ): ToolsOzoneQueueDefs.AssignmentView {\n const queueService = this.queueService\n\n const queue = this.queueFromJoined(row)\n const queueView = queue ? queueService.view(queue) : undefined\n if (!queueView) {\n throw new Error('Failed to hydrate queue')\n }\n\n return {\n id: row.id,\n did: row.did,\n ...(member ? { moderator: member } : {}),\n queue: queueView,\n startAt: row.startAt,\n ...(row.endAt !== null ? { endAt: row.endAt } : {}),\n }\n }\n\n viewReportAssignment(\n row: AssignmentRowWithQueue,\n member?: TeamMember,\n ): ToolsOzoneReportDefs.AssignmentView {\n const queueService = this.queueService\n\n const queue = this.queueFromJoined(row)\n const queueView = queue ? queueService.view(queue) : undefined\n\n return {\n id: row.id,\n did: row.did,\n ...(member ? { moderator: member } : {}),\n reportId: row.reportId!,\n ...(queueView ? { queue: queueView } : {}),\n startAt: row.startAt,\n ...(row.endAt !== null ? { endAt: row.endAt } : {}),\n }\n }\n}\n"]}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/assignment/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Selectable } from 'kysely'\nimport { ToolsOzoneQueueDefs } from '@atproto/api'\nimport { InvalidRequestError } from '@atproto/xrpc-server'\nimport { Database } from '../db/index.js'\nimport { EndAtIdKeyset, paginate } from '../db/pagination.js'\nimport { ModeratorAssignment } from '../db/schema/moderator_assignment.js'\nimport { ReportQueue } from '../db/schema/report_queue.js'\nimport type * as ToolsOzoneReportDefs from '../lexicon/types/tools/ozone/report/defs.js'\nimport type { Member as TeamMember } from '../lexicon/types/tools/ozone/team/defs.js'\nimport { QueueService, QueueServiceCreator } from '../queue/service.js'\nimport { createReportActivity } from '../report/activity.js'\nimport { TeamService, TeamServiceCreator } from '../team/index.js'\n\nexport interface AssignmentServiceOpts {\n queueDurationMs: number\n reportDurationMs: number\n}\n\n// Queue\nexport interface GetQueueAssignmentsInput {\n onlyActive?: boolean\n queueIds?: number[]\n dids?: string[]\n limit?: number\n cursor?: string\n}\nexport interface AssignQueueInput {\n did: string\n queueId: number\n}\nexport interface UnassignQueueInput {\n did: string\n queueId: number\n}\n\n// Report\nexport interface GetReportAssignmentsInput {\n onlyActive?: boolean\n reportIds?: number[]\n queueIds?: number[]\n dids?: string[]\n limit?: number\n cursor?: string\n}\nexport interface AssignReportInput {\n did: string\n reportId: number\n queueId?: number | null\n isPermanent?: boolean\n createdBy?: string\n}\nexport interface UnassignReportInput {\n reportId: number\n createdBy?: string\n}\n\ntype AssignmentRowWithQueue = Selectable<ModeratorAssignment> & {\n queueName: string | null\n queueSubjectTypes: string[] | null\n queueCollection: string | null\n queueDescription: string | null\n queueReportTypes: string[] | null\n queueCreatedBy: string | null\n queueCreatedAt: string | null\n queueUpdatedAt: string | null\n queueEnabled: boolean | null\n queueDeletedAt: string | null\n}\n\nexport type AssignmentServiceCreator = (db: Database) => AssignmentService\n\nexport class AssignmentService {\n constructor(\n public db: Database,\n public opts: AssignmentServiceOpts,\n private queueService: QueueService,\n private teamService: TeamService,\n ) {}\n\n static creator(\n opts: AssignmentServiceOpts,\n queueServiceCreator: QueueServiceCreator,\n teamServiceCreator: TeamServiceCreator,\n ): AssignmentServiceCreator {\n return (db: Database) =>\n new AssignmentService(\n db,\n opts,\n queueServiceCreator(db),\n teamServiceCreator(db),\n )\n }\n\n private async fetchMemberViews(\n dids: string[],\n ): Promise<Map<string, TeamMember>> {\n return this.teamService.viewByDids(dids)\n }\n\n async getQueueAssignments(input: GetQueueAssignmentsInput): Promise<{\n assignments: ToolsOzoneQueueDefs.AssignmentView[]\n cursor?: string\n }> {\n const { onlyActive, queueIds, dids, limit, cursor } = input\n const { ref } = this.db.db.dynamic\n\n let query = this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('reportId', 'is', null)\n .where('queueId', 'is not', null)\n\n if (onlyActive) {\n const now = new Date().toISOString()\n query = query.where((eb) =>\n eb.or([eb('endAt', 'is', null), eb('endAt', '>', now)]),\n )\n }\n\n if (queueIds?.length) {\n query = query.where('queueId', 'in', queueIds)\n }\n\n if (dids?.length) {\n query = query.where('did', 'in', dids)\n }\n\n // use endAt to take advantage of indexes\n // Qualify column refs to avoid ambiguity with the report_queue join\n const keyset = new EndAtIdKeyset(\n ref('moderator_assignment.endAt'),\n ref('moderator_assignment.id'),\n )\n const paginatedQuery = paginate(query, {\n limit,\n cursor,\n keyset,\n direction: 'desc',\n tryIndex: true,\n })\n\n const results = await paginatedQuery.execute()\n const memberViews = await this.fetchMemberViews(results.map((r) => r.did))\n\n return {\n assignments: results.map((row) =>\n this.viewQueueAssignment(row, memberViews.get(row.did)),\n ),\n cursor: keyset.packFromResult(results),\n }\n }\n\n async getReportAssignments(input: GetReportAssignmentsInput): Promise<{\n assignments: ToolsOzoneReportDefs.AssignmentView[]\n cursor?: string\n }> {\n const { onlyActive, reportIds, queueIds, dids, limit, cursor } = input\n const { ref } = this.db.db.dynamic\n\n let query = this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('reportId', 'is not', null)\n\n if (onlyActive) {\n const now = new Date().toISOString()\n query = query.where((eb) =>\n eb.or([eb('endAt', '>', now), eb('endAt', 'is', null)]),\n )\n }\n\n if (reportIds?.length) {\n query = query.where('reportId', 'in', reportIds)\n }\n\n if (queueIds?.length) {\n query = query.where('queueId', 'in', queueIds)\n }\n\n if (dids?.length) {\n query = query.where('did', 'in', dids)\n }\n\n // Qualify column refs to avoid ambiguity with the report_queue join\n const keyset = new EndAtIdKeyset(\n ref('moderator_assignment.endAt'),\n ref('moderator_assignment.id'),\n )\n const paginatedQuery = paginate(query, {\n limit,\n cursor,\n keyset,\n direction: 'desc',\n tryIndex: true,\n })\n\n const results = await paginatedQuery.execute()\n const memberViews = await this.fetchMemberViews(results.map((r) => r.did))\n\n return {\n assignments: results.map((row) =>\n this.viewReportAssignment(row, memberViews.get(row.did)),\n ),\n cursor: keyset.packFromResult(results),\n }\n }\n\n async assignQueue(\n input: AssignQueueInput,\n ): Promise<ToolsOzoneQueueDefs.AssignmentView> {\n const { did, queueId } = input\n const now = new Date()\n\n // Check queue since we aren't using foreign keys\n const queue = await this.db.db\n .selectFrom('report_queue')\n .selectAll()\n .where('id', '=', queueId)\n .where('enabled', '=', true)\n .where('deletedAt', 'is', null)\n .executeTakeFirst()\n if (!queue) {\n throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')\n }\n\n // Queue assignments are permanent (endAt = null). Upgrade any existing\n // active expiry-based row to permanent; otherwise insert a new permanent row.\n const result = await this.db.transaction(async (dbTxn) => {\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('did', '=', did)\n .where('queueId', '=', queueId)\n .where('reportId', 'is', null)\n .where((eb) =>\n eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]),\n )\n .executeTakeFirst()\n if (existing) {\n if (existing.endAt === null) {\n return existing\n }\n const updated = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ endAt: null })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n return updated\n }\n const created = await dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n queueId,\n startAt: now.toISOString(),\n endAt: null,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n return created\n })\n\n if (result.queueId === null || result.reportId !== null) {\n throw new Error('Failed to assign moderator to queue')\n }\n\n const row = await this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('moderator_assignment.id', '=', result.id)\n .executeTakeFirstOrThrow()\n\n const memberViews = await this.fetchMemberViews([result.did])\n return this.viewQueueAssignment(row, memberViews.get(result.did))\n }\n\n async unassignQueue(input: UnassignQueueInput): Promise<void> {\n const { did, queueId } = input\n const now = new Date()\n\n const existing = await this.db.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('did', '=', did)\n .where('queueId', '=', queueId)\n .where('reportId', 'is', null)\n .where((eb) =>\n eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]),\n )\n .executeTakeFirst()\n\n if (!existing) {\n throw new InvalidRequestError(\n 'No active assignment found',\n 'InvalidAssignment',\n )\n }\n\n await this.db.db\n .updateTable('moderator_assignment')\n .set({ endAt: now.toISOString() })\n .where('id', '=', existing.id)\n .execute()\n }\n\n async assignReport(\n input: AssignReportInput,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const { did, reportId, queueId, isPermanent = false } = input\n const now = new Date()\n\n // Check report and queue since we aren't using foreign keys\n await this.checkReport(reportId)\n if (queueId !== undefined && queueId !== null) {\n await this.checkQueue(queueId)\n }\n\n // Make assignment\n const result = await this.db.transaction(async (dbTxn) => {\n if (isPermanent) {\n // Check for an existing permanent assignment (endAt IS NULL)\n const permanentExisting = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where('endAt', 'is', null)\n .executeTakeFirst()\n\n let result: Selectable<ModeratorAssignment>\n\n if (permanentExisting) {\n if (permanentExisting.did !== did) {\n throw new InvalidRequestError(\n 'Report already assigned',\n 'AlreadyAssigned',\n )\n }\n // Same user — update queueId if provided\n result = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ queueId: queueId ?? permanentExisting.queueId ?? null })\n .where('id', '=', permanentExisting.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n } else {\n // Upgrade an existing active (non-permanent) assignment to permanent\n const activeExisting = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where('endAt', '>', now.toISOString())\n .executeTakeFirst()\n\n if (activeExisting) {\n result = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({\n did,\n endAt: null,\n queueId: queueId ?? activeExisting.queueId ?? null,\n })\n .where('id', '=', activeExisting.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n } else {\n // Create new permanent assignment\n result = await dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n reportId,\n queueId: queueId,\n startAt: now.toISOString(),\n endAt: null,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n }\n }\n\n // Sync denormalized assignment fields on report table\n await dbTxn.db\n .updateTable('report')\n .set({ assignedTo: did, assignedAt: now.toISOString() })\n .where('id', '=', reportId)\n .execute()\n\n return result\n }\n\n // Non-permanent: find any active or permanent assignment\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where((eb) =>\n eb.or([eb('endAt', '>', now.toISOString()), eb('endAt', 'is', null)]),\n )\n .executeTakeFirst()\n\n if (existing) {\n if (existing.did !== did) {\n throw new InvalidRequestError(\n 'Report already assigned',\n 'AlreadyAssigned',\n )\n }\n // Refresh the expiry unless the assignment is already permanent\n const newEndAt =\n existing.endAt === null\n ? null\n : new Date(now.getTime() + this.opts.reportDurationMs).toISOString()\n return dbTxn.db\n .updateTable('moderator_assignment')\n .set({\n endAt: newEndAt,\n queueId: queueId ?? existing.queueId ?? null,\n })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n }\n\n const endAt = new Date(\n now.getTime() + this.opts.reportDurationMs,\n ).toISOString()\n return dbTxn.db\n .insertInto('moderator_assignment')\n .values({\n did,\n reportId,\n queueId: queueId,\n startAt: now.toISOString(),\n endAt,\n })\n .returningAll()\n .executeTakeFirstOrThrow()\n })\n\n // Log an assignmentActivity ONLY for permanent assignments. Swallow AlreadyInTargetState\n // so that re-assignments (e.g. refreshing expiry) don't throw.\n if (input.isPermanent) {\n try {\n await createReportActivity(this.db, {\n reportId,\n activityType: 'assignmentActivity',\n isAutomated: false,\n createdBy: input.createdBy ?? did,\n meta: { assignedTo: did },\n })\n } catch (err) {\n if (\n err instanceof InvalidRequestError &&\n err.customErrorName === 'AlreadyInTargetState'\n ) {\n // no-op — report already assigned, no state change to record\n } else {\n throw err\n }\n }\n }\n\n return this.hydrateReportAssignment(result.id)\n }\n\n async unassignReport(\n input: UnassignReportInput,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const { reportId, createdBy } = input\n const now = new Date()\n\n await this.checkReport(reportId)\n\n const { result, reportStatus } = await this.db.transaction(\n async (dbTxn) => {\n const existing = await dbTxn.db\n .selectFrom('moderator_assignment')\n .selectAll()\n .where('reportId', '=', reportId)\n .where((eb) =>\n eb.or([\n eb('endAt', '>', now.toISOString()),\n eb('endAt', 'is', null),\n ]),\n )\n .executeTakeFirst()\n\n if (!existing) {\n throw new InvalidRequestError(\n 'Report is not assigned',\n 'InvalidAssignment',\n )\n }\n\n const updated = await dbTxn.db\n .updateTable('moderator_assignment')\n .set({ endAt: now.toISOString() })\n .where('id', '=', existing.id)\n .returningAll()\n .executeTakeFirstOrThrow()\n\n // Capture status before any update so we can decide on the next status.\n const reportRow = await dbTxn.db\n .selectFrom('report')\n .select('status')\n .where('id', '=', reportId)\n .forUpdate()\n .executeTakeFirstOrThrow()\n\n // If the report had moved to 'assigned' and the assignment was not tied\n // to a queue, send it back to 'open' so it isn't stuck in 'assigned'\n // after the moderator releases it. The 'queued' transition (when there\n // is a queueId) is handled below via createReportActivity.\n const updateSet: Record<string, string | null> = {\n assignedTo: null,\n assignedAt: null,\n }\n if (reportRow.status === 'assigned' && existing.queueId === null) {\n updateSet.status = 'open'\n updateSet.updatedAt = now.toISOString()\n }\n await dbTxn.db\n .updateTable('report')\n .set(updateSet)\n .where('id', '=', reportId)\n .execute()\n\n return { result: updated, reportStatus: reportRow.status }\n },\n )\n\n // If unassigning from a queued report (status moved to 'assigned' on\n // permanent assign) before any other status change, send it back to\n // 'queued' so other moderators can pick it up.\n if (reportStatus === 'assigned' && result.queueId !== null) {\n try {\n await createReportActivity(this.db, {\n reportId,\n activityType: 'queueActivity',\n isAutomated: false,\n createdBy: createdBy ?? result.did,\n })\n } catch (err) {\n if (\n err instanceof InvalidRequestError &&\n (err.customErrorName === 'AlreadyInTargetState' ||\n err.customErrorName === 'InvalidStateTransition')\n ) {\n // no-op — status changed concurrently; leave it alone\n } else {\n throw err\n }\n }\n }\n\n return this.hydrateReportAssignment(result.id)\n }\n\n private async checkReport(reportId: number): Promise<void> {\n const report = await this.db.db\n .selectFrom('report')\n .selectAll()\n .where('id', '=', reportId)\n .executeTakeFirst()\n if (!report) {\n throw new InvalidRequestError('Invalid report', 'InvalidAssignment')\n }\n }\n\n private async checkQueue(queueId: number): Promise<void> {\n const queue = await this.db.db\n .selectFrom('report_queue')\n .selectAll()\n .where('id', '=', queueId)\n .where('enabled', '=', true)\n .where('deletedAt', 'is', null)\n .executeTakeFirst()\n if (!queue) {\n throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')\n }\n }\n\n private async hydrateReportAssignment(\n assignmentId: number,\n ): Promise<ToolsOzoneReportDefs.AssignmentView> {\n const row = await this.db.db\n .selectFrom('moderator_assignment')\n .leftJoin(\n 'report_queue',\n 'report_queue.id',\n 'moderator_assignment.queueId',\n )\n .selectAll('moderator_assignment')\n .select([\n 'report_queue.name as queueName',\n 'report_queue.subjectTypes as queueSubjectTypes',\n 'report_queue.collection as queueCollection',\n 'report_queue.description as queueDescription',\n 'report_queue.reportTypes as queueReportTypes',\n 'report_queue.createdBy as queueCreatedBy',\n 'report_queue.createdAt as queueCreatedAt',\n 'report_queue.updatedAt as queueUpdatedAt',\n 'report_queue.enabled as queueEnabled',\n 'report_queue.deletedAt as queueDeletedAt',\n ])\n .where('moderator_assignment.id', '=', assignmentId)\n .executeTakeFirstOrThrow()\n\n const memberViews = await this.fetchMemberViews([row.did])\n return this.viewReportAssignment(row, memberViews.get(row.did))\n }\n\n queueFromJoined(\n row: AssignmentRowWithQueue,\n ): Selectable<ReportQueue> | undefined {\n if (row.queueId === null || row.queueName === null) {\n return undefined\n }\n\n return {\n id: row.queueId,\n name: row.queueName,\n subjectTypes: row.queueSubjectTypes ?? [],\n collection: row.queueCollection,\n reportTypes: row.queueReportTypes ?? [],\n description: row.queueDescription ?? null,\n createdBy: row.queueCreatedBy ?? '',\n createdAt: row.queueCreatedAt ?? '',\n updatedAt: row.queueUpdatedAt ?? '',\n enabled: row.queueEnabled ?? false,\n deletedAt: row.queueDeletedAt,\n }\n }\n\n viewQueueAssignment(\n row: AssignmentRowWithQueue,\n member?: TeamMember,\n ): ToolsOzoneQueueDefs.AssignmentView {\n const queueService = this.queueService\n\n const queue = this.queueFromJoined(row)\n const queueView = queue ? queueService.view(queue) : undefined\n if (!queueView) {\n throw new Error('Failed to hydrate queue')\n }\n\n return {\n id: row.id,\n did: row.did,\n ...(member ? { moderator: member } : {}),\n queue: queueView,\n startAt: row.startAt,\n ...(row.endAt !== null ? { endAt: row.endAt } : {}),\n }\n }\n\n viewReportAssignment(\n row: AssignmentRowWithQueue,\n member?: TeamMember,\n ): ToolsOzoneReportDefs.AssignmentView {\n const queueService = this.queueService\n\n const queue = this.queueFromJoined(row)\n const queueView = queue ? queueService.view(queue) : undefined\n\n return {\n id: row.id,\n did: row.did,\n ...(member ? { moderator: member } : {}),\n reportId: row.reportId!,\n ...(queueView ? { queue: queueView } : {}),\n startAt: row.startAt,\n ...(row.endAt !== null ? { endAt: row.endAt } : {}),\n }\n }\n}\n"]}
@@ -48,7 +48,13 @@ export declare class EventPusher {
48
48
  attemptRecordEvent(id: number): Promise<void>;
49
49
  attemptBlobEvent(id: number): Promise<void>;
50
50
  markBlobEventAttempt(dbTxn: Database, event: Selectable<BlobPushEvent>, succeeded: boolean): Promise<void>;
51
- logBlobPushEvent(blobValues: Insertable<BlobPushEvent>[], takedownRef?: string | null): Promise<import("kysely").Selection<import("../db/schema/index.js").DatabaseSchemaType, "blob_push_event", "eventType" | "id" | "subjectBlobCid" | "subjectDid" | "subjectUri">[]>;
51
+ logBlobPushEvent(blobValues: Insertable<BlobPushEvent>[], takedownRef?: string | null): Promise<{
52
+ eventType: import("../db/schema/blob_push_event.js").BlobPushEventType;
53
+ id: number;
54
+ subjectBlobCid: string;
55
+ subjectDid: string;
56
+ subjectUri: string | null;
57
+ }[]>;
52
58
  }
53
59
  export {};
54
60
  //# sourceMappingURL=event-pusher.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"event-pusher.d.ts","sourceRoot":"","sources":["../../src/daemon/event-pusher.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAA;AAC/C,OAAO,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAA;AAEvC,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAA;AACzC,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAA;AAC/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAA;AAQnE,KAAK,SAAS,GAAG;IACf,KAAK,CAAC,EAAE,MAAM,CAAC,OAAO,CAAA;IACtB,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC,CAAA;CACvB,CAAA;AAED,KAAK,WAAW,GAAG;IACjB,OAAO,EAAE;QACP,aAAa,EAAE,MAAM,CAAA;KACtB,CAAA;CACF,CAAA;AAED,KAAK,OAAO,GAAG;IACb,KAAK,EAAE,QAAQ,CAAA;IACf,GAAG,EAAE,MAAM,CAAA;CACZ,CAAA;AAED,qBAAa,WAAW;IAiBb,EAAE,EAAE,QAAQ;IACZ,iBAAiB,EAAE,CACxB,GAAG,EAAE,MAAM,EACX,MAAM,EAAE,MAAM,KACX,OAAO,CAAC,WAAW,CAAC;IApB3B,SAAS,UAAQ;IAEjB,aAAa,EAAE,SAAS,CAEvB;IACD,eAAe,EAAE,SAAS,CAEzB;IACD,aAAa,EAAE,SAAS,CAEvB;IAED,OAAO,EAAE,OAAO,GAAG,SAAS,CAAA;IAC5B,GAAG,EAAE,OAAO,GAAG,SAAS,CAAA;IAExB,YACS,EAAE,EAAE,QAAQ,EACZ,iBAAiB,EAAE,CACxB,GAAG,EAAE,MAAM,EACX,MAAM,EAAE,MAAM,KACX,OAAO,CAAC,WAAW,CAAC,EACzB,QAAQ,EAAE;QACR,OAAO,CAAC,EAAE;YACR,GAAG,EAAE,MAAM,CAAA;YACX,GAAG,EAAE,MAAM,CAAA;SACZ,CAAA;QACD,GAAG,CAAC,EAAE;YACJ,GAAG,EAAE,MAAM,CAAA;YACX,GAAG,EAAE,MAAM,CAAA;SACZ,CAAA;KACF,EAcF;IAED,KAAK,SAIJ;IAKD,mBAAmB,CAAC,cAAc,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,iBAAiB,EAAE,CAmBpE;IAED,IAAI,CAAC,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,QAS7C;IAEK,UAAU,kBASf;IAEK,OAAO,kBAaZ;IAEK,cAAc,kBAUnB;IAEK,gBAAgB,kBAUrB;IAEK,cAAc,kBAUnB;YAEa,sBAAsB;IAgC9B,gBAAgB,CAAC,EAAE,EAAE,MAAM,iBAoChC;IAEK,kBAAkB,CAAC,EAAE,EAAE,MAAM,iBAqClC;IAEK,gBAAgB,CAAC,EAAE,EAAE,MAAM,iBA0BhC;IAEK,oBAAoB,CACxB,KAAK,EAAE,QAAQ,EACf,KAAK,EAAE,UAAU,CAAC,aAAa,CAAC,EAChC,SAAS,EAAE,OAAO,iBAgBnB;IAEK,gBAAgB,CACpB,UAAU,EAAE,UAAU,CAAC,aAAa,CAAC,EAAE,EACvC,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,qLAqB5B;CACF"}
1
+ {"version":3,"file":"event-pusher.d.ts","sourceRoot":"","sources":["../../src/daemon/event-pusher.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAA;AAC/C,OAAO,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAA;AAEvC,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAA;AACzC,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAA;AAC/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAA;AAQnE,KAAK,SAAS,GAAG;IACf,KAAK,CAAC,EAAE,MAAM,CAAC,OAAO,CAAA;IACtB,OAAO,EAAE,OAAO,CAAC,IAAI,CAAC,CAAA;CACvB,CAAA;AAED,KAAK,WAAW,GAAG;IACjB,OAAO,EAAE;QACP,aAAa,EAAE,MAAM,CAAA;KACtB,CAAA;CACF,CAAA;AAED,KAAK,OAAO,GAAG;IACb,KAAK,EAAE,QAAQ,CAAA;IACf,GAAG,EAAE,MAAM,CAAA;CACZ,CAAA;AAED,qBAAa,WAAW;IAiBb,EAAE,EAAE,QAAQ;IACZ,iBAAiB,EAAE,CACxB,GAAG,EAAE,MAAM,EACX,MAAM,EAAE,MAAM,KACX,OAAO,CAAC,WAAW,CAAC;IApB3B,SAAS,UAAQ;IAEjB,aAAa,EAAE,SAAS,CAEvB;IACD,eAAe,EAAE,SAAS,CAEzB;IACD,aAAa,EAAE,SAAS,CAEvB;IAED,OAAO,EAAE,OAAO,GAAG,SAAS,CAAA;IAC5B,GAAG,EAAE,OAAO,GAAG,SAAS,CAAA;IAExB,YACS,EAAE,EAAE,QAAQ,EACZ,iBAAiB,EAAE,CACxB,GAAG,EAAE,MAAM,EACX,MAAM,EAAE,MAAM,KACX,OAAO,CAAC,WAAW,CAAC,EACzB,QAAQ,EAAE;QACR,OAAO,CAAC,EAAE;YACR,GAAG,EAAE,MAAM,CAAA;YACX,GAAG,EAAE,MAAM,CAAA;SACZ,CAAA;QACD,GAAG,CAAC,EAAE;YACJ,GAAG,EAAE,MAAM,CAAA;YACX,GAAG,EAAE,MAAM,CAAA;SACZ,CAAA;KACF,EAcF;IAED,KAAK,SAIJ;IAKD,mBAAmB,CAAC,cAAc,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,iBAAiB,EAAE,CAmBpE;IAED,IAAI,CAAC,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,QAS7C;IAEK,UAAU,kBASf;IAEK,OAAO,kBAaZ;IAEK,cAAc,kBAUnB;IAEK,gBAAgB,kBAUrB;IAEK,cAAc,kBAUnB;YAEa,sBAAsB;IAgC9B,gBAAgB,CAAC,EAAE,EAAE,MAAM,iBAoChC;IAEK,kBAAkB,CAAC,EAAE,EAAE,MAAM,iBAqClC;IAEK,gBAAgB,CAAC,EAAE,EAAE,MAAM,iBA0BhC;IAEK,oBAAoB,CACxB,KAAK,EAAE,QAAQ,EACf,KAAK,EAAE,UAAU,CAAC,aAAa,CAAC,EAChC,SAAS,EAAE,OAAO,iBAgBnB;IAEK,gBAAgB,CACpB,UAAU,EAAE,UAAU,CAAC,aAAa,CAAC,EAAE,EACvC,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI;;;;;;SAqB5B;CACF"}
@@ -1,6 +1,5 @@
1
- import { Migrator } from 'kysely';
2
- import pg from 'pg';
3
- declare const PgPool: typeof pg.Pool;
1
+ import { Migrator } from 'kysely/migration';
2
+ declare const PgPool: typeof import("pg").Pool;
4
3
  type PgPool = InstanceType<typeof PgPool>;
5
4
  import type TypedEmitter from 'typed-emitter';
6
5
  import { DatabaseSchema } from './schema/index.js';
@@ -24,8 +23,8 @@ export declare class Database {
24
23
  transaction<T>(fn: (db: Database) => Promise<T>): Promise<T>;
25
24
  onCommit(fn: () => void): void;
26
25
  close(): Promise<void>;
27
- migrateToOrThrow(migration: string): Promise<import("kysely").MigrationResult[]>;
28
- migrateToLatestOrThrow(): Promise<import("kysely").MigrationResult[]>;
26
+ migrateToOrThrow(migration: string): Promise<import("kysely/migration").MigrationResult[]>;
27
+ migrateToLatestOrThrow(): Promise<import("kysely/migration").MigrationResult[]>;
29
28
  }
30
29
  export default Database;
31
30
  type TxnEmitter = TypedEmitter.default<TxnEvents>;
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"AAEA,OAAO,EAGL,QAAQ,EAOT,MAAM,QAAQ,CAAA;AAEf,OAAO,EAAE,MAAM,IAAI,CAAA;AAEnB,QAAA,MAAc,MAAM,gBAAuB,CAAA;AAC3C,KAAK,MAAM,GAAG,YAAY,CAAC,OAAO,MAAM,CAAC,CAAA;AACzC,OAAO,KAAK,YAAY,MAAM,eAAe,CAAA;AAI7C,OAAO,EAAE,cAAc,EAAsB,MAAM,mBAAmB,CAAA;AACtE,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAA;AAEtC,qBAAa,QAAQ;IASV,IAAI,EAAE,SAAS;IARxB,IAAI,EAAE,MAAM,CAAA;IACZ,EAAE,EAAE,cAAc,CAAA;IAClB,QAAQ,EAAE,QAAQ,CAAA;IAClB,KAAK,EAAyB,UAAU,CAAA;IACxC,SAAS,UAAQ;IACjB,SAAS,UAAQ;IAEjB,YACS,IAAI,EAAE,SAAS,EACtB,SAAS,CAAC,EAAE;QAAE,EAAE,EAAE,cAAc,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,EAkDjD;IAED,IAAI,MAAM,IAAI,MAAM,GAAG,SAAS,CAE/B;IAED,IAAI,aAAa,YAEhB;IAED,iBAAiB,SAEhB;IAED,oBAAoB,SAEnB;IAEK,WAAW,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAsBjE;IAED,QAAQ,CAAC,EAAE,EAAE,MAAM,IAAI,QAGtB;IAEK,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAI3B;IAEK,gBAAgB,CAAC,SAAS,EAAE,MAAM,+CAYvC;IAEK,sBAAsB,gDAY3B;CACF;eAEc,QAAQ;AA6BvB,KAAK,UAAU,GAAG,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,CAAA;AAEjD,KAAK,SAAS,GAAG;IACf,MAAM,EAAE,MAAM,IAAI,CAAA;CACnB,CAAA"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"AAYA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAA;AAI3C,QAAA,MAAc,MAAM,0BAAuB,CAAA;AAC3C,KAAK,MAAM,GAAG,YAAY,CAAC,OAAO,MAAM,CAAC,CAAA;AACzC,OAAO,KAAK,YAAY,MAAM,eAAe,CAAA;AAI7C,OAAO,EAAE,cAAc,EAAsB,MAAM,mBAAmB,CAAA;AACtE,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAA;AAEtC,qBAAa,QAAQ;IASV,IAAI,EAAE,SAAS;IARxB,IAAI,EAAE,MAAM,CAAA;IACZ,EAAE,EAAE,cAAc,CAAA;IAClB,QAAQ,EAAE,QAAQ,CAAA;IAClB,KAAK,EAAyB,UAAU,CAAA;IACxC,SAAS,UAAQ;IACjB,SAAS,UAAQ;IAEjB,YACS,IAAI,EAAE,SAAS,EACtB,SAAS,CAAC,EAAE;QAAE,EAAE,EAAE,cAAc,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,EAkDjD;IAED,IAAI,MAAM,IAAI,MAAM,GAAG,SAAS,CAE/B;IAED,IAAI,aAAa,YAEhB;IAED,iBAAiB,SAEhB;IAED,oBAAoB,SAEnB;IAEK,WAAW,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAsBjE;IAED,QAAQ,CAAC,EAAE,EAAE,MAAM,IAAI,QAGtB;IAEK,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAI3B;IAEK,gBAAgB,CAAC,SAAS,EAAE,MAAM,yDAYvC;IAEK,sBAAsB,0DAY3B;CACF;eAEc,QAAQ;AA6BvB,KAAK,UAAU,GAAG,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,CAAA;AAEjD,KAAK,SAAS,GAAG;IACf,MAAM,EAAE,MAAM,IAAI,CAAA;CACnB,CAAA"}
package/dist/db/index.js CHANGED
@@ -1,6 +1,7 @@
1
1
  import assert from 'node:assert';
2
2
  import { EventEmitter } from 'node:events';
3
- import { Kysely, Migrator, PostgresDialect, } from 'kysely';
3
+ import { Kysely, PostgresDialect, } from 'kysely';
4
+ import { Migrator } from 'kysely/migration';
4
5
  // eslint-disable-next-line import/default
5
6
  import pg from 'pg';
6
7
  // eslint-disable-next-line import/no-named-as-default-member
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'node:assert'\nimport { EventEmitter } from 'node:events'\nimport {\n Kysely,\n KyselyPlugin,\n Migrator,\n PluginTransformQueryArgs,\n PluginTransformResultArgs,\n PostgresDialect,\n QueryResult,\n RootOperationNode,\n UnknownRow,\n} from 'kysely'\n// eslint-disable-next-line import/default\nimport pg from 'pg'\n// eslint-disable-next-line import/no-named-as-default-member\nconst { Pool: PgPool, types: pgTypes } = pg\ntype PgPool = InstanceType<typeof PgPool>\nimport type TypedEmitter from 'typed-emitter'\nimport { dbLogger } from '../logger.js'\nimport * as migrations from './migrations/index.js'\nimport { CtxMigrationProvider } from './migrations/provider.js'\nimport { DatabaseSchema, DatabaseSchemaType } from './schema/index.js'\nimport { PgOptions } from './types.js'\n\nexport class Database {\n pool: PgPool\n db: DatabaseSchema\n migrator: Migrator\n txEvt = new EventEmitter() as TxnEmitter\n destroyed = false\n isPrimary = false\n\n constructor(\n public opts: PgOptions,\n instances?: { db: DatabaseSchema; pool: PgPool },\n ) {\n // if instances are provided, use those\n if (instances) {\n this.db = instances.db\n this.pool = instances.pool\n } else {\n // else create a pool & connect\n const { schema, url } = opts\n const pool =\n opts.pool ??\n new PgPool({\n connectionString: url,\n max: opts.poolSize,\n maxUses: opts.poolMaxUses,\n idleTimeoutMillis: opts.poolIdleTimeoutMs,\n })\n\n // Select count(*) and other pg bigints as js integer\n pgTypes.setTypeParser(pgTypes.builtins.INT8, (n) => parseInt(n, 10))\n\n // Setup schema usage, primarily for test parallelism (each test suite runs in its own pg schema)\n if (schema && !/^[a-z_]+$/i.test(schema)) {\n throw new Error(\n `Postgres schema must only contain [A-Za-z_]: ${schema}`,\n )\n }\n\n pool.on('error', onPoolError)\n pool.on('connect', (client) => {\n client.on('error', onClientError)\n // Used for trigram indexes, e.g. on actor search\n client.query('SET pg_trgm.word_similarity_threshold TO .4;')\n if (schema) {\n // Shared objects such as extensions will go in the public schema\n client.query(`SET search_path TO \"${schema}\",public;`)\n }\n })\n\n this.pool = pool\n this.db = new Kysely<DatabaseSchemaType>({\n dialect: new PostgresDialect({ pool }),\n })\n }\n\n this.migrator = new Migrator({\n db: this.db,\n migrationTableSchema: opts.schema,\n provider: new CtxMigrationProvider(migrations, 'pg'),\n })\n }\n\n get schema(): string | undefined {\n return this.opts.schema\n }\n\n get isTransaction() {\n return this.db.isTransaction\n }\n\n assertTransaction() {\n assert(this.isTransaction, 'Transaction required')\n }\n\n assertNotTransaction() {\n assert(!this.isTransaction, 'Cannot be in a transaction')\n }\n\n async transaction<T>(fn: (db: Database) => Promise<T>): Promise<T> {\n const leakyTxPlugin = new LeakyTxPlugin()\n const { dbTxn, txRes } = await this.db\n .withPlugin(leakyTxPlugin)\n .transaction()\n .execute(async (txn) => {\n const dbTxn = new Database(this.opts, {\n db: txn,\n pool: this.pool,\n })\n const txRes = await fn(dbTxn)\n .catch(async (err) => {\n leakyTxPlugin.endTx()\n // ensure that all in-flight queries are flushed & the connection is open\n await dbTxn.db.getExecutor().provideConnection(noopAsync)\n throw err\n })\n .finally(() => leakyTxPlugin.endTx())\n return { dbTxn, txRes }\n })\n dbTxn?.txEvt.emit('commit')\n return txRes\n }\n\n onCommit(fn: () => void) {\n this.assertTransaction()\n this.txEvt.once('commit', fn)\n }\n\n async close(): Promise<void> {\n if (this.destroyed) return\n await this.db.destroy()\n this.destroyed = true\n }\n\n async migrateToOrThrow(migration: string) {\n if (this.schema) {\n await this.db.schema.createSchema(this.schema).ifNotExists().execute()\n }\n const { error, results } = await this.migrator.migrateTo(migration)\n if (error) {\n throw error\n }\n if (!results) {\n throw new Error('An unknown failure occurred while migrating')\n }\n return results\n }\n\n async migrateToLatestOrThrow() {\n if (this.schema) {\n await this.db.schema.createSchema(this.schema).ifNotExists().execute()\n }\n const { error, results } = await this.migrator.migrateToLatest()\n if (error) {\n throw error\n }\n if (!results) {\n throw new Error('An unknown failure occurred while migrating')\n }\n return results\n }\n}\n\nexport default Database\n\nconst onPoolError = (err: Error) => dbLogger.error({ err }, 'db pool error')\nconst onClientError = (err: Error) => dbLogger.error({ err }, 'db client error')\n\n// utils\n// -------\n\nclass LeakyTxPlugin implements KyselyPlugin {\n private txOver = false\n\n endTx() {\n this.txOver = true\n }\n\n transformQuery(args: PluginTransformQueryArgs): RootOperationNode {\n if (this.txOver) {\n throw new Error('tx already failed')\n }\n return args.node\n }\n\n async transformResult(\n args: PluginTransformResultArgs,\n ): Promise<QueryResult<UnknownRow>> {\n return args.result\n }\n}\n\ntype TxnEmitter = TypedEmitter.default<TxnEvents>\n\ntype TxnEvents = {\n commit: () => void\n}\n\nconst noopAsync = async () => {}\n"]}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/db/index.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'node:assert'\nimport { EventEmitter } from 'node:events'\nimport {\n Kysely,\n KyselyPlugin,\n PluginTransformQueryArgs,\n PluginTransformResultArgs,\n PostgresDialect,\n QueryResult,\n RootOperationNode,\n UnknownRow,\n} from 'kysely'\nimport { Migrator } from 'kysely/migration'\n// eslint-disable-next-line import/default\nimport pg from 'pg'\n// eslint-disable-next-line import/no-named-as-default-member\nconst { Pool: PgPool, types: pgTypes } = pg\ntype PgPool = InstanceType<typeof PgPool>\nimport type TypedEmitter from 'typed-emitter'\nimport { dbLogger } from '../logger.js'\nimport * as migrations from './migrations/index.js'\nimport { CtxMigrationProvider } from './migrations/provider.js'\nimport { DatabaseSchema, DatabaseSchemaType } from './schema/index.js'\nimport { PgOptions } from './types.js'\n\nexport class Database {\n pool: PgPool\n db: DatabaseSchema\n migrator: Migrator\n txEvt = new EventEmitter() as TxnEmitter\n destroyed = false\n isPrimary = false\n\n constructor(\n public opts: PgOptions,\n instances?: { db: DatabaseSchema; pool: PgPool },\n ) {\n // if instances are provided, use those\n if (instances) {\n this.db = instances.db\n this.pool = instances.pool\n } else {\n // else create a pool & connect\n const { schema, url } = opts\n const pool =\n opts.pool ??\n new PgPool({\n connectionString: url,\n max: opts.poolSize,\n maxUses: opts.poolMaxUses,\n idleTimeoutMillis: opts.poolIdleTimeoutMs,\n })\n\n // Select count(*) and other pg bigints as js integer\n pgTypes.setTypeParser(pgTypes.builtins.INT8, (n) => parseInt(n, 10))\n\n // Setup schema usage, primarily for test parallelism (each test suite runs in its own pg schema)\n if (schema && !/^[a-z_]+$/i.test(schema)) {\n throw new Error(\n `Postgres schema must only contain [A-Za-z_]: ${schema}`,\n )\n }\n\n pool.on('error', onPoolError)\n pool.on('connect', (client) => {\n client.on('error', onClientError)\n // Used for trigram indexes, e.g. on actor search\n client.query('SET pg_trgm.word_similarity_threshold TO .4;')\n if (schema) {\n // Shared objects such as extensions will go in the public schema\n client.query(`SET search_path TO \"${schema}\",public;`)\n }\n })\n\n this.pool = pool\n this.db = new Kysely<DatabaseSchemaType>({\n dialect: new PostgresDialect({ pool }),\n })\n }\n\n this.migrator = new Migrator({\n db: this.db,\n migrationTableSchema: opts.schema,\n provider: new CtxMigrationProvider(migrations, 'pg'),\n })\n }\n\n get schema(): string | undefined {\n return this.opts.schema\n }\n\n get isTransaction() {\n return this.db.isTransaction\n }\n\n assertTransaction() {\n assert(this.isTransaction, 'Transaction required')\n }\n\n assertNotTransaction() {\n assert(!this.isTransaction, 'Cannot be in a transaction')\n }\n\n async transaction<T>(fn: (db: Database) => Promise<T>): Promise<T> {\n const leakyTxPlugin = new LeakyTxPlugin()\n const { dbTxn, txRes } = await this.db\n .withPlugin(leakyTxPlugin)\n .transaction()\n .execute(async (txn) => {\n const dbTxn = new Database(this.opts, {\n db: txn,\n pool: this.pool,\n })\n const txRes = await fn(dbTxn)\n .catch(async (err) => {\n leakyTxPlugin.endTx()\n // ensure that all in-flight queries are flushed & the connection is open\n await dbTxn.db.getExecutor().provideConnection(noopAsync)\n throw err\n })\n .finally(() => leakyTxPlugin.endTx())\n return { dbTxn, txRes }\n })\n dbTxn?.txEvt.emit('commit')\n return txRes\n }\n\n onCommit(fn: () => void) {\n this.assertTransaction()\n this.txEvt.once('commit', fn)\n }\n\n async close(): Promise<void> {\n if (this.destroyed) return\n await this.db.destroy()\n this.destroyed = true\n }\n\n async migrateToOrThrow(migration: string) {\n if (this.schema) {\n await this.db.schema.createSchema(this.schema).ifNotExists().execute()\n }\n const { error, results } = await this.migrator.migrateTo(migration)\n if (error) {\n throw error\n }\n if (!results) {\n throw new Error('An unknown failure occurred while migrating')\n }\n return results\n }\n\n async migrateToLatestOrThrow() {\n if (this.schema) {\n await this.db.schema.createSchema(this.schema).ifNotExists().execute()\n }\n const { error, results } = await this.migrator.migrateToLatest()\n if (error) {\n throw error\n }\n if (!results) {\n throw new Error('An unknown failure occurred while migrating')\n }\n return results\n }\n}\n\nexport default Database\n\nconst onPoolError = (err: Error) => dbLogger.error({ err }, 'db pool error')\nconst onClientError = (err: Error) => dbLogger.error({ err }, 'db client error')\n\n// utils\n// -------\n\nclass LeakyTxPlugin implements KyselyPlugin {\n private txOver = false\n\n endTx() {\n this.txOver = true\n }\n\n transformQuery(args: PluginTransformQueryArgs): RootOperationNode {\n if (this.txOver) {\n throw new Error('tx already failed')\n }\n return args.node\n }\n\n async transformResult(\n args: PluginTransformResultArgs,\n ): Promise<QueryResult<UnknownRow>> {\n return args.result\n }\n}\n\ntype TxnEmitter = TypedEmitter.default<TxnEvents>\n\ntype TxnEvents = {\n commit: () => void\n}\n\nconst noopAsync = async () => {}\n"]}
@@ -1,5 +1,4 @@
1
1
  import { Kysely } from 'kysely';
2
- import { DatabaseSchemaType } from '../schema/index.js';
3
2
  export declare function up(db: Kysely<any>): Promise<void>;
4
- export declare function down(db: Kysely<DatabaseSchemaType>): Promise<void>;
3
+ export declare function down(db: Kysely<any>): Promise<void>;
5
4
  //# sourceMappingURL=20241220T144630860Z-stats-materialized-views.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"20241220T144630860Z-stats-materialized-views.d.ts","sourceRoot":"","sources":["../../../src/db/migrations/20241220T144630860Z-stats-materialized-views.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAO,MAAM,QAAQ,CAAA;AAMpC,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAA;AAKvD,wBAAsB,EAAE,CAAC,EAAE,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,CAqMvD;AAED,wBAAsB,IAAI,CAAC,EAAE,EAAE,MAAM,CAAC,kBAAkB,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,CAKxE"}
1
+ {"version":3,"file":"20241220T144630860Z-stats-materialized-views.d.ts","sourceRoot":"","sources":["../../../src/db/migrations/20241220T144630860Z-stats-materialized-views.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAO,MAAM,QAAQ,CAAA;AAUpC,wBAAsB,EAAE,CAAC,EAAE,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,CAqMvD;AAED,wBAAsB,IAAI,CAAC,EAAE,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,CAKzD"}
@@ -1 +1 @@
1
- {"version":3,"file":"20241220T144630860Z-stats-materialized-views.js","sourceRoot":"","sources":["../../../src/db/migrations/20241220T144630860Z-stats-materialized-views.ts"],"names":[],"mappings":"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","sourcesContent":["import { Kysely, sql } from 'kysely'\nimport { REASONAPPEAL } from '../../lexicon/types/com/atproto/moderation/defs.js'\nimport {\n REVIEWESCALATED,\n REVIEWOPEN,\n} from '../../lexicon/types/tools/ozone/moderation/defs.js'\nimport { DatabaseSchemaType } from '../schema/index.js'\nimport * as modEvent from '../schema/moderation_event.js'\nimport * as modStatus from '../schema/moderation_subject_status.js'\nimport * as recordEventsStats from '../schema/record_events_stats.js'\n\nexport async function up(db: Kysely<any>): Promise<void> {\n // Used by \"tools.ozone.moderation.queryStatuses\". Reduces query cost by two\n // order of magnitudes when sorting using \"reportedRecordsCount\" or\n // \"takendownRecordsCount\" and filtering by \"reviewState\".\n await db.schema\n .createIndex('moderation_subject_status_did_id_review_state_idx')\n .on('moderation_subject_status')\n .column('did')\n .expression(sql`\"id\" ASC NULLS FIRST`)\n .column('reviewState')\n .execute()\n\n // ~6sec for 16M events\n await db.schema\n .createView('account_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modEvent.PartialDB>)\n .selectFrom('moderation_event')\n .where('subjectType', '=', 'com.atproto.admin.defs#repoRef')\n .where('subjectUri', 'is', null)\n .select('subjectDid')\n .select([\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventTakedown'\n AND ${eb.ref('durationInHours')} IS NULL\n )`.as('takedownCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventTakedown'\n AND ${eb.ref('durationInHours')} IS NOT NULL\n )`.as('suspendCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventEscalate'\n )`.as('escalateCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport'\n AND ${eb.ref('meta')} ->> 'reportType' != ${REASONAPPEAL}\n )`.as('reportCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport'\n AND ${eb.ref('meta')} ->> 'reportType' = ${REASONAPPEAL}\n )`.as('appealCount'),\n ])\n .groupBy('subjectDid'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_events_stats_did_idx')\n .unique()\n .on('account_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createIndex('account_events_stats_suspend_count_idx')\n .on('account_events_stats')\n .expression(sql`\"suspendCount\" ASC NULLS FIRST`)\n .column('subjectDid')\n .execute()\n\n // ~50sec for 16M events\n await db.schema\n .createView('record_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modEvent.PartialDB>)\n .selectFrom('moderation_event')\n .select([\n 'subjectDid',\n 'subjectUri',\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventEscalate')`.as(\n 'escalateCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport' AND ${eb.ref('meta')} ->> 'reportType' != 'com.atproto.moderation.defs#reasonAppeal')`.as(\n 'reportCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport' AND ${eb.ref('meta')} ->> 'reportType' = 'com.atproto.moderation.defs#reasonAppeal')`.as(\n 'appealCount',\n ),\n ])\n .where('subjectType', '=', 'com.atproto.repo.strongRef')\n .where('subjectUri', 'is not', null)\n .groupBy(['subjectDid', 'subjectUri']),\n )\n .execute()\n\n await db.schema\n .createIndex('record_events_stats_uri_idx')\n .unique()\n .on('record_events_stats')\n .column('subjectUri')\n .execute()\n\n await db.schema\n .createIndex('record_events_stats_did_idx')\n .on('record_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createView('account_record_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<recordEventsStats.PartialDB>)\n .selectFrom('record_events_stats')\n .select([\n 'subjectDid',\n (eb) =>\n // Casting to \"bigint\" because \"numeric\" gets casted to a string\n // by default by postgres-node.\n sql<number>`SUM(${eb.ref('reportCount')})::bigint`.as(\n 'totalReports',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reportCount')} > 0)`.as(\n 'reportedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('escalateCount')} > 0)`.as(\n 'escalatedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('appealCount')} > 0)`.as(\n 'appealedCount',\n ),\n ])\n .groupBy('subjectDid'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_record_events_stats_did_idx')\n .unique()\n .on('account_record_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createIndex('account_record_events_stats_reported_count_idx')\n .on('account_record_events_stats')\n .expression(sql`\"reportedCount\" ASC NULLS FIRST`)\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createView('account_record_status_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modStatus.PartialDB>)\n .selectFrom('moderation_subject_status')\n .select('did')\n .select([\n sql<number>`COUNT(*)`.as('subjectCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reviewState')} IN (${REVIEWOPEN}, ${REVIEWESCALATED}))`.as(\n 'pendingCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reviewState')} NOT IN (${REVIEWOPEN}, ${REVIEWESCALATED}))`.as(\n 'processedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('takendown')})`.as(\n 'takendownCount',\n ),\n ])\n .where('recordPath', '!=', '')\n .groupBy('did'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_record_status_stats_did_idx')\n .unique()\n .on('account_record_status_stats')\n .column('did')\n .execute()\n\n await db.schema\n .createIndex('account_record_status_stats_takendown_count_idx')\n .on('account_record_status_stats')\n .expression(sql`\"takendownCount\" ASC NULLS FIRST`)\n .column('did')\n .execute()\n}\n\nexport async function down(db: Kysely<DatabaseSchemaType>): Promise<void> {\n db.schema.dropView('account_record_status_stats').materialized().execute()\n db.schema.dropView('account_record_events_stats').materialized().execute()\n db.schema.dropView('record_events_stats').materialized().execute()\n db.schema.dropView('account_events_stats').materialized().execute()\n}\n"]}
1
+ {"version":3,"file":"20241220T144630860Z-stats-materialized-views.js","sourceRoot":"","sources":["../../../src/db/migrations/20241220T144630860Z-stats-materialized-views.ts"],"names":[],"mappings":"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","sourcesContent":["import { Kysely, sql } from 'kysely'\nimport { REASONAPPEAL } from '../../lexicon/types/com/atproto/moderation/defs.js'\nimport {\n REVIEWESCALATED,\n REVIEWOPEN,\n} from '../../lexicon/types/tools/ozone/moderation/defs.js'\nimport * as modEvent from '../schema/moderation_event.js'\nimport * as modStatus from '../schema/moderation_subject_status.js'\nimport * as recordEventsStats from '../schema/record_events_stats.js'\n\nexport async function up(db: Kysely<any>): Promise<void> {\n // Used by \"tools.ozone.moderation.queryStatuses\". Reduces query cost by two\n // order of magnitudes when sorting using \"reportedRecordsCount\" or\n // \"takendownRecordsCount\" and filtering by \"reviewState\".\n await db.schema\n .createIndex('moderation_subject_status_did_id_review_state_idx')\n .on('moderation_subject_status')\n .column('did')\n .expression(sql`\"id\" ASC NULLS FIRST`)\n .column('reviewState')\n .execute()\n\n // ~6sec for 16M events\n await db.schema\n .createView('account_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modEvent.PartialDB>)\n .selectFrom('moderation_event')\n .where('subjectType', '=', 'com.atproto.admin.defs#repoRef')\n .where('subjectUri', 'is', null)\n .select('subjectDid')\n .select([\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventTakedown'\n AND ${eb.ref('durationInHours')} IS NULL\n )`.as('takedownCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventTakedown'\n AND ${eb.ref('durationInHours')} IS NOT NULL\n )`.as('suspendCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventEscalate'\n )`.as('escalateCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport'\n AND ${eb.ref('meta')} ->> 'reportType' != ${REASONAPPEAL}\n )`.as('reportCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER(\n WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport'\n AND ${eb.ref('meta')} ->> 'reportType' = ${REASONAPPEAL}\n )`.as('appealCount'),\n ])\n .groupBy('subjectDid'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_events_stats_did_idx')\n .unique()\n .on('account_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createIndex('account_events_stats_suspend_count_idx')\n .on('account_events_stats')\n .expression(sql`\"suspendCount\" ASC NULLS FIRST`)\n .column('subjectDid')\n .execute()\n\n // ~50sec for 16M events\n await db.schema\n .createView('record_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modEvent.PartialDB>)\n .selectFrom('moderation_event')\n .select([\n 'subjectDid',\n 'subjectUri',\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventEscalate')`.as(\n 'escalateCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport' AND ${eb.ref('meta')} ->> 'reportType' != 'com.atproto.moderation.defs#reasonAppeal')`.as(\n 'reportCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('action')} = 'tools.ozone.moderation.defs#modEventReport' AND ${eb.ref('meta')} ->> 'reportType' = 'com.atproto.moderation.defs#reasonAppeal')`.as(\n 'appealCount',\n ),\n ])\n .where('subjectType', '=', 'com.atproto.repo.strongRef')\n .where('subjectUri', 'is not', null)\n .groupBy(['subjectDid', 'subjectUri']),\n )\n .execute()\n\n await db.schema\n .createIndex('record_events_stats_uri_idx')\n .unique()\n .on('record_events_stats')\n .column('subjectUri')\n .execute()\n\n await db.schema\n .createIndex('record_events_stats_did_idx')\n .on('record_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createView('account_record_events_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<recordEventsStats.PartialDB>)\n .selectFrom('record_events_stats')\n .select([\n 'subjectDid',\n (eb) =>\n // Casting to \"bigint\" because \"numeric\" gets casted to a string\n // by default by postgres-node.\n sql<number>`SUM(${eb.ref('reportCount')})::bigint`.as(\n 'totalReports',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reportCount')} > 0)`.as(\n 'reportedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('escalateCount')} > 0)`.as(\n 'escalatedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('appealCount')} > 0)`.as(\n 'appealedCount',\n ),\n ])\n .groupBy('subjectDid'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_record_events_stats_did_idx')\n .unique()\n .on('account_record_events_stats')\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createIndex('account_record_events_stats_reported_count_idx')\n .on('account_record_events_stats')\n .expression(sql`\"reportedCount\" ASC NULLS FIRST`)\n .column('subjectDid')\n .execute()\n\n await db.schema\n .createView('account_record_status_stats')\n .materialized()\n .ifNotExists()\n .as(\n (db as Kysely<modStatus.PartialDB>)\n .selectFrom('moderation_subject_status')\n .select('did')\n .select([\n sql<number>`COUNT(*)`.as('subjectCount'),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reviewState')} IN (${REVIEWOPEN}, ${REVIEWESCALATED}))`.as(\n 'pendingCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('reviewState')} NOT IN (${REVIEWOPEN}, ${REVIEWESCALATED}))`.as(\n 'processedCount',\n ),\n (eb) =>\n sql<number>`COUNT(*) FILTER (WHERE ${eb.ref('takendown')})`.as(\n 'takendownCount',\n ),\n ])\n .where('recordPath', '!=', '')\n .groupBy('did'),\n )\n .execute()\n\n await db.schema\n .createIndex('account_record_status_stats_did_idx')\n .unique()\n .on('account_record_status_stats')\n .column('did')\n .execute()\n\n await db.schema\n .createIndex('account_record_status_stats_takendown_count_idx')\n .on('account_record_status_stats')\n .expression(sql`\"takendownCount\" ASC NULLS FIRST`)\n .column('did')\n .execute()\n}\n\nexport async function down(db: Kysely<any>): Promise<void> {\n db.schema.dropView('account_record_status_stats').materialized().execute()\n db.schema.dropView('account_record_events_stats').materialized().execute()\n db.schema.dropView('record_events_stats').materialized().execute()\n db.schema.dropView('account_events_stats').materialized().execute()\n}\n"]}
@@ -1,5 +1,4 @@
1
1
  import { Kysely } from 'kysely';
2
- import { DatabaseSchemaType } from '../schema/index.js';
3
2
  export declare function up(db: Kysely<any>): Promise<void>;
4
- export declare function down(db: Kysely<DatabaseSchemaType>): Promise<void>;
3
+ export declare function down(db: Kysely<any>): Promise<void>;
5
4
  //# sourceMappingURL=20250718T150931000Z-update-appeal-reason-stats.d.ts.map