/redux-grim

Primary LanguageJavaScriptBSD 3-Clause "New" or "Revised" LicenseBSD-3-Clause

GRiM (Generate Redux in Memory)

Table of contents

What is GRiM?

GRiM is small library to reduce the boiler plate involved in making Redux action creators, and reducers. It's specifically intended for actions which retrieve remote restful data, and provides normalization support.

Action creators and reducers can be created with a single line of code. E.g.

export fetchItem = makeActionCreator('item', 'get', '/parent/(parentId)/item/(itemId)'); 
export itemReducer = makeReducer('item');

Background

We had a lot Redux code at Cloudflare, the bulk of which of which was nearly identical. Very little of it was hand written - most was generated from a simple block of json and the resultant actions and reducers etc. were written out. (This was GRiM's predecessor Grod - Generate Redux on Disk). The theory initially was that people might want to manually edit the generated code, but this never came to passed, and there were a few disadvantages:

  • any changes to the generator would result in a vast number of files being changed
  • having to manually run the process to generate the code was awkward.

It was a perfectly workable solution, one that saved a lot of time, but that that no-one was entirely happy with.

Here's a truncated example of the sort of code that would be generated:

// CRUD actions for 'item'

import * as ActionTypes from '../constants/ActionTypes';
import * as api from '../api/itemApi';

function createItemRequest() {
  return {
    type: ActionTypes.CREATE_ITEM
  };
}

function createItemSuccess(response) {
  return {
    type: ActionTypes.CREATE_ITEM_SUCCESS,
    result: response.body.result,
    reduxKey: 'item',
    method: 'POST'
  };
}

function createItemFailure(error) {
  return {
    type: ActionTypes.CREATE_ITEM_FAILURE,
    errors: error.body && error.body.errors
  };
}

export function createItem(parentId, item, callback, options) {
  return function(dispatch) {
    dispatch(createItemRequest());
    return api.createItem(parentId, item, (error, response) => {
      if (response) {
        dispatch(createItemSuccess(response));
        if (callback) {
          callback(null, response.body.result);
        }
      } else {
        dispatch(createItemFailure(error));
        if (callback) {
          callback(error.body && error.body.errors);
        }
      }
    }, options);
  };
}

//
// For the sake of brevity, the nearly identical methods for GET, PUT, PATCH 
// and DELETE haved been omitted.
//
//  Reducer for 'item'

import * as ActionTypes from '../constants/ActionTypes';
import { static as Immutable } from 'seamless-immutable';

const initialState = Immutable.from({
  data: undefined,
  errors: null,
  isRequesting: false,
  isErrored: false
});

export default function itemReducer(state = initialState, action) {
  switch (action.type) {
    case ActionTypes.CREATE_ITEM:
    case ActionTypes.SAVE_ITEM:
    case ActionTypes.PATCH_ITEM:
    case ActionTypes.GET_ITEM:
    case ActionTypes.DELETE_ITEM:
      return Immutable.merge(state, {
        isRequesting: true,
        isErrored: false,
        errors: null
      });

    case ActionTypes.CREATE_ITEM_SUCCESS:
    case ActionTypes.SAVE_ITEM_SUCCESS:
    case ActionTypes.PATCH_ITEM_SUCCESS:
    case ActionTypes.GET_ITEM_SUCCESS:
    case ActionTypes.DELETE_ITEM_SUCCESS:
      return Immutable.merge(state, {
        data: action.result,
        isRequesting: false,
        isErrored: false
      });

    case ActionTypes.CREATE_ITEM_FAILURE:
    case ActionTypes.SAVE_ITEM_FAILURE:
    case ActionTypes.PATCH_ITEM_FAILURE:
    case ActionTypes.GET_ITEM_FAILURE:
    case ActionTypes.DELETE_ITEM_FAILURE:
      return Immutable.merge(state, {
        isRequesting: false,
        isErrored: true,
        errors: action.errors
      });

    case ActionTypes.RATELIMIT_BOOTSTRAP:
      return Immutable.merge(state, {
        data: action.result,
        isRequesting: false,
        isErrored: false,
        errors: null
      });

    case ActionTypes.RESET_TO_INITIAL_STATE:
      const { includes, excludes } = action;
      if (includes) {
        if (includes.indexOf('item') !== -1) {
          return initialState;
        }
      } else if (!excludes || excludes.indexOf('item') === -1) {
        return initialState;
      }

    default:
      return state;
  }
}


// Api functions:

import * as http from 'cf-util-http';

export function createItem(parentId, item, callback, options = {}) {
  const apiOptions = Object.assign({}, {
    body: item
  }, options);
  http.post('/parent/' + parentId + '/rate_limits', apiOptions, callback);
}

export function saveItem(parentId, item, callback, options = {}) {
  const apiOptions = Object.assign({}, {
    body: item
  }, options);
  http.put('/parent/' + parentId + '/item/' + item.id, apiOptions, callback);
}

export function patchItem(parentId, item, callback, options = {}) {
  const apiOptions = Object.assign({}, {
    body: item
  }, options);
  http.patch('/parent/' + parentId + '/item/' + item.id, apiOptions, callback);
}

export function fetchItem(parentId, itemId, callback, options = {}) {
  const apiOptions = Object.assign({}, null, options);
  http.get('/parent/' + parentId + '/item/' + itemId, apiOptions, callback);
}

export function deleteItem(parentId, item, callback, options = {}) {
  const apiOptions = Object.assign({}, null, options);
  http.del('/parent/' + parentId + '/item/' + item.id, apiOptions, callback);
}

And this is what we have now with GRiM:

// Crud actions for 'item'
export fetchItem = makeActionCreator('item', 'get', '/parent/(parentId)/item/(itemId)');
export createItem = makeActionCreator('item', 'post', '/parent/(parentId)/item');
export saveItem =  makeActionCreator('item', 'put', '/parent/(parentId)/item/[item.id]');
export patchItem =  makeActionCreator('item', 'patch', '/parent/(parentId)/item/[item.id]');
export deleteItem =  makeActionCreator('item', 'delete', '/parent/(parentId)/item/[item.id]');

// Reducer for 'item' 
export itemReducer = makeReducer('item');

It's important to note that although the boilerplate has been abstracted away, GRiM's action creators still produce normal Redux actions, in Flux Standard Action format.

Action lifecycle

As you can see from the original generated code above, executing one of GRiM's action creators begins a sequence of events that will produce two of three possible actions.

  • A start action is dispatched
  • An http request is made for a remote resource
  • If the request succeeds a success event is dispatched
  • if the request fails, an error event is dispatched
const getThing = makeActionCreator('thing', 'get', '/thing');

const action = getThing();

// Start action
{
  type: 'thing.start'
  meta: {
    entityType: 'thing',
    method: 'get' 
  }
}

// Start action if request to /thing succeeds
{
  type: 'thing.success',
  payload: {}, // Api response
  meta: {
    entityType: 'thing',
    method: 'get' 
  }
}

// Error action if request to /thing errors
{
  type: 'thing.error',
  payload: {}, // Api Error response 
  error: true,
  meta: {
    entityType: 'thing',
    method: 'get' 
  }
}

About action types

From the example actions above, you can see that the type property of an action depends solely on the type parameter that was passed to makeActionCreator and the phase of the action (e.g. 'start', 'success' or 'error').

This means that if several action creators share the same key, they will generate the action types, regardless of whether they are GET, POST, PUT etc. requests. The method is added to the meta property of action and actions and reducers can be further customized (see below)

This works well for the restful API's we use at Cloudflare which in which POST, PUT and PATCH methods all return the full representation of the object. If you're not using using a restful API, GRiM may not be the best tool for you.

Usage

GRiM exposes two functions, makeActionCreator and makeReducer

makeActionCreator

const action = makeActionCreator(entityType, method, templateUrl, options)

makeActionCreator is used to generate asynchronous restful actions.

Parameters:

  • entityType: used to generate action types, and indicate the type of data that the url returns
  • method: the http method (lowercase). E.g. 'get', 'put', 'post' etc
  • templateUrl: a string which indicates the url used to fetch the data, and also the fields which will be mapped from the action's parameters. See below.
  • options: options are passed to the various hooks (see below) and can be used to enable debugging information

templateUrls, methods, and action creator parameters

The named parameters are those required by an action creator. They're derived from the template url and the http method. If you pass { debug: true } in the options, makeActionCreator will log the named parameters the action expects.

A get method for basic template url results in an action which has no named parameters.

const action = makeActionCreator('item', 'get', '/foo/bar');
action();

// get request is made to '/foo/bar';

A get method for template url in which parts are delineated by parentheses results in an action where the contents of each set of parentheses are added to the named parameters, and are replaced when the action is called.

const action = makeActionCreator('item', 'get', '/foo/(fooId)/bar/(barId)');
action('abc', 123);

// get request is made to '/foo/abc/bar/123'

The contents of the parentheses are evaluated and replaced

const action = makeActionCreator('item', 'get',
'/foo/(item.parentId)/bar/(item.id)');
action({ parentId: 'abc', id: 123 });

// get request is made to '/foo/abc/bar/123'

For put, patch, and post methods, the previous rules apply, except an additional parameter is required. This is a javascript object which will be sent as a the request body. The body is always the last of the named parameters.

const action = makeActionCreator('item', 'post', '/foo/bar');
action({ value: 1 });

// { value: 1 } is the body posted to '/foo/bar';
const action = makeActionCreator('item', 'post', '/foo/(fooId)/bar/(barId)');
action('abc', 123, {value: 2});

// { value: 2 } is the body posted to '/foo/abc/bar/123'

For methods that require a body, [] syntax can also be used to substitute a value from the object being sent as the body.

const action = makeActionCreator('item', 'patch', '/foo/(fooId)/bar/[id]');
action('abc', { id: 123 });

// { id: 123 } is posted to '/foo/abc/bar/123'

Further action configuration

Actions, as well as being functions, also have methods which allow additional, optional, configuration.

The available methods are:

  • apiFetch
  • on
apiFetch

By default the fetch api is used to make http requests. apiFetch can be used to change or modify this behaviour. Given the limited browser support for fetch, this is recommended.

const action = makeActionCreator('item', 'get', '/item').apiFetch(
  (method, url, body, ...restArgs) => Promise;  
)

// restArgs is array of any parameters that were passed to the action, beyond
// thme named parameters.
on(phase, (action, ...)) => action)

on is used to hook into action creation and enables actions to be customised. It requires two parameters:

  • phase: one of four values corresponding to the action that's been created - start, success, error and all.
  • function: the function that will be executed. The parameters passed to the function depend on the action that's been created

All functions passed to on must return an action object.

on('start', (startAction, namedParams, restArgs, options) => action)

  • startAction - the start action that's just been created.
  • namedParams - an object mapping from the templateUrl's named parameters to their values.
  • restArgs - an array of the remaining arguments (those after the named parameters) that were passed to the action.
  • options - the options object that was passed to makeActionCreator

on('success', (successAction, namedParams, restArgs, options, response) => action)

  • successAction - the success action that's just been created.
  • namedParams - an object mapping from the templateUrl's named parameters to their values.
  • restArgs - an array of the remaining arguments (those after the named parameters) that were passed to the action.
  • options - the options object that was passed to makeActionCreator.
  • response - the response from the api call.

on('error', (errorAction, namedParams, restArgs, options, error) => action

  • errorAction - the error action that's just been created.
  • namedParams - an object mapping from the named parameters to their values.
  • restArgs - an array of the remaining arguments (those after the named parameters) that were passed to the action.
  • options - the options object that was passed to makeActionCreator
  • error - the error that was thrown by the api fetch

on('all', (action, namedParams, restArgs, options, [response])

  • action - that's just been created
  • namedParams - an object mapping from the named parameters to their values.
  • restArgs - an array of the arguments that were passed to the action.
  • options - the options object that was passed to makeActionCreator
  • response (optional) - dependant on the phase. Will be undefined for a start action, the response for a success action, and the error for the error action.

Multiple functions can be added for each phase. They will be executed in the order that they were added, and the action that's passed as the first parameter to each is the result of the preceding call.

mock(function | value => payload)

The mock function allows the endpoint request to be bypassed and a predefined response to be returned instead.

If a function passed to mock, when the action is dispatched the function is executed with the same parameters as the action, and the return value is used as the success action payload.

If the function returns undefined, the endpoint request will be made, so that mocking can be performed selectively, depending on the action parameters.

If the function throws an exception, the value thrown is used as the payload of the error action.

If an object or value is passed to mock, it will be used as the success action payload whenever the action is dispatched.

const action = makeAction('item', 'post', '/endpoint/(id)')
  .mock((id, value) => ({ id, value });

action('abcd', 'elephants');

// The action payload is { id: 'abcd', value: 'elephants' }

unmock()

This will clear the function passed to mock.

makeReducer

makeReducer(entityType, options) is used to generate a reducer which process actions created with the same type.

Parameters:

  • entityType: used to determine which action types this reducer will process.
  • options: options are passed to the various hooks. See below

Further reducer configuration

The reducer produced by calling this function also provides additional functions that can be used for configuration. All these functions return the reducer and can be chained.

  • modifyInitialState
  • on
modifyInitialState(state => state)

Change the initial state of the reducer

on(phase, (state, ...)) => state)

Similarly to how action.on is used to modify actions, the reducer equivalent can be used to modify state. The hook functions are called after the reducer has processed the state, and can be used to return a new state.

It requires two parameters:

  • phase: one of four values corresponding to the action that's been created - start, success, error, all and default.
  • function: the function that will be executed. The parameters passed to the function depend on the phase

All functions passed to on must return the state.

on('start', (nextState, prevState, action, options) on('success', (nextState, prevState, action, options) on('error', (nextState, prevState, action, options) on('all', (nextState, prevState, action, options)

  • nextState - the new state that was created after processing the action
  • prevState - the state that existed prior to processing the action.
  • action - the action that was processed by the reducer
  • options - the options object that was passed to makeReducer

start, success, and error are executed when processing the respective actions, all is executed for all of the above.

on('default', (nextState, prevState, action, options)

  • state - the state
  • action - the action that was received by the reducer
  • options - the options object that was passed to makeReducer

The default case is executed when no other case matches the action type.

Multiple functions can be added for each phase. They will be executed in the order that they were added, and the state that's passed as the first parameter to each is the result of the preceding call.

Dependencies

Seamless-immutable

All the reducers created by GRiM use seamless-immutable to create immutable results. This is not bundled with GRiM and must be included in your project. See https://github.com/rtfeldman/seamless-immutable for details.

Redux-thunk

The redux-thunk middleware must be installed in order to process the actions created by GRiM

Normalization

This library also includes an additional set of functions for dealing with normalization with Redux.

Why Normalize?

Normalization ensures that entities are only stored once in the state tree. Individual reducers will record ids (or arrays of ids), rather than complete objects. Combined with React, this means that any change to an individual entity will be reflected wherever than entity is rendered.

Several functions are provided to deal with normalization.

  • getNormalizerMiddleware: A function which creates Redux middleware which normalizes the results of actions created by GRiM
  • normalizationReducer: A reducer which stores normalized entities in the state tree
  • createSelector: Creates memoized selectors which return denormalized entities.

Rules

The rules array specifies which entity types will be normalized. Each entry requires at least an 'entityType' field, and other values describe how they are normalized.

const rules = [
  { entityType: 'normalizedType' },
  { entityType: 'aliasType', to: 'normalizedType' },
  { 
    entityType: 'nestedType',
    nestedProps: {
      prop1: 'normalizedType',
      prop2: 'normalizedType'
    }
  },
  { entityType: 'notIdType', idProp: 'otherId' }
];

An object with only an entityType field indicates that the object will be normalized under that entityType. E.g. state.entities[entityType].

Aliases

If the rule has a to property, the value of that property is used to look up the actual entity type under which items will be normalized. It must resolve to another entity type defined in the rules array. This allows entities controlled by different sets of actions and reducers to be normalized to the same place in the state tree.

Typically it's used when an endpoint returns an array, to ensure that objects in the array are normalized to the same place as individual items.

Note, aliases can't be used with any other configuration options, such as nestedProp or idProp

Normalizing child properties

If the rule value has a nestedProps property, it's value should be an object whose keys denote the child properties to also be normalized. The values indicate which entity type it will be normalized under.

In the example above, nestedType has two properties, 'prop1', and 'prop2', which will be normalized under 'normalizedType'. The process is recursive, so that if 'normalizedType' also had properties to be normalized or denormalized, this will be managed automatically.

Cyclic dependencies aren't handled, so I don't recommend you created any.

Normalizing by properties other than the id

By default, it's assumed that objects will be normalized using their id properties. This can be overridden by specifying a idProp property in the rule configuration. In the example above, nestedType objects are normalized using their name properties.

Functions

getNormalizerMiddleware(rules, callback)

getNormalizerMiddleware creates a Redux middleware function which takes action payload values and, if they are normalized, replaces them with their ids (or arrays of ids). The normalized data is added to the action under an entities property

The callback function is executed when entities are normalized. It's passed the entityType and the original denormalized value. This is used for legacy integration with older code at Cloudflare. It shouldn't be used otherwise.

Note: Only the payloads of success and set actions are normalized.

normalizationReducer(state, action)

The normalizationReducer records the normalized data created by the middleware, and also removes deleted data.

createSelector(rules, entitiesSelector, entityType, selector)

createSelector is a selector factory which simplifies working with normalized entities. It creates memoized selectors which return denormalized objects. The return value is recomputed when relevant parts of the state tree changes, making it suitable for use with PureComponent.

It's passed the following parameters:

  • rules - the rules array which specifies which entities are normalized and how
  • entitiesSelector - a selector function which is passed the state tree and returns the part of the state tree where the normalized data is stored
  • entityType - the type of the entity being selected
  • selector - a selector function which returns the data from the state tree to be denormalized.

The selector function can return either an id, an array of ids, or an object whose property values are ids. Alternatively it can return an object with a data property in one of the previous formats.

Note: Unlike getNormalizerMiddleware and normalizationReducer, createSelector is almost completely independent of GRiM. Almost. With GRiM, api responses are stored under a 'data' property, alongside a few other fields like 'isRequesting', and 'errors'. Rather than insisting the selector to point directly to the data property, I instead first test if the result of the selector is a data property and then denormalize that. In general I strongly recommend that the selectors you pass to createSelector point directly to the data you want to denormalize rather than the containing object that GRiM uses.

Usage:

const rules = {...};
const entitiesSelector = state => state.entities;
const itemSelector = state => state.path.to.the.normalized.item;
const itemEntitySelector = createSelector(rules, entitiesSelector, 'itemEntityType', itemSelector);
const denormalizedItem = itemEntitySelector(state);