Musique/node_modules/ow/dist/predicates/object.js

174 lines
6.3 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.ObjectPredicate = void 0;
const is_1 = require("@sindresorhus/is");
const dotProp = require("dot-prop");
const isEqual = require("lodash.isequal");
const has_items_1 = require("../utils/has-items");
const of_type_1 = require("../utils/of-type");
const of_type_deep_1 = require("../utils/of-type-deep");
const match_shape_1 = require("../utils/match-shape");
const predicate_1 = require("./predicate");
class ObjectPredicate extends predicate_1.Predicate {
/**
@hidden
*/
constructor(options) {
super('object', options);
}
/**
Test if an Object is a plain object.
*/
get plain() {
return this.addValidator({
message: (_, label) => `Expected ${label} to be a plain object`,
validator: object => is_1.default.plainObject(object)
});
}
/**
Test an object to be empty.
*/
get empty() {
return this.addValidator({
message: (object, label) => `Expected ${label} to be empty, got \`${JSON.stringify(object)}\``,
validator: object => Object.keys(object).length === 0
});
}
/**
Test an object to be not empty.
*/
get nonEmpty() {
return this.addValidator({
message: (_, label) => `Expected ${label} to not be empty`,
validator: object => Object.keys(object).length > 0
});
}
/**
Test all the values in the object to match the provided predicate.
@param predicate - The predicate that should be applied against every value in the object.
*/
valuesOfType(predicate) {
return this.addValidator({
message: (_, label, error) => `(${label}) ${error}`,
validator: object => of_type_1.default(Object.values(object), 'values', predicate)
});
}
/**
Test all the values in the object deeply to match the provided predicate.
@param predicate - The predicate that should be applied against every value in the object.
*/
deepValuesOfType(predicate) {
return this.addValidator({
message: (_, label, error) => `(${label}) ${error}`,
validator: object => of_type_deep_1.default(object, predicate)
});
}
/**
Test an object to be deeply equal to the provided object.
@param expected - Expected object to match.
*/
deepEqual(expected) {
return this.addValidator({
message: (object, label) => `Expected ${label} to be deeply equal to \`${JSON.stringify(expected)}\`, got \`${JSON.stringify(object)}\``,
validator: object => isEqual(object, expected)
});
}
/**
Test an object to be of a specific instance type.
@param instance - The expected instance type of the object.
*/
instanceOf(instance) {
return this.addValidator({
message: (object, label) => {
var _a;
let { name } = (_a = object === null || object === void 0 ? void 0 : object.constructor) !== null && _a !== void 0 ? _a : {};
if (!name || name === 'Object') {
name = JSON.stringify(object);
}
return `Expected ${label} \`${name}\` to be of type \`${instance.name}\``;
},
validator: object => object instanceof instance
});
}
/**
Test an object to include all the provided keys. You can use [dot-notation](https://github.com/sindresorhus/dot-prop) in a key to access nested properties.
@param keys - The keys that should be present in the object.
*/
hasKeys(...keys) {
return this.addValidator({
message: (_, label, missingKeys) => `Expected ${label} to have keys \`${JSON.stringify(missingKeys)}\``,
validator: object => has_items_1.default({
has: item => dotProp.has(object, item)
}, keys)
});
}
/**
Test an object to include any of the provided keys. You can use [dot-notation](https://github.com/sindresorhus/dot-prop) in a key to access nested properties.
@param keys - The keys that could be a key in the object.
*/
hasAnyKeys(...keys) {
return this.addValidator({
message: (_, label) => `Expected ${label} to have any key of \`${JSON.stringify(keys)}\``,
validator: object => keys.some(key => dotProp.has(object, key))
});
}
/**
Test an object to match the `shape` partially. This means that it ignores unexpected properties. The shape comparison is deep.
The shape is an object which describes how the tested object should look like. The keys are the same as the source object and the values are predicates.
@param shape - Shape to test the object against.
@example
```
import ow from 'ow';
const object = {
unicorn: '🦄',
rainbow: '🌈'
};
ow(object, ow.object.partialShape({
unicorn: ow.string
}));
```
*/
partialShape(shape) {
return this.addValidator({
// TODO: Improve this when message handling becomes smarter
message: (_, label, message) => `${message.replace('Expected', 'Expected property')} in ${label}`,
validator: object => match_shape_1.partial(object, shape)
});
}
/**
Test an object to match the `shape` exactly. This means that will fail if it comes across unexpected properties. The shape comparison is deep.
The shape is an object which describes how the tested object should look like. The keys are the same as the source object and the values are predicates.
@param shape - Shape to test the object against.
@example
```
import ow from 'ow';
ow({unicorn: '🦄'}, ow.object.exactShape({
unicorn: ow.string
}));
```
*/
exactShape(shape) {
// TODO [typescript@>=5] If higher-kinded types are supported natively by typescript, refactor `addValidator` to use them to avoid the usage of `any`. Otherwise, bump or remove this TODO.
return this.addValidator({
// TODO: Improve this when message handling becomes smarter
message: (_, label, message) => `${message.replace('Expected', 'Expected property')} in ${label}`,
validator: object => match_shape_1.exact(object, shape)
});
}
}
exports.ObjectPredicate = ObjectPredicate;