npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

ducktype

v1.2.2

Published

Flexible data validation using a duck type interface

Downloads

1,995

Readme

ducktype

Flexible data validation using a duck type interface for JavaScript and Node.js.

As JavaScript is a loosely typed language, any variable can contain any type of data, and any type of data can be passed as arguments to any function. When dealing with data inputs coming from external sources, there is a need to validate the type and contents of the data. Ducktype offers an easy way to validate both basic data types as well as complex structured data types in a flexible way.

Replace this kind of type checking mess:

function save (contact) {
  if (contact && isInteger(contact.id) && (contact.id > 0) && isString(contact.name) &&
      contact.address && isString(contact.address.city) && isString(contact.address.street)) {
    // ... save contact
  }
  else {
    throw new Error('Invalid contact');
  }
}

with this:

var contactType = ducktype({
  id: ducktype(Number, {integer: true, min: 0}),
  name: String,
  address: {
    city: String,
    street: String,
  }
});

function save (contact) {
  contactType.validate(contact);
  // ... save contact
}

Install

npm

npm install ducktype

bower

bower install ducktype

Load

Node.js

var ducktype = require('ducktype');

browser

<!DOCTYPE HTML>
<html>
<head>
    <script src="ducktype.js" type="text/javascript"></script>
</head>
<body>
    <script type="text/javascript">
        // use ducktype...
    </script>
</body>
</html>

Use

Basic types

// use built-in types
ducktype.number.test(2.3);      // true
ducktype.number.test('hi');     // false
ducktype.number.test(true);     // false
ducktype.date.test(new Date()); // true
ducktype.date.test(2.3);        // false
ducktype.string.test('hello');  // true

// create a ducktype
var type = ducktype(Number);
type.test(2.3);                 // true
type.test('hi');                // false
type.test(true);                // false

// create a ducktype with options
var nullableString = ducktype(String, {nullable: true});
nullableString.test('string');  // true
nullableString.test(null);      // true
nullableString.test(2.3);       // false

Combined types

// combination of types
var combi = ducktype(Number, String);
combi.test(2.3);   // true
combi.test('hi');  // true
combi.test(true);  // false

Structured objects

// structured object
var person = ducktype({
  name: String,
  age: Number,
  address: {
    city: String,
    street: String,
    country: String
  },
  email: ducktype(String, {optional: true})
});

person.test({
  name: 'John',
  age: 32,
  address: {
    city: 'Sunnyvale, CA 95125',
    street: '701 First Ave.',
    country: 'United States'
  }
}); // true

person.test({
  name: 'Mary',
  age: 26
}); // false

Structured Arrays

// structured arrays
var numberArray = ducktype([Number]);
numberArray.test([1, 2, 3]);        // true
numberArray.test([1, 'string', 3]); // false

// structured object and array
var family = ducktype({
  name: String,
  age: ducktype(Number, {optional: true}),
  children: [
    {
      name: String,
      age: ducktype(Number, {optional: true})
    }
  ]
});

family.test({
  name: 'John',
  children: [
    {
      'name': 'Mary',
      'age': 6
    },
    {
      'name': 'Grant'
    }
  ]
}); // true

family.test({
  name: 'John',
  children: [
    {
      'firstName': 'Mary',
      'age': 6
    },
    {
      'firstName': 'Grant'
    }
  ]
}); // false

Function arguments

var type = ducktype([Number, Number]);

function add (a, b) {
  type.validate(arguments);
  return a + b;
}

var sum = add(2, 3);        // ok
var sum = add(2, 'string'); // will throw a TypeError

Alternatively, a ducktype wrapper can be created which validates the function arguments against the ducktype:

var add = ducktype([Number, Number]).wrap(function add (a, b) {
  return a + b;
});

var sum = add(2, 3);        // ok
var sum = add(2, 'string'); // will throw a TypeError

API

construction

A ducktype can be constructed as:

ducktype(type)
ducktype(type, options)
ducktype(type1, type2, ...)
ducktype(type1, type2, ..., options)

Where:

  • type can be:

    • A basic type. Choose from Array, Boolean, Date, Function, Number, Object, RegExp, String, null, undefined.
    • Another ducktype.
    • An object. All properties of the object will be checked. Each property can be a basic type, ducktype, object, or array.
    • An array. An array can have zero, one or multiple elements which can be a basic type, ducktype, object, or array. Providing an array with zero elements will just return a ducktype(Array). Providing an array with one element will return a ducktype which will test each of tested arrays elements against the given type, for example ducktype([Number]).test(1, 2, 3). Providing an array with multiple elements will validate the length of the tested array, and validate each of the array elements one to one against the provided types. This can be used to test the number and type of function arguments. Example: ducktype([Number, String]).test(2, 'str').
  • options is an object which can contain properties:

    • A string name
    • A boolean optional
    • A boolean nullable
    • A boolean integer. Test whether a number has an integer value. Only applicable for Numbers.
    • A number min. Test whether a number is larger or equal to a minimum value. Only applicable for Numbers.
    • A number max. Test whether a number is smaller or equal to a maximum value. Only applicable for Numbers.

A created ducktype has functions:

  • test(object). A function which returns true when provided object matches the ducktype, and false otherwise.
  • validate(object). A function which will throw a TypeError when the provided object does not match the ducktype.
  • wrap(fn). Creates a wrapper function around the provided function, which validates the function arguments against the ducktype. Only applicable for ducktypes containing an array, as the ducktype is tested against an array with the function arguments.

Built-in types

Ducktype comes with a set of built-in types:

  • ducktype.array
  • ducktype.boolean
  • ducktype.date
  • ducktype.email
  • ducktype.integer
  • ducktype.function
  • ducktype.number
  • ducktype.object
  • ducktype.regexp
  • ducktype.string
  • ducktype.null
  • ducktype.undefined
  • ducktype.url

The built-in types can be used as:

ducktype.number.test(2.3); // true
ducktype.string.test(2.3); // false

Test

To execute tests for the library, run:

npm test

Roadmap

  • Implement more options for specific types:
    • Number: finite, odd, even, positive, negative, nan, ...
    • String: lowercase, uppercase, alpha, alphanumeric, empty, ...
    • Array: length, length.min, length.max, ...
  • Implement a parser accepting a string describing a type in annotations.
  • Implement support to define your own tests for custom types.
  • Implement non-strict type checking: when an object can be converted to the desired type, it is ok. For example a string containing a numeric value can be considered a valid Number, or a string containing an ISO date can be considered a valid Date.

License

Copyright (C) 2013-2018 Jos de Jong [email protected]

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.