Blork! Mini runtime type checking in Javascript
A mini type checker for locking down the external edges of your code. Mainly for use in modules when you don"t know who'll be using the code. Minimal boilerplate code keeps your functions hyper readable and lets them be their beautiful minimal best selves (...or something?)
Blork is fully unit tested and 100% covered (if you're into that!). Heaps of love has been put into the niceness and consistency of error messages, so hopefully you'll enjoy that too.
Installation
npm install blork
Usage
check(): Check individual values
The check()
function allows you to test that individual values correspond to a type, and throw a TypeError
if not. This is primarily designed for checking function arguments but can be used for any purpose.
check()
accepts four arguments:
value
The value to checktype
The type to check the value against (full reference list of types is available below)prefix
An optional string name/prefix for the value, which is prepended to any error message thrown to help debuggingerror
An optional custom error type to throw if the check fails
import { check } from "blork";
check("Sally", "string");
check("Sally", String);
check("Sally", "number");
check("Sally", Boolean);
check("Sally", "num", "name");
check(true, "str", "status");
check(123, "str", "num", ReferenceError);
Type modifiers
type
will mostly be specified with a type string (a full list of string types is available below), and these string types can also be modified using other characters:
- Appending
?
question mark to any type string makes it optional (which means it also allows undefined
). - Prepending a
!
exclaimation mark to any type string makes it inverted (e.g. !string
means anything except string). - Multiple types can be combined with
|
and &
for OR and AND conditions (optionally grouped with ()
parens to resolve ambiguity). - Appending a
+
means non-empty (e.g. arr+
str+
means non-empty arrays and strings respectively).
check(undefined, "number");
check(undefined, "number?");
check(null, "number?");
check(123, "!str");
check(123, "!int");
check(1234, "int | str");
check(null, "int | str");
check("abc", "string & !falsy");
check("", "string & !falsy");
check("abc", "str+");
check("", "str+");
check([1, 2, 3], "arr{2,4}");
check([1], "arr{2,3}");
check([1, 3, 3, 4], "arr{,3}");
check([1, 2], "arr{3,}");
check([1, 2, 3], "num[]");
check(["a", "b"], "num[]");
check([1, "a"], "[int, str]");
check([1, false], "[int, str]");
check({ a: 1 }, "{ camel: integer }");
check({ "$": 1 }, "{ camel: integer }");
Checking objects and arrays
Blork can also perform deep checks on objects and arrays to ensure the schema is correct deeply. You can use literal arrays or literal objects with check()
or args()
to do so:
check({ name: "Sally" }, { name: "string" });
check(["Sally", "John", "Sonia"], ["str"]);
check([1029, "Sonia"], ["number", "string"]);
check({ name: "Sally" }, { name: "string" });
check(["Sally", "John", "Sonia"], ["str"]);
check([1029, "Sonia"], ["number", "string"]);
check([1029, "Sonia", true], ["number", "string"]);
Arrays and objects can be deeply nested within each other and Blork will recursively check the schema all the way down:
check(
[
{ id: 1028, name: "Sally", status: [1, 2, 3] },
{ id: 1062, name: "Bobby", status: [1, 2, 3] }
],
[
{ id: Number, name: String, status: [Number] }
]
);
check(
[
{ id: 1028, name: "Sally", status: [1, 2, 3] },
{ id: 1062, name: "Bobby", status: [1, 2, "not_a_number"] }
],
[
{ id: Number, name: String, status: [Number] }
]
);
args(): Check function arguments
The primary use case of Blork is validating function input arguments. The args()
function is provided for this purpose and can be passed four arguments:
arguments
| The arguments object provided automatically to functions in Javascripttypes
| An array identifying the types for the arguments (list of types is available below)prefix
An optional string name/prefix for the value, which is prepended to any error message thrown to help debuggingerror
An optional custom error type to throw if the check fails
import { args } from "blork";
export default function myFunc(definitelyString, optionalNumber)
{
args(arguments, ["string", "number?"]);
return "It passed!";
}
myFunc("abc", 123);
myFunc("abc");
myFunc(123);
myFunc("abc", "abc");
myFunc();
myFunc("abc", 123, true);
assert(): Check a random true/false statement.
Check a random true/false statement using the assert()
function. This allows you to make other assertions with a similar argument order to check()
. This is mainly just syntactic sugar, but is neater than messy if (x) throw new X;
type statements.
Takes up to four arguments:
assertion
The true/false value that is the assertion.description
A description of the positive assertion. Must fit the phrase Must ${description}
, e.g. "be unique" or "be equal to dog".prefix
An optional string name/prefix for the value, which is prepended to any error message thrown to help debuggingerror
An optional custom error type to throw if the check fails
import { assert } from "blork";
assert(isUnique(val1), "unique");
assert(isUnique(val2), "be unique");
assert(isUnique(val2), "be unique", "val2");
assert(isUnique(val2), "be unique", "val2", ReferenceError);
add(): Add a custom checker type
Register your own checker using the add()
function. This is great if 1) you're going to be applying the same check over and over, or 2) want to integrate your own checks with Blork's built-in types so your code looks clean.
add()
accepts four arguments:
name
The name of the custom checker (only kebab-case strings allowed).checker
A function that accepts a single argument, value
, and returns true
or false
.description
A description of the type of value that's valid. Must fit the phrase Must be ${description}
, e.g. "positive number" or "unique string". Defaults to name.error=undefined
A custom class that is thrown when this checker fails (can be [VALUES]_ class, not just classes extending Error
). An error set with add() takes precedence for this checker over the error set through
throws()`.
import { add, check } from "blork";
add(
"catty",
(v) => typeof v === "string" && v.strToLower().indexOf("cat") >= 0,
"string containing 'cat'"
);
check("That cat is having fun", "catty");
check("That CAT is having fun", "catty");
check("A dog sits on the chair", "catty");
check("A CAT SAT ON THE MAT", "upper+ & catty");
check("A DOG SAT ON THE MAT", "upper+ & catty");
import { add, args } from "blork";
function myFunc(str)
{
args(arguments, ["catty"]);
return "It passed!";
}
myFunc("That cat is chasing string");
myFunc("A dog sits over there");
throws(): Set a custom error constructor
To change the error object Blork throws when a type doesn't match, use the throws()
function. It accepts a single argument a custom class (can be [VALUES]_ class, not just classes extending Error
).
import { throws, check } from "blork";
class MyError extends Error {};
throws(MyError);
check(true, "false");
blork(): Create an independent instance of Blork
To create an instance of Blork with an independent set of checkers (added with add()
) and an independently set throws()
error object, use the blork()
function.
This functionality is provided so you can ensure multiple versions of Blork in submodules of the same project don't interfere with each other, even if they have been (possibly purposefully) deduped in npm. This is how you can ensure if you've set a custom error for a set of checks, that custom error type is always thrown.
import { blork } from "blork";
const { check, args, add, throws } = blork();
throws(class CustomError extends ValueError);
add("mychecker", v => v === "abc", "'abc'");
check("123", "mychecker");
debug(): Debug any value as a string.
Blork exposes its debugger helper function debug()
, which it uses to format error messages correctly. debug()
accepts any argument and will return a clear string interpretation of the value.
debug()
deals well with large and nested objects/arrays by inserting linebreaks and tabs if line length would be unreasonable. Output is also kept cleanish by only debugging 3 levels deep, truncating long strings, and not recursing into circular references.
import debug from "blork";
debug(undefined);
debug(null);
debug(true);
debug(false);
debug(123);
debug("abc");
debug(Symbol("abc"));
debug(function dog() {});
debug(function() {});
debug({});
debug({ a: 123 });
debug(new Promise());
debug(new MyClass());
debug(new class {}());
ValueError: Great debuggable error class
Internally, when there's a problem with a value, Blork will throw a ValueError
. This value extends TypeError
and standardises error message formats, so errors are consistent and provide the detail a developer should need to debug the issue error quickly and easily.
It accepts three values:
message
The error message describing the issue with the value, e.g. "Must be string"
value
The actual value that was incorrect so a debugged version of this value can appear in the error message, e.g. (received 123)
prefix
A string prefix for the error that should identify the location the error occurred and the name of the value, e.g. "myFunction(): name"
import { ValueError } from "blork";
function myFunc(name) {
if (typeof name !== "string") throw new ValueError("Must be string", name, "myFunc(): name");
}
myFunc(123);
Types
This section lists all types that are available in Blork. A number of different formats can be used for types:
- String types (e.g.
"promise"
and "integer"
) - String modifiers that modify those string types (e.g.
"?"
and "!"
) - Constant and constructor shorthand types (e.g.
null
and String
) - Object and Array literal types (e.g.
{}
and []
)
String types
Type string | Description |
---|
primitive | Any primitive value (undefined, null, booleans, strings, finite numbers) |
null | Value is null |
undefined , undef , void | Value is undefined |
defined , def | Value is not undefined |
boolean , bool | Value is true or false |
true | Value is true |
false | Value is false |
truthy | Any truthy values (i.e. == true) |
falsy | Any falsy values (i.e. == false) |
zero | Value is 0 |
one | Value is 1 |
nan | Value is NaN |
number , num | Any numbers except NaN/Infinity (using Number.isFinite()) |
+number , +num , | Numbers more than or equal to zero |
-number , -num | Numbers less than or equal to zero |
integer , int | Integers (using Number.isInteger()) |
+integer , +int | Positive integers including zero |
-integer , -int | Negative integers including zero |
string , str | Any strings (using typeof) |
alphabetic | alphabetic string (non-empty and alphabetic only) |
numeric | numeric strings (non-empty and numerals 0-9 only) |
alphanumeric | alphanumeric strings (non-empty and alphanumeric only) |
lower | lowercase strings (non-empty and lowercase alphabetic only) |
upper | UPPERCASE strings (non-empty and UPPERCASE alphabetic only) |
camel | camelCase strings e.g. variable/function names (non-empty alphanumeric with lowercase first letter) |
pascal | PascalCase strings e.g. class names (non-empty alphanumeric with uppercase first letter) |
snake | snake_case strings (non-empty alphanumeric lowercase) |
screaming | SCREAMING_SNAKE_CASE strings e.g. environment vars (non-empty uppercase alphanumeric) |
kebab , slug | kebab-case strings e.g. URL slugs (non-empty alphanumeric lowercase) |
train | Train-Case strings e.g. HTTP-Headers (non-empty with uppercase first letters) |
function , func | Functions (using instanceof Function) |
object , obj | Plain objects (using typeof && !null and constructor check) |
objectlike | Any object-like object (using typeof && !null) |
iterable | Objects with a Symbol.iterator method (that can be used with for..of loops) |
circular | Objects with one or more circular references (use !circular to disallow circular references) |
array , arr | Plain arrays (using instanceof Array and constructor check) |
arraylike , arguments , args | Array-like objects, e.g. arguments (any object with numeric .length property, not just arrays) |
map | Instances of Map |
weakmap | Instances of WeakMap |
set | Instances of Set |
weakset | Instances of WeakSet |
promise | Instances of Promise |
date | Instances of Date |
future | Instances of Date with a value in the future |
past | Instances of Date with a value in the past |
regex , regexp | Instances of RegExp (regular expressions) |
symbol | Value is Symbol (using typeof) |
empty | Value is empty (e.g. v.length === 0 (string/array), v.size === 0 (Map/Set), Object.keys(v) === 0 (objects), or !v (anything else) |
any , mixed | Allow any value (transparently passes through with no error) |
json , jsonable | Values that can be successfully converted to JSON and back again! (null, true, false, finite numbers, strings, plain objects, plain arrays) |
String modifiers
String modifier types can be applied to any string type from the list above to modify that type's behaviour.
Type modifier | Description |
---|
(type) | Grouped type, e.g. `(num |
type1 & type2 | AND combined type, e.g. str & upper |
`type1 | type2` |
type[] | Array type (all array entries must match type) |
[type1, type2] | Tuple type (must match tuple exactly) |
{ type } | Object value type (all own props must match type |
{ keyType: type } | Object key:value type (keys and own props must match types) |
!type | Inverted type (opposite is allowed), e.g. !str |
type? | Optional type (allows type or undefined ), e.g. str? |
type+ | Non-empty type, e.g. str+ or num[]+ |
Any string type can be made into an array of that type by appending []
brackets to the type reference. This means the check looks for a plain array whose contents only include the specified type.
check(["a", "b"], "str[]");
check([1, 2, 3], "int[]");
check([], "int[]");
check([1], "int[]+");
check([1, 2], "str[]");
check(["a"], "int[]");
check([], "int[]+");
Array tuples can be specified by surrounding types in []
brackets.
check([true, false], "[bool, bool]")
check(["a", "b"], "[str, str]")
check([1, 2, 3], "[num, num, num]");
check([true, true], "[str, str]")
check([true], "[bool, bool]")
check(["a", "b", "c"], "[str, str]")
Check for objects only containing strings of a specified type by surrounding the type in {}
braces. This means the check looks for a plain object whose contents only include the specified type (whitespace is optional).
check({ a: "a", b: "b" }, "{str}");
check({ a: 1, b: 2 }, "{int}");
check({}, "{int}");
check({ a: 1 }, "{int}+");
check({ a: 1, b: 2 }, "{str}");
check({ a: "a" }, "{int}");
check({}, "{int}+");
A type for the keys can also be specified by using { key: value }
format.
check({ myVar: 123 }, "{ camel: integer }");
check({ "my-var": 123 }, "{ kebab: integer }");
Any string type can be made optional by appending a ?
question mark to the type reference. This means the check will also accept undefined
in addition to the specified type.
check(undefined, "str?");
check(undefined, "lower?");
check(undefined, "int?");
check([undefined, undefined, 123], ["number?"]);
check(123, "str?");
check(null, "str?");
Any type can be made non-empty by appending a +
plus sign to the type reference. This means the check will only pass if the value is non-empty. Specifically this works as follows:
- Strings:
.length
is more than 0 - Map and Set objects:
.size
is more than 0 - Objects and arrays: If it has a
.length
property Number of own properties is not zero (using typeof === "object"
&& Object.keys()
) - Booleans and numbers: Use truthyness (e.g.
true
is non-empty, false
and 0
is empty)
This is equivalent to the inverse of the empty
type.
check("abc", "str+");
check([1], "arr+");
check({ a: 1 }, "obj+");
check(123, "str+");
check([], "arr+");
check({}, "obj+");
To specify a size for the type, you can prepend minimum/maximum with e.g. {12}
, {4,8}
, {4,}
or {,8}
(e.g. RegExp style quantifiers). This allows you to specify e.g. a string with 12 characters, an array with between 10 and 20 items, or an integer with a minimum value of 4.
check("abc", "str{3}");
check(4, "num{,4}");
check(["a", "b"], "arr{1,}");
check([1, 2, 3], "num[]{2,4}");
check("ab", "str{3}");
check(4, "num{,4}");
check(["a", "b"], "arr{1,}");
check([1, 2, 3], "num[]{2,4}");
Any string type can inverted by prepending a !
exclamation mark to the type reference. This means the check will only pass if the inverse of its type is true.
check(undefined, "!str");
check("Abc", "!lower");
check(123.456, "!integer");
check([undefined, "abc", true, false], ["!number"]);
check(123, "!str");
check(true, "!bool");
check([undefined, "abc", true, 123], ["!number"]);
You can use &
and |
to join string types together, to form AND and OR chains of allowed types. This allows you to compose together more complex types like number | string
or date | number | null
or string && custom-checker
|
is used to create an OR type, meaning any of the values is valid, e.g. number|string
or string | null
check(123, "str|num");
check("a", "str|num");
check(null, "str|num");
check(null, "str|num|bool|func|obj");
&
is used to create an AND type, meaning the value must pass all of the checks to be valid. This is primarily useful for custom checkers e.g. lower & username-unique
.
add("catty", v => v.toLowerCase().indexOf("cat") >= 0);
check("this cat is crazy!", "lower & catty");
check("THIS CAT IS CRAZY", "upper & catty");
check("THIS CAT IS CRAZY", "lower & catty");
check("THIS DOG IS CRAZY", "string & catty");
Note: Built in checkers like lower
or int+
already check the basic type of a value (e.g. string and number), so there's no need to use string & lower
or number & int+
— internally the value will be checked twice. Spaces around the &
or |
are optional.
()
parentheses can be used to create a 'grouped type'. This is useful to specify an array that allows several types, to make an invert/optional type of several types, or to state an explicit precence order for &
and |
.
check([123, "abc"], "(str|num)[]");
check({ a: 123, b: "abc" }, "!(str|num)");
check("", "(int & truthy) | (str & falsy)");
check(12, "(int & truthy) | (str & falsy)");
Constructor and constant types
For convenience some constructors (e.g. String
) and constants (e.g. null
) can be used as types in args()
and check()
. The following built-in objects and constants are supported:
Type | Description |
---|
Boolean | Same as 'boolean' type |
String | Same as 'string' type |
Number | Same as 'number' type |
true | Same as 'true' type |
false | Same as 'false' type |
null | Same as 'null' type |
undefined | Same as 'undefined' type |
You can pass in any class name, and Blork will check the value using instanceof
and generate a corresponding error message if the type doesn't match.
Using Object
and Array
constructors will work also and will allow any object that is instanceof Object
or instanceof Array
. Note: this is not the same as e.g. the 'object'
and 'array'
string types, which only allow plain objects and arrays.
check(true, Boolean);
check("abc", String);
check(123, Number);
check(new Date, Date);
check(new MyClass, MyClass);
check(Promise.resolved(true), Promise);
check([true, true, false], [Boolean]);
check({ name: 123 }, { name: Number });
check("abc", Boolean);
check("abc", String);
check("abc", String, "myVar");
check(new MyClass, OtherClass);
check({ name: 123 }, { name: String });
check({ name: 123 }, { name: String }, "myObj");
Object literal type
To check the types of object properties, use a literal object as a type. You can also deeply nest these properties and the types will be checked recursively and will generate useful debuggable error messages.
Note: it is fine for objects to contain additional properties that don't have a type specified.
check({ name: "abc" }, { name: "str" });
check({ name: "abc" }, { name: "str?", age: "num?" });
check({ name: "abc", additional: true }, { name: "str" });
check({ age: "apple" }, { age: "num" });
check({ size: { height: 10, width: "abc" } }, { size: { height: "num", width: "num" } });
To check that the type of any properties conform to a single type, use the VALUES
symbol and create a [VALUES]
key. This allows you to check objects that don't have known keys (e.g. from user generated data). This is similar to how indexer keys work in Flow or Typescript.
import { check, VALUES } from "blork";
check(
{ a: 1, b: 2, c: 3 },
{ [VALUES]: "num" }
);
check(
{ name: "Dan", a: 1, b: 2, c: 3 },
{ name: "str", [VALUES]: "num" }
);
check(
{ a: 1, b: 2, c: "abc" },
{ [VALUES]: "num" }
);
check(
{ name: "Dan", a: 1, b: 2, c: 3 },
{ name: "str", [VALUES]: "bool" }
);
You can use this functionality with the undefined
type to ensure objects do not contain additional properties (object literal types by default are allowed to contain additional properties).
check(
{ name: "Carl" },
{ name: "str", [VALUES]: "undefined" }
);
check(
{ name: "Jess", another: 28 },
{ name: "str", [VALUES]: "undefined" }
);
To check that the keys of any additional properties conform to a single type, use the KEYS
symbol and create a [KEYS]
key. This allows you to ensure that keys conform to a specific string type, e.g. camelCase, kebab-case or UPPERCASE (see string types above).
import { check, VALUES } from "blork";
check({ MYVAL: 1 }, { [KEYS]: "upper" });
check({ myVal: 1 }, { [KEYS]: "camel" });
check({ MyVal: 1 }, { [KEYS]: "pascal" });
check({ my-val: 1 }, { [KEYS]: "kebab" });
check({ MYVAL: 1 }, { [KEYS]: "upper" });
check({ myVal: 1 }, { [KEYS]: "camel" });
check({ MyVal: 1 }, { [KEYS]: "pascal" });
check({ my-val: 1 }, { [KEYS]: "kebab" });
Normally object literal types check that the object is a plain object. If you wish to allow the object to be a different object (in order to check specific keys on that object at the same time), use the CLASS
symbol and create a [CLASS]
key.
import { check, CLASS } from "blork";
class MyClass {
constructor () {
this.num = 123;
}
}
check(
new MyClass,
{ num: 123, [CLASS]: MyClass }
);
check(
{ num: 123, },
{ num: 123, [CLASS]: MyClass }
);
Array literal type
To check an array where all items conform to a specific type, pass an array as the type. Arrays and objects can be deeply nested to check types recursively.
check(["abc", "abc"], ["str"]);
check([123, 123], ["num"]);
check([{ names: ["Alice", "John"] }], [{ names: ["str"] }]);
check(["abc", "abc", 123], ["str"]);
check(["abc", "abc", 123], ["number"]);
Array tuple type
Similarly, to check the format of tuples, pass an array with two or more items as the type. If two or more types are in an type array, it is considered a tuple type and will be rejected if it does not conform exactly to the tuple.
check([123, "abc"], ["num", "str"]);
check([123, "abc"], ["num", "str", "str?"]);
check([123], ["num", "str"]);
check([123, 123], ["num", "str"]);
check([123, "abc", true], ["num", "str"]);
Contributing
Please see (CONTRIBUTING.md)
Roadmap
Changelog
- 8.1.0
- Add min/max size constraints on types via e.g.
{4,8}
suffix
- 8.0.0
- Remove
props()
functionality (bloat) - Prepend function name to
ValueError
errors, e.g. MyClass.myFunc(): Must be string...
- Add
destack()
method that parses Error.stack
across major browsers
- 7.6.0
- Allow
prefix
and error
arguments for check()
and args()
- Add
assert()
function
- 7.5.0
- Enable tuple arrays via
[type1, type2]
syntax
- 7.4.0
- Make properties created with
props()
enumerable - Return the original object from
props()
(for chaining)
- 7.2.0
- Add grouping for string types via parentheses, e.g.
(str | num)
- Add
empty
type to detect emptiness in strings, arrays, Map, Set, and objects - Add
alphabetic
, numeric
and alphanumeric
string types for specific strings
- 7.1.0
- Add object and array string modifiers (using
type[]
, {type}
and { keyType: type }
syntax)
- 7.0.0
- Add
VALUES
, KEYS
, and CLASS
symbol constants - Remove
_any
key and use VALUES
to provide the same functionality - Add
KEYS
functionality to check type or case of object keys, e.g. camelCase or kebab-case - Add
CLASS
functionality to check the class of an object - Add string case checkers for e.g. variable names (kebab-case, camelCase, snake_case etc)
upper
and lower
checkers work differently (all characters must be UPPERCASE/lowercase)- Rename
int+
, int-
checkers to +int
and -int
- Add '+' modifier to check for non-empty values with any checker
- Remove hardcoded '+' checkers like
lower+
, object+
- Remove
uppercase
and lowercase
checkers for consistency
- 6.0.0
- Remove
prop()
function and add props()
function instead (prop()
was impossible to type with Flow)
- 5.1.0
- Add
prop()
function that defines a locked object property that must match a Blork type
- 5.0.0
- Change from symbol
[ANY]
key to [VALUES]
key for indexer property (for convenience and better Flow compatibility)
- 4.5.0
- Add
checker()
function to return the boolean checker function itself.
- 4.4.0
- Add
json
checker to check for JSON-friendly values (null, true, false, finite numbers, strings, plain objects, plain arrays)
- 4.3.0
- Add
circular
checker to check for objects with circular references - Add
!
modifier to enable invert checking, e.g. !num
(don't allow numbers) or !circular
(don't allow circular references)
- 4.2.2
- Use
.
dot notation in error message prefix when recursing into objects
- 4.2.1
- Fix bug where optional types were throwing an incorrect error message
- 4.2.0
- Rename
FormattedError
to ValueError
(more descriptive and reusable name) - Make
ValueError
the default error thrown by Blork (not ValueError)
- 4.1.0
- Allow custom error to be set for custom checkers via
add()
- Export
debug()
which allows any value to be converted to a string in a clean and clear format - Export
format()
which takes three arguments (message, value, prefix) and returns a consistently and beautifully formatted error message. - Export
FormattedError
which takes the same three arguments and applies format()
so it always has beautiful errors - Export
BlorkError
(which is thrown when you're using Blork wrong) for the purposes of checking thrown errors against it
- 4.0.0
- Major internal rewrite with API kept almost the same
- Add support for combining checkers with
|
and &
syntax check()
and args()
no longer return anything (previously returned the number of passing values)- Custom checkers should now return
boolean
(message/description for the checker can be passed in as third field to add()
)