@atproto/ozone 0.1.172 → 0.1.174

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 (450) hide show
  1. package/CHANGELOG.md +18 -0
  2. package/dist/api/index.d.ts.map +1 -1
  3. package/dist/api/index.js +40 -0
  4. package/dist/api/index.js.map +1 -1
  5. package/dist/api/moderation/emitEvent.d.ts.map +1 -1
  6. package/dist/api/moderation/emitEvent.js +31 -0
  7. package/dist/api/moderation/emitEvent.js.map +1 -1
  8. package/dist/api/queue/assignModerator.d.ts +4 -0
  9. package/dist/api/queue/assignModerator.d.ts.map +1 -0
  10. package/dist/api/queue/assignModerator.js +28 -0
  11. package/dist/api/queue/assignModerator.js.map +1 -0
  12. package/dist/api/queue/createQueue.d.ts +4 -0
  13. package/dist/api/queue/createQueue.d.ts.map +1 -0
  14. package/dist/api/queue/createQueue.js +44 -0
  15. package/dist/api/queue/createQueue.js.map +1 -0
  16. package/dist/api/queue/deleteQueue.d.ts +4 -0
  17. package/dist/api/queue/deleteQueue.d.ts.map +1 -0
  18. package/dist/api/queue/deleteQueue.js +40 -0
  19. package/dist/api/queue/deleteQueue.js.map +1 -0
  20. package/dist/api/queue/getAssignments.d.ts +4 -0
  21. package/dist/api/queue/getAssignments.d.ts.map +1 -0
  22. package/dist/api/queue/getAssignments.js +19 -0
  23. package/dist/api/queue/getAssignments.js.map +1 -0
  24. package/dist/api/queue/listQueues.d.ts +4 -0
  25. package/dist/api/queue/listQueues.d.ts.map +1 -0
  26. package/dist/api/queue/listQueues.js +29 -0
  27. package/dist/api/queue/listQueues.js.map +1 -0
  28. package/dist/api/queue/routeReports.d.ts +4 -0
  29. package/dist/api/queue/routeReports.d.ts.map +1 -0
  30. package/dist/api/queue/routeReports.js +33 -0
  31. package/dist/api/queue/routeReports.js.map +1 -0
  32. package/dist/api/queue/unassignModerator.d.ts +4 -0
  33. package/dist/api/queue/unassignModerator.d.ts.map +1 -0
  34. package/dist/api/queue/unassignModerator.js +24 -0
  35. package/dist/api/queue/unassignModerator.js.map +1 -0
  36. package/dist/api/queue/updateQueue.d.ts +4 -0
  37. package/dist/api/queue/updateQueue.d.ts.map +1 -0
  38. package/dist/api/queue/updateQueue.js +39 -0
  39. package/dist/api/queue/updateQueue.js.map +1 -0
  40. package/dist/api/report/assignModerator.d.ts +4 -0
  41. package/dist/api/report/assignModerator.d.ts.map +1 -0
  42. package/dist/api/report/assignModerator.js +33 -0
  43. package/dist/api/report/assignModerator.js.map +1 -0
  44. package/dist/api/report/createActivity.d.ts +4 -0
  45. package/dist/api/report/createActivity.d.ts.map +1 -0
  46. package/dist/api/report/createActivity.js +44 -0
  47. package/dist/api/report/createActivity.js.map +1 -0
  48. package/dist/api/report/getAssignments.d.ts +4 -0
  49. package/dist/api/report/getAssignments.d.ts.map +1 -0
  50. package/dist/api/report/getAssignments.js +19 -0
  51. package/dist/api/report/getAssignments.js.map +1 -0
  52. package/dist/api/report/getHistoricalStats.d.ts +4 -0
  53. package/dist/api/report/getHistoricalStats.d.ts.map +1 -0
  54. package/dist/api/report/getHistoricalStats.js +32 -0
  55. package/dist/api/report/getHistoricalStats.js.map +1 -0
  56. package/dist/api/report/getLatestReport.d.ts +4 -0
  57. package/dist/api/report/getLatestReport.d.ts.map +1 -0
  58. package/dist/api/report/getLatestReport.js +31 -0
  59. package/dist/api/report/getLatestReport.js.map +1 -0
  60. package/dist/api/report/getLiveStats.d.ts +4 -0
  61. package/dist/api/report/getLiveStats.d.ts.map +1 -0
  62. package/dist/api/report/getLiveStats.js +25 -0
  63. package/dist/api/report/getLiveStats.js.map +1 -0
  64. package/dist/api/report/getReport.d.ts +4 -0
  65. package/dist/api/report/getReport.d.ts.map +1 -0
  66. package/dist/api/report/getReport.js +35 -0
  67. package/dist/api/report/getReport.js.map +1 -0
  68. package/dist/api/report/listActivities.d.ts +4 -0
  69. package/dist/api/report/listActivities.d.ts.map +1 -0
  70. package/dist/api/report/listActivities.js +25 -0
  71. package/dist/api/report/listActivities.js.map +1 -0
  72. package/dist/api/report/queryReports.d.ts +4 -0
  73. package/dist/api/report/queryReports.d.ts.map +1 -0
  74. package/dist/api/report/queryReports.js +29 -0
  75. package/dist/api/report/queryReports.js.map +1 -0
  76. package/dist/api/report/reassignQueue.d.ts +4 -0
  77. package/dist/api/report/reassignQueue.d.ts.map +1 -0
  78. package/dist/api/report/reassignQueue.js +45 -0
  79. package/dist/api/report/reassignQueue.js.map +1 -0
  80. package/dist/api/report/refreshStats.d.ts +4 -0
  81. package/dist/api/report/refreshStats.d.ts.map +1 -0
  82. package/dist/api/report/refreshStats.js +26 -0
  83. package/dist/api/report/refreshStats.js.map +1 -0
  84. package/dist/api/report/unassignModerator.d.ts +4 -0
  85. package/dist/api/report/unassignModerator.d.ts.map +1 -0
  86. package/dist/api/report/unassignModerator.js +21 -0
  87. package/dist/api/report/unassignModerator.js.map +1 -0
  88. package/dist/api/util.d.ts +2 -0
  89. package/dist/api/util.d.ts.map +1 -1
  90. package/dist/api/util.js +9 -1
  91. package/dist/api/util.js.map +1 -1
  92. package/dist/assignment/index.d.ts +89 -0
  93. package/dist/assignment/index.d.ts.map +1 -0
  94. package/dist/assignment/index.js +537 -0
  95. package/dist/assignment/index.js.map +1 -0
  96. package/dist/config/config.d.ts +14 -0
  97. package/dist/config/config.d.ts.map +1 -1
  98. package/dist/config/config.js +9 -0
  99. package/dist/config/config.js.map +1 -1
  100. package/dist/config/env.d.ts +3 -0
  101. package/dist/config/env.d.ts.map +1 -1
  102. package/dist/config/env.js +3 -0
  103. package/dist/config/env.js.map +1 -1
  104. package/dist/context.d.ts +9 -0
  105. package/dist/context.d.ts.map +1 -1
  106. package/dist/context.js +31 -10
  107. package/dist/context.js.map +1 -1
  108. package/dist/daemon/context.d.ts +6 -0
  109. package/dist/daemon/context.d.ts.map +1 -1
  110. package/dist/daemon/context.js +28 -4
  111. package/dist/daemon/context.js.map +1 -1
  112. package/dist/daemon/event-reverser.d.ts +1 -0
  113. package/dist/daemon/event-reverser.d.ts.map +1 -1
  114. package/dist/daemon/event-reverser.js +42 -1
  115. package/dist/daemon/event-reverser.js.map +1 -1
  116. package/dist/daemon/job-cursor.d.ts +5 -0
  117. package/dist/daemon/job-cursor.d.ts.map +1 -0
  118. package/dist/daemon/job-cursor.js +28 -0
  119. package/dist/daemon/job-cursor.js.map +1 -0
  120. package/dist/daemon/queue-router.d.ts +17 -0
  121. package/dist/daemon/queue-router.d.ts.map +1 -0
  122. package/dist/daemon/queue-router.js +114 -0
  123. package/dist/daemon/queue-router.js.map +1 -0
  124. package/dist/daemon/stats-computer.d.ts +51 -0
  125. package/dist/daemon/stats-computer.d.ts.map +1 -0
  126. package/dist/daemon/stats-computer.js +117 -0
  127. package/dist/daemon/stats-computer.js.map +1 -0
  128. package/dist/daemon/strike-expiry-processor.d.ts.map +1 -1
  129. package/dist/daemon/strike-expiry-processor.js +4 -19
  130. package/dist/daemon/strike-expiry-processor.js.map +1 -1
  131. package/dist/db/migrations/20260219T164523000Z-create-report-table.d.ts +4 -0
  132. package/dist/db/migrations/20260219T164523000Z-create-report-table.d.ts.map +1 -0
  133. package/dist/db/migrations/20260219T164523000Z-create-report-table.js +126 -0
  134. package/dist/db/migrations/20260219T164523000Z-create-report-table.js.map +1 -0
  135. package/dist/db/migrations/20260219T165302248Z-moderator-assignment.d.ts +4 -0
  136. package/dist/db/migrations/20260219T165302248Z-moderator-assignment.d.ts.map +1 -0
  137. package/dist/db/migrations/20260219T165302248Z-moderator-assignment.js +35 -0
  138. package/dist/db/migrations/20260219T165302248Z-moderator-assignment.js.map +1 -0
  139. package/dist/db/migrations/20260225T000000000Z-add-report-queue-table.d.ts +4 -0
  140. package/dist/db/migrations/20260225T000000000Z-add-report-queue-table.d.ts.map +1 -0
  141. package/dist/db/migrations/20260225T000000000Z-add-report-queue-table.js +36 -0
  142. package/dist/db/migrations/20260225T000000000Z-add-report-queue-table.js.map +1 -0
  143. package/dist/db/migrations/20260313T000000000Z-add-report-activity-table.d.ts +4 -0
  144. package/dist/db/migrations/20260313T000000000Z-add-report-activity-table.d.ts.map +1 -0
  145. package/dist/db/migrations/20260313T000000000Z-add-report-activity-table.js +39 -0
  146. package/dist/db/migrations/20260313T000000000Z-add-report-activity-table.js.map +1 -0
  147. package/dist/db/migrations/20260318T152058935Z-add-report-stat.d.ts +4 -0
  148. package/dist/db/migrations/20260318T152058935Z-add-report-stat.d.ts.map +1 -0
  149. package/dist/db/migrations/20260318T152058935Z-add-report-stat.js +34 -0
  150. package/dist/db/migrations/20260318T152058935Z-add-report-stat.js.map +1 -0
  151. package/dist/db/migrations/20260428T000000000Z-add-expiring-tag-table.d.ts +4 -0
  152. package/dist/db/migrations/20260428T000000000Z-add-expiring-tag-table.d.ts.map +1 -0
  153. package/dist/db/migrations/20260428T000000000Z-add-expiring-tag-table.js +32 -0
  154. package/dist/db/migrations/20260428T000000000Z-add-expiring-tag-table.js.map +1 -0
  155. package/dist/db/migrations/index.d.ts +6 -0
  156. package/dist/db/migrations/index.d.ts.map +1 -1
  157. package/dist/db/migrations/index.js +7 -1
  158. package/dist/db/migrations/index.js.map +1 -1
  159. package/dist/db/pagination.d.ts +31 -0
  160. package/dist/db/pagination.d.ts.map +1 -1
  161. package/dist/db/pagination.js +74 -1
  162. package/dist/db/pagination.js.map +1 -1
  163. package/dist/db/schema/expiring_tag.d.ts +15 -0
  164. package/dist/db/schema/expiring_tag.d.ts.map +1 -0
  165. package/dist/db/schema/expiring_tag.js +5 -0
  166. package/dist/db/schema/expiring_tag.js.map +1 -0
  167. package/dist/db/schema/index.d.ts +7 -1
  168. package/dist/db/schema/index.d.ts.map +1 -1
  169. package/dist/db/schema/index.js.map +1 -1
  170. package/dist/db/schema/moderator_assignment.d.ts +14 -0
  171. package/dist/db/schema/moderator_assignment.d.ts.map +1 -0
  172. package/dist/db/schema/moderator_assignment.js +5 -0
  173. package/dist/db/schema/moderator_assignment.js.map +1 -0
  174. package/dist/db/schema/report.d.ts +25 -0
  175. package/dist/db/schema/report.d.ts.map +1 -0
  176. package/dist/db/schema/report.js +5 -0
  177. package/dist/db/schema/report.js.map +1 -0
  178. package/dist/db/schema/report_activity.d.ts +18 -0
  179. package/dist/db/schema/report_activity.d.ts.map +1 -0
  180. package/dist/db/schema/report_activity.js +5 -0
  181. package/dist/db/schema/report_activity.js.map +1 -0
  182. package/dist/db/schema/report_queue.d.ts +19 -0
  183. package/dist/db/schema/report_queue.d.ts.map +1 -0
  184. package/dist/db/schema/report_queue.js +5 -0
  185. package/dist/db/schema/report_queue.js.map +1 -0
  186. package/dist/db/schema/report_stat.d.ts +20 -0
  187. package/dist/db/schema/report_stat.d.ts.map +1 -0
  188. package/dist/db/schema/report_stat.js +5 -0
  189. package/dist/db/schema/report_stat.js.map +1 -0
  190. package/dist/lexicon/index.d.ts +50 -0
  191. package/dist/lexicon/index.d.ts.map +1 -1
  192. package/dist/lexicon/index.js +120 -2
  193. package/dist/lexicon/index.js.map +1 -1
  194. package/dist/lexicon/lexicons.d.ts +11255 -7885
  195. package/dist/lexicon/lexicons.d.ts.map +1 -1
  196. package/dist/lexicon/lexicons.js +1900 -120
  197. package/dist/lexicon/lexicons.js.map +1 -1
  198. package/dist/lexicon/types/app/bsky/embed/external.d.ts +2 -0
  199. package/dist/lexicon/types/app/bsky/embed/external.d.ts.map +1 -1
  200. package/dist/lexicon/types/app/bsky/embed/external.js.map +1 -1
  201. package/dist/lexicon/types/chat/bsky/actor/defs.d.ts +8 -2
  202. package/dist/lexicon/types/chat/bsky/actor/defs.d.ts.map +1 -1
  203. package/dist/lexicon/types/chat/bsky/actor/defs.js +9 -0
  204. package/dist/lexicon/types/chat/bsky/actor/defs.js.map +1 -1
  205. package/dist/lexicon/types/chat/bsky/convo/defs.d.ts +37 -10
  206. package/dist/lexicon/types/chat/bsky/convo/defs.d.ts.map +1 -1
  207. package/dist/lexicon/types/chat/bsky/convo/defs.js +9 -0
  208. package/dist/lexicon/types/chat/bsky/convo/defs.js.map +1 -1
  209. package/dist/lexicon/types/chat/bsky/convo/getMessages.d.ts +3 -0
  210. package/dist/lexicon/types/chat/bsky/convo/getMessages.d.ts.map +1 -1
  211. package/dist/lexicon/types/chat/bsky/convo/getMessages.js.map +1 -1
  212. package/dist/lexicon/types/tools/ozone/moderation/defs.d.ts +2 -0
  213. package/dist/lexicon/types/tools/ozone/moderation/defs.d.ts.map +1 -1
  214. package/dist/lexicon/types/tools/ozone/moderation/defs.js.map +1 -1
  215. package/dist/lexicon/types/tools/ozone/moderation/emitEvent.d.ts +19 -0
  216. package/dist/lexicon/types/tools/ozone/moderation/emitEvent.d.ts.map +1 -1
  217. package/dist/lexicon/types/tools/ozone/moderation/emitEvent.js +9 -0
  218. package/dist/lexicon/types/tools/ozone/moderation/emitEvent.js.map +1 -1
  219. package/dist/lexicon/types/tools/ozone/queue/assignModerator.d.ts +27 -0
  220. package/dist/lexicon/types/tools/ozone/queue/assignModerator.d.ts.map +1 -0
  221. package/dist/lexicon/types/tools/ozone/queue/assignModerator.js +7 -0
  222. package/dist/lexicon/types/tools/ozone/queue/assignModerator.js.map +1 -0
  223. package/dist/lexicon/types/tools/ozone/queue/createQueue.d.ts +35 -0
  224. package/dist/lexicon/types/tools/ozone/queue/createQueue.d.ts.map +1 -0
  225. package/dist/lexicon/types/tools/ozone/queue/createQueue.js +7 -0
  226. package/dist/lexicon/types/tools/ozone/queue/createQueue.js.map +1 -0
  227. package/dist/lexicon/types/tools/ozone/queue/defs.d.ts +62 -0
  228. package/dist/lexicon/types/tools/ozone/queue/defs.d.ts.map +1 -0
  229. package/dist/lexicon/types/tools/ozone/queue/defs.js +34 -0
  230. package/dist/lexicon/types/tools/ozone/queue/defs.js.map +1 -0
  231. package/dist/lexicon/types/tools/ozone/queue/deleteQueue.d.ts +29 -0
  232. package/dist/lexicon/types/tools/ozone/queue/deleteQueue.d.ts.map +1 -0
  233. package/dist/lexicon/types/tools/ozone/queue/deleteQueue.js +7 -0
  234. package/dist/lexicon/types/tools/ozone/queue/deleteQueue.js.map +1 -0
  235. package/dist/lexicon/types/tools/ozone/queue/getAssignments.d.ts +30 -0
  236. package/dist/lexicon/types/tools/ozone/queue/getAssignments.d.ts.map +1 -0
  237. package/dist/lexicon/types/tools/ozone/queue/getAssignments.js +7 -0
  238. package/dist/lexicon/types/tools/ozone/queue/getAssignments.js.map +1 -0
  239. package/dist/lexicon/types/tools/ozone/queue/listQueues.d.ts +32 -0
  240. package/dist/lexicon/types/tools/ozone/queue/listQueues.d.ts.map +1 -0
  241. package/dist/lexicon/types/tools/ozone/queue/listQueues.js +7 -0
  242. package/dist/lexicon/types/tools/ozone/queue/listQueues.js.map +1 -0
  243. package/dist/lexicon/types/tools/ozone/queue/routeReports.d.ts +31 -0
  244. package/dist/lexicon/types/tools/ozone/queue/routeReports.d.ts.map +1 -0
  245. package/dist/lexicon/types/tools/ozone/queue/routeReports.js +7 -0
  246. package/dist/lexicon/types/tools/ozone/queue/routeReports.js.map +1 -0
  247. package/dist/lexicon/types/tools/ozone/queue/unassignModerator.d.ts +18 -0
  248. package/dist/lexicon/types/tools/ozone/queue/unassignModerator.d.ts.map +1 -0
  249. package/dist/lexicon/types/tools/ozone/queue/unassignModerator.js +7 -0
  250. package/dist/lexicon/types/tools/ozone/queue/unassignModerator.js.map +1 -0
  251. package/dist/lexicon/types/tools/ozone/queue/updateQueue.d.ts +32 -0
  252. package/dist/lexicon/types/tools/ozone/queue/updateQueue.d.ts.map +1 -0
  253. package/dist/lexicon/types/tools/ozone/queue/updateQueue.js +7 -0
  254. package/dist/lexicon/types/tools/ozone/queue/updateQueue.js.map +1 -0
  255. package/dist/lexicon/types/tools/ozone/report/assignModerator.d.ts +31 -0
  256. package/dist/lexicon/types/tools/ozone/report/assignModerator.d.ts.map +1 -0
  257. package/dist/lexicon/types/tools/ozone/report/assignModerator.js +7 -0
  258. package/dist/lexicon/types/tools/ozone/report/assignModerator.js.map +1 -0
  259. package/dist/lexicon/types/tools/ozone/report/createActivity.d.ts +37 -0
  260. package/dist/lexicon/types/tools/ozone/report/createActivity.d.ts.map +1 -0
  261. package/dist/lexicon/types/tools/ozone/report/createActivity.js +7 -0
  262. package/dist/lexicon/types/tools/ozone/report/createActivity.js.map +1 -0
  263. package/dist/lexicon/types/tools/ozone/report/defs.d.ts +185 -0
  264. package/dist/lexicon/types/tools/ozone/report/defs.d.ts.map +1 -1
  265. package/dist/lexicon/types/tools/ozone/report/defs.js +108 -0
  266. package/dist/lexicon/types/tools/ozone/report/defs.js.map +1 -1
  267. package/dist/lexicon/types/tools/ozone/report/getAssignments.d.ts +30 -0
  268. package/dist/lexicon/types/tools/ozone/report/getAssignments.d.ts.map +1 -0
  269. package/dist/lexicon/types/tools/ozone/report/getAssignments.js +7 -0
  270. package/dist/lexicon/types/tools/ozone/report/getAssignments.js.map +1 -0
  271. package/dist/lexicon/types/tools/ozone/report/getHistoricalStats.d.ts +36 -0
  272. package/dist/lexicon/types/tools/ozone/report/getHistoricalStats.d.ts.map +1 -0
  273. package/dist/lexicon/types/tools/ozone/report/getHistoricalStats.js +7 -0
  274. package/dist/lexicon/types/tools/ozone/report/getHistoricalStats.js.map +1 -0
  275. package/dist/lexicon/types/tools/ozone/report/getLatestReport.d.ts +21 -0
  276. package/dist/lexicon/types/tools/ozone/report/getLatestReport.d.ts.map +1 -0
  277. package/dist/lexicon/types/tools/ozone/report/getLatestReport.js +7 -0
  278. package/dist/lexicon/types/tools/ozone/report/getLatestReport.js.map +1 -0
  279. package/dist/lexicon/types/tools/ozone/report/getLiveStats.d.ts +27 -0
  280. package/dist/lexicon/types/tools/ozone/report/getLiveStats.d.ts.map +1 -0
  281. package/dist/lexicon/types/tools/ozone/report/getLiveStats.js +7 -0
  282. package/dist/lexicon/types/tools/ozone/report/getLiveStats.js.map +1 -0
  283. package/dist/lexicon/types/tools/ozone/report/getReport.d.ts +22 -0
  284. package/dist/lexicon/types/tools/ozone/report/getReport.d.ts.map +1 -0
  285. package/dist/lexicon/types/tools/ozone/report/getReport.js +7 -0
  286. package/dist/lexicon/types/tools/ozone/report/getReport.js.map +1 -0
  287. package/dist/lexicon/types/tools/ozone/report/listActivities.d.ts +26 -0
  288. package/dist/lexicon/types/tools/ozone/report/listActivities.d.ts.map +1 -0
  289. package/dist/lexicon/types/tools/ozone/report/listActivities.js +7 -0
  290. package/dist/lexicon/types/tools/ozone/report/listActivities.js.map +1 -0
  291. package/dist/lexicon/types/tools/ozone/report/queryReports.d.ts +48 -0
  292. package/dist/lexicon/types/tools/ozone/report/queryReports.d.ts.map +1 -0
  293. package/dist/lexicon/types/tools/ozone/report/queryReports.js +7 -0
  294. package/dist/lexicon/types/tools/ozone/report/queryReports.js.map +1 -0
  295. package/dist/lexicon/types/tools/ozone/report/reassignQueue.d.ts +31 -0
  296. package/dist/lexicon/types/tools/ozone/report/reassignQueue.d.ts.map +1 -0
  297. package/dist/lexicon/types/tools/ozone/report/reassignQueue.js +7 -0
  298. package/dist/lexicon/types/tools/ozone/report/reassignQueue.js.map +1 -0
  299. package/dist/lexicon/types/tools/ozone/report/refreshStats.d.ts +28 -0
  300. package/dist/lexicon/types/tools/ozone/report/refreshStats.d.ts.map +1 -0
  301. package/dist/lexicon/types/tools/ozone/report/refreshStats.js +7 -0
  302. package/dist/lexicon/types/tools/ozone/report/refreshStats.js.map +1 -0
  303. package/dist/lexicon/types/tools/ozone/report/unassignModerator.d.ts +25 -0
  304. package/dist/lexicon/types/tools/ozone/report/unassignModerator.d.ts.map +1 -0
  305. package/dist/lexicon/types/tools/ozone/report/unassignModerator.js +7 -0
  306. package/dist/lexicon/types/tools/ozone/report/unassignModerator.js.map +1 -0
  307. package/dist/mod-service/expiring-tags.d.ts +27 -0
  308. package/dist/mod-service/expiring-tags.d.ts.map +1 -0
  309. package/dist/mod-service/expiring-tags.js +62 -0
  310. package/dist/mod-service/expiring-tags.js.map +1 -0
  311. package/dist/mod-service/index.d.ts +3 -1
  312. package/dist/mod-service/index.d.ts.map +1 -1
  313. package/dist/mod-service/index.js +61 -2
  314. package/dist/mod-service/index.js.map +1 -1
  315. package/dist/mod-service/report.d.ts +64 -0
  316. package/dist/mod-service/report.d.ts.map +1 -0
  317. package/dist/mod-service/report.js +282 -0
  318. package/dist/mod-service/report.js.map +1 -0
  319. package/dist/mod-service/status.d.ts +24 -0
  320. package/dist/mod-service/status.d.ts.map +1 -1
  321. package/dist/queue/service.d.ts +86 -0
  322. package/dist/queue/service.d.ts.map +1 -0
  323. package/dist/queue/service.js +430 -0
  324. package/dist/queue/service.js.map +1 -0
  325. package/dist/report/activity.d.ts +77 -0
  326. package/dist/report/activity.d.ts.map +1 -0
  327. package/dist/report/activity.js +141 -0
  328. package/dist/report/activity.js.map +1 -0
  329. package/dist/report/handle-report-update.d.ts +47 -0
  330. package/dist/report/handle-report-update.d.ts.map +1 -0
  331. package/dist/report/handle-report-update.js +178 -0
  332. package/dist/report/handle-report-update.js.map +1 -0
  333. package/dist/report/reassign.d.ts +10 -0
  334. package/dist/report/reassign.d.ts.map +1 -0
  335. package/dist/report/reassign.js +75 -0
  336. package/dist/report/reassign.js.map +1 -0
  337. package/dist/report/stats.d.ts +105 -0
  338. package/dist/report/stats.d.ts.map +1 -0
  339. package/dist/report/stats.js +619 -0
  340. package/dist/report/stats.js.map +1 -0
  341. package/dist/report/views.d.ts +111 -0
  342. package/dist/report/views.d.ts.map +1 -0
  343. package/dist/report/views.js +156 -0
  344. package/dist/report/views.js.map +1 -0
  345. package/dist/team/index.d.ts +1 -0
  346. package/dist/team/index.d.ts.map +1 -1
  347. package/dist/team/index.js +11 -0
  348. package/dist/team/index.js.map +1 -1
  349. package/package.json +3 -3
  350. package/src/api/index.ts +40 -0
  351. package/src/api/moderation/emitEvent.ts +38 -0
  352. package/src/api/queue/assignModerator.ts +31 -0
  353. package/src/api/queue/createQueue.ts +62 -0
  354. package/src/api/queue/deleteQueue.ts +56 -0
  355. package/src/api/queue/getAssignments.ts +19 -0
  356. package/src/api/queue/listQueues.ts +39 -0
  357. package/src/api/queue/routeReports.ts +44 -0
  358. package/src/api/queue/unassignModerator.ts +26 -0
  359. package/src/api/queue/updateQueue.ts +54 -0
  360. package/src/api/report/assignModerator.ts +36 -0
  361. package/src/api/report/createActivity.ts +57 -0
  362. package/src/api/report/getAssignments.ts +20 -0
  363. package/src/api/report/getHistoricalStats.ts +41 -0
  364. package/src/api/report/getLatestReport.ts +44 -0
  365. package/src/api/report/getLiveStats.ts +26 -0
  366. package/src/api/report/getReport.ts +55 -0
  367. package/src/api/report/listActivities.ts +34 -0
  368. package/src/api/report/queryReports.ts +44 -0
  369. package/src/api/report/reassignQueue.ts +68 -0
  370. package/src/api/report/refreshStats.ts +27 -0
  371. package/src/api/report/unassignModerator.ts +21 -0
  372. package/src/api/util.ts +12 -0
  373. package/src/assignment/index.ts +731 -0
  374. package/src/config/config.ts +27 -0
  375. package/src/config/env.ts +8 -0
  376. package/src/context.ts +31 -0
  377. package/src/daemon/context.ts +34 -0
  378. package/src/daemon/event-reverser.ts +50 -1
  379. package/src/daemon/job-cursor.ts +33 -0
  380. package/src/daemon/queue-router.ts +101 -0
  381. package/src/daemon/stats-computer.ts +101 -0
  382. package/src/daemon/strike-expiry-processor.ts +4 -20
  383. package/src/db/migrations/20260219T164523000Z-create-report-table.ts +155 -0
  384. package/src/db/migrations/20260219T165302248Z-moderator-assignment.ts +42 -0
  385. package/src/db/migrations/20260225T000000000Z-add-report-queue-table.ts +41 -0
  386. package/src/db/migrations/20260313T000000000Z-add-report-activity-table.ts +48 -0
  387. package/src/db/migrations/20260318T152058935Z-add-report-stat.ts +35 -0
  388. package/src/db/migrations/20260428T000000000Z-add-expiring-tag-table.ts +32 -0
  389. package/src/db/migrations/index.ts +6 -0
  390. package/src/db/pagination.ts +85 -0
  391. package/src/db/schema/expiring_tag.ts +17 -0
  392. package/src/db/schema/index.ts +13 -1
  393. package/src/db/schema/moderator_assignment.ts +16 -0
  394. package/src/db/schema/report.ts +27 -0
  395. package/src/db/schema/report_activity.ts +22 -0
  396. package/src/db/schema/report_queue.ts +21 -0
  397. package/src/db/schema/report_stat.ts +27 -0
  398. package/src/lexicon/index.ts +280 -0
  399. package/src/lexicon/lexicons.ts +2083 -214
  400. package/src/lexicon/types/app/bsky/embed/external.ts +2 -0
  401. package/src/lexicon/types/chat/bsky/actor/defs.ts +17 -1
  402. package/src/lexicon/types/chat/bsky/convo/defs.ts +50 -10
  403. package/src/lexicon/types/chat/bsky/convo/getMessages.ts +3 -0
  404. package/src/lexicon/types/tools/ozone/moderation/defs.ts +2 -0
  405. package/src/lexicon/types/tools/ozone/moderation/emitEvent.ts +24 -0
  406. package/src/lexicon/types/tools/ozone/queue/assignModerator.ts +46 -0
  407. package/src/lexicon/types/tools/ozone/queue/createQueue.ts +54 -0
  408. package/src/lexicon/types/tools/ozone/queue/defs.ts +99 -0
  409. package/src/lexicon/types/tools/ozone/queue/deleteQueue.ts +48 -0
  410. package/src/lexicon/types/tools/ozone/queue/getAssignments.ts +48 -0
  411. package/src/lexicon/types/tools/ozone/queue/listQueues.ts +50 -0
  412. package/src/lexicon/types/tools/ozone/queue/routeReports.ts +50 -0
  413. package/src/lexicon/types/tools/ozone/queue/unassignModerator.ts +37 -0
  414. package/src/lexicon/types/tools/ozone/queue/updateQueue.ts +51 -0
  415. package/src/lexicon/types/tools/ozone/report/assignModerator.ts +50 -0
  416. package/src/lexicon/types/tools/ozone/report/createActivity.ts +60 -0
  417. package/src/lexicon/types/tools/ozone/report/defs.ts +327 -0
  418. package/src/lexicon/types/tools/ozone/report/getAssignments.ts +48 -0
  419. package/src/lexicon/types/tools/ozone/report/getHistoricalStats.ts +54 -0
  420. package/src/lexicon/types/tools/ozone/report/getLatestReport.ts +39 -0
  421. package/src/lexicon/types/tools/ozone/report/getLiveStats.ts +45 -0
  422. package/src/lexicon/types/tools/ozone/report/getReport.ts +38 -0
  423. package/src/lexicon/types/tools/ozone/report/listActivities.ts +44 -0
  424. package/src/lexicon/types/tools/ozone/report/queryReports.ts +72 -0
  425. package/src/lexicon/types/tools/ozone/report/reassignQueue.ts +55 -0
  426. package/src/lexicon/types/tools/ozone/report/refreshStats.ts +46 -0
  427. package/src/lexicon/types/tools/ozone/report/unassignModerator.ts +44 -0
  428. package/src/mod-service/expiring-tags.ts +98 -0
  429. package/src/mod-service/index.ts +71 -3
  430. package/src/mod-service/report.ts +408 -0
  431. package/src/queue/service.ts +599 -0
  432. package/src/report/activity.ts +234 -0
  433. package/src/report/handle-report-update.ts +209 -0
  434. package/src/report/reassign.ts +109 -0
  435. package/src/report/stats.ts +850 -0
  436. package/src/report/views.ts +241 -0
  437. package/src/team/index.ts +11 -0
  438. package/tests/expiring-tags.test.ts +231 -0
  439. package/tests/get-report.test.ts +136 -0
  440. package/tests/query-reports.test.ts +608 -0
  441. package/tests/queue-assignment.test.ts +428 -0
  442. package/tests/queue-router.test.ts +306 -0
  443. package/tests/queues.test.ts +690 -0
  444. package/tests/report-action.test.ts +308 -0
  445. package/tests/report-activity.test.ts +567 -0
  446. package/tests/report-assignment.test.ts +517 -0
  447. package/tests/report-reassign-queue.test.ts +340 -0
  448. package/tests/report-routing.test.ts +245 -0
  449. package/tests/report-stats.test.ts +545 -0
  450. package/tsconfig.build.tsbuildinfo +1 -1
@@ -0,0 +1,241 @@
1
+ import { Selectable } from 'kysely'
2
+ import {
3
+ AppBskyActorDefs,
4
+ ToolsOzoneModerationDefs,
5
+ ToolsOzoneQueueDefs,
6
+ ToolsOzoneReportDefs,
7
+ } from '@atproto/api'
8
+ import { addAccountInfoToRepoViewDetail } from '../api/util'
9
+ import { ReportStat } from '../db/schema/report_stat'
10
+ import { AccountView } from '../lexicon/types/com/atproto/admin/defs'
11
+ import {
12
+ RecordViewDetail,
13
+ RepoView,
14
+ } from '../lexicon/types/tools/ozone/moderation/defs'
15
+ import { Member as TeamMember } from '../lexicon/types/tools/ozone/team/defs'
16
+ import { ReportWithEvent } from '../mod-service/report'
17
+ import { ParsedLabelers } from '../util'
18
+
19
+ type ReportViews = {
20
+ repoDetails(
21
+ dids: string[],
22
+ labelers?: ParsedLabelers,
23
+ ): Promise<Map<string, RepoView>>
24
+ recordDetails(
25
+ subjects: { uri: string }[],
26
+ labelers?: ParsedLabelers,
27
+ ): Promise<Map<string, RecordViewDetail>>
28
+ getProfiles(
29
+ dids: string[],
30
+ ): Promise<Map<string, AppBskyActorDefs.ProfileViewDetailed>>
31
+ }
32
+
33
+ export type HydratedReport = {
34
+ partialRepos: Map<string, RepoView>
35
+ accountInfo: Map<string, AccountView | null>
36
+ recordInfo: Map<string, RecordViewDetail>
37
+ profiles: Map<string, AppBskyActorDefs.ProfileViewDetailed>
38
+ queues: Map<number, ToolsOzoneQueueDefs.QueueView>
39
+ memberViews: Map<string, TeamMember>
40
+ }
41
+
42
+ export async function hydrateReportInfo(
43
+ reports: ReportWithEvent[],
44
+ views: ReportViews,
45
+ getAccountInfos: (dids: string[]) => Promise<Map<string, AccountView | null>>,
46
+ getQueues: (
47
+ queueIds: number[],
48
+ ) => Promise<Map<number, ToolsOzoneQueueDefs.QueueView>>,
49
+ getTeamMembers: (dids: string[]) => Promise<Map<string, TeamMember>>,
50
+ labelers: ParsedLabelers,
51
+ ): Promise<HydratedReport> {
52
+ const dids = new Set<string>()
53
+ const uris = new Set<string>()
54
+ const queueIds = new Set<number>()
55
+ const assignmentDids: string[] = []
56
+
57
+ for (const report of reports) {
58
+ dids.add(report.subjectDid)
59
+ dids.add(report.reportedBy)
60
+ if (report.subjectUri) uris.add(report.subjectUri)
61
+ if (report.queueId && report.queueId > 0) queueIds.add(report.queueId)
62
+ if (report.assignedTo) {
63
+ dids.add(report.assignedTo)
64
+ assignmentDids.push(report.assignedTo)
65
+ }
66
+ }
67
+
68
+ const didsArray = Array.from(dids)
69
+ const [partialRepos, accountInfo, recordInfo, profiles, queues, memberViews] =
70
+ await Promise.all([
71
+ views.repoDetails(didsArray, labelers),
72
+ getAccountInfos(didsArray),
73
+ views.recordDetails(
74
+ Array.from(uris).map((uri) => ({ uri })),
75
+ labelers,
76
+ ),
77
+ views.getProfiles(didsArray),
78
+ getQueues(Array.from(queueIds)),
79
+ getTeamMembers(assignmentDids),
80
+ ])
81
+
82
+ return {
83
+ partialRepos,
84
+ accountInfo,
85
+ recordInfo,
86
+ profiles,
87
+ queues,
88
+ memberViews,
89
+ }
90
+ }
91
+
92
+ export function buildReportView(
93
+ report: ReportWithEvent,
94
+ hydrated: HydratedReport,
95
+ isModerator: boolean,
96
+ actions?: ToolsOzoneModerationDefs.ModEventView[],
97
+ ) {
98
+ const {
99
+ partialRepos,
100
+ accountInfo,
101
+ recordInfo,
102
+ profiles,
103
+ queues,
104
+ memberViews,
105
+ } = hydrated
106
+ const isRecord = !!report.subjectUri
107
+ const did = report.subjectDid
108
+ const partialRepo = partialRepos.get(did)
109
+ const repo = partialRepo
110
+ ? addAccountInfoToRepoViewDetail(
111
+ partialRepo,
112
+ accountInfo.get(did) || null,
113
+ isModerator,
114
+ )
115
+ : undefined
116
+ const profile = profiles.get(did)
117
+ const record = isRecord ? recordInfo.get(report.subjectUri!) : undefined
118
+ const status = isRecord
119
+ ? record?.moderation.subjectStatus
120
+ : repo?.moderation.subjectStatus
121
+
122
+ const reportType = report.meta?.reportType as string
123
+
124
+ const subject = isRecord ? report.subjectUri! : report.subjectDid
125
+ const subjectView = {
126
+ type: isRecord ? 'record' : 'account',
127
+ subject,
128
+ repo,
129
+ record,
130
+ profile: profile
131
+ ? {
132
+ $type: 'app.bsky.actor.defs#profileViewDetailed' as const,
133
+ ...profile,
134
+ }
135
+ : undefined,
136
+ status,
137
+ }
138
+
139
+ const reporterDid = report.reportedBy
140
+ const reporterPartialRepo = partialRepos.get(reporterDid)
141
+ const reporterRepo = reporterPartialRepo
142
+ ? addAccountInfoToRepoViewDetail(
143
+ reporterPartialRepo,
144
+ accountInfo.get(reporterDid) || null,
145
+ isModerator,
146
+ )
147
+ : undefined
148
+ const reporterProfile = profiles.get(reporterDid)
149
+ const reporterStatus = reporterRepo?.moderation.subjectStatus
150
+
151
+ const reporterView = {
152
+ type: 'account',
153
+ subject: reporterDid,
154
+ repo: reporterRepo,
155
+ profile: reporterProfile
156
+ ? {
157
+ $type: 'app.bsky.actor.defs#profileViewDetailed' as const,
158
+ ...reporterProfile,
159
+ }
160
+ : undefined,
161
+ status: reporterStatus,
162
+ }
163
+
164
+ const assignmentView =
165
+ report.assignedTo && report.assignedAt
166
+ ? {
167
+ did: report.assignedTo,
168
+ moderator: memberViews.get(report.assignedTo),
169
+ assignedAt: report.assignedAt,
170
+ }
171
+ : undefined
172
+
173
+ return {
174
+ id: report.id,
175
+ eventId: report.eventId,
176
+ status: report.status,
177
+ subject: subjectView,
178
+ reportType,
179
+ reportedBy: report.reportedBy,
180
+ reporter: reporterView,
181
+ comment: report.comment ?? undefined,
182
+ createdAt: report.createdAt,
183
+ updatedAt: report.updatedAt,
184
+ queuedAt: report.queuedAt ?? undefined,
185
+ actionEventIds:
186
+ report.actionEventIds && Array.isArray(report.actionEventIds)
187
+ ? (report.actionEventIds as number[])
188
+ : undefined,
189
+ actions: actions && actions.length ? actions : undefined,
190
+ actionNote: report.actionNote ?? undefined,
191
+ assignment: assignmentView,
192
+ queue:
193
+ report.queueId && report.queueId > 0
194
+ ? queues.get(report.queueId)
195
+ : undefined,
196
+ isMuted: report.isMuted,
197
+ }
198
+ }
199
+
200
+ export function viewQueueStats(
201
+ row?: Selectable<ReportStat>,
202
+ ): ToolsOzoneQueueDefs.QueueStats {
203
+ return {
204
+ pendingCount: row?.pendingCount ?? undefined,
205
+ actionedCount: row?.actionedCount ?? undefined,
206
+ escalatedCount: row?.escalatedCount ?? undefined,
207
+ inboundCount: row?.inboundCount ?? undefined,
208
+ actionRate: row?.actionRate ?? undefined,
209
+ avgHandlingTimeSec: row?.avgHandlingTimeSec ?? undefined,
210
+ lastUpdated: row?.computedAt,
211
+ }
212
+ }
213
+
214
+ export function viewLiveStats(
215
+ row?: Selectable<ReportStat>,
216
+ ): ToolsOzoneReportDefs.LiveStats {
217
+ return {
218
+ pendingCount: row?.pendingCount ?? undefined,
219
+ actionedCount: row?.actionedCount ?? undefined,
220
+ escalatedCount: row?.escalatedCount ?? undefined,
221
+ inboundCount: row?.inboundCount ?? undefined,
222
+ actionRate: row?.actionRate ?? undefined,
223
+ avgHandlingTimeSec: row?.avgHandlingTimeSec ?? undefined,
224
+ lastUpdated: row?.computedAt,
225
+ }
226
+ }
227
+
228
+ export function viewHistoricalStats(
229
+ row: Selectable<ReportStat>,
230
+ ): ToolsOzoneReportDefs.HistoricalStats {
231
+ return {
232
+ date: row.date,
233
+ computedAt: row.computedAt,
234
+ pendingCount: row.pendingCount ?? undefined,
235
+ actionedCount: row.actionedCount ?? undefined,
236
+ escalatedCount: row.escalatedCount ?? undefined,
237
+ inboundCount: row.inboundCount ?? undefined,
238
+ actionRate: row.actionRate ?? undefined,
239
+ avgHandlingTimeSec: row.avgHandlingTimeSec ?? undefined,
240
+ }
241
+ }
package/src/team/index.ts CHANGED
@@ -267,6 +267,17 @@ export class TeamService {
267
267
  } while (lastDid)
268
268
  }
269
269
 
270
+ async viewByDids(dids: string[]): Promise<Map<string, TeamMember>> {
271
+ if (!dids.length) return new Map()
272
+ const members = await this.db.db
273
+ .selectFrom('member')
274
+ .selectAll()
275
+ .where('did', 'in', dids)
276
+ .execute()
277
+ const memberViews = await this.view(members)
278
+ return new Map(memberViews.map((m) => [m.did, m]))
279
+ }
280
+
270
281
  async view(members: Selectable<Member>[]): Promise<TeamMember[]> {
271
282
  const profiles = await this.getProfiles(members.map(({ did }) => did))
272
283
  return members.map((member) => {
@@ -0,0 +1,231 @@
1
+ import {
2
+ ModeratorClient,
3
+ SeedClient,
4
+ TestNetwork,
5
+ basicSeed,
6
+ } from '@atproto/dev-env'
7
+ import { EventReverser } from '../src/daemon/event-reverser'
8
+ import { REASONSPAM } from '../src/lexicon/types/com/atproto/moderation/defs'
9
+
10
+ describe('expiring tags', () => {
11
+ let network: TestNetwork
12
+ let sc: SeedClient
13
+ let modClient: ModeratorClient
14
+
15
+ beforeAll(async () => {
16
+ network = await TestNetwork.create({
17
+ dbPostgresSchema: 'ozone_expiring_tags',
18
+ })
19
+ sc = network.getSeedClient()
20
+ modClient = network.ozone.getModClient()
21
+ await basicSeed(sc)
22
+ await network.processAll()
23
+ })
24
+
25
+ afterAll(async () => {
26
+ await network.close()
27
+ })
28
+
29
+ const emitTagEvent = (
30
+ did: string,
31
+ add: string[],
32
+ remove: string[],
33
+ durationInHours?: number,
34
+ ) =>
35
+ modClient.emitEvent({
36
+ subject: { $type: 'com.atproto.admin.defs#repoRef', did },
37
+ event: {
38
+ $type: 'tools.ozone.moderation.defs#modEventTag',
39
+ add,
40
+ remove,
41
+ ...(durationInHours !== undefined ? { durationInHours } : {}),
42
+ },
43
+ })
44
+
45
+ const getSubjectTags = async (did: string): Promise<string[]> => {
46
+ const result = await modClient.queryStatuses({ subject: did })
47
+ return result.subjectStatuses[0]?.tags ?? []
48
+ }
49
+
50
+ const createReverser = () =>
51
+ new EventReverser(
52
+ network.ozone.ctx.db,
53
+ // @ts-expect-error Error due to circular dependency with the dev-env package
54
+ network.ozone.ctx.modService,
55
+ )
56
+
57
+ it('creates expiring_tag rows when durationInHours is set', async () => {
58
+ // Create a report so the subject exists in moderation_subject_status
59
+ await sc.createReport({
60
+ reasonType: REASONSPAM,
61
+ reason: 'test',
62
+ subject: { $type: 'com.atproto.admin.defs#repoRef', did: sc.dids.bob },
63
+ reportedBy: sc.dids.alice,
64
+ })
65
+
66
+ await emitTagEvent(sc.dids.bob, ['temp-tag-1', 'temp-tag-2'], [], 1)
67
+
68
+ const tags = await getSubjectTags(sc.dids.bob)
69
+ expect(tags).toContain('temp-tag-1')
70
+ expect(tags).toContain('temp-tag-2')
71
+
72
+ // Verify expiring_tag rows were created
73
+ const expiringRows = await network.ozone.ctx.db.db
74
+ .selectFrom('expiring_tag')
75
+ .where('did', '=', sc.dids.bob)
76
+ .selectAll()
77
+ .execute()
78
+
79
+ expect(expiringRows).toHaveLength(2)
80
+ expect(expiringRows.map((r) => r.tag).sort()).toEqual([
81
+ 'temp-tag-1',
82
+ 'temp-tag-2',
83
+ ])
84
+ expect(expiringRows[0].expiresAt).toBeTruthy()
85
+ })
86
+
87
+ it('does not create expiring_tag rows without durationInHours', async () => {
88
+ await emitTagEvent(sc.dids.bob, ['permanent-tag'], [])
89
+
90
+ const tags = await getSubjectTags(sc.dids.bob)
91
+ expect(tags).toContain('permanent-tag')
92
+
93
+ const expiringRows = await network.ozone.ctx.db.db
94
+ .selectFrom('expiring_tag')
95
+ .where('did', '=', sc.dids.bob)
96
+ .where('tag', '=', 'permanent-tag')
97
+ .selectAll()
98
+ .execute()
99
+
100
+ expect(expiringRows).toHaveLength(0)
101
+ })
102
+
103
+ it('daemon reverts expired tags', async () => {
104
+ // Manually expire the tags in the DB
105
+ await network.ozone.ctx.db.db
106
+ .updateTable('expiring_tag')
107
+ .set({ expiresAt: new Date(Date.now() - 1000).toISOString() })
108
+ .where('did', '=', sc.dids.bob)
109
+ .execute()
110
+
111
+ const reverser = createReverser()
112
+ await reverser.findAndRevertDueActions()
113
+
114
+ const tags = await getSubjectTags(sc.dids.bob)
115
+ expect(tags).not.toContain('temp-tag-1')
116
+ expect(tags).not.toContain('temp-tag-2')
117
+ // Permanent tag should still be there
118
+ expect(tags).toContain('permanent-tag')
119
+
120
+ // Verify expiring_tag rows are cleaned up
121
+ const remainingRows = await network.ozone.ctx.db.db
122
+ .selectFrom('expiring_tag')
123
+ .where('did', '=', sc.dids.bob)
124
+ .selectAll()
125
+ .execute()
126
+
127
+ expect(remainingRows).toHaveLength(0)
128
+ })
129
+
130
+ it('daemon emits a modEventTag removal event', async () => {
131
+ const events = await modClient.queryEvents({
132
+ subject: sc.dids.bob,
133
+ types: ['tools.ozone.moderation.defs#modEventTag'],
134
+ })
135
+
136
+ const lastTagEvent = events.events[0]
137
+ expect(lastTagEvent.event).toMatchObject({
138
+ $type: 'tools.ozone.moderation.defs#modEventTag',
139
+ add: [],
140
+ remove: expect.arrayContaining(['temp-tag-1', 'temp-tag-2']),
141
+ comment:
142
+ '[SCHEDULED_REVERSAL] Reverting temporary tags as originally scheduled',
143
+ })
144
+ })
145
+
146
+ it('cleans up expiring_tag rows when tags are manually removed', async () => {
147
+ await sc.createReport({
148
+ reasonType: REASONSPAM,
149
+ reason: 'test',
150
+ subject: {
151
+ $type: 'com.atproto.admin.defs#repoRef',
152
+ did: sc.dids.carol,
153
+ },
154
+ reportedBy: sc.dids.alice,
155
+ })
156
+
157
+ await emitTagEvent(sc.dids.carol, ['auto-remove-tag'], [], 24)
158
+
159
+ // Verify the expiring_tag row exists
160
+ let expiringRows = await network.ozone.ctx.db.db
161
+ .selectFrom('expiring_tag')
162
+ .where('did', '=', sc.dids.carol)
163
+ .where('tag', '=', 'auto-remove-tag')
164
+ .selectAll()
165
+ .execute()
166
+ expect(expiringRows).toHaveLength(1)
167
+
168
+ // Manually remove the tag
169
+ await emitTagEvent(sc.dids.carol, [], ['auto-remove-tag'])
170
+
171
+ // Verify the expiring_tag row is cleaned up
172
+ expiringRows = await network.ozone.ctx.db.db
173
+ .selectFrom('expiring_tag')
174
+ .where('did', '=', sc.dids.carol)
175
+ .where('tag', '=', 'auto-remove-tag')
176
+ .selectAll()
177
+ .execute()
178
+ expect(expiringRows).toHaveLength(0)
179
+ })
180
+
181
+ it('daemon skips tags already manually removed', async () => {
182
+ await emitTagEvent(sc.dids.carol, ['skip-tag'], [], 1)
183
+
184
+ // Manually remove the tag first
185
+ await emitTagEvent(sc.dids.carol, [], ['skip-tag'])
186
+
187
+ // Re-insert an expiring_tag row to simulate the race condition
188
+ // (row wasn't cleaned up for some reason)
189
+ await network.ozone.ctx.db.db
190
+ .insertInto('expiring_tag')
191
+ .values({
192
+ eventId: 0,
193
+ did: sc.dids.carol,
194
+ recordPath: '',
195
+ tag: 'skip-tag',
196
+ expiresAt: new Date(Date.now() - 1000).toISOString(),
197
+ createdBy: sc.dids.alice,
198
+ })
199
+ .execute()
200
+
201
+ const tagsBefore = await getSubjectTags(sc.dids.carol)
202
+ expect(tagsBefore).not.toContain('skip-tag')
203
+
204
+ const reverser = createReverser()
205
+ await reverser.findAndRevertDueActions()
206
+
207
+ // Verify no removal event was emitted for a tag that's already gone
208
+ const events = await modClient.queryEvents({
209
+ subject: sc.dids.carol,
210
+ types: ['tools.ozone.moderation.defs#modEventTag'],
211
+ })
212
+
213
+ // The last tag event should be the manual removal, not a scheduled reversal
214
+ const lastEvent = events.events[0]
215
+ expect(lastEvent.event).toMatchObject({
216
+ $type: 'tools.ozone.moderation.defs#modEventTag',
217
+ remove: ['skip-tag'],
218
+ })
219
+ const comment = (lastEvent.event as { comment?: string }).comment ?? ''
220
+ expect(comment).not.toContain('SCHEDULED_REVERSAL')
221
+
222
+ // Verify the expiring_tag row is still cleaned up
223
+ const remainingRows = await network.ozone.ctx.db.db
224
+ .selectFrom('expiring_tag')
225
+ .where('did', '=', sc.dids.carol)
226
+ .where('tag', '=', 'skip-tag')
227
+ .selectAll()
228
+ .execute()
229
+ expect(remainingRows).toHaveLength(0)
230
+ })
231
+ })
@@ -0,0 +1,136 @@
1
+ import AtpAgent from '@atproto/api'
2
+ import {
3
+ ModeratorClient,
4
+ SeedClient,
5
+ TestNetwork,
6
+ basicSeed,
7
+ } from '@atproto/dev-env'
8
+ import { ids } from '../src/lexicon/lexicons'
9
+ import { REASONSPAM } from '../src/lexicon/types/com/atproto/moderation/defs'
10
+
11
+ describe('ozone-get-report', () => {
12
+ let network: TestNetwork
13
+ let agent: AtpAgent
14
+ let sc: SeedClient
15
+ let modClient: ModeratorClient
16
+
17
+ const modHeaders = async (nsid: string) =>
18
+ network.ozone.modHeaders(nsid, 'admin')
19
+
20
+ beforeAll(async () => {
21
+ network = await TestNetwork.create({
22
+ dbPostgresSchema: 'ozone_get_report',
23
+ })
24
+ agent = network.ozone.getAgent()
25
+ sc = network.getSeedClient()
26
+ modClient = network.ozone.getModClient()
27
+ await basicSeed(sc)
28
+ await network.processAll()
29
+ })
30
+
31
+ afterAll(async () => {
32
+ await network.close()
33
+ })
34
+
35
+ it('returns a single report by id', async () => {
36
+ // Create a report on Alice's account
37
+ await sc.createReport({
38
+ reasonType: REASONSPAM,
39
+ subject: {
40
+ $type: 'com.atproto.admin.defs#repoRef',
41
+ did: sc.dids.alice,
42
+ },
43
+ reportedBy: sc.dids.bob,
44
+ })
45
+ await network.processAll()
46
+
47
+ // Fetch the report list to get the report ID
48
+ const { data: list } = await agent.tools.ozone.report.queryReports(
49
+ { status: 'open', subject: sc.dids.alice },
50
+ { headers: await modHeaders(ids.ToolsOzoneReportQueryReports) },
51
+ )
52
+ expect(list.reports.length).toBeGreaterThan(0)
53
+ const reportId = list.reports[0].id
54
+
55
+ // Fetch the single report by ID
56
+ const { data: report } = await agent.tools.ozone.report.getReport(
57
+ { id: reportId },
58
+ { headers: await modHeaders(ids.ToolsOzoneReportGetReport) },
59
+ )
60
+
61
+ expect(report.id).toBe(reportId)
62
+ expect(report.subject.type).toBe('account')
63
+ expect(report.subject.subject).toBe(sc.dids.alice)
64
+ expect(report.reportType).toBe('com.atproto.moderation.defs#reasonSpam')
65
+ expect(report.status).toBe('open')
66
+ })
67
+
68
+ it('hydrates actions for reports that have been actioned', async () => {
69
+ const bobsAccount = {
70
+ $type: 'com.atproto.admin.defs#repoRef' as const,
71
+ did: sc.dids.bob,
72
+ }
73
+
74
+ await sc.createReport({
75
+ reasonType: REASONSPAM,
76
+ subject: bobsAccount,
77
+ reportedBy: sc.dids.alice,
78
+ })
79
+ await network.processAll()
80
+
81
+ const { data: list } = await agent.tools.ozone.report.queryReports(
82
+ { status: 'open', subject: sc.dids.bob },
83
+ { headers: await modHeaders(ids.ToolsOzoneReportQueryReports) },
84
+ )
85
+ const reportId = list.reports[0].id
86
+
87
+ const event = await modClient.emitEvent({
88
+ event: { $type: 'tools.ozone.moderation.defs#modEventAcknowledge' },
89
+ subject: bobsAccount,
90
+ reportAction: {
91
+ ids: [reportId],
92
+ note: 'Reviewed',
93
+ },
94
+ })
95
+
96
+ const { data: report } = await agent.tools.ozone.report.getReport(
97
+ { id: reportId },
98
+ { headers: await modHeaders(ids.ToolsOzoneReportGetReport) },
99
+ )
100
+
101
+ expect(report.status).toBe('closed')
102
+ expect(report.actionEventIds).toEqual([event.id])
103
+ expect(report.actions).toBeDefined()
104
+ expect(report.actions).toHaveLength(1)
105
+ expect(report.actions![0].id).toBe(event.id)
106
+ expect(report.actions![0].event.$type).toBe(
107
+ 'tools.ozone.moderation.defs#modEventAcknowledge',
108
+ )
109
+ })
110
+
111
+ it('omits actions when the report has not been actioned', async () => {
112
+ await sc.createReport({
113
+ reasonType: REASONSPAM,
114
+ subject: {
115
+ $type: 'com.atproto.admin.defs#repoRef',
116
+ did: sc.dids.carol,
117
+ },
118
+ reportedBy: sc.dids.alice,
119
+ })
120
+ await network.processAll()
121
+
122
+ const { data: list } = await agent.tools.ozone.report.queryReports(
123
+ { status: 'open', subject: sc.dids.carol },
124
+ { headers: await modHeaders(ids.ToolsOzoneReportQueryReports) },
125
+ )
126
+ const reportId = list.reports[0].id
127
+
128
+ const { data: report } = await agent.tools.ozone.report.getReport(
129
+ { id: reportId },
130
+ { headers: await modHeaders(ids.ToolsOzoneReportGetReport) },
131
+ )
132
+
133
+ expect(report.actionEventIds).toBeFalsy()
134
+ expect(report.actions).toBeUndefined()
135
+ })
136
+ })