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

efate

v1.5.1

Published

javascript test fixture generator. Autofixturejs 2.0

Downloads

22,282

Readme

Efate

A Test Fixture Builder Image of Efate

Why use a fixture builder

If you have a moderate to large web app and are using JSON files to store your test fixtures, a test fixture library can solve several issues you may be experiencing.

Test Fixture Change Transference

When you load the json file into a test file, you have a single instance of that test data for all of the tests in that file. If you need to change a field in your test data, all subsequent tests will run with that changed value. This can lead to confusing test failures.

Inconsistent Test Data

Sometimes, when you need to vary the data for a test, it's tempting to create a new object with the information needed just for that single test. This leads to a proliferation of unstructured test data that can be difficult and time consuming to update when data structures change.

Features

efate provides a typesafe way to define and create your fixtures. You can create the fixtures for each test, each with unique but understandable values. You can also set specific values of fields of interest for the current test.

Creating a fixture

  1. Use createFixtureFactory to get the function to build fixtures. This function is used for fixture extension.
const createFixture = createFixtureFactory() // returns a function that will create fixtures.
  1. Use the the function to create the fixture, using the fixture definition to specify the field fields and how the values should be generated.
const userFixture = createFixture<User>((t) => {
  t.id.asNumber(); // id field will be numberical value that increments as you create user objects
  t.firstName.asString();
  t.lastName.asString();
});
  1. Create the mock object using the fixture, overriding any values that you need to for the test.
const user = userFixture.create({ title: 'myUserName' });

When you create a this fixture it will create an object that has the firstName and lastName fields with string values.

Using the fixture

// generatate a fixture with system generated values
const user = UserFixture.create();
// {firstName: 'firstName1', lastName: 'lastName1'

As you create more fixtures, the values will vary by incrementing the number for each field

// generatate a fixture with system generated values
const user = userFixture.create();
// {firstName: 'firstName1', lastName: 'lastName1'
const user = userFixture.create();
// {firstName: 'firstName2', lastName: 'lastName2'

You can override specific fields of the test fixture to test specific cases in your tests and still have complete objects

const user = userFixture.create({ firstName: 'George', lastName: 'Washington' });
// {firstName: 'George', lastName: 'Washington'}

If you need to make more complicated changes to the fixture, you can pass a function

const user = UserFixture.create((user) => {
  // make changes to the user fixture then return back the user
  user.firstName = 'George';
});
// {firstName: 'George', lastName: 'lastName1'

Creating Arrays of Fixtures

You can generate an array of fixtures by using the UserFixture.createArrayWith. The function accepts an override parameter that will either:

  • single Partial of the object that will override all objects in the array
  • a subset array that will override the matching index in the return array
  • a function that allows you to determine how the array will be overridden
  • no value and all arrays are created with default values

Override All elements

const users = UserFixture.createArrayWith(2, {firstName: 'George'});
[
    {firstName:'George', lastName: 'lastName1', ...},
    {firstName:'George', lastName: 'lastName1', ...}
]

** Override just the first element **

const users = UserFixture.createArrayWith(2, [{firstName: 'George'}]);
[
    {firstName:'George', lastName: 'lastName1', ...},
    {firstName:'George', lastName: 'lastName1', ...}
]

** Override with a function ** const users = UserFixture.createArrayWith(2, (idx, create) => { if (idx === 0){ return create({firstName: 'George'}); } else { return create(); } }); [ {firstName:'George', lastName: 'George', ...}, {firstName:'George', lastName: 'lastName1', ...} ]



```typescript
### Specifying field types and special values
The function passed to the `createFixture` allows you to determine how each field will be generated when you create a new fixture

The type specified will be user as the return type when you `create` the fixture.

These two features together allow for the type-safe creation of fixtures, plus great auto-complete in your editor.
```typescript
const userFixture = createFixture<User>(t => {
  t.firstName.asString();
  t.lastName.asString();
  t.dateOfBirth.asDate();
})

Provided Type Generators

All of the type generators behavior is described in the generated Spec file.

  • withValues() uses specified text to generate values

    const userFixture = createFixture<{ foo: string }>((t) => {
      t.foo.withValue('bar');
    });
    const user = userFixture.create();
    // {foo: 'bar1'}
  • asConstant() does not increment the value of the field for all fixtures

  • as((increment)=> val) custom function to generate values. The increment parameter is the count of usages of the fixture

    new Fixture('email'.as(increment => `email${increment}@company.com`);
    // {email: '[email protected]'}```
    
  • asNumber() generates auto incrementing number values for the field

  • asDate({incrementDay: boolean}) generates a date value for the field. If incrementDay is true the day will increment for each fixture created. Otherwise the same date is used for all fixtures.

  • asBoolean() generates a boolean value for the field

  • asArray(length = 1) generates an array for the field. The length parameter specifies length of the array, defaulted to 1

    const fixture = createFixture<{role:string[]}>(t => {
      t.role.asArray()
    })
    // {roles: ['roles1']}
  • pickFrom([possible values]) randomly picks one of the possible values provided to set the field value.

    const fixture = createFixture<{role:string[]}>(t => {
      t.role.pickFrom(['user', 'admin'])
    })
    
    //will generate {role: 'user'|'admin'}
    • fromFixture(Fixture) When you need to nest an object created from another fixture
      const userFixture = createFixture<User>(t => {
      ...
      });
      const OrderFixture = createFixture<Order>(t => {
        t.orderId.asNumber();
        t.customer.fromFixture(UserFixture)
      );
      // {orderId: 1, customer: {username: 'username1', email: 'email1'}}```
  • arrayOfFixture(Fixture, length = 3) generates and array of object created from the fixture passed in.

  • asEmail() creates an email address [email protected]

  • asFirstName() uses value from list of possible names to create more realistic data

  • asLastName() uses value from list of possible names to create more realistic data

  • asFullName() combines first name and last name values

  • asLoremIpsum({minLength: 10, maxLength: 25}) generates ispsum lorem text for generating longer text. Useful when using this tool for seed data.

Extending Efate with your own type builder

You can extend the efate with your own custom type builder. An example is the efate-uuid builder that creates valid UUID values

There are 2 parts to extending efate:

  1. Define an interface that describes value definition
  2. Create a function that will be used during fixture creation

Creating the interface

When you call createFixtureFactory, you can pass an interface as generic parameter to the function that will attach your extension to the object fields when defining a fixture. This interface takes the form of:

interface CustomDomainEmail {
  asCustomDomainEmail(domainName: string): void;
}

Create the builder

You pass an object to createFixtureFactory that has a fieldName that matches the name of the interface that returns a curried function with the following signature, fieldName: string, options: ?

{
  asCustomDomainEmail: (fieldName: string, domainName: string) => (increment: number) => Field;
}

Example builder Function

const customDomainEmailExtension = {
  asCustomDomainEmail: (fieldName: string, domainName: string = 'example.com') =>
          (increment: number) =>  return new Field(fieldName, `email${increment}@${domainName}`);
}

You then call the factory method with this interface and object

const createFixture = createFixtureFactory<CustomDomainEmail>(customDomainEmailExtension);

If you have multiple extensions, union the interfaces and pass botth object to the factory.

const createFixture = createFixtureFactory<CustomDomainEmail & AnotherExtention>(
  customDomainEmailExtension,
  otherExtension,
);

These extensions will now be available when you define your fixtures.