From af5c3fd5609f5e5fe444ecf15cfc8ebb199e972b Mon Sep 17 00:00:00 2001
From: Szymon Marczak <36894700+szmarczak@users.noreply.github.com>
Date: Thu, 2 Aug 2018 10:38:44 +0200
Subject: [PATCH] Improve merging options (#539)
---
advanced-creation.md | 2 +-
readme.md | 36 +++++++++++-----------
source/create.js | 6 ++--
source/{merge-options.js => merge.js} | 28 ++++++++---------
test/create.js | 43 ++++++++++++++-------------
test/headers.js | 16 ++++++++--
6 files changed, 70 insertions(+), 61 deletions(-)
rename source/{merge-options.js => merge.js} (52%)
diff --git a/advanced-creation.md b/advanced-creation.md
index e781a91e9..0bef1fce7 100644
--- a/advanced-creation.md
+++ b/advanced-creation.md
@@ -11,7 +11,7 @@ Configure a new `got` instance with the provided settings.
##### [options](readme.md#options)
-To inherit from parent, set it as `got.defaults.options` or use [`got.mergeOptions(defaults.options, options)`](readme.md#gotmergeOptionsparentoptions-newoptions).
+To inherit from parent, set it as `got.defaults.options` or use [`got.mergeOptions(defaults.options, options)`](readme.md#gotmergeoptionsparentoptions-newoptions).
**Note**: Avoid using [object spread](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#Spread_in_object_literals) as it doesn't work recursively.
##### methods
diff --git a/readme.md b/readme.md
index 9982cda27..b028cb98b 100644
--- a/readme.md
+++ b/readme.md
@@ -124,7 +124,7 @@ Default: `{}`
Request headers.
-Existing headers will be overwritten. Headers set to `null` or `undefined` will be omitted.
+Existing headers will be overwritten. Headers set to `null` will be omitted.
###### stream
@@ -358,7 +358,7 @@ Sets `options.method` to the method name and makes a request.
#### got.extend([options])
-Configure a new `got` instance with default `options`. `options` are merged with the extended instance's `defaults.options` as described in [`got.mergeOptions`](#gotmergeoptionsparentoptions-newoptions).
+Configure a new `got` instance with default `options`. `options` are merged with the parent instance's `defaults.options` using [`got.mergeOptions`](#gotmergeoptionsparentoptions-newoptions).
```js
@@ -403,30 +403,28 @@ client.get('/demo');
*Need more control over the behavior of Got? Check out the [`got.create()`](advanced-creation.md).*
-**Both `got.extend(options)` and `got.create(options)` will freeze the instance's default options. For `got.extend()`, the instance's default options are the result of `got.mergeOptions`, which effectively copies plain `Object` and `Array` values. Therefore, you should treat objects passed to these methods as immutable.**
-
#### got.mergeOptions(parentOptions, newOptions)
-Extends parent options. The options objects are deeply merged to a new object. The value of each property is determined as follows:
-
-- If the new value is `undefined` the parent value is preserved.
-- If the parent value is an instance of `URL` and the new value is a `string` or `URL`, a new URL instance is created, using the parent value as the base: `new URL(new, parent)`.
-- If the new value is an `Array`, the new value is recursively merged into an empty array (the source value is discarded). `undefined` elements in the source array are not assigned during the merge, so the resulting array will have an empty item where the source array had an `undefined` item.
-- If the new value is a plain `Object`
- - If the parent value is a plain `Object`, both values are merged recursively into a new `Object`.
- - Otherwise, only the new value is merged recursively into a new `Object`.
-- Otherwise, the new value is assigned to the property.
-
-Avoid using [object spread](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#Spread_in_object_literals) as it doesn't work recursively:
+Extends parent options. Avoid using [object spread](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#Spread_in_object_literals) as it doesn't work recursively:
```js
-const a = {headers: {cat: 'meow', habitat: ['house', 'alley']}};
-const b = {headers: {cow: 'moo', habitat: ['barn']}};
+const a = {headers: {cat: 'meow', wolf: ['bark', 'wrrr']}};
+const b = {headers: {cow: 'moo', wolf: ['auuu']}};
-{...a, ...b} // => {headers: {cow: 'moo'}}
-got.mergeOptions(a, b) // => {headers: {cat: 'meow', cow: 'moo', habitat: ['barn']}}
+{...a, ...b} // => {headers: {cow: 'moo', wolf: ['auuu']}}
+got.mergeOptions(a, b) // => {headers: {cat: 'meow', cow: 'moo', wolf: ['auuu']}}
```
+Options are deeply merged to a new object. The value of each key is determined as follows:
+
+- If the new property is set to `undefined`, it keeps the old one.
+- If the parent property is an instance of `URL` and the new value is a `string` or `URL`, a new URL instance is created: [`new URL(new, parent)`](https://developer.mozilla.org/en-US/docs/Web/API/URL/URL#Syntax).
+- If the new property is a plain `Object`:
+ - If the parent property is a plain `Object` too, both values are merged recursively into a new `Object`.
+ - Otherwise, only the new value is deeply cloned.
+- If the new property is an `Array`, it overwrites the old one with a deep clone of the new property.
+- Otherwise, the new value is assigned to the key.
+
## Errors
Each error contains (if available) `statusCode`, `statusMessage`, `host`, `hostname`, `method`, `path`, `protocol` and `url` properties to make debugging easier.
diff --git a/source/create.js b/source/create.js
index 2cca4a7dd..1d86b8a09 100644
--- a/source/create.js
+++ b/source/create.js
@@ -1,16 +1,18 @@
'use strict';
const errors = require('./errors');
-const mergeOptions = require('./merge-options');
const asStream = require('./as-stream');
const asPromise = require('./as-promise');
const normalizeArguments = require('./normalize-arguments');
+const merge = require('./merge');
const deepFreeze = require('./deep-freeze');
const next = options => options.stream ? asStream(options) : asPromise(options);
+const mergeOptions = (defaults, options = {}) => merge({}, defaults, options);
const create = defaults => {
+ defaults = merge({}, defaults);
if (!defaults.handler) {
- defaults.handler = (options, next) => next(options);
+ defaults.handler = next;
}
function got(url, options) {
diff --git a/source/merge-options.js b/source/merge.js
similarity index 52%
rename from source/merge-options.js
rename to source/merge.js
index 91d231c68..140230740 100644
--- a/source/merge-options.js
+++ b/source/merge.js
@@ -1,25 +1,16 @@
+'use strict';
const {URL} = require('url');
const is = require('@sindresorhus/is');
-module.exports = (defaults, options = {}) => {
- return merge({}, defaults, options);
-};
-
-function merge(target, ...sources) {
+const merge = (target, ...sources) => {
for (const source of sources) {
- const sourceIter = is.array(source) ?
- source.entries() :
- Object.entries(source);
- for (const [key, sourceValue] of sourceIter) {
- const targetValue = target[key];
+ for (const [key, sourceValue] of Object.entries(source)) {
if (is.undefined(sourceValue)) {
continue;
}
- if (is.array(sourceValue)) {
- target[key] = merge(new Array(sourceValue.length), sourceValue);
- } else if (is.urlInstance(targetValue) && (
- is.urlInstance(sourceValue) || is.string(sourceValue)
- )) {
+
+ const targetValue = target[key];
+ if (is.urlInstance(targetValue) && (is.urlInstance(sourceValue) || is.string(sourceValue))) {
target[key] = new URL(sourceValue, targetValue);
} else if (is.plainObject(sourceValue)) {
if (is.plainObject(targetValue)) {
@@ -27,10 +18,15 @@ function merge(target, ...sources) {
} else {
target[key] = merge({}, sourceValue);
}
+ } else if (is.array(sourceValue)) {
+ target[key] = merge([], sourceValue);
} else {
target[key] = sourceValue;
}
}
}
+
return target;
-}
+};
+
+module.exports = merge;
diff --git a/test/create.js b/test/create.js
index 4004757b3..bcd60c51d 100644
--- a/test/create.js
+++ b/test/create.js
@@ -77,26 +77,18 @@ test('custom headers (extend)', async t => {
t.is(headers.unicorn, 'rainbow');
});
-test('extend overwrites arrays', t => {
+test('extend overwrites arrays with a deep clone', t => {
const statusCodes = [408];
const a = got.extend({retry: {statusCodes}});
- t.deepEqual(a.defaults.options.retry.statusCodes, statusCodes);
+ statusCodes[0] = 500;
+ t.deepEqual(a.defaults.options.retry.statusCodes, [408]);
t.not(a.defaults.options.retry.statusCodes, statusCodes);
});
-test('extend overwrites null', t => {
- const statusCodes = null;
- const a = got.extend({retry: {statusCodes}});
- t.is(a.defaults.options.retry.statusCodes, statusCodes);
-});
-
-test('extend ignores source values set to undefined', t => {
- const a = got.extend({
- headers: {foo: undefined, 'user-agent': undefined}
- });
- const b = a.extend({headers: {foo: undefined}});
+test('extend keeps the old value if the new one is undefined', t => {
+ const a = got.extend({headers: undefined});
t.deepEqual(
- b.defaults.options.headers,
+ a.defaults.options.headers,
got.defaults.options.headers
);
});
@@ -107,12 +99,6 @@ test('extend merges URL instances', t => {
t.is(b.defaults.options.baseUrl.toString(), 'https://example.com/foo');
});
-test('extend ignores object values set to undefined (root keys)', t => {
- t.true(Reflect.has(got.defaults.options, 'headers'));
- const a = got.extend({headers: undefined});
- t.deepEqual(a.defaults.options, got.defaults.options);
-});
-
test('create', async t => {
const instance = got.create({
options: {},
@@ -161,3 +147,20 @@ test('no tampering with defaults', t => {
t.is(instance.defaults.options.baseUrl, 'example');
t.is(instance2.defaults.options.baseUrl, 'example');
});
+
+test('defaults are cloned on instance creation', t => {
+ const options = {foo: 'bar'};
+ const methods = ['get'];
+ const instance = got.create({
+ methods,
+ options
+ });
+
+ t.notThrows(() => {
+ options.foo = 'foo';
+ methods[0] = 'post';
+ });
+
+ t.not(options.foo, instance.defaults.options.foo);
+ t.not(methods[0], instance.defaults.methods[0]);
+});
diff --git a/test/headers.js b/test/headers.js
index c95a726d5..8c8319a63 100644
--- a/test/headers.js
+++ b/test/headers.js
@@ -142,12 +142,22 @@ test('remove null value headers', async t => {
t.false(Reflect.has(headers, 'user-agent'));
});
-test('remove undefined value headers', async t => {
+test('setting a header to undefined keeps the old value', async t => {
const {body} = await got(s.url, {
headers: {
- foo: undefined
+ 'user-agent': undefined
}
});
const headers = JSON.parse(body);
- t.false(Reflect.has(headers, 'foo'));
+ t.not(headers['user-agent'], undefined);
+});
+
+test('non-existent headers set to undefined are omitted', async t => {
+ const {body} = await got(s.url, {
+ headers: {
+ blah: undefined
+ }
+ });
+ const headers = JSON.parse(body);
+ t.false(Reflect.has(headers, 'blah'));
});