Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

tiny-decoders

Package Overview
Dependencies
Maintainers
1
Versions
30
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

tiny-decoders

Type-safe data decoding for the minimalist.

  • 2.0.0
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
5.8K
decreased by-32.48%
Maintainers
1
Weekly downloads
 
Created
Source

tiny-decoders Build Status no dependencies minified size

Type-safe data decoding for the minimalist, inspired by nvie/decoders and Elm’s JSON Decoders.

Supports Flow and TypeScript.

Contents

Installation

npm install tiny-decoders

Example

import {
  array,
  boolean,
  either,
  number,
  optional,
  record,
  string,
} from "tiny-decoders";

type User = {|
  name: string,
  active: boolean,
  age: ?number,
  interests: Array<string>,
  id: string | number,
|};

const userDecoder: (mixed) => User = record({
  name: string,
  active: boolean,
  age: optional(number),
  interests: array(string),
  id: either(string, number),
});

const payload: mixed = getSomeJSON();

const user: User = userDecoder(payload);

/*
If we get here, `user` is now a valid `User`!
Otherwise, a `TypeError` is thrown.
The error can look like this:

    TypeError: object["age"]: (optional) Expected a number, but got: "30"
    at "age" in {"age": "30", "name": "John Doe", "active": true, (2 more)}
*/

Full example

Intro

The central concept in tiny-decoders is the decoder. It’s a function that turns mixed into some narrower type, or throws an error.

For example, there’s a decoder called string ((mixed) => string) that returns a string if the input is a string, and throws a TypeError otherwise. That’s all there is to a decoder!

tiny-decoders contains:

  • A bunch of decoders (such as (mixed) => string).
  • A bunch of functions that return a decoder (such as array: ((mixed) => T) => (mixed) => Array<T>).

Composing those functions together, you can describe the shape of your objects and let tiny-decoders verify that a given input matches that description.

API

Decoding primitive values

Booleans, numbers and strings, plus constant.

boolean

(value: mixed) => boolean

Returns value if it is a boolean and throws a TypeError otherwise.

number

(value: mixed) => number

Returns value if it is a number and throws a TypeError otherwise.

string

(value: mixed) => string

Returns value if it is a string and throws a TypeError otherwise.

constant

(constantValue: T) => (value: mixed) => T

T must be one of boolean | number | string | undefined | null.

Returns a decoder. That decoder returns value if value === constantValue and throws a TypeError otherwise.

Commonly used when Decoding by type name.

Decoding combined values

Arrays, objects and optional values.

For an array, you need to not just make sure that the value is an array, but also that every item inside the array has the correct type. Same thing for objects (the values need to be checked). For this kind of cases you need to combine decoders. This is done through functions that take a decoder as input and returns a new decoder. For example, array(string) returns a decoder that handles arrays of strings.

Note that there is no object decoder, because there are two ways of decoding objects:

  • If you know all the keys, use record.
  • If the keys are dynamic and all values have the same type, use dict.

Related:

array

(decoder: (mixed) => T) => (value: mixed) => Array<T>

Takes a decoder as input, and returns a new decoder. The new decoder checks that value is an array, and then runs the input decoder on every item. If all of that succeeds it returns Array<T>, otherwise it throws a TypeError.

Example:

import { array, string } from "tiny-decoders";

const arrayOfStringsDecoder: (mixed) => Array<string> = array(string);
dict

(decoder: (mixed) => T) => (value: mixed) => { [string]: T }

Takes a decoder as input, and returns a new decoder. The new decoder checks that value is an object, and then goes through all keys in the object and runs the input decoder on every value. If all of that succeeds it returns { [string]: T }, otherwise it throws a TypeError.

import { dict, string } from "tiny-decoders";

const dictOfStringsDecoder: (mixed) => { [string]: T } = dict(string);
record

(mapping: Mapping) => (value: mixed) => Result

  • Mapping:

    {
      key1: (mixed) => A,
      key2: (mixed) => B,
      ...
      keyN: (mixed) => C,
    }
    
  • Result:

    {
      key1: A,
      key2: B,
      ...
      keyN: C,
    }
    

Takes a “Mapping” as input, and returns a decoder. The new decoder checks that value is an object, and then goes through all the key-decoder pairs in the Mapping. For every key, the value of value[key] must match the key’s decoder. If all of that succeeds it returns “Result,” otherwise it throws a TypeError. The Result is identical to the Mapping, except all of the (mixed) => are gone, so to speak.

Example:

import { record, string, number, boolean } from "tiny-decoders";

type User = {|
  name: string,
  age: number,
  active: boolean,
|};

const userDecoder: (mixed) => User = record({
  name: string,
  age: number,
  active: boolean,
});

Notes:

  • record is a convenience function around group and field. Check those out if you need more flexibility, such as renaming fields!

  • The value we’re decoding is allowed to have extra keys not mentioned in the record mapping. I haven’t found a use case where it is useful to disallow extra keys. This package is about extracting data in a type-safe way, not validation.

  • Want to add some extra keys? Checkout the extra fields example.

  • There’s a way to let Flow infer types from your record decoders (or any decoder actually) if you want to take the DRY principle to the extreme – see the inference example.

  • The actual type annotation for this function is a bit weird but does its job (with good error messages!) – check out the source code if you’re interested.

optional

(decoder: (mixed) => T, defaultValue?: U) => (value: mixed) => Array<T | U>

Takes a decoder as input, and returns a new decoder. The new decoder returns defaultValue if value is undefined or null, and runs the input decoder on value otherwise. (If you don’t supply defaultValue, undefined is used as the default.)

This is especially useful to mark fields as optional in a record:

import { optional, record, string, number, boolean } from "tiny-decoders";

type User = {|
  name: string,
  age: ?number,
  active: boolean,
|};

const userDecoder: (mixed) => User = record({
  name: string,
  age: optional(number),
  active: optional(boolean, true),
});

In the above example:

  • .name must be a string.
  • .age is allowed to be undefined, null or missing (defaults to undefined).
  • .active defaults to true if it is undefined, null or missing.

If the need should ever arise, check out the example on how to distinguish between undefined, null and missing values. tiny-decoders treats undefined, null and missing values the same by default, to keep things simple.

Decoding specific fields

Parts of objects and arrays, plus group.

field

(key: string | number, decoder: (mixed) => T) => (value: mixed) => T

Takes a key (object key, or array index) and a decoder as input, and returns a new decoder. The new decoder checks that value is an object (if key is a string) or an array (if key is a number), and runs the input decoder on value[key]. If both of those checks succeed it returns T, otherwise it throws a TypeError.

This lets you pick a single value out of an object or array.

field is typically used with group.

Examples:

import { field, group, string, number } from "tiny-decoders";

type Person = {|
  firstName: string,
  lastName: string,
|};

// You can use `field` with `group` to rename keys on a record.
const personDecoder: (mixed) => Person = group({
  firstName: field("first_name", string),
  lastName: field("last_name", string),
});

type Point = {|
  x: number,
  y: number,
|};

// If you want to pick out items at certain indexes of an array, treating it
// is a tuple, use `field` and save the results in a `group`.
// This will decode `[4, 7]` into `{ x: 4, y: 7 }`.
const pointDecoder: (mixed) => Point = group({
  x: field(0, number),
  y: field(1, number),
});

Full examples:

fieldDeep

(keys: Array<string | number>, decoder: (mixed) => T) => (value: mixed) => T

Takes an array of keys (object keys, and array indexes) and a decoder as input, and returns a new decoder. It works like field, but repeatedly goes deeper and deeper using the given keys. If all of those checks succeed it returns T, otherwise it throws a TypeError.

fieldDeep is used to pick a one-off value from a deep structure, rather than having to decode each level manually with record and array.

Note that fieldDeep([], decoder) is equivalent to just decoder.

You probably want to combine fieldDeep with either since reaching deeply into structures is likely to fail.

Examples:

import { fieldDeep, number, either } from "tiny-decoders";

const accessoryPriceDecoder: (mixed) => number = fieldDeep(
  ["store", "products", 0, "accessories", 0, "price"],
  number
);

const accessoryPriceDecoderWithDefault: (mixed) => number = either(
  accessoryPriceDecoder,
  () => 0
);
group

(mapping: Mapping) => (value: mixed) => Result

  • Mapping:

    {
      key1: (mixed) => A,
      key2: (mixed) => B,
      ...
      keyN: (mixed) => C,
    }
    
  • Result:

    {
      key1: A,
      key2: B,
      ...
      keyN: C,
    }
    

Takes a “Mapping” as input, and returns a decoder. The new decoder goes through all the key-decoder pairs in the Mapping. For every key-decoder pair, value must match the decoder. (The keys don’t matter – all their decoders are run on the same value). If all of that succeeds it returns “Result,” otherwise it throws a TypeError. The Result is identical to the Mapping, except all of the (mixed) => are gone, so to speak.

As you might have noticed, group has the exact same type annotation as record. So what’s the difference? record is all about decoding objects with certain keys. group is all about running several decoders on the same value and saving the results. If all of the decoders in the Mapping succeed, an object with named values is returned. Otherwise, a TypeError is thrown.

If you’re familiar with Elm’s mapping functions, group plus map replaces all of those. For example, Elm’s map3 function lets you run three decoders. You are then given the result values in the same order, allowing you to do something with them. With group you combine any number of decoders, and it lets you refer to the result values by name instead of order (reducing the risk of mix-ups).

group is typically used with field to decode objects where you want to rename the fields.

Example:

import { group, field, string, number, boolean } from "tiny-decoders";

const userDecoder = group({
  firstName: field("first_name", string),
  lastName: field("last_name", string),
  age: field("age", number),
  active: field("active", boolean),
});

It’s also possible to rename only some fields without repetition if you’d like.

The actual type annotation for this function is a bit weird but does its job (with good error messages!) – check out the source code if you’re interested.

Chaining

Two decoders chained together in different ways, plus map.

either

(decoder1: (mixed) => T, decoder2: (mixed) => U) => (value: mixed) => T | U

Takes two decoders as input, and returns a new decoder. The new decoder tries to run the first input decoder on value. If that succeeds, it returns T, otherwise it tries the second input decoder. If that succeeds it returns U, otherwise it throws a TypeError.

Example:

import { either, string, number } from "tiny-decoders";

const stringOrNumberDecoder: (mixed) => string | number = either(
  string,
  number
);

What if you want to try three (or more) decoders? You’ll need to nest another either:

import { either, string, number, boolean } from "tiny-decoders";

const weirdDecoder: (mixed) => string | number | boolean = either(
  string,
  either(number, boolean)
);

That’s perhaps not very pretty, but not very common either. It’s possible to make either2, either3, etc functions, but I don’t think it’s worth it.

You can also use either to allow decoders to fail and to distinguish between undefined, null and missing values.

map

(decoder: (mixed) => T, fn: (T) => U): (value: mixed) => U

Takes a decoder and a function as input, and returns a new decoder. The new decoder runs the input decoder on value, and then passes the result to the provided function. That function can return a transformed result. It can also be another decoder. If all of that succeeds it returns U (the return value of fn), otherwise it throws a TypeError.

Example:

import { map, array, number } from "tiny-decoders";

const numberSetDecoder: (mixed) => Set<number> = map(
  array(number),
  (arr) => new Set(arr)
);

const nameDecoder: (mixed) => string = map(
  record({
    firstName: string,
    lastName: string,
  }),
  ({ firstName, lastName }) => `${firstName} ${lastName}`
);

Full examples:

andThen

(decoder: (mixed) => T, fn: (T) => (mixed) => U): (value: mixed) => U

Takes a decoder and a function as input, and returns a new decoder. The new decoder runs the input decoder on value, and then passes the result to the provided function. That function must return yet another decoder. That final decoder is then run on the same value as before. If all of that succeeds it returns U, otherwise it throws a TypeError.

This is used when you need to decode a value a little bit, and then decode it some more based on the first decoding result.

The most common case is to first decode a “type” field of an object, and then choose a decoder based on that. Since that is so common, there’s actually a special decoder for that – fieldAndThen – with a better error message.

So when do you need andThen? Here are some examples:

fieldAndThen

(key: string | number, decoder: (mixed) => T, fn: (T) => (mixed) => U) => (value: mixed) => U

fieldAndThen(key, decoder, fn) is equivalent to andThen(field(key, decoder), fn) but has a better error message. In other words, it takes the combined parameters of field and andThen and returns a new decoder.

See Decoding by type name for an example and comparison with andThen(field(key, decoder), fn).

Less common decoders

Recursive structures, and less precise objects and arrays.

Related:

lazy

(fn: () => (mixed) => T) => (value: mixed) => T

Takes a function that returns a decoder as input, and returns a new decoder. The new decoder runs the function to get the input decoder, and then runs the input decoder on value. If that succeeds it returns T (the return value of the input decoder), otherwise it throws a TypeError.

lazy lets you decode recursive structures. lazy(() => decoder) is equivalent to just decoder, but let’s you use decoder before it has been defined yet (which is the case for recursive structures). So all lazy is doing is allowing you to wrap a decoder in an “unnecessary” arrow function, delaying the reference to the decoder until it’s safe to access in JavaScript. In other words, you make a lazy reference – one that does not evaluate until the last minute.

Examples:

import { lazy, record, array, string } from "tiny-decoders";

// A recursive data structure:
type Person = {|
  name: string,
  friends: Array<Person>,
|};

// Attempt one:
const personDecoder: (mixed) => Person = record({
  name: string,
  friends: array(personDecoder), // ReferenceError: personDecoder is not defined
});

// Attempt two:
const personDecoder: (mixed) => Person = record({
  name: string,
  friends: lazy(() => array(personDecoder)), // No errors!
});

Full recursive example

If you use the no-use-before-define ESLint rule, you need to disable it for the lazy line:

const personDecoder: (mixed) => Person = record({
  name: string,
  // eslint-disable-next-line no-use-before-define
  friends: lazy(() => array(personDecoder)),
});
mixedArray

(value: mixed) => $ReadOnlyArray<mixed>

Usually you want to use array instead. array actually uses this decoder behind the scenes, to verify that value is an array (before proceeding to decode every item of the array). mixedArray only checks that value is an array, but does not care about what’s inside the array – all those values stay as mixed.

This can be useful for custom decoders, such as when distinguishing between undefined, null and missing values.

mixedDict

(value: mixed) => { +[string]: mixed }

Usually you want to use dict or record instead. dict and record actually use this decoder behind the scenes, to verify that value is an object (before proceeding to decode values of the object). mixedDict only checks that value is an object, but does not care about what’s inside the object – all the keys remain unknown and their values stay as mixed.

This can be useful for custom decoders, such as when distinguishing between undefined, null and missing values.

repr

(value: mixed, options?: Options) => string

Takes any value, and returns a string representation of it for use in error messages. Useful when making custom decoders.

Options:

nametypedefaultdescription
keystring | number | voidundefinedAn object key or array index to highlight when repring objects or arrays.
recursebooleantrueWhether to recursively call repr on array items and object values. It only recurses once.
maxArrayChildrennumber5The number of array items to print (when recurse is true.)
maxObjectChildrennumber3The number of object key-values to print (when recurse is true.)

Example:

import { repr } from "tiny-decoders";

type Alignment = "top" | "right" | "bottom" | "left";

function alignmentDecoder(value: string): Alignment {
  switch (value) {
    case "top":
    case "right":
    case "bottom":
    case "left":
      return value;
    default:
      throw new TypeError(`Expected an Alignment, but got: ${repr(value)}`);
  }
}

This function returns a string, but what that string looks like is not part of the public API.

Comparison with nvie/decoders

nvie/decoderstiny-decoders
Sizeminified size
minzipped size
minified size
minzipped size
Dependencieshas dependenciesno dependencies
Error messagesReally fancyKinda good (size tradeoff)
Built-in functionsType checking + validation (regex, email)Type checking only (validation can be plugged in)
Decoders……return Results or throw errors…only throw errors

In other words:

  • nvie/decoders: Larger API, fancier error messages, larger size.
  • tiny-decoders: Smaller (slightly different) API, kinda good error messages, smaller size.

Error messages

The error messages of nvie/decoders are really nice, but also quite verbose:

Decoding error:
[
  {
    "id": "512971",
    "name": "Ergonomic Mouse",
    "image": "",
    "price": 499,
    "accessories": [],
  },
  {
    "id": "382973",
    "name": "Ergonomic Keyboard",
    "image": "",
    "price": 998,
    "accessories": [
      {
        "name": "Keycap Puller",
        "image": "",
        "discount": "5%",
                    ^^^^
                    Either:
                    - Must be null
                    - Must be number
      },
      ^ Missing key: "id" (at index 0)
      {
        "id": 892873,
        "name": "Keycap Set",
        "image": "",
        "discount": null,
      },
    ],
  },
  ^ index 1
  {
    "id": "493673",
    "name": "Foot Pedals",
    "image": "",
    "price": 299,
    "accessories": [],
  },
]

The errors of tiny-decoders are shorter and a little bit more cryptic. As opposed to nvie/decoders, it stops at the first error in a record (instead of showing them all). First, the missing “id” field:

TypeError: array[1]["accessories"][0]["id"]: Expected a string, but got: undefined
at "id" (missing) in {"name": "Keycap Puller", "image": "data:imag…AkQBADs=", "discount": "5%"}
at 0 in [(index 0) Object(3), Object(4)]
at "accessories" in {"accessories": Array(2), "id": "382973", "name": "Ergonomic Keyboard", (2 more)}
at 1 in [Object(5), (index 1) Object(5), Object(5)]

And if we add an “id” we get the “discount” error:

TypeError: array[1]["accessories"][0]["discount"]: (optional) Expected a number, but got: "5%"
at "discount" in {"discount": "5%", "id": "489382", "name": "Keycap Puller", (1 more)}
at 0 in [(index 0) Object(4), Object(4)]
at "accessories" in {"accessories": Array(2), "id": "382973", "name": "Ergonomic Keyboard", (2 more)}
at 1 in [Object(5), (index 1) Object(5), Object(5)]

If you read the “stack trace” of tiny-decoders from bottom to top, it’s a bit like expanding objects and arrays in the browser devtools (but in your head):

[Object(5), (index 1) Object(5), Object(5)]
                      |
                      v
                      {"accessories": Array(2), "id": "382973", "name": "Ergonomic Keyboard", (2 more)}
                                      |
                                      v
                                      [(index 0) Object(4), Object(4)]
                                                 |
                                                 v
                                                 {"discount": "5%", "id": "489382", "name": "Keycap Puller", (1 more)}
                                                              ^^^^

Development

You need Node.js 10 and npm 6.

npm scripts

  • npm run flow: Run Flow.
  • npm run eslint: Run ESLint (including Flow and Prettier).
  • npm run eslint:fix: Autofix ESLint errors.
  • npm run dtslint: Run dtslint.
  • npm run prettier: Run Prettier for files other than JS.
  • npm run doctoc: Run doctoc on README.md.
  • npm run jest: Run unit tests. During development, npm run jest -- --watch is nice.
  • npm run coverage: Run unit tests with code coverage.
  • npm build: Compile with Babel.
  • npm test: Check that everything works.
  • npm publish: Publish to npm, but only if npm test passes.

Directories

  • src/: Source code.
  • examples/: Examples, in the form of Jest tests.
  • test/: Jest tests.
  • flow/: Flow typechecking tests. Turn off “ExpectError” in .flowconfig to see what the errors look like.
  • typescript/: TypeScript type definitions, config and typechecking tests.
  • dist/: Compiled code, built by npm run build. This is what is published in the npm package.

License

MIT

Keywords

FAQs

Package last updated on 07 Jun 2019

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc