@atproto/pds 0.4.117 → 0.4.119
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +23 -0
- package/dist/account-manager/account-manager.d.ts +5 -2
- package/dist/account-manager/account-manager.d.ts.map +1 -1
- package/dist/account-manager/account-manager.js.map +1 -1
- package/dist/actor-store/blob/transactor.d.ts +2 -1
- package/dist/actor-store/blob/transactor.d.ts.map +1 -1
- package/dist/actor-store/blob/transactor.js +19 -5
- package/dist/actor-store/blob/transactor.js.map +1 -1
- package/dist/api/com/atproto/repo/uploadBlob.d.ts.map +1 -1
- package/dist/api/com/atproto/repo/uploadBlob.js +1 -0
- package/dist/api/com/atproto/repo/uploadBlob.js.map +1 -1
- package/dist/config/env.d.ts +1 -0
- package/dist/config/env.d.ts.map +1 -1
- package/dist/config/env.js +1 -0
- package/dist/config/env.js.map +1 -1
- package/dist/config/secrets.d.ts +1 -0
- package/dist/config/secrets.d.ts.map +1 -1
- package/dist/config/secrets.js +1 -0
- package/dist/config/secrets.js.map +1 -1
- package/dist/context.d.ts +3 -0
- package/dist/context.d.ts.map +1 -1
- package/dist/context.js +19 -0
- package/dist/context.js.map +1 -1
- package/dist/lexicon/index.d.ts +10 -0
- package/dist/lexicon/index.d.ts.map +1 -1
- package/dist/lexicon/index.js +20 -0
- package/dist/lexicon/index.js.map +1 -1
- package/dist/lexicon/lexicons.d.ts +516 -0
- package/dist/lexicon/lexicons.d.ts.map +1 -1
- package/dist/lexicon/lexicons.js +277 -0
- package/dist/lexicon/lexicons.js.map +1 -1
- package/dist/lexicon/types/app/bsky/unspecced/defs.d.ts +27 -0
- package/dist/lexicon/types/app/bsky/unspecced/defs.d.ts.map +1 -1
- package/dist/lexicon/types/app/bsky/unspecced/defs.js +18 -0
- package/dist/lexicon/types/app/bsky/unspecced/defs.js.map +1 -1
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.d.ts +36 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.d.ts.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.js +7 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.js.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.d.ts +37 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.d.ts.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.js +7 -0
- package/dist/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.js.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrends.d.ts +36 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrends.d.ts.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrends.js +7 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrends.js.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.d.ts +38 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.d.ts.map +1 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.js +7 -0
- package/dist/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.js.map +1 -0
- package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.d.ts +31 -0
- package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.d.ts.map +1 -0
- package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.js +7 -0
- package/dist/lexicon/types/com/atproto/admin/updateAccountSigningKey.js.map +1 -0
- package/dist/repo/prepare.d.ts.map +1 -1
- package/dist/repo/prepare.js +1 -0
- package/dist/repo/prepare.js.map +1 -1
- package/dist/repo/types.d.ts +1 -0
- package/dist/repo/types.d.ts.map +1 -1
- package/dist/repo/types.js.map +1 -1
- package/dist/scripts/index.d.ts +8 -1
- package/dist/scripts/index.d.ts.map +1 -1
- package/dist/scripts/index.js +11 -0
- package/dist/scripts/index.js.map +1 -1
- package/dist/scripts/publish-identity.d.ts +8 -0
- package/dist/scripts/publish-identity.d.ts.map +1 -0
- package/dist/scripts/publish-identity.js +46 -0
- package/dist/scripts/publish-identity.js.map +1 -0
- package/dist/scripts/rebuild-repo.d.ts +10 -2
- package/dist/scripts/rebuild-repo.d.ts.map +1 -1
- package/dist/scripts/rebuild-repo.js +21 -12
- package/dist/scripts/rebuild-repo.js.map +1 -1
- package/dist/scripts/rotate-keys.d.ts +18 -0
- package/dist/scripts/rotate-keys.d.ts.map +1 -0
- package/dist/scripts/rotate-keys.js +115 -0
- package/dist/scripts/rotate-keys.js.map +1 -0
- package/dist/scripts/sequencer-recovery/index.d.ts +3 -0
- package/dist/scripts/sequencer-recovery/index.d.ts.map +1 -0
- package/dist/scripts/sequencer-recovery/index.js +17 -0
- package/dist/scripts/sequencer-recovery/index.js.map +1 -0
- package/dist/scripts/sequencer-recovery/recoverer.d.ts +27 -0
- package/dist/scripts/sequencer-recovery/recoverer.d.ts.map +1 -0
- package/dist/scripts/sequencer-recovery/recoverer.js +235 -0
- package/dist/scripts/sequencer-recovery/recoverer.js.map +1 -0
- package/dist/scripts/sequencer-recovery/recovery-db.d.ts +18 -0
- package/dist/scripts/sequencer-recovery/recovery-db.d.ts.map +1 -0
- package/dist/scripts/sequencer-recovery/recovery-db.js +45 -0
- package/dist/scripts/sequencer-recovery/recovery-db.js.map +1 -0
- package/dist/scripts/sequencer-recovery/repair-repos.d.ts +3 -0
- package/dist/scripts/sequencer-recovery/repair-repos.d.ts.map +1 -0
- package/dist/scripts/sequencer-recovery/repair-repos.js +45 -0
- package/dist/scripts/sequencer-recovery/repair-repos.js.map +1 -0
- package/dist/scripts/sequencer-recovery/user-queues.d.ts +12 -0
- package/dist/scripts/sequencer-recovery/user-queues.d.ts.map +1 -0
- package/dist/scripts/sequencer-recovery/user-queues.js +54 -0
- package/dist/scripts/sequencer-recovery/user-queues.js.map +1 -0
- package/dist/scripts/util.d.ts +2 -0
- package/dist/scripts/util.d.ts.map +1 -0
- package/dist/scripts/util.js +12 -0
- package/dist/scripts/util.js.map +1 -0
- package/dist/sequencer/sequencer.d.ts +2 -0
- package/dist/sequencer/sequencer.d.ts.map +1 -1
- package/dist/sequencer/sequencer.js +52 -42
- package/dist/sequencer/sequencer.js.map +1 -1
- package/package.json +11 -11
- package/src/account-manager/account-manager.ts +6 -2
- package/src/actor-store/blob/transactor.ts +25 -7
- package/src/api/com/atproto/repo/uploadBlob.ts +1 -0
- package/src/config/env.ts +2 -0
- package/src/config/secrets.ts +2 -0
- package/src/context.ts +23 -0
- package/src/lexicon/index.ts +62 -0
- package/src/lexicon/lexicons.ts +285 -0
- package/src/lexicon/types/app/bsky/unspecced/defs.ts +45 -0
- package/src/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacks.ts +54 -0
- package/src/lexicon/types/app/bsky/unspecced/getSuggestedStarterPacksSkeleton.ts +55 -0
- package/src/lexicon/types/app/bsky/unspecced/getTrends.ts +54 -0
- package/src/lexicon/types/app/bsky/unspecced/getTrendsSkeleton.ts +56 -0
- package/src/lexicon/types/com/atproto/admin/updateAccountSigningKey.ts +48 -0
- package/src/repo/prepare.ts +1 -0
- package/src/repo/types.ts +1 -0
- package/src/scripts/README.md +40 -0
- package/src/scripts/index.ts +15 -0
- package/src/scripts/publish-identity.ts +54 -0
- package/src/scripts/rebuild-repo.ts +39 -12
- package/src/scripts/rotate-keys.ts +141 -0
- package/src/scripts/sequencer-recovery/index.ts +23 -0
- package/src/scripts/sequencer-recovery/recoverer.ts +289 -0
- package/src/scripts/sequencer-recovery/recovery-db.ts +64 -0
- package/src/scripts/sequencer-recovery/repair-repos.ts +48 -0
- package/src/scripts/sequencer-recovery/user-queues.ts +41 -0
- package/src/scripts/util.ts +7 -0
- package/src/sequencer/sequencer.ts +43 -40
- package/tests/recovery.test.ts +178 -0
- package/tsconfig.build.tsbuildinfo +1 -1
- 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
|
package/src/repo/prepare.ts
CHANGED
package/src/repo/types.ts
CHANGED
@@ -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.
|
package/src/scripts/index.ts
CHANGED
@@ -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 {
|
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
|
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
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
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
|
-
|
59
|
-
|
60
|
-
|
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
|
+
}
|