You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

152 lines
4.2 KiB

import * as isEqual from 'lodash.isequal';
import {ow} from '../../ow';
import {Predicate, Context} from './predicate';
export class ArrayPredicate extends Predicate<any[]> {
constructor(context?: Context) {
super('array', context);
}
/**
* Test an array to have a specific length.
*
* @param length The length of the array.
*/
length(length: number) {
return this.addValidator({
message: value => `Expected array to have length \`${length}\`, got \`${value.length}\``,
validator: value => value.length === length
});
}
/**
* Test an array to have a minimum length.
*
* @param length The minimum length of the array.
*/
minLength(length: number) {
return this.addValidator({
message: value => `Expected array to have a minimum length of \`${length}\`, got \`${value.length}\``,
validator: value => value.length >= length
});
}
/**
* Test an array to have a maximum length.
*
* @param length The maximum length of the array.
*/
maxLength(length: number) {
return this.addValidator({
message: value => `Expected array to have a maximum length of \`${length}\`, got \`${value.length}\``,
validator: value => value.length <= length
});
}
/**
* Test an array to start with a specific value. The value is tested by identity, not structure.
*
* @param searchElement The value that should be the start of the array.
*/
startsWith(searchElement: any) {
return this.addValidator({
message: value => `Expected array to start with \`${searchElement}\`, got \`${value[0]}\``,
validator: value => value[0] === searchElement
});
}
/**
* Test an array to end with a specific value. The value is tested by identity, not structure.
*
* @param searchElement The value that should be the end of the array.
*/
endsWith(searchElement: any) {
return this.addValidator({
message: value => `Expected array to end with \`${searchElement}\`, got \`${value[value.length - 1]}\``,
validator: value => value[value.length - 1] === searchElement
});
}
/**
* Test an array to include all the provided elements. The values are tested by identity, not structure.
*
* @param searchElements The values that should be included in the array.
*/
includes(...searchElements: any[]) {
return this.addValidator({
message: value => `Expected array to include all elements of \`${JSON.stringify(searchElements)}\`, got \`${JSON.stringify(value)}\``,
validator: value => searchElements.every(el => value.indexOf(el) !== -1)
});
}
/**
* Test an array to include any of the provided elements. The values are tested by identity, not structure.
*
* @param searchElements The values that should be included in the array.
*/
includesAny(...searchElements: any[]) {
return this.addValidator({
message: value => `Expected array to include any element of \`${JSON.stringify(searchElements)}\`, got \`${JSON.stringify(value)}\``,
validator: value => searchElements.some(el => value.indexOf(el) !== -1)
});
}
/**
* Test an array to be empty.
*/
get empty() {
return this.addValidator({
message: value => `Expected array to be empty, got \`${JSON.stringify(value)}\``,
validator: value => value.length === 0
});
}
/**
* Test an array to be not empty.
*/
get nonEmpty() {
return this.addValidator({
message: () => 'Expected array to not be empty',
validator: value => value.length > 0
});
}
/**
* Test an array to be deeply equal to the provided array.
*
* @param expected Expected value to match.
*/
deepEqual(expected: any[]) {
return this.addValidator({
message: value => `Expected array to be deeply equal to \`${JSON.stringify(expected)}\`, got \`${JSON.stringify(value)}\``,
validator: value => isEqual(value, expected)
});
}
/**
* Test all elements in the array to match to provided predicate.
*
* @param predicate The predicate that should be applied against every individual item.
*/
ofType<T>(predicate: Predicate<T>) {
let error: string;
return this.addValidator({
message: () => error,
validator: value => {
try {
for (const item of value) {
ow(item, predicate);
}
return true;
} catch (err) {
error = err.message;
return false;
}
}
});
}
}