@atproto/pds 0.4.117 → 0.4.118

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 (137) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/dist/account-manager/account-manager.d.ts +5 -2
  3. package/dist/account-manager/account-manager.d.ts.map +1 -1
  4. package/dist/account-manager/account-manager.js.map +1 -1
  5. package/dist/actor-store/blob/transactor.d.ts +2 -1
  6. package/dist/actor-store/blob/transactor.d.ts.map +1 -1
  7. package/dist/actor-store/blob/transactor.js +19 -5
  8. package/dist/actor-store/blob/transactor.js.map +1 -1
  9. package/dist/api/com/atproto/repo/uploadBlob.d.ts.map +1 -1
  10. package/dist/api/com/atproto/repo/uploadBlob.js +1 -0
  11. package/dist/api/com/atproto/repo/uploadBlob.js.map +1 -1
  12. package/dist/config/env.d.ts +1 -0
  13. package/dist/config/env.d.ts.map +1 -1
  14. package/dist/config/env.js +1 -0
  15. package/dist/config/env.js.map +1 -1
  16. package/dist/config/secrets.d.ts +1 -0
  17. package/dist/config/secrets.d.ts.map +1 -1
  18. package/dist/config/secrets.js +1 -0
  19. package/dist/config/secrets.js.map +1 -1
  20. package/dist/context.d.ts +3 -0
  21. package/dist/context.d.ts.map +1 -1
  22. package/dist/context.js +19 -0
  23. package/dist/context.js.map +1 -1
  24. package/dist/lexicon/index.d.ts +10 -0
  25. package/dist/lexicon/index.d.ts.map +1 -1
  26. package/dist/lexicon/index.js +20 -0
  27. package/dist/lexicon/index.js.map +1 -1
  28. package/dist/lexicon/lexicons.d.ts +516 -0
  29. package/dist/lexicon/lexicons.d.ts.map +1 -1
  30. package/dist/lexicon/lexicons.js +277 -0
  31. package/dist/lexicon/lexicons.js.map +1 -1
  32. package/dist/lexicon/types/app/bsky/unspecced/defs.d.ts +27 -0
  33. package/dist/lexicon/types/app/bsky/unspecced/defs.d.ts.map +1 -1
  34. package/dist/lexicon/types/app/bsky/unspecced/defs.js +18 -0
  35. package/dist/lexicon/types/app/bsky/unspecced/defs.js.map +1 -1
  36. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.d.ts +36 -0
  37. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.d.ts.map +1 -0
  38. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.js +7 -0
  39. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.js.map +1 -0
  40. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.d.ts +37 -0
  41. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.d.ts.map +1 -0
  42. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.js +7 -0
  43. package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.js.map +1 -0
  44. package/dist/lexicon/types/app/bsky/unspecced/getTrends.d.ts +36 -0
  45. package/dist/lexicon/types/app/bsky/unspecced/getTrends.d.ts.map +1 -0
  46. package/dist/lexicon/types/app/bsky/unspecced/getTrends.js +7 -0
  47. package/dist/lexicon/types/app/bsky/unspecced/getTrends.js.map +1 -0
  48. package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.d.ts +38 -0
  49. package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.d.ts.map +1 -0
  50. package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.js +7 -0
  51. package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.js.map +1 -0
  52. package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.d.ts +31 -0
  53. package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.d.ts.map +1 -0
  54. package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.js +7 -0
  55. package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.js.map +1 -0
  56. package/dist/repo/prepare.d.ts.map +1 -1
  57. package/dist/repo/prepare.js +1 -0
  58. package/dist/repo/prepare.js.map +1 -1
  59. package/dist/repo/types.d.ts +1 -0
  60. package/dist/repo/types.d.ts.map +1 -1
  61. package/dist/repo/types.js.map +1 -1
  62. package/dist/scripts/index.d.ts +8 -1
  63. package/dist/scripts/index.d.ts.map +1 -1
  64. package/dist/scripts/index.js +11 -0
  65. package/dist/scripts/index.js.map +1 -1
  66. package/dist/scripts/publish-identity.d.ts +8 -0
  67. package/dist/scripts/publish-identity.d.ts.map +1 -0
  68. package/dist/scripts/publish-identity.js +46 -0
  69. package/dist/scripts/publish-identity.js.map +1 -0
  70. package/dist/scripts/rebuild-repo.d.ts +10 -2
  71. package/dist/scripts/rebuild-repo.d.ts.map +1 -1
  72. package/dist/scripts/rebuild-repo.js +21 -12
  73. package/dist/scripts/rebuild-repo.js.map +1 -1
  74. package/dist/scripts/rotate-keys.d.ts +18 -0
  75. package/dist/scripts/rotate-keys.d.ts.map +1 -0
  76. package/dist/scripts/rotate-keys.js +115 -0
  77. package/dist/scripts/rotate-keys.js.map +1 -0
  78. package/dist/scripts/sequencer-recovery/index.d.ts +3 -0
  79. package/dist/scripts/sequencer-recovery/index.d.ts.map +1 -0
  80. package/dist/scripts/sequencer-recovery/index.js +17 -0
  81. package/dist/scripts/sequencer-recovery/index.js.map +1 -0
  82. package/dist/scripts/sequencer-recovery/recoverer.d.ts +27 -0
  83. package/dist/scripts/sequencer-recovery/recoverer.d.ts.map +1 -0
  84. package/dist/scripts/sequencer-recovery/recoverer.js +235 -0
  85. package/dist/scripts/sequencer-recovery/recoverer.js.map +1 -0
  86. package/dist/scripts/sequencer-recovery/recovery-db.d.ts +18 -0
  87. package/dist/scripts/sequencer-recovery/recovery-db.d.ts.map +1 -0
  88. package/dist/scripts/sequencer-recovery/recovery-db.js +45 -0
  89. package/dist/scripts/sequencer-recovery/recovery-db.js.map +1 -0
  90. package/dist/scripts/sequencer-recovery/repair-repos.d.ts +3 -0
  91. package/dist/scripts/sequencer-recovery/repair-repos.d.ts.map +1 -0
  92. package/dist/scripts/sequencer-recovery/repair-repos.js +45 -0
  93. package/dist/scripts/sequencer-recovery/repair-repos.js.map +1 -0
  94. package/dist/scripts/sequencer-recovery/user-queues.d.ts +12 -0
  95. package/dist/scripts/sequencer-recovery/user-queues.d.ts.map +1 -0
  96. package/dist/scripts/sequencer-recovery/user-queues.js +54 -0
  97. package/dist/scripts/sequencer-recovery/user-queues.js.map +1 -0
  98. package/dist/scripts/util.d.ts +2 -0
  99. package/dist/scripts/util.d.ts.map +1 -0
  100. package/dist/scripts/util.js +12 -0
  101. package/dist/scripts/util.js.map +1 -0
  102. package/dist/sequencer/sequencer.d.ts +2 -0
  103. package/dist/sequencer/sequencer.d.ts.map +1 -1
  104. package/dist/sequencer/sequencer.js +52 -42
  105. package/dist/sequencer/sequencer.js.map +1 -1
  106. package/package.json +10 -10
  107. package/src/account-manager/account-manager.ts +6 -2
  108. package/src/actor-store/blob/transactor.ts +25 -7
  109. package/src/api/com/atproto/repo/uploadBlob.ts +1 -0
  110. package/src/config/env.ts +2 -0
  111. package/src/config/secrets.ts +2 -0
  112. package/src/context.ts +23 -0
  113. package/src/lexicon/index.ts +62 -0
  114. package/src/lexicon/lexicons.ts +285 -0
  115. package/src/lexicon/types/app/bsky/unspecced/defs.ts +45 -0
  116. package/src/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.ts +54 -0
  117. package/src/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.ts +55 -0
  118. package/src/lexicon/types/app/bsky/unspecced/getTrends.ts +54 -0
  119. package/src/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.ts +56 -0
  120. package/src/lexicon/types/com/atproto/admin/updateAccountSigningKey.ts +48 -0
  121. package/src/repo/prepare.ts +1 -0
  122. package/src/repo/types.ts +1 -0
  123. package/src/scripts/README.md +40 -0
  124. package/src/scripts/index.ts +15 -0
  125. package/src/scripts/publish-identity.ts +54 -0
  126. package/src/scripts/rebuild-repo.ts +39 -12
  127. package/src/scripts/rotate-keys.ts +141 -0
  128. package/src/scripts/sequencer-recovery/index.ts +23 -0
  129. package/src/scripts/sequencer-recovery/recoverer.ts +289 -0
  130. package/src/scripts/sequencer-recovery/recovery-db.ts +64 -0
  131. package/src/scripts/sequencer-recovery/repair-repos.ts +48 -0
  132. package/src/scripts/sequencer-recovery/user-queues.ts +41 -0
  133. package/src/scripts/util.ts +7 -0
  134. package/src/sequencer/sequencer.ts +43 -40
  135. package/tests/recovery.test.ts +178 -0
  136. package/tsconfig.build.tsbuildinfo +1 -1
  137. package/tsconfig.tests.tsbuildinfo +1 -1
@@ -0,0 +1,48 @@
1
+ /**
2
+ * GENERATED CODE - DO NOT MODIFY
3
+ */
4
+ import express from 'express'
5
+ import { type ValidationResult, BlobRef } from '@atproto/lexicon'
6
+ import { CID } from 'multiformats/cid'
7
+ import { validate as _validate } from '../../../../lexicons'
8
+ import {
9
+ type $Typed,
10
+ is$typed as _is$typed,
11
+ type OmitKey,
12
+ } from '../../../../util'
13
+ import { HandlerAuth, HandlerPipeThrough } from '@atproto/xrpc-server'
14
+
15
+ const is$typed = _is$typed,
16
+ validate = _validate
17
+ const id = 'com.atproto.admin.updateAccountSigningKey'
18
+
19
+ export interface QueryParams {}
20
+
21
+ export interface InputSchema {
22
+ did: string
23
+ /** Did-key formatted public key */
24
+ signingKey: string
25
+ }
26
+
27
+ export interface HandlerInput {
28
+ encoding: 'application/json'
29
+ body: InputSchema
30
+ }
31
+
32
+ export interface HandlerError {
33
+ status: number
34
+ message?: string
35
+ }
36
+
37
+ export type HandlerOutput = HandlerError | void
38
+ export type HandlerReqCtx<HA extends HandlerAuth = never> = {
39
+ auth: HA
40
+ params: QueryParams
41
+ input: HandlerInput
42
+ req: express.Request
43
+ res: express.Response
44
+ resetRouteRateLimits: () => Promise<void>
45
+ }
46
+ export type Handler<HA extends HandlerAuth = never> = (
47
+ ctx: HandlerReqCtx<HA>,
48
+ ) => Promise<HandlerOutput> | HandlerOutput
@@ -278,6 +278,7 @@ export const blobsForWrite = (
278
278
  }
279
279
 
280
280
  return refs.map(({ ref, path }) => ({
281
+ size: ref.size,
281
282
  cid: ref.ref,
282
283
  mimeType: ref.mimeType,
283
284
  constraints:
package/src/repo/types.ts CHANGED
@@ -11,6 +11,7 @@ export type BlobConstraint = {
11
11
  }
12
12
 
13
13
  export type PreparedBlobRef = {
14
+ size: number
14
15
  cid: CID
15
16
  mimeType: string
16
17
  constraints: BlobConstraint
@@ -0,0 +1,40 @@
1
+ # PDS Scripts
2
+
3
+ This directory includes some low-level administrative scripts primarily meant to help recover from situations of data loss or repository corruption.
4
+
5
+ These scripts are included in the Docker image. The recommended way to run them is to shell into the PDS container (`docker exec -it pds /bin/sh`) and run the relevant script using `node run-script.js SCRIPT_NAME`.
6
+
7
+ ### rebuild-repo
8
+
9
+ Rebuild a repo's MST and sign a new commit based on data stored in the actor's record table. Intended to be used if a repository is corrupted and there are missing MST blocks.
10
+
11
+ `node run-script.js rebuild-repo DID`
12
+
13
+ ### publish-identity
14
+
15
+ Publishes an identity event on the PDS's outgoing firehose for the relevant DID. Intended to be used if a user's identity is out of date and a refresh of their identity needs to be pushed through the system.
16
+
17
+ `node run-script.js publish-identity DID`
18
+ `node run-script.js publish-identity DID1 DID2 DID3 ...`
19
+ `node run-script.js publish-identity-file dids.txt` (where `dids.txt` is a `\n` delimited text file of dids)
20
+
21
+ ### rotate-keys
22
+
23
+ Ensures that an account's signing key in their PLC DID document matches the signing key that the PDS is holidng for them locally. If not, then update their PLC document. Does not work for `did:web`s. Intended to be used in recovery situations where an accounts' signing key is lost and it needs to be re-generated. This script _does not_ regenerate the key.
24
+
25
+ `node run-script.js rotate-keys DID`
26
+ `node run-script.js rotate-keys DID1 DID2 DID3 ...`
27
+ `node run-script.js rotate-keys-file dids.txt` (where `dids.txt` is a `\n` delimited text file of dids)
28
+ `node run-script.js rotate-keys-recovery` (to be used after `sequencer-recovery` script)
29
+
30
+ ### sequencer-recovery
31
+
32
+ Replays the sequencer file on top of actor stores. Creates new actor stores & keys for actors that do not exist but are in the sequencer. Deletes actor stores & entries in the accounts DB when processing an account deletion on the stream. This script is meant to be re-runnable. Though because it processes events in parallel, it is important to be discerning about the cursor you pick up from if you stop & start the script.
33
+
34
+ Does _not_ rotate signing keys even if it generates them. Signing keys that need to be rotated are stored in a recovery DB and can be actually rotated with the `rotate-keys-recovery` script.
35
+
36
+ Intended to be used for recovery from data loss.
37
+
38
+ `node run-script.js sequencer-recovery START_CURSOR CONCURRENCY` (both params are optional & default to `0` and `10` respectively)
39
+
40
+ Failures are also tracked in the recovery DB and can be recovered from with `node run-script.js recovery-repair-repos`. Which will rebuild repos (a la `rebuild-repo`) and then play back the events from the sequencer only pertaining to the recovered DIDs.
@@ -1,5 +1,20 @@
1
+ import { publishIdentity, publishIdentityFromFile } from './publish-identity'
1
2
  import { rebuildRepo } from './rebuild-repo'
3
+ import {
4
+ rotateKeys,
5
+ rotateKeysFromFile,
6
+ rotateKeysRecovery,
7
+ } from './rotate-keys'
8
+ import { sequencerRecovery } from './sequencer-recovery'
9
+ import { repairRepos } from './sequencer-recovery/repair-repos'
2
10
 
3
11
  export const scripts = {
4
12
  'rebuild-repo': rebuildRepo,
13
+ 'sequencer-recovery': sequencerRecovery,
14
+ 'recovery-repair-repos': repairRepos,
15
+ 'rotate-keys': rotateKeys,
16
+ 'rotate-keys-file': rotateKeysFromFile,
17
+ 'rotate-keys-recovery': rotateKeysRecovery,
18
+ 'publish-identity': publishIdentity,
19
+ 'publish-identity-file': publishIdentityFromFile,
5
20
  }
@@ -0,0 +1,54 @@
1
+ import fs from 'node:fs/promises'
2
+ import { wait } from '@atproto/common'
3
+ import { Sequencer } from '../sequencer'
4
+ import { parseIntArg } from './util'
5
+
6
+ export type PublishIdentityContext = {
7
+ sequencer: Sequencer
8
+ }
9
+
10
+ export const publishIdentity = async (
11
+ ctx: PublishIdentityContext,
12
+ args: string[],
13
+ ) => {
14
+ const dids = args
15
+ await publishIdentityEvtForDids(ctx, dids)
16
+ console.log('DONE')
17
+ }
18
+
19
+ export const publishIdentityFromFile = async (
20
+ ctx: PublishIdentityContext,
21
+ args: string[],
22
+ ) => {
23
+ const filepath = args[0]
24
+ if (!filepath) {
25
+ throw new Error('Expected filepath as argument')
26
+ }
27
+ const timeBetween = args[1] ? parseIntArg(args[1]) : 5
28
+ const file = await fs.readFile(filepath)
29
+ const dids = file
30
+ .toString()
31
+ .split('\n')
32
+ .map((did) => did.trim())
33
+
34
+ await publishIdentityEvtForDids(ctx, dids, timeBetween)
35
+ console.log('DONE')
36
+ }
37
+
38
+ export const publishIdentityEvtForDids = async (
39
+ ctx: PublishIdentityContext,
40
+ dids: string[],
41
+ timeBetween = 0,
42
+ ) => {
43
+ for (const did of dids) {
44
+ try {
45
+ await ctx.sequencer.sequenceIdentityEvt(did)
46
+ console.log(`published identity evt for ${did}`)
47
+ } catch (err) {
48
+ console.error(`failed to sequence new identity evt for ${did}: ${err}`)
49
+ }
50
+ if (timeBetween > 0) {
51
+ await wait(timeBetween)
52
+ }
53
+ }
54
+ }
@@ -7,21 +7,46 @@ import {
7
7
  MemoryBlockstore,
8
8
  signCommit,
9
9
  } from '@atproto/repo'
10
- import { AppContext } from '../context'
10
+ import { AccountManager } from '../account-manager/account-manager'
11
+ import { ActorStore } from '../actor-store/actor-store'
12
+ import { Sequencer } from '../sequencer'
11
13
 
12
- export const rebuildRepo = async (ctx: AppContext, args: string[]) => {
14
+ export interface RebuildContext {
15
+ sequencer: Sequencer
16
+ accountManager: AccountManager
17
+ actorStore: ActorStore
18
+ }
19
+
20
+ export const rebuildRepoScript = async (
21
+ ctx: RebuildContext,
22
+ args: string[],
23
+ ) => {
13
24
  const did = args[0]
14
25
  if (!did || !did.startsWith('did:')) {
15
26
  throw new Error('Expected DID as argument')
16
27
  }
28
+ return rebuildRepo(ctx, did, true)
29
+ }
17
30
 
31
+ export const rebuildRepo = async (
32
+ ctx: RebuildContext,
33
+ did: string,
34
+ promptUser: boolean,
35
+ ) => {
18
36
  const memoryStore = new MemoryBlockstore()
19
- const rev = TID.nextStr()
20
37
  const commit = await ctx.actorStore.transact(did, async (store) => {
21
- const [records, existingCids] = await Promise.all([
38
+ const [rootDetails, records, existingCids] = await Promise.all([
39
+ store.repo.storage.getRootDetailed(),
22
40
  store.record.listAll(),
23
41
  store.record.listExistingBlocks(),
24
42
  ])
43
+ // increment existing rev by 1 ms
44
+ const revTid = TID.fromStr(rootDetails.rev)
45
+ const rev = TID.fromTime(
46
+ revTid.timestamp() + 1,
47
+ revTid.clockid(),
48
+ ).toString()
49
+
25
50
  let mst = await MST.create(memoryStore)
26
51
  for (const record of records) {
27
52
  mst = await mst.add(record.path, record.cid)
@@ -50,14 +75,16 @@ export const rebuildRepo = async (ctx: AppContext, args: string[]) => {
50
75
  )
51
76
  const commitCid = await newBlocks.add(newCommit)
52
77
 
53
- console.log('Record count: ', records.length)
54
- console.log('Existing blocks: ', existingCids.toList().length)
55
- console.log('Deleting blocks:', toDelete.toList().length)
56
- console.log('Adding blocks: ', newBlocks.size)
78
+ if (promptUser) {
79
+ console.log('Record count: ', records.length)
80
+ console.log('Existing blocks: ', existingCids.toList().length)
81
+ console.log('Deleting blocks:', toDelete.toList().length)
82
+ console.log('Adding blocks: ', newBlocks.size)
57
83
 
58
- const shouldContinue = await promptContinue()
59
- if (!shouldContinue) {
60
- throw new Error('Aborted')
84
+ const shouldContinue = await promptContinue()
85
+ if (!shouldContinue) {
86
+ throw new Error('Aborted')
87
+ }
61
88
  }
62
89
 
63
90
  await store.repo.storage.deleteMany(toDelete.toList())
@@ -76,7 +103,7 @@ export const rebuildRepo = async (ctx: AppContext, args: string[]) => {
76
103
  prevData: null,
77
104
  }
78
105
  })
79
- await ctx.accountManager.updateRepoRoot(did, commit.cid, rev)
106
+ await ctx.accountManager.updateRepoRoot(did, commit.cid, commit.rev)
80
107
  const syncData = await ctx.actorStore.read(did, (store) =>
81
108
  store.repo.getSyncEventData(),
82
109
  )
@@ -0,0 +1,141 @@
1
+ import fs from 'node:fs/promises'
2
+ import * as plc from '@did-plc/lib'
3
+ import PQueue from 'p-queue'
4
+ import AtpAgent from '@atproto/api'
5
+ import { Keypair } from '@atproto/crypto'
6
+ import { IdResolver } from '@atproto/identity'
7
+ import { ActorStore } from '../actor-store/actor-store'
8
+ import { SyncEvtData } from '../repo'
9
+ import { Sequencer } from '../sequencer'
10
+ import { getRecoveryDbFromSequencerLoc } from './sequencer-recovery/recovery-db'
11
+ import { parseIntArg } from './util'
12
+
13
+ export type RotateKeysContext = {
14
+ sequencer: Sequencer
15
+ actorStore: ActorStore
16
+ idResolver: IdResolver
17
+ plcClient: plc.Client
18
+ plcRotationKey: Keypair
19
+ entrywayAdminAgent?: AtpAgent
20
+ }
21
+
22
+ export const rotateKeys = async (ctx: RotateKeysContext, args: string[]) => {
23
+ const dids = args
24
+ await rotateKeysForRepos(ctx, dids, 10)
25
+ }
26
+
27
+ export const rotateKeysFromFile = async (
28
+ ctx: RotateKeysContext,
29
+ args: string[],
30
+ ) => {
31
+ const filepath = args[0]
32
+ if (!filepath) {
33
+ throw new Error('Expected filepath as argument')
34
+ }
35
+ const concurrency = args[1] ? parseIntArg(args[1]) : 25
36
+ const file = await fs.readFile(filepath)
37
+ const dids = file
38
+ .toString()
39
+ .split('\n')
40
+ .map((did) => did.trim())
41
+ .filter((did) => did.startsWith('did:plc'))
42
+
43
+ await rotateKeysForRepos(ctx, dids, concurrency)
44
+ }
45
+
46
+ export const rotateKeysRecovery = async (
47
+ ctx: RotateKeysContext,
48
+ args: string[],
49
+ ) => {
50
+ const concurrency = args[1] ? parseIntArg(args[0]) : 10
51
+
52
+ const recoveryDb = await getRecoveryDbFromSequencerLoc(
53
+ ctx.sequencer.dbLocation,
54
+ )
55
+ const rows = await recoveryDb.db
56
+ .selectFrom('new_account')
57
+ .select('did')
58
+ .where('new_account.published', '=', 0)
59
+ .execute()
60
+ const dids = rows.map((r) => r.did)
61
+
62
+ await rotateKeysForRepos(ctx, dids, concurrency, async (did) => {
63
+ await recoveryDb.db
64
+ .updateTable('new_account')
65
+ .set({ published: 1 })
66
+ .where('did', '=', did)
67
+ .execute()
68
+ })
69
+ }
70
+
71
+ const rotateKeysForRepos = async (
72
+ ctx: RotateKeysContext,
73
+ dids: string[],
74
+ concurrency: number,
75
+ onSuccess?: (did: string) => Promise<void>,
76
+ ) => {
77
+ const queue = new PQueue({ concurrency })
78
+ let completed = 0
79
+ for (const did of dids) {
80
+ queue.add(async () => {
81
+ try {
82
+ await updatePlcSigningKey(ctx, did)
83
+ } catch (err) {
84
+ console.error(`failed to update key for ${did}: ${err}`)
85
+ return
86
+ }
87
+ let syncData: SyncEvtData
88
+ try {
89
+ syncData = await ctx.actorStore.transact(did, async (actorTxn) => {
90
+ await actorTxn.repo.processWrites([])
91
+ return actorTxn.repo.getSyncEventData()
92
+ })
93
+ } catch (err) {
94
+ console.error(`failed to write new commit for ${did}: ${err}`)
95
+ return
96
+ }
97
+ try {
98
+ await ctx.sequencer.sequenceIdentityEvt(did)
99
+ } catch (err) {
100
+ console.error(`failed to sequence new identity evt for ${did}: ${err}`)
101
+ return
102
+ }
103
+ try {
104
+ await ctx.sequencer.sequenceSyncEvt(did, syncData)
105
+ } catch (err) {
106
+ console.error(`failed to sequence for ${did}: ${err}`)
107
+ return
108
+ }
109
+ if (onSuccess) {
110
+ await onSuccess(did)
111
+ }
112
+ completed++
113
+ if (completed % 10 === 0) {
114
+ console.log(`${completed}/${dids.length}`)
115
+ }
116
+ })
117
+ }
118
+ await queue.onIdle()
119
+ console.log('DONE')
120
+ }
121
+
122
+ const updatePlcSigningKey = async (ctx: RotateKeysContext, did: string) => {
123
+ const updateTo = await ctx.actorStore.keypair(did)
124
+ const currSigningKey = await ctx.idResolver.did.resolveAtprotoKey(did, true)
125
+ if (updateTo.did() === currSigningKey) {
126
+ // already up to date
127
+ return
128
+ }
129
+ if (ctx.entrywayAdminAgent) {
130
+ await ctx.entrywayAdminAgent.api.com.atproto.admin.updateAccountSigningKey({
131
+ did,
132
+ signingKey: updateTo.did(),
133
+ })
134
+ } else {
135
+ await ctx.plcClient.updateAtprotoKey(
136
+ did,
137
+ ctx.plcRotationKey,
138
+ updateTo.did(),
139
+ )
140
+ }
141
+ }
@@ -0,0 +1,23 @@
1
+ import { parseIntArg } from '../util'
2
+ import { Recoverer, RecovererContextNoDb } from './recoverer'
3
+ import { getRecoveryDbFromSequencerLoc } from './recovery-db'
4
+
5
+ export const sequencerRecovery = async (
6
+ ctx: RecovererContextNoDb,
7
+ args: string[],
8
+ ) => {
9
+ const cursor = args[0] ? parseIntArg(args[0]) : 0
10
+ const concurrency = args[1] ? parseIntArg(args[1]) : 10
11
+
12
+ const recoveryDb = await getRecoveryDbFromSequencerLoc(
13
+ ctx.sequencer.dbLocation,
14
+ )
15
+
16
+ const recover = new Recoverer(
17
+ { ...ctx, recoveryDb },
18
+ {
19
+ concurrency,
20
+ },
21
+ )
22
+ await recover.run(cursor)
23
+ }