Skip to content

Commit

Permalink
feat(client): add ._request_id property to object responses
Browse files Browse the repository at this point in the history
  • Loading branch information
RobertCraigie committed Nov 8, 2024
1 parent 7572e48 commit 8ac70a5
Show file tree
Hide file tree
Showing 4 changed files with 197 additions and 21 deletions.
12 changes: 12 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -227,6 +227,18 @@ Error codes are as followed:
| >=500 | `InternalServerError` |
| N/A | `APIConnectionError` |

## Request IDs

> For more information on debugging requests, see [these docs](https://docs.anthropic.com/en/api/errors#request-id)
All object responses in the SDK provide a `_request_id` property which is added from the `request-id` response header so that you can quickly log failing requests and report them back to Anthropic.

```ts
const message = await client.messages.create({ max_tokens: 1024, messages: [{ role: 'user', content: 'Hello, Claude' }], model: 'claude-3-opus-20240229' });
console.log(completion._request_id) // req_018EeWyXxfu5pfWkrYcMdjWG
```


### Retries

Certain errors will be automatically retried 2 times by default, with a short exponential backoff.
Expand Down
66 changes: 46 additions & 20 deletions src/core.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ type APIResponseProps = {
controller: AbortController;
};

async function defaultParseResponse<T>(props: APIResponseProps): Promise<T> {
async function defaultParseResponse<T>(props: APIResponseProps): Promise<WithRequestID<T>> {
const { response } = props;
if (props.options.stream) {
debug('response', response.status, response.url, response.headers, response.body);
Expand All @@ -54,11 +54,11 @@ async function defaultParseResponse<T>(props: APIResponseProps): Promise<T> {

// fetch refuses to read the body when the status code is 204.
if (response.status === 204) {
return null as T;
return null as WithRequestID<T>;
}

if (props.options.__binaryResponse) {
return response as unknown as T;
return response as unknown as WithRequestID<T>;
}

const contentType = response.headers.get('content-type');
Expand All @@ -69,26 +69,44 @@ async function defaultParseResponse<T>(props: APIResponseProps): Promise<T> {

debug('response', response.status, response.url, response.headers, json);

return json as T;
return _addRequestID(json as T, response);
}

const text = await response.text();
debug('response', response.status, response.url, response.headers, text);

// TODO handle blob, arraybuffer, other content types, etc.
return text as unknown as T;
return text as unknown as WithRequestID<T>;
}

type WithRequestID<T> =
T extends Array<any> | Response | AbstractPage<any> ? T
: T extends Record<string, any> ? T & { _request_id?: string | null }
: T;

function _addRequestID<T>(value: T, response: Response): WithRequestID<T> {
if (!value || typeof value !== 'object' || Array.isArray(value)) {
return value as WithRequestID<T>;
}

return Object.defineProperty(value, '_request_id', {
value: response.headers.get('request-id'),
enumerable: false,
}) as WithRequestID<T>;
}

/**
* A subclass of `Promise` providing additional helper methods
* for interacting with the SDK.
*/
export class APIPromise<T> extends Promise<T> {
private parsedPromise: Promise<T> | undefined;
export class APIPromise<T> extends Promise<WithRequestID<T>> {
private parsedPromise: Promise<WithRequestID<T>> | undefined;

constructor(
private responsePromise: Promise<APIResponseProps>,
private parseResponse: (props: APIResponseProps) => PromiseOrValue<T> = defaultParseResponse,
private parseResponse: (
props: APIResponseProps,
) => PromiseOrValue<WithRequestID<T>> = defaultParseResponse,
) {
super((resolve) => {
// this is maybe a bit weird but this has to be a no-op to not implicitly
Expand All @@ -100,7 +118,7 @@ export class APIPromise<T> extends Promise<T> {

_thenUnwrap<U>(transform: (data: T, props: APIResponseProps) => U): APIPromise<U> {
return new APIPromise(this.responsePromise, async (props) =>
transform(await this.parseResponse(props), props),
_addRequestID(transform(await this.parseResponse(props), props), props.response),
);
}

Expand All @@ -120,45 +138,47 @@ export class APIPromise<T> extends Promise<T> {
asResponse(): Promise<Response> {
return this.responsePromise.then((p) => p.response);
}

/**
* Gets the parsed response data and the raw `Response` instance.
* Gets the parsed response data, the raw `Response` instance and the ID of the request,
* returned vie the `request-id` header which is useful for debugging requests and resporting
* issues to Anthropic.
*
* If you just want to get the raw `Response` instance without parsing it,
* you can use {@link asResponse()}.
*
*
* 👋 Getting the wrong TypeScript type for `Response`?
* Try setting `"moduleResolution": "NodeNext"` if you can,
* or add one of these imports before your first `import … from '@anthropic-ai/sdk'`:
* - `import '@anthropic-ai/sdk/shims/node'` (if you're running on Node)
* - `import '@anthropic-ai/sdk/shims/web'` (otherwise)
*/
async withResponse(): Promise<{ data: T; response: Response }> {
async withResponse(): Promise<{ data: T; response: Response; request_id: string | null | undefined }> {
const [data, response] = await Promise.all([this.parse(), this.asResponse()]);
return { data, response };
return { data, response, request_id: response.headers.get('request-id') };
}

private parse(): Promise<T> {
private parse(): Promise<WithRequestID<T>> {
if (!this.parsedPromise) {
this.parsedPromise = this.responsePromise.then(this.parseResponse);
this.parsedPromise = this.responsePromise.then(this.parseResponse) as any as Promise<WithRequestID<T>>;
}
return this.parsedPromise;
}

override then<TResult1 = T, TResult2 = never>(
onfulfilled?: ((value: T) => TResult1 | PromiseLike<TResult1>) | undefined | null,
override then<TResult1 = WithRequestID<T>, TResult2 = never>(
onfulfilled?: ((value: WithRequestID<T>) => TResult1 | PromiseLike<TResult1>) | undefined | null,
onrejected?: ((reason: any) => TResult2 | PromiseLike<TResult2>) | undefined | null,
): Promise<TResult1 | TResult2> {
return this.parse().then(onfulfilled, onrejected);
}

override catch<TResult = never>(
onrejected?: ((reason: any) => TResult | PromiseLike<TResult>) | undefined | null,
): Promise<T | TResult> {
): Promise<WithRequestID<T> | TResult> {
return this.parse().catch(onrejected);
}

override finally(onfinally?: (() => void) | undefined | null): Promise<T> {
override finally(onfinally?: (() => void) | undefined | null): Promise<WithRequestID<T>> {
return this.parse().finally(onfinally);
}
}
Expand Down Expand Up @@ -724,7 +744,13 @@ export class PagePromise<
) {
super(
request,
async (props) => new Page(client, props.response, await defaultParseResponse(props), props.options),
async (props) =>
new Page(
client,
props.response,
await defaultParseResponse(props),
props.options,
) as WithRequestID<PageClass>,
);
}

Expand Down
131 changes: 130 additions & 1 deletion tests/responses.test.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
import { createResponseHeaders } from '@anthropic-ai/sdk/core';
import { APIPromise, createResponseHeaders } from '@anthropic-ai/sdk/core';
import Anthropic from '@anthropic-ai/sdk/index';
import { Headers } from '@anthropic-ai/sdk/_shims/index';
import { Response } from 'node-fetch';
import { compareType } from './utils/typing';

describe('response parsing', () => {
// TODO: test unicode characters
Expand All @@ -23,3 +26,129 @@ describe('response parsing', () => {
expect(headers['content-type']).toBe('text/xml, application/json');
});
});

describe('request id', () => {
test('types', () => {
compareType<Awaited<APIPromise<string>>, string>(true);
compareType<Awaited<APIPromise<number>>, number>(true);
compareType<Awaited<APIPromise<null>>, null>(true);
compareType<Awaited<APIPromise<void>>, void>(true);
compareType<Awaited<APIPromise<Response>>, Response>(true);
compareType<Awaited<APIPromise<Response>>, Response>(true);
compareType<Awaited<APIPromise<{ foo: string }>>, { foo: string } & { _request_id?: string | null }>(
true,
);
compareType<Awaited<APIPromise<Array<{ foo: string }>>>, Array<{ foo: string }>>(true);
});

test('withResponse', async () => {
const client = new Anthropic({
apiKey: 'dummy',
fetch: async () =>
new Response(JSON.stringify({ id: 'bar' }), {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/json' },
}),
});

const {
data: message,
response,
request_id,
} = await client.messages
.create({ messages: [], model: 'claude-3-opus-20240229', max_tokens: 1024 })
.withResponse();

expect(request_id).toBe('req_xxx');
expect(response.headers.get('x-request-id')).toBe('req_xxx');
expect(message.id).toBe('bar');
expect(JSON.stringify(message)).toBe('{"id":"bar"}');
});

test('object response', async () => {
const client = new Anthropic({
apiKey: 'dummy',
fetch: async () =>
new Response(JSON.stringify({ id: 'bar' }), {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/json' },
}),
});

const rsp = await client.messages.create({
messages: [],
model: 'claude-3-opus-20240229',
max_tokens: 1024,
});
expect(rsp.id).toBe('bar');
expect(rsp._request_id).toBe('req_xxx');
expect(JSON.stringify(rsp)).toBe('{"id":"bar"}');
});

test('envelope response', async () => {
const promise = new APIPromise<{ data: { foo: string } }>(
(async () => {
return {
response: new Response(JSON.stringify({ data: { foo: 'bar' } }), {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/json' },
}),
controller: {} as any,
options: {} as any,
};
})(),
)._thenUnwrap((d) => d.data);

const rsp = await promise;
expect(rsp.foo).toBe('bar');
expect(rsp._request_id).toBe('req_xxx');
});

test('page response', async () => {
const client = new Anthropic({
apiKey: 'dummy',
fetch: async () =>
new Response(JSON.stringify({ data: [{ foo: 'bar' }] }), {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/json' },
}),
});

const page = await client.beta.messages.batches.list();
expect(page.data).toMatchObject([{ foo: 'bar' }]);
expect((page as any)._request_id).toBeUndefined();
});

test('array response', async () => {
const promise = new APIPromise<Array<{ foo: string }>>(
(async () => {
return {
response: new Response(JSON.stringify([{ foo: 'bar' }]), {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/json' },
}),
controller: {} as any,
options: {} as any,
};
})(),
);

const rsp = await promise;
expect(rsp.length).toBe(1);
expect(rsp[0]).toMatchObject({ foo: 'bar' });
expect((rsp as any)._request_id).toBeUndefined();
});

test('string response', async () => {
const promise = new APIPromise<string>(
(async () => {
return {
response: new Response('hello world', {
headers: { 'request-id': 'req_xxx', 'content-type': 'application/text' },
}),
controller: {} as any,
options: {} as any,
};
})(),
);

const result = await promise;
expect(result).toBe('hello world');
expect((result as any)._request_id).toBeUndefined();
});
});
9 changes: 9 additions & 0 deletions tests/utils/typing.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
type Equal<X, Y> = (<T>() => T extends X ? 1 : 2) extends <T>() => T extends Y ? 1 : 2 ? true : false;

export const expectType = <T>(_expression: T): void => {
return;
};

export const compareType = <T1, T2>(_expression: Equal<T1, T2>): void => {
return;
};

0 comments on commit 8ac70a5

Please sign in to comment.