atproto/packages/bsky/tests/data-plane/handle-invalidation.test.ts
Daniel Holmgren 50c0ec176c
Service auth method binding (lxm) (#2663)
* add scopes to service auth impl

* add error to getServiceAuth

* send scoped tokens from pds

* clean up privileged access scopes & allow simple service auth tokens for app passwords

* integration into ozone

* fix up bsky tests

* cleanup xrpc-server tests

* fix up tests & types

* one more test

* fix read after write tests

* fix mod auth test

* convert scopes to be a single method name

* add scope check callback for auth verifier

* pds changes only

* fix feed generation tests

* use scope for ozone service profile

* dont verify scopes on pds yet

* tidy

* tidy imports

* changeset

* add tests

* tidy

* another changeset

* scope -> lxm

* tidy

* clean up scope references

* update nonce size

* pr feedback

* trim trailing slash

* nonce -> jti

* fix xrpc-server test

* allow service auth on uploadBlob

* fix build error

* changeset

* build, tidy

* xrpc-server: update lxm claim check error

* appview: temporarily permit labeler service calls to omit lxm claim

* xrpc-server: fix test

* changeset

* fix merged tests

---------

Co-authored-by: Devin Ivy <devinivy@gmail.com>
2024-08-18 15:46:07 -04:00

152 lines
4.1 KiB
TypeScript

import { DAY } from '@atproto/common'
import { TestNetwork, SeedClient, usersSeed } from '@atproto/dev-env'
import { AtpAgent } from '@atproto/api'
import { ids } from '../../src/lexicon/lexicons'
describe('handle invalidation', () => {
let network: TestNetwork
let agent: AtpAgent
let pdsAgent: AtpAgent
let sc: SeedClient
let alice: string
let bob: string
const mockHandles = {}
beforeAll(async () => {
network = await TestNetwork.create({
dbPostgresSchema: 'bsky_handle_invalidation',
})
agent = network.bsky.getClient()
pdsAgent = network.pds.getClient()
sc = network.getSeedClient()
await usersSeed(sc)
await network.processAll()
alice = sc.dids.alice
bob = sc.dids.bob
const origResolve = network.bsky.dataplane.idResolver.handle.resolve
network.bsky.dataplane.idResolver.handle.resolve = async (
handle: string,
) => {
if (mockHandles[handle] === null) {
return undefined
} else if (mockHandles[handle]) {
return mockHandles[handle]
}
return origResolve(handle)
}
})
afterAll(async () => {
await network.close()
})
const backdateIndexedAt = async (did: string) => {
const TWO_DAYS_AGO = new Date(Date.now() - 2 * DAY).toISOString()
await network.bsky.db.db
.updateTable('actor')
.set({ indexedAt: TWO_DAYS_AGO })
.where('did', '=', did)
.execute()
}
it('indexes an account with no proper handle', async () => {
mockHandles['eve.test'] = null
const eveAccnt = await sc.createAccount('eve', {
handle: 'eve.test',
email: 'eve@test.com',
password: 'eve-pass',
})
await network.processAll()
const res = await agent.api.app.bsky.actor.getProfile(
{ actor: eveAccnt.did },
{
headers: await network.serviceHeaders(
alice,
ids.AppBskyActorGetProfile,
),
},
)
expect(res.data.handle).toEqual('handle.invalid')
})
it('invalidates out of date handles', async () => {
await backdateIndexedAt(alice)
const aliceHandle = sc.accounts[alice].handle
// alice's handle no longer resolves
mockHandles[aliceHandle] = null
await sc.post(alice, 'blah')
await network.processAll()
const res = await agent.api.app.bsky.actor.getProfile(
{ actor: alice },
{
headers: await network.serviceHeaders(
alice,
ids.AppBskyActorGetProfile,
),
},
)
expect(res.data.handle).toEqual('handle.invalid')
})
it('revalidates an out of date handle', async () => {
await backdateIndexedAt(alice)
const aliceHandle = sc.accounts[alice].handle
// alice's handle no longer resolves
delete mockHandles[aliceHandle]
await sc.post(alice, 'blah')
await network.processAll()
const res = await agent.api.app.bsky.actor.getProfile(
{ actor: alice },
{
headers: await network.serviceHeaders(
alice,
ids.AppBskyActorGetProfile,
),
},
)
expect(res.data.handle).toEqual(sc.accounts[alice].handle)
})
it('deals with handle contention', async () => {
await backdateIndexedAt(bob)
// update alices handle so that the pds will let bob take her old handle
await network.pds.ctx.accountManager.updateHandle(alice, 'not-alice.test')
await pdsAgent.api.com.atproto.identity.updateHandle(
{
handle: sc.accounts[alice].handle,
},
{ headers: sc.getHeaders(bob), encoding: 'application/json' },
)
await network.processAll()
const aliceRes = await agent.api.app.bsky.actor.getProfile(
{ actor: alice },
{
headers: await network.serviceHeaders(
alice,
ids.AppBskyActorGetProfile,
),
},
)
expect(aliceRes.data.handle).toEqual('handle.invalid')
const bobRes = await agent.api.app.bsky.actor.getProfile(
{ actor: bob },
{
headers: await network.serviceHeaders(
alice,
ids.AppBskyActorGetProfile,
),
},
)
expect(bobRes.data.handle).toEqual(sc.accounts[alice].handle)
})
})