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

jest-storybook-facade

v0.0.8

Published

Facades for Jest and Storybook so that your Jest tests become stories

Downloads

28

Readme

Jest Storybook Facade

Jest Storybook Facade allows you to return "stories" (aka React elements) from your it tests to produce your React Storybook. Each describe block is one "story", and its it blocks the sub-stories. It's fantastic! Yay! Try it.

Installation

yarn add jest-storybook-facade

storybook config.js:

import 'jest-storybook-facade/storybook-facade.js'
import { configure } from '@kadira/storybook'
...

package.json:

"jest": {
    "setupTestFrameworkScriptFile": "jest-storybook-facade/jest-facade.js",
    ...

Usage:

import React from 'react'
import renderer from 'react-test-renderer'

import MyComponent from '../src/MyComponent'
import add from '../src/add'

describe('my test', () => {
  it('does something', () => {
    const element = <MyComponent />
    const tree = renderer.create(element).toJSON()

    expect(tree).toMatchSnapshot()
    return element // will become a story in Storybook, yay!
  })

  it('does something', () => {
    const value = myFunction(1, 2)
    expect(value).toEqual(3)
    // if you don't return anything, no story is displayed
  })
})

Yay, it looks just like my regular tests! I can trust this won't break or become a nuisance to my all-import test harness!!!

...and in fact, you can get started today with your existing tests, just by returning react elements!

How It Works

  • In Storybook land, globals for describe, it, expect, etc are created by the storybook-facade.js file you import. The Jest environment is essentially mirrored. Under the hood, it creates your stories and spec tests using the specifications addon.

  • In Jest land, a few minor tweaks are made by jest-facade.js: For one, you can now return react elements without penality (typically undefined or a promise is expected as the return). In addition to describe, it, jest, etc, you have one more global: storybook. And of course it exists on both sides. Here's what it has on it in Jest land:

global.storybook = {
  action: name => () => console.log('action', name),
  linkTo: (component, story) => () => console.log('linkTo', component, story),
  withReadme: readme => null,
  knobs: {
    text: (label, value) => value,
    boolean: (label, value) => value,
    number: (label, value, options) => value,
    color: (label, value) => value,
    object: (label, value) => value,
    array: (label, value, separator = ',') => value,
    select: (label, options, value) => value,
    date: (label, value) => value,
  },
  stories: {
    addDecorator: () => null,
    add: () => null,
    addWithInfo: () => null,
  },

for this particular package, we urge you to analyze all the code. It's simple and small. You should know what's available to you, and how to add to it if we missed anything.

In this case, you can see that several Storybook addons such as knobs and withReadme is available to you, as well as the ability to addDecorator. In Storybook land, the real ones will of course be used. In general, the goal is to not obstruct the standard appearance of your tests, but our assumption is that after you get used to it, you will want some of the Storybook features back.

Here's how an example of how to use it:

import React from 'react'
import renderer from 'react-test-renderer'

import readme from '../README.md'
const { color } = storybook.knobs

describe('my test', () => {
  storybook.stories.addDecorator(withReadme(readme))

  it('does something', () => {
    const color = color('my color', 'blue')
    const element = <MyComponent color={color} />
    const tree = renderer.create(element).toJSON()

    expect(tree).toMatchSnapshot()
    return element
  })
})

Storybook Specifications

The Spec tests displayed in Storybook have one interesting difference from your tests when run by the Jest test runner:

  • you will see your potentially multiple expectations displayed as tests in Storybook.
  • This is because otherwise there would only be one test, since the system is setup to return one story component per test. You can essentially use the Jest command line tool (or perhaps Wallaby) as your primary test reporter, and then when you use Storybook drill-down even farther. This is especially useful since once a test fails you typically only see information about the first expectation that failed, whereas in Storybook you will see info about all of them!

Help us Improve storybook-facade.js

The mocks in storybook-facade.js could be improved. We are missing a few expect methods from Jest, and the jest.fn() mock tool could be better. Basically they should be copied from the Jest source. When I started this, I copied the initial way the specifications addon did this, but since have decided to make this jest-only. So that means we should use the precise tools from Jest.

Until we replace these things with code from the Jest repository, basically here's what you need to do:

const expectMethods = [
  'toBeAn',
  'toBeFalsy',
  'toBeFewerThan',
  'toBeMoreThan',
  'toBeTruthy',
  'toContain',
  'toContainKey',
  'addAnotherMethodHere',
  ...

expectReal.extend({
  toMatchSnapshot() {
    expectReal.assert(
      true,
      'expected a snapshot',
      this.actual,
    )
    return this
  },
  addAnotherMethodHere...
})

and this should be improved:

const jest = {
  fn: (implementation) => {
    implementation = implementation || function jestFn() {}

    implementation.mock = {
      calls: [[], [], [], [], []],
      instances: [{}, {}, {}, {}],
      mockClear() {},
      mockReset() {},
      mockImplementation: () => implementation,
      mockImplementationOnce: () => implementation,
      mockReturnThis: () => implementation,
      mockReturnValue: () => implementation,
      mockReturnValueOnce: () => implementation,
    }

    return implementation
  }

That all said, when your tests run in Storybook, for most use-cases you should be fine. If you need a specific enhancement, git clone it, make your enhancement, add your fork to package.json instead, and make a pull request. It should be a fairly simple process to add what you need for a package this basic.

In addition, it should be fairly simple for all of us together get this up to speed, so that we are not worrying about what methods are missing in Storybook land. Making this jest-only provides us that advantage. Our thinking is that Jest is (or will become) the one true way for React testing. So lets reduce complexity by focusing on it.