@atproto/ozone 0.2.8 → 0.2.11
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/CHANGELOG.md +33 -0
- package/package.json +25 -21
- package/bin/migration-create.ts +0 -38
- package/jest.config.cjs +0 -22
- package/src/api/chat/getActorMetadata.ts +0 -23
- package/src/api/chat/getConvo.ts +0 -23
- package/src/api/chat/getConvoMembers.ts +0 -23
- package/src/api/chat/getConvos.ts +0 -23
- package/src/api/chat/getMessageContext.ts +0 -42
- package/src/api/chat/index.ts +0 -16
- package/src/api/communication/createTemplate.ts +0 -51
- package/src/api/communication/deleteTemplate.ts +0 -23
- package/src/api/communication/listTemplates.ts +0 -31
- package/src/api/communication/updateTemplate.ts +0 -51
- package/src/api/health.ts +0 -27
- package/src/api/index.ts +0 -146
- package/src/api/label/fetchLabels.ts +0 -32
- package/src/api/label/queryLabels.ts +0 -57
- package/src/api/label/subscribeLabels.ts +0 -25
- package/src/api/moderation/cancelScheduledActions.ts +0 -72
- package/src/api/moderation/emitEvent.ts +0 -475
- package/src/api/moderation/getAccountTimeline.ts +0 -160
- package/src/api/moderation/getEvent.ts +0 -19
- package/src/api/moderation/getRecord.ts +0 -40
- package/src/api/moderation/getRecords.ts +0 -50
- package/src/api/moderation/getRepo.ts +0 -34
- package/src/api/moderation/getReporterStats.ts +0 -18
- package/src/api/moderation/getRepos.ts +0 -41
- package/src/api/moderation/getSubjects.ts +0 -101
- package/src/api/moderation/listScheduledActions.ts +0 -45
- package/src/api/moderation/queryEvents.ts +0 -72
- package/src/api/moderation/queryStatuses.ts +0 -23
- package/src/api/moderation/scheduleAction.ts +0 -129
- package/src/api/moderation/searchRepos.ts +0 -46
- package/src/api/moderation/util.ts +0 -96
- package/src/api/proxied.ts +0 -327
- package/src/api/queue/assignModerator.ts +0 -31
- package/src/api/queue/createQueue.ts +0 -62
- package/src/api/queue/deleteQueue.ts +0 -56
- package/src/api/queue/getAssignments.ts +0 -19
- package/src/api/queue/listQueues.ts +0 -39
- package/src/api/queue/routeReports.ts +0 -44
- package/src/api/queue/unassignModerator.ts +0 -26
- package/src/api/queue/updateQueue.ts +0 -54
- package/src/api/report/assignModerator.ts +0 -36
- package/src/api/report/createActivity.ts +0 -57
- package/src/api/report/createReport.ts +0 -93
- package/src/api/report/getAssignments.ts +0 -20
- package/src/api/report/getHistoricalStats.ts +0 -41
- package/src/api/report/getLatestReport.ts +0 -44
- package/src/api/report/getLiveStats.ts +0 -26
- package/src/api/report/getReport.ts +0 -55
- package/src/api/report/listActivities.ts +0 -37
- package/src/api/report/queryActivities.ts +0 -64
- package/src/api/report/queryReports.ts +0 -44
- package/src/api/report/reassignQueue.ts +0 -68
- package/src/api/report/refreshStats.ts +0 -27
- package/src/api/report/unassignModerator.ts +0 -21
- package/src/api/safelink/addRule.ts +0 -48
- package/src/api/safelink/queryEvents.ts +0 -32
- package/src/api/safelink/queryRules.ts +0 -58
- package/src/api/safelink/removeRule.ts +0 -42
- package/src/api/safelink/updateRule.ts +0 -48
- package/src/api/server/getConfig.ts +0 -35
- package/src/api/set/addValues.ts +0 -28
- package/src/api/set/deleteSet.ts +0 -34
- package/src/api/set/deleteValues.ts +0 -31
- package/src/api/set/getValues.ts +0 -42
- package/src/api/set/querySets.ts +0 -36
- package/src/api/set/upsertSet.ts +0 -38
- package/src/api/setting/listOptions.ts +0 -44
- package/src/api/setting/removeOptions.ts +0 -64
- package/src/api/setting/upsertOption.ts +0 -156
- package/src/api/team/addMember.ts +0 -51
- package/src/api/team/deleteMember.ts +0 -29
- package/src/api/team/listMembers.ts +0 -20
- package/src/api/team/updateMember.ts +0 -47
- package/src/api/util.ts +0 -265
- package/src/api/verification/grantVerifications.ts +0 -90
- package/src/api/verification/listVerifications.ts +0 -44
- package/src/api/verification/revokeVerifications.ts +0 -43
- package/src/api/well-known.ts +0 -46
- package/src/assignment/index.ts +0 -728
- package/src/auth-verifier.ts +0 -227
- package/src/background.ts +0 -183
- package/src/communication-service/template.ts +0 -110
- package/src/communication-service/util.ts +0 -8
- package/src/config/config.ts +0 -211
- package/src/config/env.ts +0 -95
- package/src/config/index.ts +0 -3
- package/src/config/secrets.ts +0 -17
- package/src/context.ts +0 -399
- package/src/daemon/blob-diverter.ts +0 -186
- package/src/daemon/context.ts +0 -247
- package/src/daemon/event-pusher.ts +0 -363
- package/src/daemon/event-reverser.ts +0 -128
- package/src/daemon/index.ts +0 -33
- package/src/daemon/job-cursor.ts +0 -33
- package/src/daemon/materialized-view-refresher.ts +0 -33
- package/src/daemon/queue-router.ts +0 -101
- package/src/daemon/scheduled-action-processor.ts +0 -304
- package/src/daemon/stats-computer.ts +0 -101
- package/src/daemon/strike-expiry-processor.ts +0 -95
- package/src/daemon/team-profile-synchronizer.ts +0 -15
- package/src/daemon/verification-listener.ts +0 -169
- package/src/db/index.ts +0 -203
- package/src/db/migrations/20231219T205730722Z-init.ts +0 -170
- package/src/db/migrations/20240116T085607200Z-communication-template.ts +0 -23
- package/src/db/migrations/20240201T051104136Z-mod-event-blobs.ts +0 -15
- package/src/db/migrations/20240208T213404429Z-add-tags-column-to-moderation-subject.ts +0 -31
- package/src/db/migrations/20240228T003647759Z-add-label-sigs.ts +0 -25
- package/src/db/migrations/20240408T192432676Z-mute-reporting.ts +0 -15
- package/src/db/migrations/20240506T225055595Z-message-subject.ts +0 -21
- package/src/db/migrations/20240521T211332580Z-member.ts +0 -17
- package/src/db/migrations/20240814T003647759Z-event-created-at-index.ts +0 -13
- package/src/db/migrations/20240903T205730722Z-add-template-lang.ts +0 -12
- package/src/db/migrations/20240904T205730722Z-add-subject-did-index.ts +0 -13
- package/src/db/migrations/20241001T205730722Z-subject-status-review-state-index.ts +0 -15
- package/src/db/migrations/20241008T205730722Z-sets.ts +0 -53
- package/src/db/migrations/20241018T205730722Z-setting.ts +0 -27
- package/src/db/migrations/20241026T205730722Z-add-hosting-status-to-subject-status.ts +0 -57
- package/src/db/migrations/20241220T144630860Z-stats-materialized-views.ts +0 -215
- package/src/db/migrations/20250204T003647759Z-add-subject-priority-score.ts +0 -22
- package/src/db/migrations/20250211T003647759Z-add-reporter-stats-index.ts +0 -38
- package/src/db/migrations/20250211T132135150Z-moderation-event-message-partial-idx.ts +0 -26
- package/src/db/migrations/20250221T132135150Z-member-details.ts +0 -14
- package/src/db/migrations/20250404T201720309Z-subject-status-sort-idxs.ts +0 -18
- package/src/db/migrations/20250415T201720309Z-verification.ts +0 -34
- package/src/db/migrations/20250417T201720309Z-firehose-cursor.ts +0 -16
- package/src/db/migrations/20250609T110704000Z-safelink.ts +0 -53
- package/src/db/migrations/20250618T180246000Z-add-mod-tool-to-moderation-event.ts +0 -18
- package/src/db/migrations/20250701T000000000Z-add-age-assurance-state.ts +0 -25
- package/src/db/migrations/20250715T000000000Z-add-mod-event-external-id.ts +0 -15
- package/src/db/migrations/20250718T150931000Z-update-appeal-reason-stats.ts +0 -310
- package/src/db/migrations/20250813T000000000Z-mod-tool-batch-id-index.ts +0 -14
- package/src/db/migrations/20250923T000000000Z-scheduled-actions.ts +0 -56
- package/src/db/migrations/20251008T120000000Z-add-strike-system.ts +0 -87
- package/src/db/migrations/20260210T154806448Z-mod-event-created-by-indexes.ts +0 -22
- package/src/db/migrations/20260219T164523000Z-create-report-table.ts +0 -155
- package/src/db/migrations/20260219T165302248Z-moderator-assignment.ts +0 -42
- package/src/db/migrations/20260225T000000000Z-add-report-queue-table.ts +0 -41
- package/src/db/migrations/20260313T000000000Z-add-report-activity-table.ts +0 -48
- package/src/db/migrations/20260318T152058935Z-add-report-stat.ts +0 -35
- package/src/db/migrations/20260428T000000000Z-add-expiring-tag-table.ts +0 -32
- package/src/db/migrations/20260513T202941104Z-add-subject-convo-id.ts +0 -114
- package/src/db/migrations/20260602T120000000Z-add-report-activity-created-index.ts +0 -17
- package/src/db/migrations/index.ts +0 -44
- package/src/db/migrations/provider.ts +0 -26
- package/src/db/pagination.ts +0 -335
- package/src/db/schema/account_events_stats.ts +0 -16
- package/src/db/schema/account_record_events_stats.ts +0 -15
- package/src/db/schema/account_record_status_stats.ts +0 -15
- package/src/db/schema/account_strike.ts +0 -13
- package/src/db/schema/blob_push_event.ts +0 -21
- package/src/db/schema/communication_template.ts +0 -19
- package/src/db/schema/expiring_tag.ts +0 -18
- package/src/db/schema/firehose_cursor.ts +0 -13
- package/src/db/schema/index.ts +0 -60
- package/src/db/schema/job_cursor.ts +0 -13
- package/src/db/schema/label.ts +0 -22
- package/src/db/schema/member.ts +0 -22
- package/src/db/schema/moderation_event.ts +0 -61
- package/src/db/schema/moderation_subject_status.ts +0 -52
- package/src/db/schema/moderator_assignment.ts +0 -16
- package/src/db/schema/ozone_set.ts +0 -24
- package/src/db/schema/record_events_stats.ts +0 -15
- package/src/db/schema/record_push_event.ts +0 -21
- package/src/db/schema/repo_push_event.ts +0 -19
- package/src/db/schema/report.ts +0 -28
- package/src/db/schema/report_activity.ts +0 -22
- package/src/db/schema/report_queue.ts +0 -21
- package/src/db/schema/report_stat.ts +0 -27
- package/src/db/schema/safelink.ts +0 -39
- package/src/db/schema/scheduled-action.ts +0 -25
- package/src/db/schema/setting.ts +0 -24
- package/src/db/schema/signing_key.ts +0 -10
- package/src/db/schema/verification.ts +0 -21
- package/src/db/types.ts +0 -24
- package/src/error.ts +0 -12
- package/src/image-invalidator.ts +0 -7
- package/src/index.ts +0 -154
- package/src/jetstream/service.ts +0 -107
- package/src/logger.ts +0 -29
- package/src/mod-service/expiring-tags.ts +0 -104
- package/src/mod-service/index.ts +0 -1842
- package/src/mod-service/profile.ts +0 -139
- package/src/mod-service/report.ts +0 -429
- package/src/mod-service/status.ts +0 -549
- package/src/mod-service/strike.ts +0 -96
- package/src/mod-service/subject.ts +0 -311
- package/src/mod-service/types.ts +0 -96
- package/src/mod-service/util.ts +0 -99
- package/src/mod-service/views.ts +0 -912
- package/src/queue/service.ts +0 -603
- package/src/report/activity.ts +0 -281
- package/src/report/handle-report-update.ts +0 -209
- package/src/report/reassign.ts +0 -109
- package/src/report/stats.ts +0 -852
- package/src/report/views.ts +0 -239
- package/src/safelink/service.ts +0 -304
- package/src/scheduled-action/service.ts +0 -281
- package/src/scheduled-action/types.ts +0 -17
- package/src/sequencer/index.ts +0 -2
- package/src/sequencer/outbox.ts +0 -123
- package/src/sequencer/sequencer.ts +0 -147
- package/src/set/service.ts +0 -230
- package/src/setting/constants.ts +0 -3
- package/src/setting/service.ts +0 -148
- package/src/setting/types.ts +0 -3
- package/src/setting/validators.ts +0 -333
- package/src/tag-service/content-tagger.ts +0 -30
- package/src/tag-service/embed-tagger.ts +0 -70
- package/src/tag-service/index.ts +0 -70
- package/src/tag-service/language-data.ts +0 -561
- package/src/tag-service/language-tagger.ts +0 -101
- package/src/tag-service/util.ts +0 -13
- package/src/team/index.ts +0 -296
- package/src/util.ts +0 -230
- package/src/verification/issuer.ts +0 -146
- package/src/verification/service.ts +0 -208
- package/src/verification/util.ts +0 -53
- package/test.env +0 -2
- package/tests/3p-labeler.test.ts +0 -288
- package/tests/__snapshots__/account-strikes.test.ts.snap +0 -159
- package/tests/__snapshots__/age-assurance.test.ts.snap +0 -66
- package/tests/__snapshots__/blob-divert.test.ts.snap +0 -219
- package/tests/__snapshots__/get-account-timeline.test.ts.snap +0 -36
- package/tests/__snapshots__/get-record.test.ts.snap +0 -271
- package/tests/__snapshots__/get-records.test.ts.snap +0 -175
- package/tests/__snapshots__/get-repo.test.ts.snap +0 -91
- package/tests/__snapshots__/get-repos.test.ts.snap +0 -127
- package/tests/__snapshots__/get-starter-pack.test.ts.snap +0 -535
- package/tests/__snapshots__/get-subjects.test.ts.snap +0 -529
- package/tests/__snapshots__/moderation-events.test.ts.snap +0 -347
- package/tests/__snapshots__/moderation-statuses.test.ts.snap +0 -276
- package/tests/__snapshots__/moderation.test.ts.snap +0 -85
- package/tests/__snapshots__/report-reason.test.ts.snap +0 -14
- package/tests/__snapshots__/safelink.test.ts.snap +0 -179
- package/tests/__snapshots__/scheduled-action.test.ts.snap +0 -61
- package/tests/__snapshots__/sets.test.ts.snap +0 -46
- package/tests/__snapshots__/settings.test.ts.snap +0 -52
- package/tests/__snapshots__/team.test.ts.snap +0 -374
- package/tests/__snapshots__/verification-listener.test.ts.snap +0 -152
- package/tests/__snapshots__/verification.test.ts.snap +0 -302
- package/tests/_util.ts +0 -242
- package/tests/account-strikes.test.ts +0 -184
- package/tests/ack-all-subjects-of-account.test.ts +0 -177
- package/tests/age-assurance.test.ts +0 -372
- package/tests/blob-divert.test.ts +0 -106
- package/tests/communication-templates.test.ts +0 -149
- package/tests/content-tagger.test.ts +0 -170
- package/tests/db.test.ts +0 -184
- package/tests/expiring-label.test.ts +0 -72
- package/tests/expiring-tags.test.ts +0 -232
- package/tests/get-account-timeline.test.ts +0 -85
- package/tests/get-config.test.ts +0 -55
- package/tests/get-lists.test.ts +0 -111
- package/tests/get-profiles.test.ts +0 -70
- package/tests/get-record.test.ts +0 -130
- package/tests/get-records.test.ts +0 -91
- package/tests/get-repo.test.ts +0 -171
- package/tests/get-report.test.ts +0 -136
- package/tests/get-reporter-stats.test.ts +0 -132
- package/tests/get-repos.test.ts +0 -91
- package/tests/get-starter-pack.test.ts +0 -115
- package/tests/get-subjects.test.ts +0 -81
- package/tests/mod-tool.test.ts +0 -268
- package/tests/moderation-appeals.test.ts +0 -260
- package/tests/moderation-events.test.ts +0 -756
- package/tests/moderation-status-tags.test.ts +0 -140
- package/tests/moderation-statuses.test.ts +0 -495
- package/tests/moderation.test.ts +0 -992
- package/tests/protected-tags.test.ts +0 -218
- package/tests/query-labels.test.ts +0 -238
- package/tests/query-reports.test.ts +0 -608
- package/tests/queue-assignment.test.ts +0 -428
- package/tests/queue-router.test.ts +0 -306
- package/tests/queues.test.ts +0 -690
- package/tests/record-and-account-events.test.ts +0 -197
- package/tests/repo-search.test.ts +0 -136
- package/tests/report-action.test.ts +0 -308
- package/tests/report-activity.test.ts +0 -711
- package/tests/report-assignment.test.ts +0 -517
- package/tests/report-muting.test.ts +0 -100
- package/tests/report-reason.test.ts +0 -154
- package/tests/report-reassign-queue.test.ts +0 -340
- package/tests/report-routing.test.ts +0 -245
- package/tests/report-stats.test.ts +0 -545
- package/tests/revoke-account-credentials.test.ts +0 -54
- package/tests/safelink.test.ts +0 -534
- package/tests/scheduled-action-processor.test.ts +0 -488
- package/tests/scheduled-action.test.ts +0 -334
- package/tests/sequencer.test.ts +0 -227
- package/tests/server.test.ts +0 -62
- package/tests/sets.test.ts +0 -246
- package/tests/settings.test.ts +0 -308
- package/tests/strike-expiry-processor.test.ts +0 -299
- package/tests/subject-priority-score.test.ts +0 -96
- package/tests/takedown.test.ts +0 -105
- package/tests/team.test.ts +0 -216
- package/tests/verification-listener.test.ts +0 -129
- package/tests/verification.test.ts +0 -186
- package/tsconfig.build.json +0 -9
- package/tsconfig.build.tsbuildinfo +0 -1
- package/tsconfig.json +0 -7
- package/tsconfig.tests.json +0 -8
package/src/assignment/index.ts
DELETED
|
@@ -1,728 +0,0 @@
|
|
|
1
|
-
import { Selectable } from 'kysely'
|
|
2
|
-
import { ToolsOzoneQueueDefs } from '@atproto/api'
|
|
3
|
-
import { InvalidRequestError } from '@atproto/xrpc-server'
|
|
4
|
-
import { Database } from '../db/index.js'
|
|
5
|
-
import { EndAtIdKeyset, paginate } from '../db/pagination.js'
|
|
6
|
-
import { ModeratorAssignment } from '../db/schema/moderator_assignment.js'
|
|
7
|
-
import { ReportQueue } from '../db/schema/report_queue.js'
|
|
8
|
-
import type * as ToolsOzoneReportDefs from '../lexicon/types/tools/ozone/report/defs.js'
|
|
9
|
-
import type { Member as TeamMember } from '../lexicon/types/tools/ozone/team/defs.js'
|
|
10
|
-
import { QueueService, QueueServiceCreator } from '../queue/service.js'
|
|
11
|
-
import { createReportActivity } from '../report/activity.js'
|
|
12
|
-
import { TeamService, TeamServiceCreator } from '../team/index.js'
|
|
13
|
-
|
|
14
|
-
export interface AssignmentServiceOpts {
|
|
15
|
-
queueDurationMs: number
|
|
16
|
-
reportDurationMs: number
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
// Queue
|
|
20
|
-
export interface GetQueueAssignmentsInput {
|
|
21
|
-
onlyActive?: boolean
|
|
22
|
-
queueIds?: number[]
|
|
23
|
-
dids?: string[]
|
|
24
|
-
limit?: number
|
|
25
|
-
cursor?: string
|
|
26
|
-
}
|
|
27
|
-
export interface AssignQueueInput {
|
|
28
|
-
did: string
|
|
29
|
-
queueId: number
|
|
30
|
-
}
|
|
31
|
-
export interface UnassignQueueInput {
|
|
32
|
-
did: string
|
|
33
|
-
queueId: number
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
// Report
|
|
37
|
-
export interface GetReportAssignmentsInput {
|
|
38
|
-
onlyActive?: boolean
|
|
39
|
-
reportIds?: number[]
|
|
40
|
-
queueIds?: number[]
|
|
41
|
-
dids?: string[]
|
|
42
|
-
limit?: number
|
|
43
|
-
cursor?: string
|
|
44
|
-
}
|
|
45
|
-
export interface AssignReportInput {
|
|
46
|
-
did: string
|
|
47
|
-
reportId: number
|
|
48
|
-
queueId?: number | null
|
|
49
|
-
isPermanent?: boolean
|
|
50
|
-
createdBy?: string
|
|
51
|
-
}
|
|
52
|
-
export interface UnassignReportInput {
|
|
53
|
-
reportId: number
|
|
54
|
-
createdBy?: string
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
type AssignmentRowWithQueue = Selectable<ModeratorAssignment> & {
|
|
58
|
-
queueName: string | null
|
|
59
|
-
queueSubjectTypes: string[] | null
|
|
60
|
-
queueCollection: string | null
|
|
61
|
-
queueDescription: string | null
|
|
62
|
-
queueReportTypes: string[] | null
|
|
63
|
-
queueCreatedBy: string | null
|
|
64
|
-
queueCreatedAt: string | null
|
|
65
|
-
queueUpdatedAt: string | null
|
|
66
|
-
queueEnabled: boolean | null
|
|
67
|
-
queueDeletedAt: string | null
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
export type AssignmentServiceCreator = (db: Database) => AssignmentService
|
|
71
|
-
|
|
72
|
-
export class AssignmentService {
|
|
73
|
-
constructor(
|
|
74
|
-
public db: Database,
|
|
75
|
-
public opts: AssignmentServiceOpts,
|
|
76
|
-
private queueService: QueueService,
|
|
77
|
-
private teamService: TeamService,
|
|
78
|
-
) {}
|
|
79
|
-
|
|
80
|
-
static creator(
|
|
81
|
-
opts: AssignmentServiceOpts,
|
|
82
|
-
queueServiceCreator: QueueServiceCreator,
|
|
83
|
-
teamServiceCreator: TeamServiceCreator,
|
|
84
|
-
): AssignmentServiceCreator {
|
|
85
|
-
return (db: Database) =>
|
|
86
|
-
new AssignmentService(
|
|
87
|
-
db,
|
|
88
|
-
opts,
|
|
89
|
-
queueServiceCreator(db),
|
|
90
|
-
teamServiceCreator(db),
|
|
91
|
-
)
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
private async fetchMemberViews(
|
|
95
|
-
dids: string[],
|
|
96
|
-
): Promise<Map<string, TeamMember>> {
|
|
97
|
-
return this.teamService.viewByDids(dids)
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
async getQueueAssignments(input: GetQueueAssignmentsInput): Promise<{
|
|
101
|
-
assignments: ToolsOzoneQueueDefs.AssignmentView[]
|
|
102
|
-
cursor?: string
|
|
103
|
-
}> {
|
|
104
|
-
const { onlyActive, queueIds, dids, limit, cursor } = input
|
|
105
|
-
const { ref } = this.db.db.dynamic
|
|
106
|
-
|
|
107
|
-
let query = this.db.db
|
|
108
|
-
.selectFrom('moderator_assignment')
|
|
109
|
-
.leftJoin(
|
|
110
|
-
'report_queue',
|
|
111
|
-
'report_queue.id',
|
|
112
|
-
'moderator_assignment.queueId',
|
|
113
|
-
)
|
|
114
|
-
.selectAll('moderator_assignment')
|
|
115
|
-
.select([
|
|
116
|
-
'report_queue.name as queueName',
|
|
117
|
-
'report_queue.subjectTypes as queueSubjectTypes',
|
|
118
|
-
'report_queue.collection as queueCollection',
|
|
119
|
-
'report_queue.description as queueDescription',
|
|
120
|
-
'report_queue.reportTypes as queueReportTypes',
|
|
121
|
-
'report_queue.createdBy as queueCreatedBy',
|
|
122
|
-
'report_queue.createdAt as queueCreatedAt',
|
|
123
|
-
'report_queue.updatedAt as queueUpdatedAt',
|
|
124
|
-
'report_queue.enabled as queueEnabled',
|
|
125
|
-
'report_queue.deletedAt as queueDeletedAt',
|
|
126
|
-
])
|
|
127
|
-
.where('reportId', 'is', null)
|
|
128
|
-
.where('queueId', 'is not', null)
|
|
129
|
-
|
|
130
|
-
if (onlyActive) {
|
|
131
|
-
const now = new Date().toISOString()
|
|
132
|
-
query = query.where((eb) =>
|
|
133
|
-
eb.or([eb('endAt', 'is', null), eb('endAt', '>', now)]),
|
|
134
|
-
)
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
if (queueIds?.length) {
|
|
138
|
-
query = query.where('queueId', 'in', queueIds)
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
if (dids?.length) {
|
|
142
|
-
query = query.where('did', 'in', dids)
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
// use endAt to take advantage of indexes
|
|
146
|
-
// Qualify column refs to avoid ambiguity with the report_queue join
|
|
147
|
-
const keyset = new EndAtIdKeyset(
|
|
148
|
-
ref('moderator_assignment.endAt'),
|
|
149
|
-
ref('moderator_assignment.id'),
|
|
150
|
-
)
|
|
151
|
-
const paginatedQuery = paginate(query, {
|
|
152
|
-
limit,
|
|
153
|
-
cursor,
|
|
154
|
-
keyset,
|
|
155
|
-
direction: 'desc',
|
|
156
|
-
tryIndex: true,
|
|
157
|
-
})
|
|
158
|
-
|
|
159
|
-
const results = await paginatedQuery.execute()
|
|
160
|
-
const memberViews = await this.fetchMemberViews(results.map((r) => r.did))
|
|
161
|
-
|
|
162
|
-
return {
|
|
163
|
-
assignments: results.map((row) =>
|
|
164
|
-
this.viewQueueAssignment(row, memberViews.get(row.did)),
|
|
165
|
-
),
|
|
166
|
-
cursor: keyset.packFromResult(results),
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
async getReportAssignments(input: GetReportAssignmentsInput): Promise<{
|
|
171
|
-
assignments: ToolsOzoneReportDefs.AssignmentView[]
|
|
172
|
-
cursor?: string
|
|
173
|
-
}> {
|
|
174
|
-
const { onlyActive, reportIds, queueIds, dids, limit, cursor } = input
|
|
175
|
-
const { ref } = this.db.db.dynamic
|
|
176
|
-
|
|
177
|
-
let query = this.db.db
|
|
178
|
-
.selectFrom('moderator_assignment')
|
|
179
|
-
.leftJoin(
|
|
180
|
-
'report_queue',
|
|
181
|
-
'report_queue.id',
|
|
182
|
-
'moderator_assignment.queueId',
|
|
183
|
-
)
|
|
184
|
-
.selectAll('moderator_assignment')
|
|
185
|
-
.select([
|
|
186
|
-
'report_queue.name as queueName',
|
|
187
|
-
'report_queue.subjectTypes as queueSubjectTypes',
|
|
188
|
-
'report_queue.collection as queueCollection',
|
|
189
|
-
'report_queue.description as queueDescription',
|
|
190
|
-
'report_queue.reportTypes as queueReportTypes',
|
|
191
|
-
'report_queue.createdBy as queueCreatedBy',
|
|
192
|
-
'report_queue.createdAt as queueCreatedAt',
|
|
193
|
-
'report_queue.updatedAt as queueUpdatedAt',
|
|
194
|
-
'report_queue.enabled as queueEnabled',
|
|
195
|
-
'report_queue.deletedAt as queueDeletedAt',
|
|
196
|
-
])
|
|
197
|
-
.where('reportId', 'is not', null)
|
|
198
|
-
|
|
199
|
-
if (onlyActive) {
|
|
200
|
-
const now = new Date().toISOString()
|
|
201
|
-
query = query.where((eb) =>
|
|
202
|
-
eb.or([eb('endAt', '>', now), eb('endAt', 'is', null)]),
|
|
203
|
-
)
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
if (reportIds?.length) {
|
|
207
|
-
query = query.where('reportId', 'in', reportIds)
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
if (queueIds?.length) {
|
|
211
|
-
query = query.where('queueId', 'in', queueIds)
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
if (dids?.length) {
|
|
215
|
-
query = query.where('did', 'in', dids)
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
// Qualify column refs to avoid ambiguity with the report_queue join
|
|
219
|
-
const keyset = new EndAtIdKeyset(
|
|
220
|
-
ref('moderator_assignment.endAt'),
|
|
221
|
-
ref('moderator_assignment.id'),
|
|
222
|
-
)
|
|
223
|
-
const paginatedQuery = paginate(query, {
|
|
224
|
-
limit,
|
|
225
|
-
cursor,
|
|
226
|
-
keyset,
|
|
227
|
-
direction: 'desc',
|
|
228
|
-
tryIndex: true,
|
|
229
|
-
})
|
|
230
|
-
|
|
231
|
-
const results = await paginatedQuery.execute()
|
|
232
|
-
const memberViews = await this.fetchMemberViews(results.map((r) => r.did))
|
|
233
|
-
|
|
234
|
-
return {
|
|
235
|
-
assignments: results.map((row) =>
|
|
236
|
-
this.viewReportAssignment(row, memberViews.get(row.did)),
|
|
237
|
-
),
|
|
238
|
-
cursor: keyset.packFromResult(results),
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
async assignQueue(
|
|
243
|
-
input: AssignQueueInput,
|
|
244
|
-
): Promise<ToolsOzoneQueueDefs.AssignmentView> {
|
|
245
|
-
const { did, queueId } = input
|
|
246
|
-
const now = new Date()
|
|
247
|
-
|
|
248
|
-
// Check queue since we aren't using foreign keys
|
|
249
|
-
const queue = await this.db.db
|
|
250
|
-
.selectFrom('report_queue')
|
|
251
|
-
.selectAll()
|
|
252
|
-
.where('id', '=', queueId)
|
|
253
|
-
.where('enabled', '=', true)
|
|
254
|
-
.where('deletedAt', 'is', null)
|
|
255
|
-
.executeTakeFirst()
|
|
256
|
-
if (!queue) {
|
|
257
|
-
throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
// Queue assignments are permanent (endAt = null). Upgrade any existing
|
|
261
|
-
// active expiry-based row to permanent; otherwise insert a new permanent row.
|
|
262
|
-
const result = await this.db.transaction(async (dbTxn) => {
|
|
263
|
-
const existing = await dbTxn.db
|
|
264
|
-
.selectFrom('moderator_assignment')
|
|
265
|
-
.selectAll()
|
|
266
|
-
.where('did', '=', did)
|
|
267
|
-
.where('queueId', '=', queueId)
|
|
268
|
-
.where('reportId', 'is', null)
|
|
269
|
-
.where((eb) =>
|
|
270
|
-
eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]),
|
|
271
|
-
)
|
|
272
|
-
.executeTakeFirst()
|
|
273
|
-
if (existing) {
|
|
274
|
-
if (existing.endAt === null) {
|
|
275
|
-
return existing
|
|
276
|
-
}
|
|
277
|
-
const updated = await dbTxn.db
|
|
278
|
-
.updateTable('moderator_assignment')
|
|
279
|
-
.set({ endAt: null })
|
|
280
|
-
.where('id', '=', existing.id)
|
|
281
|
-
.returningAll()
|
|
282
|
-
.executeTakeFirstOrThrow()
|
|
283
|
-
return updated
|
|
284
|
-
}
|
|
285
|
-
const created = await dbTxn.db
|
|
286
|
-
.insertInto('moderator_assignment')
|
|
287
|
-
.values({
|
|
288
|
-
did,
|
|
289
|
-
queueId,
|
|
290
|
-
startAt: now.toISOString(),
|
|
291
|
-
endAt: null,
|
|
292
|
-
})
|
|
293
|
-
.returningAll()
|
|
294
|
-
.executeTakeFirstOrThrow()
|
|
295
|
-
return created
|
|
296
|
-
})
|
|
297
|
-
|
|
298
|
-
if (result.queueId === null || result.reportId !== null) {
|
|
299
|
-
throw new Error('Failed to assign moderator to queue')
|
|
300
|
-
}
|
|
301
|
-
|
|
302
|
-
const row = await this.db.db
|
|
303
|
-
.selectFrom('moderator_assignment')
|
|
304
|
-
.leftJoin(
|
|
305
|
-
'report_queue',
|
|
306
|
-
'report_queue.id',
|
|
307
|
-
'moderator_assignment.queueId',
|
|
308
|
-
)
|
|
309
|
-
.selectAll('moderator_assignment')
|
|
310
|
-
.select([
|
|
311
|
-
'report_queue.name as queueName',
|
|
312
|
-
'report_queue.subjectTypes as queueSubjectTypes',
|
|
313
|
-
'report_queue.collection as queueCollection',
|
|
314
|
-
'report_queue.description as queueDescription',
|
|
315
|
-
'report_queue.reportTypes as queueReportTypes',
|
|
316
|
-
'report_queue.createdBy as queueCreatedBy',
|
|
317
|
-
'report_queue.createdAt as queueCreatedAt',
|
|
318
|
-
'report_queue.updatedAt as queueUpdatedAt',
|
|
319
|
-
'report_queue.enabled as queueEnabled',
|
|
320
|
-
'report_queue.deletedAt as queueDeletedAt',
|
|
321
|
-
])
|
|
322
|
-
.where('moderator_assignment.id', '=', result.id)
|
|
323
|
-
.executeTakeFirstOrThrow()
|
|
324
|
-
|
|
325
|
-
const memberViews = await this.fetchMemberViews([result.did])
|
|
326
|
-
return this.viewQueueAssignment(row, memberViews.get(result.did))
|
|
327
|
-
}
|
|
328
|
-
|
|
329
|
-
async unassignQueue(input: UnassignQueueInput): Promise<void> {
|
|
330
|
-
const { did, queueId } = input
|
|
331
|
-
const now = new Date()
|
|
332
|
-
|
|
333
|
-
const existing = await this.db.db
|
|
334
|
-
.selectFrom('moderator_assignment')
|
|
335
|
-
.selectAll()
|
|
336
|
-
.where('did', '=', did)
|
|
337
|
-
.where('queueId', '=', queueId)
|
|
338
|
-
.where('reportId', 'is', null)
|
|
339
|
-
.where((eb) =>
|
|
340
|
-
eb.or([eb('endAt', 'is', null), eb('endAt', '>', now.toISOString())]),
|
|
341
|
-
)
|
|
342
|
-
.executeTakeFirst()
|
|
343
|
-
|
|
344
|
-
if (!existing) {
|
|
345
|
-
throw new InvalidRequestError(
|
|
346
|
-
'No active assignment found',
|
|
347
|
-
'InvalidAssignment',
|
|
348
|
-
)
|
|
349
|
-
}
|
|
350
|
-
|
|
351
|
-
await this.db.db
|
|
352
|
-
.updateTable('moderator_assignment')
|
|
353
|
-
.set({ endAt: now.toISOString() })
|
|
354
|
-
.where('id', '=', existing.id)
|
|
355
|
-
.execute()
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
async assignReport(
|
|
359
|
-
input: AssignReportInput,
|
|
360
|
-
): Promise<ToolsOzoneReportDefs.AssignmentView> {
|
|
361
|
-
const { did, reportId, queueId, isPermanent = false } = input
|
|
362
|
-
const now = new Date()
|
|
363
|
-
|
|
364
|
-
// Check report and queue since we aren't using foreign keys
|
|
365
|
-
await this.checkReport(reportId)
|
|
366
|
-
if (queueId !== undefined && queueId !== null) {
|
|
367
|
-
await this.checkQueue(queueId)
|
|
368
|
-
}
|
|
369
|
-
|
|
370
|
-
// Make assignment
|
|
371
|
-
const result = await this.db.transaction(async (dbTxn) => {
|
|
372
|
-
if (isPermanent) {
|
|
373
|
-
// Check for an existing permanent assignment (endAt IS NULL)
|
|
374
|
-
const permanentExisting = await dbTxn.db
|
|
375
|
-
.selectFrom('moderator_assignment')
|
|
376
|
-
.selectAll()
|
|
377
|
-
.where('reportId', '=', reportId)
|
|
378
|
-
.where('endAt', 'is', null)
|
|
379
|
-
.executeTakeFirst()
|
|
380
|
-
|
|
381
|
-
let result: Selectable<ModeratorAssignment>
|
|
382
|
-
|
|
383
|
-
if (permanentExisting) {
|
|
384
|
-
if (permanentExisting.did !== did) {
|
|
385
|
-
throw new InvalidRequestError(
|
|
386
|
-
'Report already assigned',
|
|
387
|
-
'AlreadyAssigned',
|
|
388
|
-
)
|
|
389
|
-
}
|
|
390
|
-
// Same user — update queueId if provided
|
|
391
|
-
result = await dbTxn.db
|
|
392
|
-
.updateTable('moderator_assignment')
|
|
393
|
-
.set({ queueId: queueId ?? permanentExisting.queueId ?? null })
|
|
394
|
-
.where('id', '=', permanentExisting.id)
|
|
395
|
-
.returningAll()
|
|
396
|
-
.executeTakeFirstOrThrow()
|
|
397
|
-
} else {
|
|
398
|
-
// Upgrade an existing active (non-permanent) assignment to permanent
|
|
399
|
-
const activeExisting = await dbTxn.db
|
|
400
|
-
.selectFrom('moderator_assignment')
|
|
401
|
-
.selectAll()
|
|
402
|
-
.where('reportId', '=', reportId)
|
|
403
|
-
.where('endAt', '>', now.toISOString())
|
|
404
|
-
.executeTakeFirst()
|
|
405
|
-
|
|
406
|
-
if (activeExisting) {
|
|
407
|
-
result = await dbTxn.db
|
|
408
|
-
.updateTable('moderator_assignment')
|
|
409
|
-
.set({
|
|
410
|
-
did,
|
|
411
|
-
endAt: null,
|
|
412
|
-
queueId: queueId ?? activeExisting.queueId ?? null,
|
|
413
|
-
})
|
|
414
|
-
.where('id', '=', activeExisting.id)
|
|
415
|
-
.returningAll()
|
|
416
|
-
.executeTakeFirstOrThrow()
|
|
417
|
-
} else {
|
|
418
|
-
// Create new permanent assignment
|
|
419
|
-
result = await dbTxn.db
|
|
420
|
-
.insertInto('moderator_assignment')
|
|
421
|
-
.values({
|
|
422
|
-
did,
|
|
423
|
-
reportId,
|
|
424
|
-
queueId: queueId,
|
|
425
|
-
startAt: now.toISOString(),
|
|
426
|
-
endAt: null,
|
|
427
|
-
})
|
|
428
|
-
.returningAll()
|
|
429
|
-
.executeTakeFirstOrThrow()
|
|
430
|
-
}
|
|
431
|
-
}
|
|
432
|
-
|
|
433
|
-
// Sync denormalized assignment fields on report table
|
|
434
|
-
await dbTxn.db
|
|
435
|
-
.updateTable('report')
|
|
436
|
-
.set({ assignedTo: did, assignedAt: now.toISOString() })
|
|
437
|
-
.where('id', '=', reportId)
|
|
438
|
-
.execute()
|
|
439
|
-
|
|
440
|
-
return result
|
|
441
|
-
}
|
|
442
|
-
|
|
443
|
-
// Non-permanent: find any active or permanent assignment
|
|
444
|
-
const existing = await dbTxn.db
|
|
445
|
-
.selectFrom('moderator_assignment')
|
|
446
|
-
.selectAll()
|
|
447
|
-
.where('reportId', '=', reportId)
|
|
448
|
-
.where((eb) =>
|
|
449
|
-
eb.or([eb('endAt', '>', now.toISOString()), eb('endAt', 'is', null)]),
|
|
450
|
-
)
|
|
451
|
-
.executeTakeFirst()
|
|
452
|
-
|
|
453
|
-
if (existing) {
|
|
454
|
-
if (existing.did !== did) {
|
|
455
|
-
throw new InvalidRequestError(
|
|
456
|
-
'Report already assigned',
|
|
457
|
-
'AlreadyAssigned',
|
|
458
|
-
)
|
|
459
|
-
}
|
|
460
|
-
// Refresh the expiry unless the assignment is already permanent
|
|
461
|
-
const newEndAt =
|
|
462
|
-
existing.endAt === null
|
|
463
|
-
? null
|
|
464
|
-
: new Date(now.getTime() + this.opts.reportDurationMs).toISOString()
|
|
465
|
-
return dbTxn.db
|
|
466
|
-
.updateTable('moderator_assignment')
|
|
467
|
-
.set({
|
|
468
|
-
endAt: newEndAt,
|
|
469
|
-
queueId: queueId ?? existing.queueId ?? null,
|
|
470
|
-
})
|
|
471
|
-
.where('id', '=', existing.id)
|
|
472
|
-
.returningAll()
|
|
473
|
-
.executeTakeFirstOrThrow()
|
|
474
|
-
}
|
|
475
|
-
|
|
476
|
-
const endAt = new Date(
|
|
477
|
-
now.getTime() + this.opts.reportDurationMs,
|
|
478
|
-
).toISOString()
|
|
479
|
-
return dbTxn.db
|
|
480
|
-
.insertInto('moderator_assignment')
|
|
481
|
-
.values({
|
|
482
|
-
did,
|
|
483
|
-
reportId,
|
|
484
|
-
queueId: queueId,
|
|
485
|
-
startAt: now.toISOString(),
|
|
486
|
-
endAt,
|
|
487
|
-
})
|
|
488
|
-
.returningAll()
|
|
489
|
-
.executeTakeFirstOrThrow()
|
|
490
|
-
})
|
|
491
|
-
|
|
492
|
-
// Log an assignmentActivity ONLY for permanent assignments. Swallow AlreadyInTargetState
|
|
493
|
-
// so that re-assignments (e.g. refreshing expiry) don't throw.
|
|
494
|
-
if (input.isPermanent) {
|
|
495
|
-
try {
|
|
496
|
-
await createReportActivity(this.db, {
|
|
497
|
-
reportId,
|
|
498
|
-
activityType: 'assignmentActivity',
|
|
499
|
-
isAutomated: false,
|
|
500
|
-
createdBy: input.createdBy ?? did,
|
|
501
|
-
meta: { assignedTo: did },
|
|
502
|
-
})
|
|
503
|
-
} catch (err) {
|
|
504
|
-
if (
|
|
505
|
-
err instanceof InvalidRequestError &&
|
|
506
|
-
err.customErrorName === 'AlreadyInTargetState'
|
|
507
|
-
) {
|
|
508
|
-
// no-op — report already assigned, no state change to record
|
|
509
|
-
} else {
|
|
510
|
-
throw err
|
|
511
|
-
}
|
|
512
|
-
}
|
|
513
|
-
}
|
|
514
|
-
|
|
515
|
-
return this.hydrateReportAssignment(result.id)
|
|
516
|
-
}
|
|
517
|
-
|
|
518
|
-
async unassignReport(
|
|
519
|
-
input: UnassignReportInput,
|
|
520
|
-
): Promise<ToolsOzoneReportDefs.AssignmentView> {
|
|
521
|
-
const { reportId, createdBy } = input
|
|
522
|
-
const now = new Date()
|
|
523
|
-
|
|
524
|
-
await this.checkReport(reportId)
|
|
525
|
-
|
|
526
|
-
const { result, reportStatus } = await this.db.transaction(
|
|
527
|
-
async (dbTxn) => {
|
|
528
|
-
const existing = await dbTxn.db
|
|
529
|
-
.selectFrom('moderator_assignment')
|
|
530
|
-
.selectAll()
|
|
531
|
-
.where('reportId', '=', reportId)
|
|
532
|
-
.where((eb) =>
|
|
533
|
-
eb.or([
|
|
534
|
-
eb('endAt', '>', now.toISOString()),
|
|
535
|
-
eb('endAt', 'is', null),
|
|
536
|
-
]),
|
|
537
|
-
)
|
|
538
|
-
.executeTakeFirst()
|
|
539
|
-
|
|
540
|
-
if (!existing) {
|
|
541
|
-
throw new InvalidRequestError(
|
|
542
|
-
'Report is not assigned',
|
|
543
|
-
'InvalidAssignment',
|
|
544
|
-
)
|
|
545
|
-
}
|
|
546
|
-
|
|
547
|
-
const updated = await dbTxn.db
|
|
548
|
-
.updateTable('moderator_assignment')
|
|
549
|
-
.set({ endAt: now.toISOString() })
|
|
550
|
-
.where('id', '=', existing.id)
|
|
551
|
-
.returningAll()
|
|
552
|
-
.executeTakeFirstOrThrow()
|
|
553
|
-
|
|
554
|
-
// Capture status before any update so we can decide on the next status.
|
|
555
|
-
const reportRow = await dbTxn.db
|
|
556
|
-
.selectFrom('report')
|
|
557
|
-
.select('status')
|
|
558
|
-
.where('id', '=', reportId)
|
|
559
|
-
.forUpdate()
|
|
560
|
-
.executeTakeFirstOrThrow()
|
|
561
|
-
|
|
562
|
-
// If the report had moved to 'assigned' and the assignment was not tied
|
|
563
|
-
// to a queue, send it back to 'open' so it isn't stuck in 'assigned'
|
|
564
|
-
// after the moderator releases it. The 'queued' transition (when there
|
|
565
|
-
// is a queueId) is handled below via createReportActivity.
|
|
566
|
-
const updateSet: Record<string, string | null> = {
|
|
567
|
-
assignedTo: null,
|
|
568
|
-
assignedAt: null,
|
|
569
|
-
}
|
|
570
|
-
if (reportRow.status === 'assigned' && existing.queueId === null) {
|
|
571
|
-
updateSet.status = 'open'
|
|
572
|
-
updateSet.updatedAt = now.toISOString()
|
|
573
|
-
}
|
|
574
|
-
await dbTxn.db
|
|
575
|
-
.updateTable('report')
|
|
576
|
-
.set(updateSet)
|
|
577
|
-
.where('id', '=', reportId)
|
|
578
|
-
.execute()
|
|
579
|
-
|
|
580
|
-
return { result: updated, reportStatus: reportRow.status }
|
|
581
|
-
},
|
|
582
|
-
)
|
|
583
|
-
|
|
584
|
-
// If unassigning from a queued report (status moved to 'assigned' on
|
|
585
|
-
// permanent assign) before any other status change, send it back to
|
|
586
|
-
// 'queued' so other moderators can pick it up.
|
|
587
|
-
if (reportStatus === 'assigned' && result.queueId !== null) {
|
|
588
|
-
try {
|
|
589
|
-
await createReportActivity(this.db, {
|
|
590
|
-
reportId,
|
|
591
|
-
activityType: 'queueActivity',
|
|
592
|
-
isAutomated: false,
|
|
593
|
-
createdBy: createdBy ?? result.did,
|
|
594
|
-
})
|
|
595
|
-
} catch (err) {
|
|
596
|
-
if (
|
|
597
|
-
err instanceof InvalidRequestError &&
|
|
598
|
-
(err.customErrorName === 'AlreadyInTargetState' ||
|
|
599
|
-
err.customErrorName === 'InvalidStateTransition')
|
|
600
|
-
) {
|
|
601
|
-
// no-op — status changed concurrently; leave it alone
|
|
602
|
-
} else {
|
|
603
|
-
throw err
|
|
604
|
-
}
|
|
605
|
-
}
|
|
606
|
-
}
|
|
607
|
-
|
|
608
|
-
return this.hydrateReportAssignment(result.id)
|
|
609
|
-
}
|
|
610
|
-
|
|
611
|
-
private async checkReport(reportId: number): Promise<void> {
|
|
612
|
-
const report = await this.db.db
|
|
613
|
-
.selectFrom('report')
|
|
614
|
-
.selectAll()
|
|
615
|
-
.where('id', '=', reportId)
|
|
616
|
-
.executeTakeFirst()
|
|
617
|
-
if (!report) {
|
|
618
|
-
throw new InvalidRequestError('Invalid report', 'InvalidAssignment')
|
|
619
|
-
}
|
|
620
|
-
}
|
|
621
|
-
|
|
622
|
-
private async checkQueue(queueId: number): Promise<void> {
|
|
623
|
-
const queue = await this.db.db
|
|
624
|
-
.selectFrom('report_queue')
|
|
625
|
-
.selectAll()
|
|
626
|
-
.where('id', '=', queueId)
|
|
627
|
-
.where('enabled', '=', true)
|
|
628
|
-
.where('deletedAt', 'is', null)
|
|
629
|
-
.executeTakeFirst()
|
|
630
|
-
if (!queue) {
|
|
631
|
-
throw new InvalidRequestError('Invalid queue', 'InvalidAssignment')
|
|
632
|
-
}
|
|
633
|
-
}
|
|
634
|
-
|
|
635
|
-
private async hydrateReportAssignment(
|
|
636
|
-
assignmentId: number,
|
|
637
|
-
): Promise<ToolsOzoneReportDefs.AssignmentView> {
|
|
638
|
-
const row = await this.db.db
|
|
639
|
-
.selectFrom('moderator_assignment')
|
|
640
|
-
.leftJoin(
|
|
641
|
-
'report_queue',
|
|
642
|
-
'report_queue.id',
|
|
643
|
-
'moderator_assignment.queueId',
|
|
644
|
-
)
|
|
645
|
-
.selectAll('moderator_assignment')
|
|
646
|
-
.select([
|
|
647
|
-
'report_queue.name as queueName',
|
|
648
|
-
'report_queue.subjectTypes as queueSubjectTypes',
|
|
649
|
-
'report_queue.collection as queueCollection',
|
|
650
|
-
'report_queue.description as queueDescription',
|
|
651
|
-
'report_queue.reportTypes as queueReportTypes',
|
|
652
|
-
'report_queue.createdBy as queueCreatedBy',
|
|
653
|
-
'report_queue.createdAt as queueCreatedAt',
|
|
654
|
-
'report_queue.updatedAt as queueUpdatedAt',
|
|
655
|
-
'report_queue.enabled as queueEnabled',
|
|
656
|
-
'report_queue.deletedAt as queueDeletedAt',
|
|
657
|
-
])
|
|
658
|
-
.where('moderator_assignment.id', '=', assignmentId)
|
|
659
|
-
.executeTakeFirstOrThrow()
|
|
660
|
-
|
|
661
|
-
const memberViews = await this.fetchMemberViews([row.did])
|
|
662
|
-
return this.viewReportAssignment(row, memberViews.get(row.did))
|
|
663
|
-
}
|
|
664
|
-
|
|
665
|
-
queueFromJoined(
|
|
666
|
-
row: AssignmentRowWithQueue,
|
|
667
|
-
): Selectable<ReportQueue> | undefined {
|
|
668
|
-
if (row.queueId === null || row.queueName === null) {
|
|
669
|
-
return undefined
|
|
670
|
-
}
|
|
671
|
-
|
|
672
|
-
return {
|
|
673
|
-
id: row.queueId,
|
|
674
|
-
name: row.queueName,
|
|
675
|
-
subjectTypes: row.queueSubjectTypes ?? [],
|
|
676
|
-
collection: row.queueCollection,
|
|
677
|
-
reportTypes: row.queueReportTypes ?? [],
|
|
678
|
-
description: row.queueDescription ?? null,
|
|
679
|
-
createdBy: row.queueCreatedBy ?? '',
|
|
680
|
-
createdAt: row.queueCreatedAt ?? '',
|
|
681
|
-
updatedAt: row.queueUpdatedAt ?? '',
|
|
682
|
-
enabled: row.queueEnabled ?? false,
|
|
683
|
-
deletedAt: row.queueDeletedAt,
|
|
684
|
-
}
|
|
685
|
-
}
|
|
686
|
-
|
|
687
|
-
viewQueueAssignment(
|
|
688
|
-
row: AssignmentRowWithQueue,
|
|
689
|
-
member?: TeamMember,
|
|
690
|
-
): ToolsOzoneQueueDefs.AssignmentView {
|
|
691
|
-
const queueService = this.queueService
|
|
692
|
-
|
|
693
|
-
const queue = this.queueFromJoined(row)
|
|
694
|
-
const queueView = queue ? queueService.view(queue) : undefined
|
|
695
|
-
if (!queueView) {
|
|
696
|
-
throw new Error('Failed to hydrate queue')
|
|
697
|
-
}
|
|
698
|
-
|
|
699
|
-
return {
|
|
700
|
-
id: row.id,
|
|
701
|
-
did: row.did,
|
|
702
|
-
...(member ? { moderator: member } : {}),
|
|
703
|
-
queue: queueView,
|
|
704
|
-
startAt: row.startAt,
|
|
705
|
-
...(row.endAt !== null ? { endAt: row.endAt } : {}),
|
|
706
|
-
}
|
|
707
|
-
}
|
|
708
|
-
|
|
709
|
-
viewReportAssignment(
|
|
710
|
-
row: AssignmentRowWithQueue,
|
|
711
|
-
member?: TeamMember,
|
|
712
|
-
): ToolsOzoneReportDefs.AssignmentView {
|
|
713
|
-
const queueService = this.queueService
|
|
714
|
-
|
|
715
|
-
const queue = this.queueFromJoined(row)
|
|
716
|
-
const queueView = queue ? queueService.view(queue) : undefined
|
|
717
|
-
|
|
718
|
-
return {
|
|
719
|
-
id: row.id,
|
|
720
|
-
did: row.did,
|
|
721
|
-
...(member ? { moderator: member } : {}),
|
|
722
|
-
reportId: row.reportId!,
|
|
723
|
-
...(queueView ? { queue: queueView } : {}),
|
|
724
|
-
startAt: row.startAt,
|
|
725
|
-
...(row.endAt !== null ? { endAt: row.endAt } : {}),
|
|
726
|
-
}
|
|
727
|
-
}
|
|
728
|
-
}
|