Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
@jackfranklin/test-data-bot
Advanced tools
IMPORTANT: @jackfranklin/test-data-bot
is the new version of this package, written in TypeScript and initially released as version 1.0.0.
The old package, test-data-bot
(not scoped to my username on npm) was last released as 0.8.0 and is not being updated any more. It is recommended to upgrade to 1.0.0, which has some breaking changes documented below.
If you want to find the old documentation for 0.8.0
, you can do so via an old README on GitHub.
test-data-bot was inspired by Factory Bot, a Ruby library that makes it really easy to generate fake yet realistic looking data for your unit tests.
Rather than creating random objects each time you want to test something in your system you can instead use a factory that can create fake data. This keeps your tests consistent and means that they always use data that replicates the real thing. If your tests work off objects close to the real thing they are more useful and there's a higher chance of them finding bugs.
Rather than the term factory
, test-data-bot uses builder
.
test-data-bot makes no assumptions about frameworks or libraries, and can be used with any test runner too. test-data-bot is written in TypeScript, so if you use that you'll get nice type safety (see the TypeScript section of this README) but you can use it in JavaScript with no problems.
npm install --save-dev @jackfranklin/test-data-bot
yarn add --dev @jackfranklin/test-data-bot
We use the build
function to create a builder. You give a builder an object of fields you want to define:
const { build } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: 'jack',
},
});
const user = userBuilder();
console.log(user);
// => { name: 'jack'}
While the examples in this README use require
, you can also use import {build} from '@jackfranklin/test-data-bot'
.
Note: if you're using Version 1.2 or higher you can leave the name of the factory and pass in only the configuration object:
const userBuilder = build({
fields: {
name: 'jack',
},
});
Feel free to use the name property if you like, but it's not used for anything in test-data-bot. It will probably get removed in a future major version.
Once you've created a builder, you can call it to generate an instance of that object - in this case, a user
.
It would be boring though if each user had the same name
- so test-data-bot lets you generate data via some API methods:
fake
test-data-bot provides an API for calling into Faker.js which is a library that lets you generate all sorts of fake data:
const { build, fake } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: fake(f => f.name.findName()),
},
});
const userOne = userBuilder();
const userTwo = userBuilder();
Each user will have a random name as generated by the FakerJS library. You can use any of the Faker API methods that you'd like.
sequence
Often you will be creating objects that have an ID that comes from a database, so you need to guarantee that it's unique. You can use sequence
, which increments every time it's called:
const { build, sequence } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
id: sequence(),
},
});
const userOne = userBuilder();
const userTwo = userBuilder();
// userOne.id === 1
// userTwo.id === 2
If you need more control, you can pass sequence
a function that will be called with the number. This is useful to ensure completely unique emails, for example:
const { build, sequence } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
email: sequence(x => `jack${x}@gmail.com`),
},
});
const userOne = userBuilder();
const userTwo = userBuilder();
// userOne.email === jack1@gmail.com
// userTwo.email === jack2@gmail.com
If you want an object to have a random value, picked from a list you control, you can use oneOf
:
const { build, oneOf } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: oneOf('alice', 'bob', 'charlie'),
},
});
bool
If you need something to be either true
or false
, you can use bool
:
const { build, bool } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
isAdmin: bool(),
},
});
perBuild
test-data-bot lets you declare a field to always be a particular value:
const { build, perBuild } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: 'jack',
details: {},
},
});
A user generated from this builder will always the same data. However, if you generate two users using the builder above, they will have exactly the same object for the details
key:
const userOne = userBuilder();
const userTwo = userBuilder();
userOne.details === userTwo.details; // true
If you want to generate a unique object every time, you can use perBuild
which takes a function and executes it when a builder is built:
const { build, perBuild } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: 'jack',
details: perBuild(() => {
return {};
}),
},
});
const userOne = userBuilder();
const userTwo = userBuilder();
userOne.details === userTwo.details; // false
postBuild
If you need to transform an object in a way that test-data-bot doesn't support out the box, you can pass a postBuild
function when creating a builder. This builder will run everytime you create an object from it.
const { build, fake } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
name: fake(f => f.name.findName()),
},
postBuild: user => {
user.name = user.name.toUpperCase();
return user;
},
});
const user = userBuilder();
// user.name will be uppercase
You'll often need to generate a random object but control one of the values directly for the purpose of testing. When you call a builder you can pass in overrides which will override the builder defaults:
const { build, fake } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
id: sequence(),
name: fake(f => f.name.findName()),
},
});
const user = userBuilder({
overrides: {
id: 1,
name: 'jack',
},
});
// user.id === 1
// user.name === 'jack'
If you need to edit the object directly, you can pass in a map
function when you call the builder:
const { build, fake } = require('@jackfranklin/test-data-bot');
const userBuilder = build('User', {
fields: {
id: sequence(),
name: fake(f => f.name.findName()),
},
});
const user = userBuilder({
map: user => {
user.name = user.name.toUpperCase();
return user;
},
});
Using overrides
and map
lets you easily customise a specific object that a builder has created.
test-data-bot is written in TypeScript and ships with the types generated so if you're using TypeScript you will get some nice type support out the box.
The builders are generic, so you can describe to test-data-bot exactly what object you're creating:
interface User {
id: number;
name: string;
}
const userBuilder = build<User>('User', {
fields: {
id: sequence(),
name: fake(f => f.name.findName()),
},
});
const users = userBuilder();
You should get TypeScript errors if the builder doesn't satisfy the interface you've given it.
I'm still quite new to TypeScript so please let me know if you don't get the errors/type hints that you expect!
test-data-bot@0.8.0
to @jackfranklin/test-data-bot@1.0.0
Firstly: there is no need to migrate immediately if the old version is doing the job for you - but it won't be improved or have bug fixes so it's recommended to migrate slowly to 1.0.0.
You can also run both versions at the same time - they won't conflict. Just make sure you rename the API methods or import everything:
const legacyTestDataBot = require('test-data-bot')
const newTestDataBot = require('@jackfranklin/test-data-bot')
const oldBuilder = legacyTestDataBot.build(...)
const newBuilder = newTestDataBot.build(...)
Previously Node 8 was supported, but now the minimum is 10.13.
Before:
const userBuilder = build('User').fields({
name: fake(f => f.name.findName()),
});
After:
const userBuilder = build('User', {
fields: {
name: fake(f => f.name.findName()),
},
});
numberBetween
has been removednumberBetween
was a shortcut around fake
. If you need it back you can easily define it:
const { fake } = require('@jackfranklin/test-data-bot');
const numberBetween = (min, max) => fake(f => f.random.number({ min, max }));
It's highly recommended to maintain a library of custom matchers that are useful for your application.
incrementingId
has been removedYou can now call sequence()
with no argument to get the same result:
Before:
id: incrementingId();
After:
id: sequence();
arrayOf
has been removedIt was hard to provide a nice API for arrayOf
that supported all cases. It's now recommended to use the postBuild
function if you always want an object to have an array of things:
const blogPostBuilder = build('BlogPost', {...})
const userBuilder = build('User', {
fields: {
name: fake(f => f.name.findName()),
blogPosts: [],
},
postBuild: user => {
user.blogPosts = Array(3).fill(undefined).map(_ => blogPostBuilder())
return user
}
});
map
function has been removedThe old version of test-data-bot provided a map function on each object that it generated. This was confusing and awkward as it meant test-data-bot placed a map
function on generated objects. If your object had a map
property, it would be overriden!
This is now replaced by the postBuild
function.
Before:
const userBuilder = build('User')
.fields({
name: fake(f => f.name.findName()),
email: sequence(x => `jack${x}@test.com`),
})
.map(user => ({
name: user.name.toUpperCase(),
email: user.email.toUpperCase(),
}));
After:
const userBuilder = build('User', {
fields: {
name: fake(f => f.name.findName()),
email: sequence(x => `jack${x}@test.com`),
},
postBuild: user => ({
name: user.name.toUpperCase(),
email: user.email.toUpperCase(),
}),
});
1.2.0 - 09 May 2020
Factories now do not need a factory name property. You can simply pass in the configuration object :
const userBuilder = build({ ... });
// rather than:
const userBuilder = build('User', { ... });
You can still pass a name if you like, but it's not required and will probably be removed in a future major version.
FAQs
Generate test data for your tests easily.
We found that @jackfranklin/test-data-bot demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
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.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.