-
-
Notifications
You must be signed in to change notification settings - Fork 382
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow adding a default response body encoding when extending ky #676
Comments
I agree this would be a nice feature. We would call it Any thoughts on the implementation or TS types? |
Sure! I tried spinning up a concept real quick, tell me what you think: type ResponseType<T = unknown> = {
json: T;
text: string;
arrayBuffer: ArrayBuffer;
blob: Blob;
};
const responseType: keyof ResponseType = "json";
export const decodeBody = <T>(response: Response) => {
type Handlers = {
[K in keyof ResponseType<T>]: () => Promise<ResponseType<T>[K]>;
};
const handlers: Handlers = {
json: response.json,
text: response.text,
arrayBuffer: response.arrayBuffer,
blob: response.blob,
};
const handler = handlers[responseType];
return handler();
};
const body = decodeBody<{ status: string }>(await fetch("example.com")); |
Nice! Looks good to me. Next step is integrating it with the internals. We can put that in |
Sounds good! export const decodeBody = async <T>(
response: Response,
responseType: keyof ResponseType<T> = "json",
) => {
type Handlers<T> = {
[K in keyof ResponseType<T>]: () => Promise<ResponseType<T>[K]>;
};
const handlers: Handlers<T> = {
json: response.json,
text: response.text,
arrayBuffer: response.arrayBuffer,
blob: response.blob,
formData: response.formData,
};
const handler = handlers[responseType];
return handler();
}; |
I found a solution that seems to work. type ResponseType<T = unknown> = {
json: () => Promise<T>;
text: () => Promise<string>;
arrayBuffer: () => Promise<ArrayBuffer>;
formData: () => Promise<FormData>;
blob: () => Promise<Blob>;
};
export const decodeBody = <T, K extends keyof ResponseType<T> = "json">(
response: Response,
responseType: K
) => {
const handlers: ResponseType<T> = {
json: response.json,
text: response.text,
arrayBuffer: response.arrayBuffer,
formData: response.formData,
blob: response.blob,
};
const handler = handlers[responseType];
return handler() as ReturnType<ResponseType<T>[K]>;
}; |
I think TypeScript is getting confused because the handler is not bound to the You can see what I mean if you run this: const response = await fetch('https://jsonplaceholder.typicode.com/todos/1');
const { json } = response;
const data = await json();
console.log(data); You might be able to fix the problem by doing this: const handlers: ResponseType<T> = {
json: response.json.bind(response),
text: response.text.bind(response),
arrayBuffer: response.arrayBuffer.bind(response),
formData: response.formData.bind(response),
blob: response.blob.bind(response),
}; But is that |
Example:
const api = ky.extend({
responseBodyEncoding: "json"
})
Now any time a request is made with ky, it will be as if you called .json() after it
api.get("example.com").json() -> api.get("example.com")
The text was updated successfully, but these errors were encountered: