@anthropic-ai/sdk
Version:
The official TypeScript library for the Anthropic API
149 lines (129 loc) • 5.11 kB
text/typescript
import type { Logger } from '../client';
import { AnthropicError } from '../core/error';
import {
BetaContentBlock,
BetaJSONOutputFormat,
BetaMessage,
BetaOutputConfig,
BetaTextBlock,
MessageCreateParams,
} from '../resources/beta/messages/messages';
// vendored from typefest just to make things look a bit nicer on hover
type Simplify<T> = { [KeyType in keyof T]: T[KeyType] } & {};
type AutoParseableBetaOutputConfig = Omit<BetaOutputConfig, 'format'> & {
format?: BetaJSONOutputFormat | AutoParseableBetaOutputFormat<any> | null;
};
export type BetaParseableMessageCreateParams = Simplify<
Omit<MessageCreateParams, 'output_format' | 'output_config'> & {
/**
* @deprecated Use `output_config.format` instead. This parameter will be removed in a future
* release.
*/
output_format?: BetaJSONOutputFormat | AutoParseableBetaOutputFormat<any> | null;
output_config?: AutoParseableBetaOutputConfig | null;
}
>;
export type ExtractParsedContentFromBetaParams<Params extends BetaParseableMessageCreateParams> =
Params['output_format'] extends AutoParseableBetaOutputFormat<infer P> ? P
: Params['output_config'] extends { format: AutoParseableBetaOutputFormat<infer P> } ? P
: null;
export type AutoParseableBetaOutputFormat<ParsedT> = BetaJSONOutputFormat & {
parse(content: string): ParsedT;
};
export type ParsedBetaMessage<ParsedT> = BetaMessage & {
content: Array<ParsedBetaContentBlock<ParsedT>>;
parsed_output: ParsedT | null;
};
export type ParsedBetaContentBlock<ParsedT> =
| (BetaTextBlock & { parsed_output: ParsedT | null })
| Exclude<BetaContentBlock, BetaTextBlock>;
function getOutputFormat(
params: BetaParseableMessageCreateParams | null,
): BetaJSONOutputFormat | AutoParseableBetaOutputFormat<any> | null | undefined {
// Prefer output_format (deprecated) over output_config.format for backward compatibility
return params?.output_format ?? params?.output_config?.format;
}
export function maybeParseBetaMessage<Params extends BetaParseableMessageCreateParams | null>(
message: BetaMessage,
params: Params,
opts: { logger: Logger },
): ParsedBetaMessage<ExtractParsedContentFromBetaParams<NonNullable<Params>>> {
const outputFormat = getOutputFormat(params);
if (!params || !('parse' in (outputFormat ?? {}))) {
return {
...message,
content: message.content.map((block) => {
if (block.type === 'text') {
const parsedBlock = Object.defineProperty({ ...block }, 'parsed_output', {
value: null,
enumerable: false,
}) as ParsedBetaContentBlock<ExtractParsedContentFromBetaParams<NonNullable<Params>>>;
return Object.defineProperty(parsedBlock, 'parsed', {
get() {
opts.logger.warn(
'The `parsed` property on `text` blocks is deprecated, please use `parsed_output` instead.',
);
return null;
},
enumerable: false,
});
}
return block;
}),
parsed_output: null,
} as ParsedBetaMessage<ExtractParsedContentFromBetaParams<NonNullable<Params>>>;
}
return parseBetaMessage(message, params, opts);
}
export function parseBetaMessage<Params extends BetaParseableMessageCreateParams>(
message: BetaMessage,
params: Params,
opts: { logger: Logger },
): ParsedBetaMessage<ExtractParsedContentFromBetaParams<Params>> {
let firstParsedOutput: ReturnType<typeof parseBetaOutputFormat<Params>> | null = null;
const content: Array<ParsedBetaContentBlock<ExtractParsedContentFromBetaParams<Params>>> =
message.content.map((block) => {
if (block.type === 'text') {
const parsedOutput = parseBetaOutputFormat(params, block.text);
if (firstParsedOutput === null) {
firstParsedOutput = parsedOutput;
}
const parsedBlock = Object.defineProperty({ ...block }, 'parsed_output', {
value: parsedOutput,
enumerable: false,
}) as ParsedBetaContentBlock<ExtractParsedContentFromBetaParams<Params>>;
return Object.defineProperty(parsedBlock, 'parsed', {
get() {
opts.logger.warn(
'The `parsed` property on `text` blocks is deprecated, please use `parsed_output` instead.',
);
return parsedOutput;
},
enumerable: false,
});
}
return block;
});
return {
...message,
content,
parsed_output: firstParsedOutput,
} as ParsedBetaMessage<ExtractParsedContentFromBetaParams<Params>>;
}
function parseBetaOutputFormat<Params extends BetaParseableMessageCreateParams>(
params: Params,
content: string,
): ExtractParsedContentFromBetaParams<Params> | null {
const outputFormat = getOutputFormat(params);
if (outputFormat?.type !== 'json_schema') {
return null;
}
try {
if ('parse' in outputFormat) {
return outputFormat.parse(content);
}
return JSON.parse(content);
} catch (error) {
throw new AnthropicError(`Failed to parse structured output: ${error}`);
}
}