@atproto/ozone 0.0.2

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 (444) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/LICENSE.txt +7 -0
  3. package/README.md +15 -0
  4. package/babel.config.js +3 -0
  5. package/bin/migration-create.ts +38 -0
  6. package/build.js +18 -0
  7. package/dist/api/admin/emitModerationEvent.d.ts +3 -0
  8. package/dist/api/admin/getModerationEvent.d.ts +3 -0
  9. package/dist/api/admin/getRecord.d.ts +3 -0
  10. package/dist/api/admin/getRepo.d.ts +3 -0
  11. package/dist/api/admin/queryModerationEvents.d.ts +3 -0
  12. package/dist/api/admin/queryModerationStatuses.d.ts +3 -0
  13. package/dist/api/admin/searchRepos.d.ts +3 -0
  14. package/dist/api/admin/util.d.ts +5 -0
  15. package/dist/api/com/atproto/admin/emitModerationEvent.d.ts +3 -0
  16. package/dist/api/com/atproto/admin/getModerationEvent.d.ts +3 -0
  17. package/dist/api/com/atproto/admin/getRecord.d.ts +3 -0
  18. package/dist/api/com/atproto/admin/getRepo.d.ts +3 -0
  19. package/dist/api/com/atproto/admin/queryModerationEvents.d.ts +3 -0
  20. package/dist/api/com/atproto/admin/queryModerationStatuses.d.ts +3 -0
  21. package/dist/api/com/atproto/admin/searchRepos.d.ts +3 -0
  22. package/dist/api/com/atproto/admin/util.d.ts +5 -0
  23. package/dist/api/com/atproto/moderation/createReport.d.ts +3 -0
  24. package/dist/api/com/atproto/moderation/util.d.ts +4 -0
  25. package/dist/api/com/atproto/temp/fetchLabels.d.ts +3 -0
  26. package/dist/api/health.d.ts +3 -0
  27. package/dist/api/index.d.ts +5 -0
  28. package/dist/api/label/queryLabels.d.ts +3 -0
  29. package/dist/api/label/subscribeLabels.d.ts +3 -0
  30. package/dist/api/moderation/createReport.d.ts +3 -0
  31. package/dist/api/moderation/util.d.ts +4 -0
  32. package/dist/api/temp/fetchLabels.d.ts +3 -0
  33. package/dist/api/util.d.ts +2 -0
  34. package/dist/api/well-known.d.ts +3 -0
  35. package/dist/auth-verifier.d.ts +47 -0
  36. package/dist/auth.d.ts +81 -0
  37. package/dist/background.d.ts +13 -0
  38. package/dist/config/config.d.ts +30 -0
  39. package/dist/config/env.d.ts +19 -0
  40. package/dist/config/index.d.ts +3 -0
  41. package/dist/config/secrets.d.ts +8 -0
  42. package/dist/config.d.ts +42 -0
  43. package/dist/context.d.ts +140 -0
  44. package/dist/daemon/config.d.ts +23 -0
  45. package/dist/daemon/context.d.ts +25 -0
  46. package/dist/daemon/event-pusher.d.ts +48 -0
  47. package/dist/daemon/event-reverser.d.ts +16 -0
  48. package/dist/daemon/index.d.ts +13 -0
  49. package/dist/db/index.d.ts +32 -0
  50. package/dist/db/index.js +33707 -0
  51. package/dist/db/index.js.map +7 -0
  52. package/dist/db/migrations/20231219T205730722Z-init.d.ts +3 -0
  53. package/dist/db/migrations/index.d.ts +1 -0
  54. package/dist/db/migrations/provider.d.ts +11 -0
  55. package/dist/db/pagination.d.ts +66 -0
  56. package/dist/db/schema/blob_push_event.d.ts +17 -0
  57. package/dist/db/schema/index.d.ts +10 -0
  58. package/dist/db/schema/label.d.ts +12 -0
  59. package/dist/db/schema/moderation_event.d.ts +22 -0
  60. package/dist/db/schema/moderation_subject_status.d.ts +25 -0
  61. package/dist/db/schema/record_push_event.d.ts +17 -0
  62. package/dist/db/schema/repo_push_event.d.ts +15 -0
  63. package/dist/db/types.d.ts +12 -0
  64. package/dist/error.d.ts +2 -0
  65. package/dist/index.d.ts +27 -0
  66. package/dist/index.js +131818 -0
  67. package/dist/index.js.map +7 -0
  68. package/dist/lexicon/index.d.ts +372 -0
  69. package/dist/lexicon/lexicons.d.ts +7553 -0
  70. package/dist/lexicon/types/app/bsky/actor/defs.d.ts +104 -0
  71. package/dist/lexicon/types/app/bsky/actor/getPreferences.d.ts +31 -0
  72. package/dist/lexicon/types/app/bsky/actor/getProfile.d.ts +29 -0
  73. package/dist/lexicon/types/app/bsky/actor/getProfiles.d.ts +32 -0
  74. package/dist/lexicon/types/app/bsky/actor/getSuggestions.d.ts +34 -0
  75. package/dist/lexicon/types/app/bsky/actor/profile.d.ts +15 -0
  76. package/dist/lexicon/types/app/bsky/actor/putPreferences.d.ts +26 -0
  77. package/dist/lexicon/types/app/bsky/actor/searchActors.d.ts +36 -0
  78. package/dist/lexicon/types/app/bsky/actor/searchActorsTypeahead.d.ts +34 -0
  79. package/dist/lexicon/types/app/bsky/embed/external.d.ts +31 -0
  80. package/dist/lexicon/types/app/bsky/embed/images.d.ts +37 -0
  81. package/dist/lexicon/types/app/bsky/embed/record.d.ts +54 -0
  82. package/dist/lexicon/types/app/bsky/embed/recordWithMedia.d.ts +24 -0
  83. package/dist/lexicon/types/app/bsky/feed/defs.d.ts +151 -0
  84. package/dist/lexicon/types/app/bsky/feed/describeFeedGenerator.d.ts +46 -0
  85. package/dist/lexicon/types/app/bsky/feed/generator.d.ts +18 -0
  86. package/dist/lexicon/types/app/bsky/feed/getActorFeeds.d.ts +35 -0
  87. package/dist/lexicon/types/app/bsky/feed/getActorLikes.d.ts +36 -0
  88. package/dist/lexicon/types/app/bsky/feed/getAuthorFeed.d.ts +37 -0
  89. package/dist/lexicon/types/app/bsky/feed/getFeed.d.ts +36 -0
  90. package/dist/lexicon/types/app/bsky/feed/getFeedGenerator.d.ts +34 -0
  91. package/dist/lexicon/types/app/bsky/feed/getFeedGenerators.d.ts +32 -0
  92. package/dist/lexicon/types/app/bsky/feed/getFeedSkeleton.d.ts +36 -0
  93. package/dist/lexicon/types/app/bsky/feed/getLikes.d.ts +47 -0
  94. package/dist/lexicon/types/app/bsky/feed/getListFeed.d.ts +36 -0
  95. package/dist/lexicon/types/app/bsky/feed/getPostThread.d.ts +38 -0
  96. package/dist/lexicon/types/app/bsky/feed/getPosts.d.ts +32 -0
  97. package/dist/lexicon/types/app/bsky/feed/getRepostedBy.d.ts +38 -0
  98. package/dist/lexicon/types/app/bsky/feed/getSuggestedFeeds.d.ts +34 -0
  99. package/dist/lexicon/types/app/bsky/feed/getTimeline.d.ts +35 -0
  100. package/dist/lexicon/types/app/bsky/feed/like.d.ts +9 -0
  101. package/dist/lexicon/types/app/bsky/feed/post.d.ts +50 -0
  102. package/dist/lexicon/types/app/bsky/feed/repost.d.ts +9 -0
  103. package/dist/lexicon/types/app/bsky/feed/searchPosts.d.ts +37 -0
  104. package/dist/lexicon/types/app/bsky/feed/threadgate.d.ts +28 -0
  105. package/dist/lexicon/types/app/bsky/graph/block.d.ts +8 -0
  106. package/dist/lexicon/types/app/bsky/graph/defs.d.ts +47 -0
  107. package/dist/lexicon/types/app/bsky/graph/follow.d.ts +8 -0
  108. package/dist/lexicon/types/app/bsky/graph/getBlocks.d.ts +34 -0
  109. package/dist/lexicon/types/app/bsky/graph/getFollowers.d.ts +36 -0
  110. package/dist/lexicon/types/app/bsky/graph/getFollows.d.ts +36 -0
  111. package/dist/lexicon/types/app/bsky/graph/getList.d.ts +36 -0
  112. package/dist/lexicon/types/app/bsky/graph/getListBlocks.d.ts +34 -0
  113. package/dist/lexicon/types/app/bsky/graph/getListMutes.d.ts +34 -0
  114. package/dist/lexicon/types/app/bsky/graph/getLists.d.ts +35 -0
  115. package/dist/lexicon/types/app/bsky/graph/getMutes.d.ts +34 -0
  116. package/dist/lexicon/types/app/bsky/graph/getSuggestedFollowsByActor.d.ts +32 -0
  117. package/dist/lexicon/types/app/bsky/graph/list.d.ts +19 -0
  118. package/dist/lexicon/types/app/bsky/graph/listblock.d.ts +8 -0
  119. package/dist/lexicon/types/app/bsky/graph/listitem.d.ts +9 -0
  120. package/dist/lexicon/types/app/bsky/graph/muteActor.d.ts +25 -0
  121. package/dist/lexicon/types/app/bsky/graph/muteActorList.d.ts +25 -0
  122. package/dist/lexicon/types/app/bsky/graph/unmuteActor.d.ts +25 -0
  123. package/dist/lexicon/types/app/bsky/graph/unmuteActorList.d.ts +25 -0
  124. package/dist/lexicon/types/app/bsky/moderation/defs.d.ts +49 -0
  125. package/dist/lexicon/types/app/bsky/moderation/getService.d.ts +29 -0
  126. package/dist/lexicon/types/app/bsky/moderation/getServices.d.ts +32 -0
  127. package/dist/lexicon/types/app/bsky/moderation/service.d.ts +17 -0
  128. package/dist/lexicon/types/app/bsky/notification/getUnreadCount.d.ts +31 -0
  129. package/dist/lexicon/types/app/bsky/notification/listNotifications.d.ts +52 -0
  130. package/dist/lexicon/types/app/bsky/notification/registerPush.d.ts +28 -0
  131. package/dist/lexicon/types/app/bsky/notification/updateSeen.d.ts +25 -0
  132. package/dist/lexicon/types/app/bsky/richtext/facet.d.ts +36 -0
  133. package/dist/lexicon/types/app/bsky/unspecced/defs.d.ts +13 -0
  134. package/dist/lexicon/types/app/bsky/unspecced/getPopular.d.ts +35 -0
  135. package/dist/lexicon/types/app/bsky/unspecced/getPopularFeedGenerators.d.ts +35 -0
  136. package/dist/lexicon/types/app/bsky/unspecced/getTimelineSkeleton.d.ts +35 -0
  137. package/dist/lexicon/types/app/bsky/unspecced/searchActorsSkeleton.d.ts +38 -0
  138. package/dist/lexicon/types/app/bsky/unspecced/searchPostsSkeleton.d.ts +37 -0
  139. package/dist/lexicon/types/com/atproto/admin/defs.d.ts +319 -0
  140. package/dist/lexicon/types/com/atproto/admin/deleteAccount.d.ts +25 -0
  141. package/dist/lexicon/types/com/atproto/admin/disableAccountInvites.d.ts +26 -0
  142. package/dist/lexicon/types/com/atproto/admin/disableInviteCodes.d.ts +26 -0
  143. package/dist/lexicon/types/com/atproto/admin/emitModerationEvent.d.ts +45 -0
  144. package/dist/lexicon/types/com/atproto/admin/enableAccountInvites.d.ts +26 -0
  145. package/dist/lexicon/types/com/atproto/admin/getAccountInfo.d.ts +29 -0
  146. package/dist/lexicon/types/com/atproto/admin/getAccountInfos.d.ts +32 -0
  147. package/dist/lexicon/types/com/atproto/admin/getInviteCodes.d.ts +35 -0
  148. package/dist/lexicon/types/com/atproto/admin/getModerationEvent.d.ts +29 -0
  149. package/dist/lexicon/types/com/atproto/admin/getRecord.d.ts +31 -0
  150. package/dist/lexicon/types/com/atproto/admin/getRepo.d.ts +30 -0
  151. package/dist/lexicon/types/com/atproto/admin/getSubjectStatus.d.ts +39 -0
  152. package/dist/lexicon/types/com/atproto/admin/queryModerationEvents.d.ts +39 -0
  153. package/dist/lexicon/types/com/atproto/admin/queryModerationStatuses.d.ts +48 -0
  154. package/dist/lexicon/types/com/atproto/admin/searchRepos.d.ts +36 -0
  155. package/dist/lexicon/types/com/atproto/admin/sendEmail.d.ts +40 -0
  156. package/dist/lexicon/types/com/atproto/admin/updateAccountEmail.d.ts +26 -0
  157. package/dist/lexicon/types/com/atproto/admin/updateAccountHandle.d.ts +26 -0
  158. package/dist/lexicon/types/com/atproto/admin/updateSubjectStatus.d.ts +46 -0
  159. package/dist/lexicon/types/com/atproto/identity/resolveHandle.d.ts +31 -0
  160. package/dist/lexicon/types/com/atproto/identity/updateHandle.d.ts +25 -0
  161. package/dist/lexicon/types/com/atproto/label/defs.d.ts +24 -0
  162. package/dist/lexicon/types/com/atproto/label/queryLabels.d.ts +36 -0
  163. package/dist/lexicon/types/com/atproto/label/subscribeLabels.d.ts +36 -0
  164. package/dist/lexicon/types/com/atproto/moderation/createReport.d.ts +52 -0
  165. package/dist/lexicon/types/com/atproto/moderation/defs.d.ts +8 -0
  166. package/dist/lexicon/types/com/atproto/repo/applyWrites.d.ts +53 -0
  167. package/dist/lexicon/types/com/atproto/repo/createRecord.d.ts +43 -0
  168. package/dist/lexicon/types/com/atproto/repo/deleteRecord.d.ts +30 -0
  169. package/dist/lexicon/types/com/atproto/repo/describeRepo.d.ts +35 -0
  170. package/dist/lexicon/types/com/atproto/repo/getRecord.d.ts +36 -0
  171. package/dist/lexicon/types/com/atproto/repo/listRecords.d.ts +47 -0
  172. package/dist/lexicon/types/com/atproto/repo/putRecord.d.ts +44 -0
  173. package/dist/lexicon/types/com/atproto/repo/strongRef.d.ts +8 -0
  174. package/dist/lexicon/types/com/atproto/repo/uploadBlob.d.ts +36 -0
  175. package/dist/lexicon/types/com/atproto/server/confirmEmail.d.ts +27 -0
  176. package/dist/lexicon/types/com/atproto/server/createAccount.d.ts +49 -0
  177. package/dist/lexicon/types/com/atproto/server/createAppPassword.d.ts +43 -0
  178. package/dist/lexicon/types/com/atproto/server/createInviteCode.d.ts +37 -0
  179. package/dist/lexicon/types/com/atproto/server/createInviteCodes.d.ts +46 -0
  180. package/dist/lexicon/types/com/atproto/server/createSession.d.ts +44 -0
  181. package/dist/lexicon/types/com/atproto/server/defs.d.ts +20 -0
  182. package/dist/lexicon/types/com/atproto/server/deleteAccount.d.ts +28 -0
  183. package/dist/lexicon/types/com/atproto/server/deleteSession.d.ts +19 -0
  184. package/dist/lexicon/types/com/atproto/server/describeServer.d.ts +41 -0
  185. package/dist/lexicon/types/com/atproto/server/getAccountInviteCodes.d.ts +34 -0
  186. package/dist/lexicon/types/com/atproto/server/getSession.d.ts +34 -0
  187. package/dist/lexicon/types/com/atproto/server/listAppPasswords.d.ts +39 -0
  188. package/dist/lexicon/types/com/atproto/server/refreshSession.d.ts +35 -0
  189. package/dist/lexicon/types/com/atproto/server/requestAccountDelete.d.ts +19 -0
  190. package/dist/lexicon/types/com/atproto/server/requestEmailConfirmation.d.ts +19 -0
  191. package/dist/lexicon/types/com/atproto/server/requestEmailUpdate.d.ts +30 -0
  192. package/dist/lexicon/types/com/atproto/server/requestPasswordReset.d.ts +25 -0
  193. package/dist/lexicon/types/com/atproto/server/reserveSigningKey.d.ts +36 -0
  194. package/dist/lexicon/types/com/atproto/server/resetPassword.d.ts +27 -0
  195. package/dist/lexicon/types/com/atproto/server/revokeAppPassword.d.ts +25 -0
  196. package/dist/lexicon/types/com/atproto/server/updateEmail.d.ts +27 -0
  197. package/dist/lexicon/types/com/atproto/sync/getBlob.d.ts +30 -0
  198. package/dist/lexicon/types/com/atproto/sync/getBlocks.d.ts +30 -0
  199. package/dist/lexicon/types/com/atproto/sync/getCheckout.d.ts +29 -0
  200. package/dist/lexicon/types/com/atproto/sync/getHead.d.ts +32 -0
  201. package/dist/lexicon/types/com/atproto/sync/getLatestCommit.d.ts +33 -0
  202. package/dist/lexicon/types/com/atproto/sync/getRecord.d.ts +32 -0
  203. package/dist/lexicon/types/com/atproto/sync/getRepo.d.ts +30 -0
  204. package/dist/lexicon/types/com/atproto/sync/listBlobs.d.ts +35 -0
  205. package/dist/lexicon/types/com/atproto/sync/listRepos.d.ts +42 -0
  206. package/dist/lexicon/types/com/atproto/sync/notifyOfUpdate.d.ts +25 -0
  207. package/dist/lexicon/types/com/atproto/sync/requestCrawl.d.ts +25 -0
  208. package/dist/lexicon/types/com/atproto/sync/subscribeRepos.d.ts +80 -0
  209. package/dist/lexicon/types/com/atproto/temp/fetchLabels.d.ts +33 -0
  210. package/dist/lexicon/types/com/atproto/temp/importRepo.d.ts +32 -0
  211. package/dist/lexicon/types/com/atproto/temp/pushBlob.d.ts +25 -0
  212. package/dist/lexicon/types/com/atproto/temp/requestPhoneVerification.d.ts +25 -0
  213. package/dist/lexicon/types/com/atproto/temp/transferAccount.d.ts +42 -0
  214. package/dist/lexicon/util.d.ts +2 -0
  215. package/dist/logger.d.ts +4 -0
  216. package/dist/mod-service/index.d.ts +155 -0
  217. package/dist/mod-service/status.d.ts +12 -0
  218. package/dist/mod-service/subject.d.ts +59 -0
  219. package/dist/mod-service/types.d.ts +19 -0
  220. package/dist/mod-service/views.d.ts +49 -0
  221. package/dist/sequencer/index.d.ts +2 -0
  222. package/dist/sequencer/outbox.d.ts +16 -0
  223. package/dist/sequencer/sequencer.d.ts +35 -0
  224. package/dist/services/index.d.ts +7 -0
  225. package/dist/services/moderation/index.d.ts +144 -0
  226. package/dist/services/moderation/status.d.ts +12 -0
  227. package/dist/services/moderation/subject.d.ts +60 -0
  228. package/dist/services/moderation/types.d.ts +19 -0
  229. package/dist/services/moderation/views.d.ts +42 -0
  230. package/dist/services/types.d.ts +2 -0
  231. package/dist/util/date.d.ts +1 -0
  232. package/dist/util/debug.d.ts +1 -0
  233. package/dist/util/retry.d.ts +3 -0
  234. package/dist/util.d.ts +3 -0
  235. package/jest.config.js +6 -0
  236. package/package.json +67 -0
  237. package/src/api/admin/emitModerationEvent.ts +130 -0
  238. package/src/api/admin/getModerationEvent.ts +19 -0
  239. package/src/api/admin/getRecord.ts +34 -0
  240. package/src/api/admin/getRepo.ts +31 -0
  241. package/src/api/admin/queryModerationEvents.ts +40 -0
  242. package/src/api/admin/queryModerationStatuses.ts +57 -0
  243. package/src/api/admin/searchRepos.ts +42 -0
  244. package/src/api/admin/util.ts +54 -0
  245. package/src/api/health.ts +20 -0
  246. package/src/api/index.ts +28 -0
  247. package/src/api/moderation/createReport.ts +41 -0
  248. package/src/api/moderation/util.ts +65 -0
  249. package/src/api/temp/fetchLabels.ts +29 -0
  250. package/src/api/well-known.ts +35 -0
  251. package/src/auth.ts +147 -0
  252. package/src/background.ts +35 -0
  253. package/src/config/config.ts +84 -0
  254. package/src/config/env.ts +41 -0
  255. package/src/config/index.ts +3 -0
  256. package/src/config/secrets.ts +23 -0
  257. package/src/context.ts +180 -0
  258. package/src/daemon/context.ts +90 -0
  259. package/src/daemon/event-pusher.ts +296 -0
  260. package/src/daemon/event-reverser.ts +74 -0
  261. package/src/daemon/index.ts +33 -0
  262. package/src/db/index.ts +197 -0
  263. package/src/db/migrations/20231219T205730722Z-init.ts +164 -0
  264. package/src/db/migrations/index.ts +5 -0
  265. package/src/db/migrations/provider.ts +25 -0
  266. package/src/db/pagination.ts +216 -0
  267. package/src/db/schema/blob_push_event.ts +21 -0
  268. package/src/db/schema/index.ts +18 -0
  269. package/src/db/schema/label.ts +12 -0
  270. package/src/db/schema/moderation_event.ts +35 -0
  271. package/src/db/schema/moderation_subject_status.ts +32 -0
  272. package/src/db/schema/record_push_event.ts +21 -0
  273. package/src/db/schema/repo_push_event.ts +19 -0
  274. package/src/db/types.ts +15 -0
  275. package/src/error.ts +12 -0
  276. package/src/index.ts +102 -0
  277. package/src/lexicon/index.ts +1626 -0
  278. package/src/lexicon/lexicons.ts +8050 -0
  279. package/src/lexicon/types/app/bsky/actor/defs.ts +235 -0
  280. package/src/lexicon/types/app/bsky/actor/getPreferences.ts +44 -0
  281. package/src/lexicon/types/app/bsky/actor/getProfile.ts +41 -0
  282. package/src/lexicon/types/app/bsky/actor/getProfiles.ts +46 -0
  283. package/src/lexicon/types/app/bsky/actor/getSuggestions.ts +48 -0
  284. package/src/lexicon/types/app/bsky/actor/profile.ts +32 -0
  285. package/src/lexicon/types/app/bsky/actor/putPreferences.ts +39 -0
  286. package/src/lexicon/types/app/bsky/actor/searchActors.ts +52 -0
  287. package/src/lexicon/types/app/bsky/actor/searchActorsTypeahead.ts +50 -0
  288. package/src/lexicon/types/app/bsky/embed/external.ts +82 -0
  289. package/src/lexicon/types/app/bsky/embed/images.ts +96 -0
  290. package/src/lexicon/types/app/bsky/embed/record.ts +120 -0
  291. package/src/lexicon/types/app/bsky/embed/recordWithMedia.ts +53 -0
  292. package/src/lexicon/types/app/bsky/feed/defs.ts +308 -0
  293. package/src/lexicon/types/app/bsky/feed/describeFeedGenerator.ts +80 -0
  294. package/src/lexicon/types/app/bsky/feed/generator.ts +35 -0
  295. package/src/lexicon/types/app/bsky/feed/getActorFeeds.ts +49 -0
  296. package/src/lexicon/types/app/bsky/feed/getActorLikes.ts +50 -0
  297. package/src/lexicon/types/app/bsky/feed/getAuthorFeed.ts +56 -0
  298. package/src/lexicon/types/app/bsky/feed/getFeed.ts +50 -0
  299. package/src/lexicon/types/app/bsky/feed/getFeedGenerator.ts +48 -0
  300. package/src/lexicon/types/app/bsky/feed/getFeedGenerators.ts +46 -0
  301. package/src/lexicon/types/app/bsky/feed/getFeedSkeleton.ts +50 -0
  302. package/src/lexicon/types/app/bsky/feed/getLikes.ts +69 -0
  303. package/src/lexicon/types/app/bsky/feed/getListFeed.ts +50 -0
  304. package/src/lexicon/types/app/bsky/feed/getPostThread.ts +53 -0
  305. package/src/lexicon/types/app/bsky/feed/getPosts.ts +46 -0
  306. package/src/lexicon/types/app/bsky/feed/getRepostedBy.ts +52 -0
  307. package/src/lexicon/types/app/bsky/feed/getSuggestedFeeds.ts +48 -0
  308. package/src/lexicon/types/app/bsky/feed/getTimeline.ts +49 -0
  309. package/src/lexicon/types/app/bsky/feed/like.ts +26 -0
  310. package/src/lexicon/types/app/bsky/feed/post.ts +102 -0
  311. package/src/lexicon/types/app/bsky/feed/repost.ts +27 -0
  312. package/src/lexicon/types/app/bsky/feed/searchPosts.ts +54 -0
  313. package/src/lexicon/types/app/bsky/feed/threadgate.ts +84 -0
  314. package/src/lexicon/types/app/bsky/graph/block.ts +26 -0
  315. package/src/lexicon/types/app/bsky/graph/defs.ts +104 -0
  316. package/src/lexicon/types/app/bsky/graph/follow.ts +26 -0
  317. package/src/lexicon/types/app/bsky/graph/getBlocks.ts +48 -0
  318. package/src/lexicon/types/app/bsky/graph/getFollowers.ts +50 -0
  319. package/src/lexicon/types/app/bsky/graph/getFollows.ts +50 -0
  320. package/src/lexicon/types/app/bsky/graph/getList.ts +50 -0
  321. package/src/lexicon/types/app/bsky/graph/getListBlocks.ts +48 -0
  322. package/src/lexicon/types/app/bsky/graph/getListMutes.ts +48 -0
  323. package/src/lexicon/types/app/bsky/graph/getLists.ts +49 -0
  324. package/src/lexicon/types/app/bsky/graph/getMutes.ts +48 -0
  325. package/src/lexicon/types/app/bsky/graph/getSuggestedFollowsByActor.ts +46 -0
  326. package/src/lexicon/types/app/bsky/graph/list.ts +36 -0
  327. package/src/lexicon/types/app/bsky/graph/listblock.ts +26 -0
  328. package/src/lexicon/types/app/bsky/graph/listitem.ts +27 -0
  329. package/src/lexicon/types/app/bsky/graph/muteActor.ts +38 -0
  330. package/src/lexicon/types/app/bsky/graph/muteActorList.ts +38 -0
  331. package/src/lexicon/types/app/bsky/graph/unmuteActor.ts +38 -0
  332. package/src/lexicon/types/app/bsky/graph/unmuteActorList.ts +38 -0
  333. package/src/lexicon/types/app/bsky/notification/getUnreadCount.ts +45 -0
  334. package/src/lexicon/types/app/bsky/notification/listNotifications.ts +87 -0
  335. package/src/lexicon/types/app/bsky/notification/registerPush.ts +41 -0
  336. package/src/lexicon/types/app/bsky/notification/updateSeen.ts +38 -0
  337. package/src/lexicon/types/app/bsky/richtext/facet.ts +97 -0
  338. package/src/lexicon/types/app/bsky/unspecced/defs.ts +41 -0
  339. package/src/lexicon/types/app/bsky/unspecced/getPopularFeedGenerators.ts +49 -0
  340. package/src/lexicon/types/app/bsky/unspecced/getTimelineSkeleton.ts +49 -0
  341. package/src/lexicon/types/app/bsky/unspecced/searchActorsSkeleton.ts +56 -0
  342. package/src/lexicon/types/app/bsky/unspecced/searchPostsSkeleton.ts +54 -0
  343. package/src/lexicon/types/com/atproto/admin/defs.ts +719 -0
  344. package/src/lexicon/types/com/atproto/admin/deleteAccount.ts +38 -0
  345. package/src/lexicon/types/com/atproto/admin/disableAccountInvites.ts +40 -0
  346. package/src/lexicon/types/com/atproto/admin/disableInviteCodes.ts +39 -0
  347. package/src/lexicon/types/com/atproto/admin/emitModerationEvent.ts +66 -0
  348. package/src/lexicon/types/com/atproto/admin/enableAccountInvites.ts +40 -0
  349. package/src/lexicon/types/com/atproto/admin/getAccountInfo.ts +41 -0
  350. package/src/lexicon/types/com/atproto/admin/getAccountInfos.ts +46 -0
  351. package/src/lexicon/types/com/atproto/admin/getInviteCodes.ts +49 -0
  352. package/src/lexicon/types/com/atproto/admin/getModerationEvent.ts +41 -0
  353. package/src/lexicon/types/com/atproto/admin/getRecord.ts +43 -0
  354. package/src/lexicon/types/com/atproto/admin/getRepo.ts +42 -0
  355. package/src/lexicon/types/com/atproto/admin/getSubjectStatus.ts +54 -0
  356. package/src/lexicon/types/com/atproto/admin/queryModerationEvents.ts +56 -0
  357. package/src/lexicon/types/com/atproto/admin/queryModerationStatuses.ts +72 -0
  358. package/src/lexicon/types/com/atproto/admin/searchRepos.ts +51 -0
  359. package/src/lexicon/types/com/atproto/admin/sendEmail.ts +54 -0
  360. package/src/lexicon/types/com/atproto/admin/updateAccountEmail.ts +40 -0
  361. package/src/lexicon/types/com/atproto/admin/updateAccountHandle.ts +39 -0
  362. package/src/lexicon/types/com/atproto/admin/updateSubjectStatus.ts +61 -0
  363. package/src/lexicon/types/com/atproto/identity/resolveHandle.ts +46 -0
  364. package/src/lexicon/types/com/atproto/identity/updateHandle.ts +38 -0
  365. package/src/lexicon/types/com/atproto/label/defs.ts +73 -0
  366. package/src/lexicon/types/com/atproto/label/queryLabels.ts +52 -0
  367. package/src/lexicon/types/com/atproto/label/subscribeLabels.ts +67 -0
  368. package/src/lexicon/types/com/atproto/moderation/createReport.ts +65 -0
  369. package/src/lexicon/types/com/atproto/moderation/defs.ts +32 -0
  370. package/src/lexicon/types/com/atproto/repo/applyWrites.ts +103 -0
  371. package/src/lexicon/types/com/atproto/repo/createRecord.ts +62 -0
  372. package/src/lexicon/types/com/atproto/repo/deleteRecord.ts +48 -0
  373. package/src/lexicon/types/com/atproto/repo/describeRepo.ts +50 -0
  374. package/src/lexicon/types/com/atproto/repo/getRecord.ts +54 -0
  375. package/src/lexicon/types/com/atproto/repo/listRecords.ts +77 -0
  376. package/src/lexicon/types/com/atproto/repo/putRecord.ts +64 -0
  377. package/src/lexicon/types/com/atproto/repo/strongRef.ts +26 -0
  378. package/src/lexicon/types/com/atproto/repo/uploadBlob.ts +47 -0
  379. package/src/lexicon/types/com/atproto/server/confirmEmail.ts +40 -0
  380. package/src/lexicon/types/com/atproto/server/createAccount.ts +69 -0
  381. package/src/lexicon/types/com/atproto/server/createAppPassword.ts +69 -0
  382. package/src/lexicon/types/com/atproto/server/createInviteCode.ts +50 -0
  383. package/src/lexicon/types/com/atproto/server/createInviteCodes.ts +72 -0
  384. package/src/lexicon/types/com/atproto/server/createSession.ts +58 -0
  385. package/src/lexicon/types/com/atproto/server/defs.ts +48 -0
  386. package/src/lexicon/types/com/atproto/server/deleteAccount.ts +41 -0
  387. package/src/lexicon/types/com/atproto/server/deleteSession.ts +31 -0
  388. package/src/lexicon/types/com/atproto/server/describeServer.ts +64 -0
  389. package/src/lexicon/types/com/atproto/server/getAccountInviteCodes.ts +48 -0
  390. package/src/lexicon/types/com/atproto/server/getSession.ts +47 -0
  391. package/src/lexicon/types/com/atproto/server/listAppPasswords.ts +62 -0
  392. package/src/lexicon/types/com/atproto/server/refreshSession.ts +48 -0
  393. package/src/lexicon/types/com/atproto/server/requestAccountDelete.ts +31 -0
  394. package/src/lexicon/types/com/atproto/server/requestEmailConfirmation.ts +31 -0
  395. package/src/lexicon/types/com/atproto/server/requestEmailUpdate.ts +43 -0
  396. package/src/lexicon/types/com/atproto/server/requestPasswordReset.ts +38 -0
  397. package/src/lexicon/types/com/atproto/server/reserveSigningKey.ts +51 -0
  398. package/src/lexicon/types/com/atproto/server/resetPassword.ts +40 -0
  399. package/src/lexicon/types/com/atproto/server/revokeAppPassword.ts +38 -0
  400. package/src/lexicon/types/com/atproto/server/updateEmail.ts +41 -0
  401. package/src/lexicon/types/com/atproto/sync/getBlob.ts +43 -0
  402. package/src/lexicon/types/com/atproto/sync/getBlocks.ts +42 -0
  403. package/src/lexicon/types/com/atproto/sync/getCheckout.ts +41 -0
  404. package/src/lexicon/types/com/atproto/sync/getHead.ts +47 -0
  405. package/src/lexicon/types/com/atproto/sync/getLatestCommit.ts +48 -0
  406. package/src/lexicon/types/com/atproto/sync/getRecord.ts +45 -0
  407. package/src/lexicon/types/com/atproto/sync/getRepo.ts +43 -0
  408. package/src/lexicon/types/com/atproto/sync/listBlobs.ts +51 -0
  409. package/src/lexicon/types/com/atproto/sync/listRepos.ts +66 -0
  410. package/src/lexicon/types/com/atproto/sync/notifyOfUpdate.ts +39 -0
  411. package/src/lexicon/types/com/atproto/sync/requestCrawl.ts +39 -0
  412. package/src/lexicon/types/com/atproto/sync/subscribeRepos.ts +161 -0
  413. package/src/lexicon/types/com/atproto/temp/fetchLabels.ts +47 -0
  414. package/src/lexicon/types/com/atproto/temp/importRepo.ts +45 -0
  415. package/src/lexicon/types/com/atproto/temp/pushBlob.ts +39 -0
  416. package/src/lexicon/types/com/atproto/temp/requestPhoneVerification.ts +38 -0
  417. package/src/lexicon/types/com/atproto/temp/transferAccount.ts +62 -0
  418. package/src/lexicon/util.ts +13 -0
  419. package/src/logger.ts +19 -0
  420. package/src/mod-service/index.ts +758 -0
  421. package/src/mod-service/status.ts +264 -0
  422. package/src/mod-service/subject.ts +136 -0
  423. package/src/mod-service/types.ts +32 -0
  424. package/src/mod-service/views.ts +531 -0
  425. package/src/util.ts +26 -0
  426. package/test.env +2 -0
  427. package/test.log +0 -0
  428. package/tests/__snapshots__/get-record.test.ts.snap +173 -0
  429. package/tests/__snapshots__/get-repo.test.ts.snap +57 -0
  430. package/tests/__snapshots__/moderation-events.test.ts.snap +168 -0
  431. package/tests/__snapshots__/moderation-statuses.test.ts.snap +64 -0
  432. package/tests/__snapshots__/moderation.test.ts.snap +55 -0
  433. package/tests/_util.ts +192 -0
  434. package/tests/db.test.ts +184 -0
  435. package/tests/get-record.test.ts +110 -0
  436. package/tests/get-repo.test.ts +131 -0
  437. package/tests/moderation-appeals.test.ts +269 -0
  438. package/tests/moderation-events.test.ts +220 -0
  439. package/tests/moderation-statuses.test.ts +144 -0
  440. package/tests/moderation.test.ts +961 -0
  441. package/tests/repo-search.test.ts +123 -0
  442. package/tests/server.test.ts +76 -0
  443. package/tsconfig.build.json +4 -0
  444. package/tsconfig.json +21 -0
@@ -0,0 +1,758 @@
1
+ import { CID } from 'multiformats/cid'
2
+ import { AtUri, INVALID_HANDLE } from '@atproto/syntax'
3
+ import { InvalidRequestError } from '@atproto/xrpc-server'
4
+ import { addHoursToDate } from '@atproto/common'
5
+ import { Database } from '../db'
6
+ import { AppviewAuth, ModerationViews } from './views'
7
+ import { Main as StrongRef } from '../lexicon/types/com/atproto/repo/strongRef'
8
+ import {
9
+ isModEventComment,
10
+ isModEventLabel,
11
+ isModEventMute,
12
+ isModEventReport,
13
+ isModEventTakedown,
14
+ isModEventEmail,
15
+ RepoRef,
16
+ RepoBlobRef,
17
+ } from '../lexicon/types/com/atproto/admin/defs'
18
+ import {
19
+ adjustModerationSubjectStatus,
20
+ getStatusIdentifierFromSubject,
21
+ } from './status'
22
+ import {
23
+ ModEventType,
24
+ ModerationEventRow,
25
+ ModerationSubjectStatusRow,
26
+ ReversibleModerationEvent,
27
+ } from './types'
28
+ import { ModerationEvent } from '../db/schema/moderation_event'
29
+ import { StatusKeyset, TimeIdKeyset, paginate } from '../db/pagination'
30
+ import AtpAgent from '@atproto/api'
31
+ import { Label } from '../lexicon/types/com/atproto/label/defs'
32
+ import { Insertable, sql } from 'kysely'
33
+ import {
34
+ ModSubject,
35
+ RecordSubject,
36
+ RepoSubject,
37
+ subjectFromStatusRow,
38
+ } from './subject'
39
+ import { BlobPushEvent } from '../db/schema/blob_push_event'
40
+ import { BackgroundQueue } from '../background'
41
+ import { EventPusher } from '../daemon'
42
+
43
+ export type ModerationServiceCreator = (db: Database) => ModerationService
44
+
45
+ export class ModerationService {
46
+ constructor(
47
+ public db: Database,
48
+ public backgroundQueue: BackgroundQueue,
49
+ public eventPusher: EventPusher,
50
+ public appviewAgent: AtpAgent,
51
+ private appviewAuth: AppviewAuth,
52
+ ) {}
53
+
54
+ static creator(
55
+ backgroundQueue: BackgroundQueue,
56
+ eventPusher: EventPusher,
57
+ appviewAgent: AtpAgent,
58
+ appviewAuth: AppviewAuth,
59
+ ) {
60
+ return (db: Database) =>
61
+ new ModerationService(
62
+ db,
63
+ backgroundQueue,
64
+ eventPusher,
65
+ appviewAgent,
66
+ appviewAuth,
67
+ )
68
+ }
69
+
70
+ views = new ModerationViews(this.db, this.appviewAgent, this.appviewAuth)
71
+
72
+ async getEvent(id: number): Promise<ModerationEventRow | undefined> {
73
+ return await this.db.db
74
+ .selectFrom('moderation_event')
75
+ .selectAll()
76
+ .where('id', '=', id)
77
+ .executeTakeFirst()
78
+ }
79
+
80
+ async getEventOrThrow(id: number): Promise<ModerationEventRow> {
81
+ const event = await this.getEvent(id)
82
+ if (!event) throw new InvalidRequestError('Moderation event not found')
83
+ return event
84
+ }
85
+
86
+ async getEvents(opts: {
87
+ subject?: string
88
+ createdBy?: string
89
+ limit: number
90
+ cursor?: string
91
+ includeAllUserRecords: boolean
92
+ types: ModerationEvent['action'][]
93
+ sortDirection?: 'asc' | 'desc'
94
+ }): Promise<{ cursor?: string; events: ModerationEventRow[] }> {
95
+ const {
96
+ subject,
97
+ createdBy,
98
+ limit,
99
+ cursor,
100
+ includeAllUserRecords,
101
+ sortDirection = 'desc',
102
+ types,
103
+ } = opts
104
+ let builder = this.db.db.selectFrom('moderation_event').selectAll()
105
+ if (subject) {
106
+ builder = builder.where((qb) => {
107
+ if (includeAllUserRecords) {
108
+ // If subject is an at-uri, we need to extract the DID from the at-uri
109
+ // otherwise, subject is probably a DID already
110
+ if (subject.startsWith('at://')) {
111
+ const uri = new AtUri(subject)
112
+ return qb.where('subjectDid', '=', uri.hostname)
113
+ }
114
+ return qb.where('subjectDid', '=', subject)
115
+ }
116
+ return qb
117
+ .where((subQb) =>
118
+ subQb
119
+ .where('subjectDid', '=', subject)
120
+ .where('subjectUri', 'is', null),
121
+ )
122
+ .orWhere('subjectUri', '=', subject)
123
+ })
124
+ }
125
+ if (types.length) {
126
+ builder = builder.where((qb) => {
127
+ if (types.length === 1) {
128
+ return qb.where('action', '=', types[0])
129
+ }
130
+
131
+ return qb.where('action', 'in', types)
132
+ })
133
+ }
134
+ if (createdBy) {
135
+ builder = builder.where('createdBy', '=', createdBy)
136
+ }
137
+
138
+ const { ref } = this.db.db.dynamic
139
+ const keyset = new TimeIdKeyset(
140
+ ref(`moderation_event.createdAt`),
141
+ ref('moderation_event.id'),
142
+ )
143
+ const paginatedBuilder = paginate(builder, {
144
+ limit,
145
+ cursor,
146
+ keyset,
147
+ direction: sortDirection,
148
+ tryIndex: true,
149
+ })
150
+
151
+ const result = await paginatedBuilder.execute()
152
+
153
+ const infos = await this.views.getAccoutInfosByDid([
154
+ ...result.map((row) => row.subjectDid),
155
+ ...result.map((row) => row.createdBy),
156
+ ])
157
+
158
+ const resultWithHandles = result.map((r) => ({
159
+ ...r,
160
+ creatorHandle: infos.get(r.createdBy)?.handle,
161
+ subjectHandle: infos.get(r.subjectDid)?.handle,
162
+ }))
163
+
164
+ return { cursor: keyset.packFromResult(result), events: resultWithHandles }
165
+ }
166
+
167
+ async getReport(id: number): Promise<ModerationEventRow | undefined> {
168
+ return await this.db.db
169
+ .selectFrom('moderation_event')
170
+ .where('action', '=', 'com.atproto.admin.defs#modEventReport')
171
+ .selectAll()
172
+ .where('id', '=', id)
173
+ .executeTakeFirst()
174
+ }
175
+
176
+ async getCurrentStatus(
177
+ subject: { did: string } | { uri: AtUri } | { cids: CID[] },
178
+ ) {
179
+ let builder = this.db.db.selectFrom('moderation_subject_status').selectAll()
180
+ if ('did' in subject) {
181
+ builder = builder.where('did', '=', subject.did)
182
+ } else if ('uri' in subject) {
183
+ builder = builder.where('recordPath', '=', subject.uri.toString())
184
+ }
185
+ // TODO: Handle the cid status
186
+ return await builder.execute()
187
+ }
188
+
189
+ async logEvent(info: {
190
+ event: ModEventType
191
+ subject: ModSubject
192
+ createdBy: string
193
+ createdAt?: Date
194
+ }): Promise<ModerationEventRow> {
195
+ this.db.assertTransaction()
196
+ const { event, subject, createdBy, createdAt = new Date() } = info
197
+
198
+ const createLabelVals =
199
+ isModEventLabel(event) && event.createLabelVals.length > 0
200
+ ? event.createLabelVals.join(' ')
201
+ : undefined
202
+ const negateLabelVals =
203
+ isModEventLabel(event) && event.negateLabelVals.length > 0
204
+ ? event.negateLabelVals.join(' ')
205
+ : undefined
206
+
207
+ const meta: Record<string, string | boolean> = {}
208
+
209
+ if (isModEventReport(event)) {
210
+ meta.reportType = event.reportType
211
+ }
212
+
213
+ if (isModEventComment(event) && event.sticky) {
214
+ meta.sticky = event.sticky
215
+ }
216
+
217
+ if (isModEventEmail(event)) {
218
+ meta.subjectLine = event.subjectLine
219
+ }
220
+
221
+ const modEvent = await this.db.db
222
+ .insertInto('moderation_event')
223
+ .values({
224
+ comment: event.comment ? `${event.comment}` : null,
225
+ action: event.$type as ModerationEvent['action'],
226
+ createdAt: createdAt.toISOString(),
227
+ createdBy,
228
+ createLabelVals,
229
+ negateLabelVals,
230
+ durationInHours: event.durationInHours
231
+ ? Number(event.durationInHours)
232
+ : null,
233
+ meta,
234
+ expiresAt:
235
+ (isModEventTakedown(event) || isModEventMute(event)) &&
236
+ event.durationInHours
237
+ ? addHoursToDate(event.durationInHours, createdAt).toISOString()
238
+ : undefined,
239
+ ...subject.info(),
240
+ })
241
+ .returningAll()
242
+ .executeTakeFirstOrThrow()
243
+
244
+ await adjustModerationSubjectStatus(this.db, modEvent, subject.blobCids)
245
+
246
+ return modEvent
247
+ }
248
+
249
+ async getLastReversibleEventForSubject(subject: ReversalSubject) {
250
+ // If the subject is neither suspended nor muted don't bother finding the last reversible event
251
+ // Ideally, this should never happen because the caller of this method should only call this
252
+ // after ensuring that the suspended or muted subjects are being reversed
253
+ if (!subject.reverseMute && !subject.reverseSuspend) {
254
+ return null
255
+ }
256
+
257
+ let builder = this.db.db
258
+ .selectFrom('moderation_event')
259
+ .where('subjectDid', '=', subject.subject.did)
260
+
261
+ if (subject.subject.recordPath) {
262
+ builder = builder.where(
263
+ 'subjectUri',
264
+ 'like',
265
+ `%${subject.subject.recordPath}%`,
266
+ )
267
+ }
268
+
269
+ // Means the subject was suspended and needs to be unsuspended
270
+ if (subject.reverseSuspend) {
271
+ builder = builder
272
+ .where('action', '=', 'com.atproto.admin.defs#modEventTakedown')
273
+ .where('durationInHours', 'is not', null)
274
+ }
275
+ if (subject.reverseMute) {
276
+ builder = builder
277
+ .where('action', '=', 'com.atproto.admin.defs#modEventMute')
278
+ .where('durationInHours', 'is not', null)
279
+ }
280
+
281
+ return await builder
282
+ .orderBy('id', 'desc')
283
+ .selectAll()
284
+ .limit(1)
285
+ .executeTakeFirst()
286
+ }
287
+
288
+ async getSubjectsDueForReversal(): Promise<ReversalSubject[]> {
289
+ const now = new Date().toISOString()
290
+ const subjects = await this.db.db
291
+ .selectFrom('moderation_subject_status')
292
+ .where('suspendUntil', '<', now)
293
+ .orWhere('muteUntil', '<', now)
294
+ .selectAll()
295
+ .execute()
296
+
297
+ return subjects.map((row) => ({
298
+ subject: subjectFromStatusRow(row),
299
+ reverseSuspend: !!row.suspendUntil && row.suspendUntil < now,
300
+ reverseMute: !!row.muteUntil && row.muteUntil < now,
301
+ }))
302
+ }
303
+
304
+ async isSubjectSuspended(did: string): Promise<boolean> {
305
+ const res = await this.db.db
306
+ .selectFrom('moderation_subject_status')
307
+ .where('did', '=', did)
308
+ .where('recordPath', '=', '')
309
+ .where('suspendUntil', '>', new Date().toISOString())
310
+ .select('did')
311
+ .limit(1)
312
+ .executeTakeFirst()
313
+ return !!res
314
+ }
315
+
316
+ async revertState({
317
+ createdBy,
318
+ createdAt,
319
+ comment,
320
+ action,
321
+ subject,
322
+ }: ReversibleModerationEvent): Promise<ModerationEventRow> {
323
+ const isRevertingTakedown =
324
+ action === 'com.atproto.admin.defs#modEventTakedown'
325
+ this.db.assertTransaction()
326
+ const result = await this.logEvent({
327
+ event: {
328
+ $type: isRevertingTakedown
329
+ ? 'com.atproto.admin.defs#modEventReverseTakedown'
330
+ : 'com.atproto.admin.defs#modEventUnmute',
331
+ comment: comment ?? undefined,
332
+ },
333
+ createdAt,
334
+ createdBy,
335
+ subject,
336
+ })
337
+
338
+ if (isRevertingTakedown) {
339
+ if (subject.isRepo()) {
340
+ await this.reverseTakedownRepo(subject)
341
+ } else if (subject.isRecord()) {
342
+ await this.reverseTakedownRecord(subject)
343
+ }
344
+ }
345
+
346
+ return result
347
+ }
348
+
349
+ async takedownRepo(
350
+ subject: RepoSubject,
351
+ takedownId: number,
352
+ isSuspend = false,
353
+ ) {
354
+ const takedownRef = `BSKY-${
355
+ isSuspend ? 'SUSPEND' : 'TAKEDOWN'
356
+ }-${takedownId}`
357
+ const values = TAKEDOWNS.map((eventType) => ({
358
+ eventType,
359
+ subjectDid: subject.did,
360
+ takedownRef,
361
+ }))
362
+ const repoEvts = await this.db.db
363
+ .insertInto('repo_push_event')
364
+ .values(values)
365
+ .onConflict((oc) =>
366
+ oc.columns(['subjectDid', 'eventType']).doUpdateSet({
367
+ takedownRef,
368
+ confirmedAt: null,
369
+ attempts: 0,
370
+ lastAttempted: null,
371
+ }),
372
+ )
373
+ .returning('id')
374
+ .execute()
375
+
376
+ this.db.onCommit(() => {
377
+ this.backgroundQueue.add(async () => {
378
+ await Promise.all(
379
+ repoEvts.map((evt) => this.eventPusher.attemptRepoEvent(evt.id)),
380
+ )
381
+ })
382
+ })
383
+ }
384
+
385
+ async reverseTakedownRepo(subject: RepoSubject) {
386
+ const repoEvts = await this.db.db
387
+ .updateTable('repo_push_event')
388
+ .where('eventType', 'in', TAKEDOWNS)
389
+ .where('subjectDid', '=', subject.did)
390
+ .set({
391
+ takedownRef: null,
392
+ confirmedAt: null,
393
+ attempts: 0,
394
+ lastAttempted: null,
395
+ })
396
+ .returning('id')
397
+ .execute()
398
+
399
+ this.db.onCommit(() => {
400
+ this.backgroundQueue.add(async () => {
401
+ await Promise.all(
402
+ repoEvts.map((evt) => this.eventPusher.attemptRepoEvent(evt.id)),
403
+ )
404
+ })
405
+ })
406
+ }
407
+
408
+ async takedownRecord(subject: RecordSubject, takedownId: number) {
409
+ this.db.assertTransaction()
410
+ const takedownRef = `BSKY-TAKEDOWN-${takedownId}`
411
+ const values = TAKEDOWNS.map((eventType) => ({
412
+ eventType,
413
+ subjectDid: subject.did,
414
+ subjectUri: subject.uri,
415
+ subjectCid: subject.cid,
416
+ takedownRef,
417
+ }))
418
+ const recordEvts = await this.db.db
419
+ .insertInto('record_push_event')
420
+ .values(values)
421
+ .onConflict((oc) =>
422
+ oc.columns(['subjectUri', 'eventType']).doUpdateSet({
423
+ takedownRef,
424
+ confirmedAt: null,
425
+ attempts: 0,
426
+ lastAttempted: null,
427
+ }),
428
+ )
429
+ .returning('id')
430
+ .execute()
431
+
432
+ this.db.onCommit(() => {
433
+ this.backgroundQueue.add(async () => {
434
+ await Promise.all(
435
+ recordEvts.map((evt) => this.eventPusher.attemptRecordEvent(evt.id)),
436
+ )
437
+ })
438
+ })
439
+
440
+ const blobCids = subject.blobCids
441
+ if (blobCids && blobCids.length > 0) {
442
+ const blobValues: Insertable<BlobPushEvent>[] = []
443
+ for (const eventType of TAKEDOWNS) {
444
+ for (const cid of blobCids) {
445
+ blobValues.push({
446
+ eventType,
447
+ subjectDid: subject.did,
448
+ subjectBlobCid: cid.toString(),
449
+ takedownRef,
450
+ })
451
+ }
452
+ }
453
+ const blobEvts = await this.db.db
454
+ .insertInto('blob_push_event')
455
+ .values(blobValues)
456
+ .onConflict((oc) =>
457
+ oc
458
+ .columns(['subjectDid', 'subjectBlobCid', 'eventType'])
459
+ .doUpdateSet({
460
+ takedownRef,
461
+ confirmedAt: null,
462
+ attempts: 0,
463
+ lastAttempted: null,
464
+ }),
465
+ )
466
+ .returning('id')
467
+ .execute()
468
+
469
+ this.db.onCommit(() => {
470
+ this.backgroundQueue.add(async () => {
471
+ await Promise.all(
472
+ blobEvts.map((evt) => this.eventPusher.attemptBlobEvent(evt.id)),
473
+ )
474
+ })
475
+ })
476
+ }
477
+ }
478
+
479
+ async reverseTakedownRecord(subject: RecordSubject) {
480
+ this.db.assertTransaction()
481
+ const recordEvts = await this.db.db
482
+ .updateTable('record_push_event')
483
+ .where('eventType', 'in', TAKEDOWNS)
484
+ .where('subjectDid', '=', subject.did)
485
+ .where('subjectUri', '=', subject.uri)
486
+ .set({
487
+ takedownRef: null,
488
+ confirmedAt: null,
489
+ attempts: 0,
490
+ lastAttempted: null,
491
+ })
492
+ .returning('id')
493
+ .execute()
494
+ this.db.onCommit(() => {
495
+ this.backgroundQueue.add(async () => {
496
+ await Promise.all(
497
+ recordEvts.map((evt) => this.eventPusher.attemptRecordEvent(evt.id)),
498
+ )
499
+ })
500
+ })
501
+
502
+ const blobCids = subject.blobCids
503
+ if (blobCids && blobCids.length > 0) {
504
+ const blobEvts = await this.db.db
505
+ .updateTable('blob_push_event')
506
+ .where('eventType', 'in', TAKEDOWNS)
507
+ .where('subjectDid', '=', subject.did)
508
+ .where(
509
+ 'subjectBlobCid',
510
+ 'in',
511
+ blobCids.map((c) => c.toString()),
512
+ )
513
+ .set({
514
+ takedownRef: null,
515
+ confirmedAt: null,
516
+ attempts: 0,
517
+ lastAttempted: null,
518
+ })
519
+ .returning('id')
520
+ .execute()
521
+
522
+ this.db.onCommit(() => {
523
+ this.backgroundQueue.add(async () => {
524
+ await Promise.all(
525
+ blobEvts.map((evt) => this.eventPusher.attemptBlobEvent(evt.id)),
526
+ )
527
+ })
528
+ })
529
+ }
530
+ }
531
+
532
+ async report(info: {
533
+ reasonType: NonNullable<ModerationEventRow['meta']>['reportType']
534
+ reason?: string
535
+ subject: ModSubject
536
+ reportedBy: string
537
+ createdAt?: Date
538
+ }): Promise<ModerationEventRow> {
539
+ const {
540
+ reasonType,
541
+ reason,
542
+ reportedBy,
543
+ createdAt = new Date(),
544
+ subject,
545
+ } = info
546
+
547
+ const event = await this.logEvent({
548
+ event: {
549
+ $type: 'com.atproto.admin.defs#modEventReport',
550
+ reportType: reasonType,
551
+ comment: reason,
552
+ },
553
+ createdBy: reportedBy,
554
+ subject,
555
+ createdAt,
556
+ })
557
+
558
+ return event
559
+ }
560
+
561
+ async getSubjectStatuses({
562
+ cursor,
563
+ limit = 50,
564
+ takendown,
565
+ appealed,
566
+ reviewState,
567
+ reviewedAfter,
568
+ reviewedBefore,
569
+ reportedAfter,
570
+ reportedBefore,
571
+ includeMuted,
572
+ ignoreSubjects,
573
+ sortDirection,
574
+ lastReviewedBy,
575
+ sortField,
576
+ subject,
577
+ }: {
578
+ cursor?: string
579
+ limit?: number
580
+ takendown?: boolean
581
+ appealed?: boolean | null
582
+ reviewedBefore?: string
583
+ reviewState?: ModerationSubjectStatusRow['reviewState']
584
+ reviewedAfter?: string
585
+ reportedAfter?: string
586
+ reportedBefore?: string
587
+ includeMuted?: boolean
588
+ subject?: string
589
+ ignoreSubjects?: string[]
590
+ sortDirection: 'asc' | 'desc'
591
+ lastReviewedBy?: string
592
+ sortField: 'lastReviewedAt' | 'lastReportedAt'
593
+ }) {
594
+ let builder = this.db.db.selectFrom('moderation_subject_status').selectAll()
595
+
596
+ if (subject) {
597
+ const subjectInfo = getStatusIdentifierFromSubject(subject)
598
+ builder = builder
599
+ .where('moderation_subject_status.did', '=', subjectInfo.did)
600
+ .where((qb) =>
601
+ subjectInfo.recordPath
602
+ ? qb.where('recordPath', '=', subjectInfo.recordPath)
603
+ : qb.where('recordPath', '=', ''),
604
+ )
605
+ }
606
+
607
+ if (ignoreSubjects?.length) {
608
+ builder = builder
609
+ .where('moderation_subject_status.did', 'not in', ignoreSubjects)
610
+ .where('recordPath', 'not in', ignoreSubjects)
611
+ }
612
+
613
+ if (reviewState) {
614
+ builder = builder.where('reviewState', '=', reviewState)
615
+ }
616
+
617
+ if (lastReviewedBy) {
618
+ builder = builder.where('lastReviewedBy', '=', lastReviewedBy)
619
+ }
620
+
621
+ if (reviewedAfter) {
622
+ builder = builder.where('lastReviewedAt', '>', reviewedAfter)
623
+ }
624
+
625
+ if (reviewedBefore) {
626
+ builder = builder.where('lastReviewedAt', '<', reviewedBefore)
627
+ }
628
+
629
+ if (reportedAfter) {
630
+ builder = builder.where('lastReviewedAt', '>', reportedAfter)
631
+ }
632
+
633
+ if (reportedBefore) {
634
+ builder = builder.where('lastReportedAt', '<', reportedBefore)
635
+ }
636
+
637
+ if (takendown) {
638
+ builder = builder.where('takendown', '=', true)
639
+ }
640
+
641
+ if (appealed !== undefined) {
642
+ builder =
643
+ appealed === null
644
+ ? builder.where('appealed', 'is', null)
645
+ : builder.where('appealed', '=', appealed)
646
+ }
647
+
648
+ if (!includeMuted) {
649
+ builder = builder.where((qb) =>
650
+ qb
651
+ .where('muteUntil', '<', new Date().toISOString())
652
+ .orWhere('muteUntil', 'is', null),
653
+ )
654
+ }
655
+
656
+ const { ref } = this.db.db.dynamic
657
+ const keyset = new StatusKeyset(
658
+ ref(`moderation_subject_status.${sortField}`),
659
+ ref('moderation_subject_status.id'),
660
+ )
661
+ const paginatedBuilder = paginate(builder, {
662
+ limit,
663
+ cursor,
664
+ keyset,
665
+ direction: sortDirection,
666
+ tryIndex: true,
667
+ nullsLast: true,
668
+ })
669
+
670
+ const results = await paginatedBuilder.execute()
671
+
672
+ const infos = await this.views.getAccoutInfosByDid(
673
+ results.map((r) => r.did),
674
+ )
675
+ const resultsWithHandles = results.map((r) => ({
676
+ ...r,
677
+ handle: infos.get(r.did)?.handle ?? INVALID_HANDLE,
678
+ }))
679
+
680
+ return {
681
+ statuses: resultsWithHandles,
682
+ cursor: keyset.packFromResult(results),
683
+ }
684
+ }
685
+
686
+ async isSubjectTakendown(subject: ModSubject): Promise<boolean> {
687
+ const builder = this.db.db
688
+ .selectFrom('moderation_subject_status')
689
+ .where('did', '=', subject.did)
690
+ .where('recordPath', '=', subject.recordPath || '')
691
+
692
+ const result = await builder.select('takendown').executeTakeFirst()
693
+
694
+ return !!result?.takendown
695
+ }
696
+
697
+ async formatAndCreateLabels(
698
+ src: string,
699
+ uri: string,
700
+ cid: string | null,
701
+ labels: { create?: string[]; negate?: string[] },
702
+ ): Promise<Label[]> {
703
+ const { create = [], negate = [] } = labels
704
+ const toCreate = create.map((val) => ({
705
+ src,
706
+ uri,
707
+ cid: cid ?? undefined,
708
+ val,
709
+ neg: false,
710
+ cts: new Date().toISOString(),
711
+ }))
712
+ const toNegate = negate.map((val) => ({
713
+ src,
714
+ uri,
715
+ cid: cid ?? undefined,
716
+ val,
717
+ neg: true,
718
+ cts: new Date().toISOString(),
719
+ }))
720
+ const formatted = [...toCreate, ...toNegate]
721
+ await this.createLabels(formatted)
722
+ return formatted
723
+ }
724
+
725
+ async createLabels(labels: Label[]) {
726
+ if (labels.length < 1) return
727
+ const dbVals = labels.map((l) => ({
728
+ ...l,
729
+ cid: l.cid ?? '',
730
+ neg: !!l.neg,
731
+ }))
732
+ const { ref } = this.db.db.dynamic
733
+ const excluded = (col: string) => ref(`excluded.${col}`)
734
+ await this.db.db
735
+ .insertInto('label')
736
+ .values(dbVals)
737
+ .onConflict((oc) =>
738
+ oc.columns(['src', 'uri', 'cid', 'val']).doUpdateSet({
739
+ neg: sql`${excluded('neg')}`,
740
+ cts: sql`${excluded('cts')}`,
741
+ }),
742
+ )
743
+ .execute()
744
+ }
745
+ }
746
+
747
+ const TAKEDOWNS = ['pds_takedown' as const, 'appview_takedown' as const]
748
+
749
+ export type TakedownSubjects = {
750
+ did: string
751
+ subjects: (RepoRef | RepoBlobRef | StrongRef)[]
752
+ }
753
+
754
+ export type ReversalSubject = {
755
+ subject: ModSubject
756
+ reverseSuspend: boolean
757
+ reverseMute: boolean
758
+ }