Skip to content

Commit 9587c9d

Browse files
authoredJul 19, 2024··
Add support for passing a function to ky.extend() (#611)
* feat(extend): support function options callback function receives parent's default options to allow extended instance to refer to modify parent options Closes #586 * test(extend): add additional function test cases * docs(extend): add JSDoc example
1 parent 8e171f5 commit 9587c9d

File tree

4 files changed

+118
-21
lines changed

4 files changed

+118
-21
lines changed
 

‎readme.md

+16
Original file line numberDiff line numberDiff line change
@@ -511,6 +511,22 @@ console.log('unicorn' in response);
511511
//=> true
512512
```
513513

514+
You can also refer to parent defaults by providing a function to `.extend()`.
515+
516+
```js
517+
import ky from 'ky';
518+
519+
const api = ky.create({prefixUrl: 'https://example.com/api'});
520+
521+
const usersApi = api.extend((options) => ({prefixUrl: `${options.prefixUrl}/users`}));
522+
523+
const response = await usersApi.get('123');
524+
//=> 'https://example.com/api/users/123'
525+
526+
const response = await api.get('version');
527+
//=> 'https://example.com/api/version'
528+
```
529+
514530
### ky.create(defaultOptions)
515531

516532
Create a new Ky instance with complete new defaults.

‎source/index.ts

+8-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,14 @@ const createInstance = (defaults?: Partial<Options>): KyInstance => {
1717
}
1818

1919
ky.create = (newDefaults?: Partial<Options>) => createInstance(validateAndMerge(newDefaults));
20-
ky.extend = (newDefaults?: Partial<Options>) => createInstance(validateAndMerge(defaults, newDefaults));
20+
ky.extend = (newDefaults?: Partial<Options> | ((parentDefaults: Partial<Options>) => Partial<Options>)) => {
21+
if (typeof newDefaults === 'function') {
22+
newDefaults = newDefaults(defaults ?? {});
23+
}
24+
25+
return createInstance(validateAndMerge(defaults, newDefaults));
26+
};
27+
2128
ky.stop = stop;
2229

2330
return ky as KyInstance;

‎source/types/ky.ts

+18-1
Original file line numberDiff line numberDiff line change
@@ -81,9 +81,26 @@ export type KyInstance = {
8181
8282
In contrast to `ky.create()`, `ky.extend()` inherits defaults from its parent.
8383
84+
You can also refer to parent defaults by providing a function to `.extend()`.
85+
86+
@example
87+
```js
88+
import ky from 'ky';
89+
90+
const api = ky.create({prefixUrl: 'https://example.com/api'});
91+
92+
const usersApi = api.extend((options) => ({prefixUrl: `${options.prefixUrl}/users`}));
93+
94+
const response = await usersApi.get('123');
95+
//=> 'https://example.com/api/users/123'
96+
97+
const response = await api.get('version');
98+
//=> 'https://example.com/api/version'
99+
```
100+
84101
@returns A new Ky instance.
85102
*/
86-
extend: (defaultOptions: Options) => KyInstance;
103+
extend: (defaultOptions: Options | ((parentOptions: Options) => Options)) => KyInstance;
87104

88105
/**
89106
A `Symbol` that can be returned by a `beforeRetry` hook to stop the retry. This will also short circuit the remaining `beforeRetry` hooks.

‎test/main.ts

+76-19
Original file line numberDiff line numberDiff line change
@@ -542,7 +542,7 @@ test('ky.create() does not mangle search params', async t => {
542542
await server.close();
543543
});
544544

545-
test('ky.extend()', async t => {
545+
const extendHooksMacro = test.macro<[{useFunction: boolean}]>(async (t, {useFunction}) => {
546546
const server = await createHttpTestServer();
547547
server.get('/', (_request, response) => {
548548
response.end();
@@ -551,25 +551,28 @@ test('ky.extend()', async t => {
551551
let isOriginBeforeRequestTrigged = false;
552552
let isExtendBeforeRequestTrigged = false;
553553

554+
const intermediateOptions = {
555+
hooks: {
556+
beforeRequest: [
557+
() => {
558+
isOriginBeforeRequestTrigged = true;
559+
},
560+
],
561+
},
562+
};
563+
const extendedOptions = {
564+
hooks: {
565+
beforeRequest: [
566+
() => {
567+
isExtendBeforeRequestTrigged = true;
568+
},
569+
],
570+
},
571+
};
572+
554573
const extended = ky
555-
.extend({
556-
hooks: {
557-
beforeRequest: [
558-
() => {
559-
isOriginBeforeRequestTrigged = true;
560-
},
561-
],
562-
},
563-
})
564-
.extend({
565-
hooks: {
566-
beforeRequest: [
567-
() => {
568-
isExtendBeforeRequestTrigged = true;
569-
},
570-
],
571-
},
572-
});
574+
.extend(useFunction ? () => intermediateOptions : intermediateOptions)
575+
.extend(useFunction ? () => extendedOptions : extendedOptions);
573576

574577
await extended(server.url);
575578

@@ -582,6 +585,60 @@ test('ky.extend()', async t => {
582585
await server.close();
583586
});
584587

588+
test('ky.extend() appends hooks', extendHooksMacro, {useFunction: false});
589+
590+
test('ky.extend() with function appends hooks', extendHooksMacro, {useFunction: false});
591+
592+
test('ky.extend() with function overrides primitives in parent defaults', async t => {
593+
const server = await createHttpTestServer();
594+
server.get('*', (request, response) => {
595+
response.end(request.url);
596+
});
597+
598+
const api = ky.create({prefixUrl: `${server.url}/api`});
599+
const usersApi = api.extend(options => ({prefixUrl: `${options.prefixUrl!.toString()}/users`}));
600+
601+
t.is(await usersApi.get('123').text(), '/api/users/123');
602+
t.is(await api.get('version').text(), '/api/version');
603+
604+
{
605+
const {ok} = await api.head(server.url);
606+
t.true(ok);
607+
}
608+
609+
{
610+
const {ok} = await usersApi.head(server.url);
611+
t.true(ok);
612+
}
613+
614+
await server.close();
615+
});
616+
617+
test('ky.extend() with function retains parent defaults when not specified', async t => {
618+
const server = await createHttpTestServer();
619+
server.get('*', (request, response) => {
620+
response.end(request.url);
621+
});
622+
623+
const api = ky.create({prefixUrl: `${server.url}/api`});
624+
const extendedApi = api.extend(() => ({}));
625+
626+
t.is(await api.get('version').text(), '/api/version');
627+
t.is(await extendedApi.get('something').text(), '/api/something');
628+
629+
{
630+
const {ok} = await api.head(server.url);
631+
t.true(ok);
632+
}
633+
634+
{
635+
const {ok} = await extendedApi.head(server.url);
636+
t.true(ok);
637+
}
638+
639+
await server.close();
640+
});
641+
585642
test('throws DOMException/Error with name AbortError when aborted by user', async t => {
586643
const server = await createHttpTestServer();
587644
// eslint-disable-next-line @typescript-eslint/no-empty-function

0 commit comments

Comments
 (0)
Please sign in to comment.