Spaces:
Sleeping
Sleeping
; | |
const Assert = require('./assert'); | |
const Clone = require('./clone'); | |
const Utils = require('./utils'); | |
const internals = {}; | |
module.exports = internals.merge = function (target, source, options) { | |
Assert(target && typeof target === 'object', 'Invalid target value: must be an object'); | |
Assert(source === null || source === undefined || typeof source === 'object', 'Invalid source value: must be null, undefined, or an object'); | |
if (!source) { | |
return target; | |
} | |
options = Object.assign({ nullOverride: true, mergeArrays: true }, options); | |
if (Array.isArray(source)) { | |
Assert(Array.isArray(target), 'Cannot merge array onto an object'); | |
if (!options.mergeArrays) { | |
target.length = 0; // Must not change target assignment | |
} | |
for (let i = 0; i < source.length; ++i) { | |
target.push(Clone(source[i], { symbols: options.symbols })); | |
} | |
return target; | |
} | |
const keys = Utils.keys(source, options); | |
for (let i = 0; i < keys.length; ++i) { | |
const key = keys[i]; | |
if (key === '__proto__' || | |
!Object.prototype.propertyIsEnumerable.call(source, key)) { | |
continue; | |
} | |
const value = source[key]; | |
if (value && | |
typeof value === 'object') { | |
if (!target[key] || | |
typeof target[key] !== 'object' || | |
(Array.isArray(target[key]) !== Array.isArray(value)) || | |
value instanceof Date || | |
(Buffer && Buffer.isBuffer(value)) || // $lab:coverage:ignore$ | |
value instanceof RegExp) { | |
target[key] = Clone(value, { symbols: options.symbols }); | |
} | |
else { | |
internals.merge(target[key], value, options); | |
} | |
} | |
else { | |
if (value !== null && | |
value !== undefined) { // Explicit to preserve empty strings | |
target[key] = value; | |
} | |
else if (options.nullOverride) { | |
target[key] = value; | |
} | |
} | |
} | |
return target; | |
}; | |