React Cosmos ✭ Dev tool for creating reusable React components
Cosmos scans your project for components and enables you to:
- Render components under any combination of props, context and state
- Mock every external dependency (eg. API responses, localStorage, etc)
- See app state evolve in real-time while interacting with running instances
Working with Cosmos improves component design because it surfaces dependencies. Cosmos forces us to define sane component inputs, making our UIs predictable and easier to debug down the road.
Read the story of React Cosmos: Fighting for Component Independence
Many other component explorers emerged in the past years. Storybook and React Styleguidist are good examples, but you can find an extensive list of options here. To decide which tool is best for you check for each project's goals, how much they match your needs, and how well the project is maintained.
Cosmos is a dev tool first, made to improve all components, big and small, not just the stateless UI bits. The fixture and proxy architecture doubles as an automated testing utility, providing a complete solution for developing robust and reusable components. Cosmos also makes it easy to create a living style guide, but it's a secondary goal and you might get more value from alternatives if this is your chief concern.
To find out more about the Cosmos project, check out Mission, Goals and Architecture.
Requirements:
- React >=0.14.9
- webpack or Browserify (or roll your own integration)
- Fixtures (you'll create them after getting started)
React Cosmos works best with webpack. Making it work with other bundlers takes extra work, but a complete Browserify example is available.
Jump to:
- Getting started
- Fixtures
- Proxies
- Integration with popular tools
- Configuration
- Exporting
- Experimental: Test helpers
Have a question or idea to share? See you on Slack.
npm install --dev react-cosmos-webpack
# or
yarn add --dev react-cosmos-webpack
Create cosmos.config.js
in your project root
module.exports = {
componentPaths: ['src/components'],
// Optionally, reuse loaders and plugins from your existing webpack config
webpackConfigPath: './config/webpack.config.dev'
};
Add script to package.json
"scripts": {
"cosmos": "NODE_ENV=development cosmos"
}
Run npm run cosmos
or yarn cosmos
and go to localhost:8989 🎉
If you rely on the default webpack config, make sure to install the Babel and webpack plugins yourself. Depending on your needs, you'll probably want
babel-preset-env babel-preset-react babel-loader style-loader css-loader html-webpack-plugin
. Finally, add.babelrc
to your project root.{ "presets": ["env", "react"] }
- See popular integrations (e.g. CRA or Next.js)
- Extend your config
- Create fixtures
- Configure or create proxies (e.g. Redux)
- Be kind and report what went wrong
A fixture is a JS object used to mock component input and external dependencies. The input can be props, children, state and context. With the help of proxies, fixtures can mock anything else a component depends on, from API responses to localStorage.
export default {
props: {
value: 'Lorem ipsum',
disabled: true,
onChange: value => console.log(`Select: ${value}`)
}
}
Check out this quick hack for getting started with fixtures.
Cosmos looks for __fixtures__
dirs next to your components. Here are a few of the supported patterns.
components/Button.jsx
components/nested/Dropdown.jsx
components/__fixtures__/Button/default.js
components/__fixtures__/Button/disabled.js
components/__fixtures__/nested/Dropdown/default.js
components/__fixtures__/nested/Dropdown/open.js
components/Button/index.jsx
components/Button/__fixtures__/default.js
components/Button/__fixtures__/disabled.js
components/nested/Dropdown/index.jsx
components/nested/Dropdown/__fixtures__/default.js
components/nested/Dropdown/__fixtures__/open.js
Named component files also work in the nested hierarchy (i.e. Button/Button.jsx
and nested/Dropdown/Dropdown.jsx
).
Mocking props is the most basic thing a fixture can do.
export default {
props: {
loggedIn: true,
user: {
name: 'Dan the Man'
}
}
}
Composition is the name of the game and many React components expect children. Components access them via props.children
, but children are not quite props so we put them under fixture.children
.
export default {
children: (
<div>
<p>Fixture ain't afraid of JSX</p>
<p>Fixture ain't afraid of nothin!</p>
</div>
)
}
Mocking state is where things get interesting. Component state is private IRL, but Cosmos allows us to inject it and simulate all the various states a component can find itself in.
export default {
state: {
searchQuery: 'Who let the dogs out?'
}
}
Proxies are Cosmos plugins, allowing fixtures to go beyond mocking props and state.
We've seen component = f(props, state)
a hundred times–the seductive promise of React and libs alike. In reality, however, it's more like component = f(props, state, context)
and most components are nothing without the context part. This is still an oversimplification. The ugly truth is components take input from many other places: API responses, localStorage and window size to name a few.
But we know developing components in isolation is The Way, so intricate inputs won't stop us! With proxies, we look the devil in the eye and mock anything components depend on. Hell, we might even simplify our components once we're aware of all the crazy things they need to work.
How do proxies work? Well duh, they're Just Components. As regular React components, proxies compose in the order they are listed in your config and decorate the loaded component, respecting the contract to render the next proxy in the chain. They can be stateless or have a life cycle, mocking before mounting and unmocking before unmounting.
Proxies have two parts:
- Configuration. Done once per project, inside cosmos.proxies.js. Import proxy packages, call their default export (always a create function) and add the result to the list of exported proxies. Some proxies require options, others work out of the box.
- Activation. Triggered by a special fixture attribute. Eg. The React Router proxy activates when
fixture.url
is defined, otherwise it's a noop. Proxies can also be always-active, but it's a best practice to make proxies opt-in to avoid useless overhead.
As soon as you're ready to add proxies to your Cosmos setup, create cosmos.proxies.js
(next to cosmos.config.js) and export a list of proxies in the order they should load–from outermost to innermost. Here's an example where we mock the Fetch API and add Redux and React Router providers:
// cosmos.proxies.js
import createFetchProxy from 'react-cosmos-fetch-proxy';
import createReduxProxy from 'react-cosmos-redux-proxy';
import createRouterProxy from 'react-cosmos-router-proxy';
// We can import app files here
import configureStore from './configureStore';
// Read more about configuring Redux in the Redux proxy section below
const ReduxProxy = createReduxProxy({
createStore: state => configureStore(state)
});
// We ensure a specific proxy order
export default [
// Not all proxies have options, and often relying on defaults is good enough
createFetchProxy(),
ReduxProxy,
createRouterProxy()
];
For details on creating proxies, see the Proxy boilerplate
Jump to:
React Context: With great power comes great responsibility.
Note: React doesn't recommend using context unless you're a lib, so most of us don't need this proxy either.
// cosmos.proxies.js
import createContextProxy from 'react-cosmos-context-proxy';
const ContextProxy = createContextProxy({
childContextTypes: {
theme: PropTypes.object.isRequired,
},
});
export default [
ContextProxy,
// ...other proxies
];
// __fixtures__/example.js
export default {
theme: {
backgroundColor: '#f1f1f1',
color: '#222'
}
}
Check out the context example to see the proxy in action.
Most components in a Redux app depend on Redux state, either they're a container or one of their descendants is. This proxy creates a store using initial data from fixtures and puts it in the context, just like the Provider does.
// cosmos.proxies.js
import createReduxProxy from 'react-cosmos-redux-proxy';
import configureStore from './configureStore';
const ReduxProxy = createReduxProxy({
createStore: state => configureStore(state)
});
export default [
ReduxProxy,
// ...other proxies
];
// __fixtures__/example.js
export default {
// An empty object will populate the store with the initial state
// returned by reducers. But we can also put any state we want here.
reduxState: {}
}
Writing Redux fixtures almost feels too easy. Because Redux state is global, once we have one state mock we can render any component we want!
Warning: react-cosmos-router-proxy is designed for React Router v4 and above
React Router is used in most React projects. Wrapping components with withRouter
makes the Router context an implicit dependency–one we need to mock.
// cosmos.proxies.js
import createRouterProxy from 'react-cosmos-router-proxy';
export default [
createRouterProxy(),
// ...other proxies
]
Simply adding a url
to your fixture will wrap the loaded component inside a Router.
// __fixtures__/example.js
export default {
url: '/about'
}
Optionally, route
can be added to also wrap the loaded component inside a Route.
// __fixtures__/example.js
export default {
url: '/users/5',
route: '/users/:userId'
}
Check out the React Router example to see the proxy in action.
If you use the React integration of Apollo Client to provide data in your app, you may want to provide mocks for isolated UI testing with GraphQL.
Your components wrapped with the graphql
higher-order component provided by react-apollo
depends on the ApolloProvider
defined at the top-level of your app. This proxy does that for you!
Provide:
- GraphQL type definitions
- A Mock object like you would with
graphql-tools
// cosmos.proxies.js
import createApolloProxy from 'react-cosmos-apollo-proxy';
const typeDefs = `
type Query {
hello(who: String): String
}
`;
const mocks = {
Query: () => ({
hello: (root, { who }) => `Hello ${who ? who : 'C O S M O S'}`,
}),
};
export default [
createApolloProxy({
typeDefs,
mocks,
}),
// ...other proxies
];
It's activated out of the box!
Check out the Apollo example to see react-cosmos-apollo-proxy
in action.
Besides client-side state, components also depend on external data. Mocking server responses allows us to completely isolate our components. This proxy makes mocking Fetch responses a breeze.
// cosmos.proxies.js
import createFetchProxy from 'react-cosmos-fetch-proxy';
export default [
createFetchProxy(),
// ...other proxies
]
// __fixtures__/example.js
export default {
fetch: [
{
matcher: '/users',
response: [
{
id: 1,
name: 'Prabu',
},
{
id: 2,
name: 'Karla',
},
{
id: 3,
name: 'Linbaba'
}
],
},
]
};
Built on top of fetch-mock. Check out the Fetch example to see the proxy in action.
Like the Fetch proxy, but for XMLHttpRequest.
// cosmos.proxies.js
import createXhrProxy from 'react-cosmos-xhr-proxy';
export default [
createXhrProxy(),
// ...other proxies
]
// __fixtures__/example.js
export default {
xhr: [
{
url: '/users',
response: (req, res) =>
res.status(200).body([
{
id: 1,
name: 'Blossom',
},
{
id: 2,
name: 'Bubbles',
},
{
id: 3,
name: 'Buttercup'
}
]),
},
],
};
Built on top of xhr-proxy. Check out the Axios example to see the proxy in action.
Overrides the global localStorage API with a replica mock.
Mocking localStorage prevents conflicts with existing browser data and enables the localStorage API in test environments like Jest.
// cosmos.proxies.js
import createLocalStorageProxy from 'react-cosmos-localstorage-proxy';
export default [
createLocalStorageProxy(),
// ...other proxies
]
// __fixtures__/example.js
export default {
localStorage: {
userToken: 'foobar-token'
}
};
What proxy would you create to improve DX?
It's preferred to use CRA's own webpack config (instead of duplicating it).
// cosmos.config.js
module.exports = {
componentPaths: ['src/components'],
containerQuerySelector: '#root',
webpackConfigPath: 'react-scripts/config/webpack.config.dev',
publicPath: 'public',
// Optional. Add this when you start using proxies
proxiesPath: 'src/cosmos.proxies'
};
Also make sure to:
- Run
cosmos
withNODE_ENV=development
- Create the
src/components
directory and place your components there, or change componentPaths option to match your existing structure - Put proxies in the
src
dir–the only place included by the CRA Babel loader
CRA + Cosmos example: Flatris
Next.js apps run on both client & server, so compilation is done via Babel plugins instead of webpack loaders. This means we can rely on Cosmos' default webpack config.
// cosmos.config.js
module.exports = {
componentPaths: ['components'],
publicPath: 'static',
publicUrl: '/static/',
};
Also make sure to:
- Add
html-webpack-plugin
to your dev dependencies - Define
.babelrc
for the Cosmos webpack config to rely on the Next.js preset:
{
"presets": ["next/babel"],
}
Next.js + Cosmos example: Illustrated Algorithms
The current version of React Boilerplate (v3.4) requires some tweaking to work with Cosmos. A PR has landed, however, which makes the integration with upcoming v3.5 as simple as this:
// cosmos.config.js
module.exports = {
componentPaths: ['app/components'],
ignore: ['tests', 'messages', /.+Styles/],
webpackConfigPath: './internals/webpack/webpack.dev.babel',
};
It's preferred to use the starter kit's own webpack config (instead of duplicating it).
// cosmos.config.js
module.exports = {
componentPaths: [
'src/routes/Counter/components',
'src/routes/Home/components'
],
webpackConfigPath: 'build/webpack.config.js'
}
Also make sure to:
- Run
cosmos
withNODE_ENV=development
- Set up the Redux proxy :)
GET http://localhost:8989/loader/index.html 404 (Not Found)
The browser console might greet you with this error when using a custom webpack config. There are two methods for configuring the Loader index.html page:
- Use html-webpack-plugin
- Put a static index.html file in your public path (see
publicPath
option below)
Using html-webpack-plugin
is recommended because it automatically injects the <script>
tag in index.html. If you create your own index.html then make sure the script tag points to "main.js".
<script src="./main.js"></script>
Use the --config
CLI arg if you prefer not placing the config in the project root.
// package.json
"scripts": {
"cosmos": "NODE_ENV=development cosmos --config path/to/cosmos.config.js"
}
The componentPaths
option supports both dir and file paths. Most of the times using a dir path is enough (e.g. "src/components"), but we might need to target the exact file paths when component files sit next to non-component files. Here's an example for this type of setup:
// cosmos.config.js
componentPaths: [
'src/components/A.jsx',
'src/components/B.jsx'
],
getComponentName: componentPath =>
componentPath.match(/src\/components\/(.+)\.jsx$/)[1]
The
getComponentName
option needs to be added when using file paths.
We can also use glob instead of adding each component by hand:
// cosmos.config.js
componentPaths: glob.sync('src/components/*.jsx')
Options supported by cosmos.config.js
.
// cosmos.config.js
module.exports = {
// Read components from multiple locations. Useful for including Redux
// containers or if you split your UI per sections.
componentPaths: [
'src/components',
'src/containers'
],
// Additional paths to search for fixtures, besides the __fixtures__ folder
// nested inside component paths. Useful if you keep fixture files separated
// from components files.
fixturePaths: [
'test/fixtures'
],
// Additional entry points that should be present along with any component.
// Sad, but inevitable.
globalImports: [
'./reset.css',
'./global.css',
],
// Components will not be loaded in the playground if their names match these.
// There's no excuse for components that can't be loaded independently, but
// if you store HoCs (which export functions) next to regular components, well,
// what are you gonna do, not use this wonderful tool?
ignore: [
/notATrueComponent/,
/itsComplicated/,
/itsNotMeItsYou/,
],
// Where to serve static files from. Like --content-base in webpack-dev-server.
publicPath: 'src/public',
// Set base URL for static assets from public folder
publicUrl: '/static/',
// Read more about proxies below
proxies: [
'./redux-proxy.js',
'./context-proxy.js',
],
// Render inside custom root element. Useful if that root element already
// has styles attached, but bad for encapsulation.
containerQuerySelector: '#app',
// Disable hot module replacement
hot: false,
// These ones are self explanatory
hostname: 'localhost',
port: 8989,
webpackConfigPath: './config/webpack.config.dev',
};
Static Component Playground? Piece of 🍰! Add this script and run npm run cosmos-export
or yarn cosmos-export
.
// package.json
"scripts": {
"cosmos-export": "NODE_ENV=production cosmos-export"
}
Fixtures can be reused inside automated tests. Along with proxies, they replace elaborate test preparation and render a component with a single JSX tag.
import { mount } from 'enzyme';
import { Loader } from 'react-cosmos-loader';
import DisplayScreen from '../';
import fixture from '../__fixtures__/hi-there';
it('should render hello message', () => {
const wrapper = mount(
<Loader component={DisplayScreen} fixture={fixture} />
);
expect(wrapper.text()).toMatch(/Hi there/);
});
You can create a snapshot of all your components with react-cosmos-telescope
. A single snapshot file for all components isn't ideal, but it makes a difference until you have time to create granular tests.
import runTests from 'react-cosmos-telescope';
runTests({
cosmosConfigPath: require.resolve('./cosmos.config.js'),
});
This project welcomes all. Check out the Contributing Guide to read about the project's mission and how to get involved. Ask anything on Slack. Let's make UI development fun!
Thanks to Kreativa Studio for the Cosmos logo.