Matthieu Sieben 8ef976d385
xrpc-server & lex-cli rework (#3999)
* xrpc-server & lex-cli rework

* codegen

* tidy

* tidy

* tidy

* Update .changeset/nasty-icons-peel.md

Co-authored-by: devin ivy <devinivy@gmail.com>

* excludeErrorResult util

* Restore export of `SkeletonHandler` from `pds`

* Make `calcKey` RateLimiter option required

* Process request input after auth

* fix tests

* changeset

---------

Co-authored-by: devin ivy <devinivy@gmail.com>
2025-07-08 13:04:11 +02:00

300 lines
8.0 KiB
TypeScript

import * as http from 'node:http'
import { AddressInfo } from 'node:net'
import { LexiconDoc } from '@atproto/lexicon'
import { XRPCError, XRPCInvalidResponseError, XrpcClient } from '@atproto/xrpc'
import * as xrpcServer from '../src'
import { closeServer, createServer } from './_util'
const UPSTREAM_LEXICONS: LexiconDoc[] = [
{
lexicon: 1,
id: 'io.example.upstreamInvalidResponse',
defs: {
main: {
type: 'query',
output: {
encoding: 'application/json',
schema: {
type: 'object',
required: ['expectedValue'],
properties: {
expectedValue: { type: 'string' },
},
},
},
},
},
},
]
const LEXICONS: LexiconDoc[] = [
{
lexicon: 1,
id: 'io.example.error',
defs: {
main: {
type: 'query',
parameters: {
type: 'params',
properties: {
which: { type: 'string', default: 'foo' },
},
},
errors: [{ name: 'Foo' }, { name: 'Bar' }],
},
},
},
{
lexicon: 1,
id: 'io.example.throwFalsyValue',
defs: {
main: {
type: 'query',
},
},
},
{
lexicon: 1,
id: 'io.example.query',
defs: {
main: {
type: 'query',
},
},
},
{
lexicon: 1,
id: 'io.example.procedure',
defs: {
main: {
type: 'procedure',
},
},
},
{
lexicon: 1,
id: 'io.example.invalidResponse',
defs: {
main: {
type: 'query',
output: {
encoding: 'application/json',
schema: {
type: 'object',
required: ['expectedValue'],
properties: {
expectedValue: { type: 'string' },
},
},
},
},
},
},
{
lexicon: 1,
id: 'io.example.invalidUpstreamResponse',
defs: {
main: {
type: 'query',
},
},
},
]
const MISMATCHED_LEXICONS: LexiconDoc[] = [
{
lexicon: 1,
id: 'io.example.query',
defs: {
main: {
type: 'procedure',
},
},
},
{
lexicon: 1,
id: 'io.example.procedure',
defs: {
main: {
type: 'query',
},
},
},
{
lexicon: 1,
id: 'io.example.doesNotExist',
defs: {
main: {
type: 'query',
},
},
},
]
describe('Errors', () => {
let upstreamS: http.Server
const upstreamServer = xrpcServer.createServer(UPSTREAM_LEXICONS, {
validateResponse: false,
}) // disable validateResponse to test client validation
upstreamServer.method('io.example.upstreamInvalidResponse', () => {
return { encoding: 'json', body: { something: 'else' } }
})
let upstreamClient: XrpcClient
let s: http.Server
const server = xrpcServer.createServer(LEXICONS, { validateResponse: false }) // disable validateResponse to test client validation
server.method('io.example.error', (ctx) => {
if (ctx.params.which === 'foo') {
throw new xrpcServer.InvalidRequestError('It was this one!', 'Foo')
} else if (ctx.params.which === 'bar') {
return { status: 400, error: 'Bar', message: 'It was that one!' }
} else {
return { status: 400 }
}
})
server.method('io.example.throwFalsyValue', () => {
throw ''
})
server.method('io.example.query', () => {
return undefined
})
// @ts-ignore We're intentionally giving the wrong response! -prf
server.method('io.example.invalidResponse', () => {
return { encoding: 'json', body: { something: 'else' } }
})
server.method('io.example.invalidUpstreamResponse', async () => {
await upstreamClient.call('io.example.upstreamInvalidResponse')
return {
encoding: 'json',
}
})
server.method('io.example.procedure', () => {
return undefined
})
let client: XrpcClient
let badClient: XrpcClient
beforeAll(async () => {
upstreamS = await createServer(upstreamServer)
const { port: upstreamPort } = upstreamS.address() as AddressInfo
upstreamClient = new XrpcClient(
`http://localhost:${upstreamPort}`,
UPSTREAM_LEXICONS,
)
s = await createServer(server)
const { port } = s.address() as AddressInfo
client = new XrpcClient(`http://localhost:${port}`, LEXICONS)
badClient = new XrpcClient(`http://localhost:${port}`, MISMATCHED_LEXICONS)
})
afterAll(async () => {
await closeServer(s)
await closeServer(upstreamS)
})
it('serves requests', async () => {
try {
await client.call('io.example.error', {
which: 'foo',
})
throw new Error('Didnt throw')
} catch (e) {
expect(e).toBeInstanceOf(XRPCError)
expect((e as XRPCError).success).toBeFalsy()
expect((e as XRPCError).error).toBe('Foo')
expect((e as XRPCError).message).toBe('It was this one!')
}
try {
await client.call('io.example.error', {
which: 'bar',
})
throw new Error('Didnt throw')
} catch (e) {
expect(e).toBeInstanceOf(XRPCError)
expect((e as XRPCError).success).toBeFalsy()
expect((e as XRPCError).error).toBe('Bar')
expect((e as XRPCError).message).toBe('It was that one!')
}
try {
await client.call('io.example.throwFalsyValue')
throw new Error('Didnt throw')
} catch (e) {
expect(e).toBeInstanceOf(XRPCError)
expect((e as XRPCError).success).toBeFalsy()
expect((e as XRPCError).error).toBe('InternalServerError')
expect((e as XRPCError).message).toBe('Internal Server Error')
}
try {
await client.call('io.example.error', {
which: 'other',
})
throw new Error('Didnt throw')
} catch (e) {
expect(e).toBeInstanceOf(XRPCError)
expect((e as XRPCError).success).toBeFalsy()
expect((e as XRPCError).error).toBe('InvalidRequest')
expect((e as XRPCError).message).toBe('Invalid Request')
}
try {
await client.call('io.example.invalidResponse')
throw new Error('Didnt throw')
} catch (e: any) {
expect(e).toBeInstanceOf(XRPCError)
expect(e).toBeInstanceOf(XRPCInvalidResponseError)
expect(e.success).toBeFalsy()
expect(e.error).toBe('Invalid Response')
expect(e.message).toBe(
'The server gave an invalid response and may be out of date.',
)
const err = e as XRPCInvalidResponseError
expect(err.validationError.message).toBe(
'Output must have the property "expectedValue"',
)
expect(err.responseBody).toStrictEqual({ something: 'else' })
}
try {
await client.call('io.example.invalidUpstreamResponse')
throw new Error('Didnt throw')
} catch (e: any) {
expect(e).toBeInstanceOf(XRPCError)
expect((e as XRPCError).status).toBe(500)
expect((e as XRPCError).success).toBeFalsy()
expect((e as XRPCError).error).toBe('InternalServerError')
expect((e as XRPCError).message).toBe('Internal Server Error')
}
})
it('serves error for missing/mismatch schemas', async () => {
await client.call('io.example.query') // No error
await client.call('io.example.procedure') // No error
try {
await badClient.call('io.example.query')
throw new Error('Didnt throw')
} catch (e: any) {
expect(e).toBeInstanceOf(XRPCError)
expect(e.success).toBeFalsy()
expect(e.error).toBe('InvalidRequest')
expect(e.message).toBe('Incorrect HTTP method (POST) expected GET')
}
try {
await badClient.call('io.example.procedure')
throw new Error('Didnt throw')
} catch (e: any) {
expect(e).toBeInstanceOf(XRPCError)
expect(e.success).toBeFalsy()
expect(e.error).toBe('InvalidRequest')
expect(e.message).toBe('Incorrect HTTP method (GET) expected POST')
}
try {
await badClient.call('io.example.doesNotExist')
throw new Error('Didnt throw')
} catch (e: any) {
expect(e).toBeInstanceOf(XRPCError)
expect(e.success).toBeFalsy()
expect(e.error).toBe('MethodNotImplemented')
expect(e.message).toBe('Method Not Implemented')
}
})
})