forked from Fediversity/Fediversity
103 lines
2.6 KiB
JavaScript
103 lines
2.6 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
const Assert = require('./assert');
|
||
|
const Clone = require('./clone');
|
||
|
const Merge = require('./merge');
|
||
|
const Reach = require('./reach');
|
||
|
|
||
|
|
||
|
const internals = {};
|
||
|
|
||
|
|
||
|
module.exports = function (defaults, source, options = {}) {
|
||
|
|
||
|
Assert(defaults && typeof defaults === 'object', 'Invalid defaults value: must be an object');
|
||
|
Assert(!source || source === true || typeof source === 'object', 'Invalid source value: must be true, falsy or an object');
|
||
|
Assert(typeof options === 'object', 'Invalid options: must be an object');
|
||
|
|
||
|
if (!source) { // If no source, return null
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
if (options.shallow) {
|
||
|
return internals.applyToDefaultsWithShallow(defaults, source, options);
|
||
|
}
|
||
|
|
||
|
const copy = Clone(defaults);
|
||
|
|
||
|
if (source === true) { // If source is set to true, use defaults
|
||
|
return copy;
|
||
|
}
|
||
|
|
||
|
const nullOverride = options.nullOverride !== undefined ? options.nullOverride : false;
|
||
|
return Merge(copy, source, { nullOverride, mergeArrays: false });
|
||
|
};
|
||
|
|
||
|
|
||
|
internals.applyToDefaultsWithShallow = function (defaults, source, options) {
|
||
|
|
||
|
const keys = options.shallow;
|
||
|
Assert(Array.isArray(keys), 'Invalid keys');
|
||
|
|
||
|
const seen = new Map();
|
||
|
const merge = source === true ? null : new Set();
|
||
|
|
||
|
for (let key of keys) {
|
||
|
key = Array.isArray(key) ? key : key.split('.'); // Pre-split optimization
|
||
|
|
||
|
const ref = Reach(defaults, key);
|
||
|
if (ref &&
|
||
|
typeof ref === 'object') {
|
||
|
|
||
|
seen.set(ref, merge && Reach(source, key) || ref);
|
||
|
}
|
||
|
else if (merge) {
|
||
|
merge.add(key);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const copy = Clone(defaults, {}, seen);
|
||
|
|
||
|
if (!merge) {
|
||
|
return copy;
|
||
|
}
|
||
|
|
||
|
for (const key of merge) {
|
||
|
internals.reachCopy(copy, source, key);
|
||
|
}
|
||
|
|
||
|
const nullOverride = options.nullOverride !== undefined ? options.nullOverride : false;
|
||
|
return Merge(copy, source, { nullOverride, mergeArrays: false });
|
||
|
};
|
||
|
|
||
|
|
||
|
internals.reachCopy = function (dst, src, path) {
|
||
|
|
||
|
for (const segment of path) {
|
||
|
if (!(segment in src)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const val = src[segment];
|
||
|
|
||
|
if (typeof val !== 'object' || val === null) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
src = val;
|
||
|
}
|
||
|
|
||
|
const value = src;
|
||
|
let ref = dst;
|
||
|
for (let i = 0; i < path.length - 1; ++i) {
|
||
|
const segment = path[i];
|
||
|
if (typeof ref[segment] !== 'object') {
|
||
|
ref[segment] = {};
|
||
|
}
|
||
|
|
||
|
ref = ref[segment];
|
||
|
}
|
||
|
|
||
|
ref[path[path.length - 1]] = value;
|
||
|
};
|