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.
@dwalter/create-reducer
Advanced tools
create-reducer
is a tiny (1kb) utility for declaring strongly typed, reusable reducer logic. If you know robodux, create-reducer
will feel like a variation on a familiar theme.
A reducer is a pure function which takes a state and an action as arguments and produces a new state. Typically, reducers also provide some sort of initial state in the form of a default state argument. By convention, reducers use a switch statement on the type
property of the action argument to determine which logic to execute.
Here's an example of how a reducer and some associated actions might be declared in typescript
without any utility libraries:
interface IncrementCounter {
type: '@counter/increment'
}
interface DecrementCounter {
type: '@counter/decrement'
}
interface SetCounter {
type: '@counter/set'
value: number
}
type CounterAction = IncrementCounter | DecrementCounter | SetCounter
const actions = {
increment(): IncrementCounter {
return { type: '@counter/increment' }
},
decrement(): DecrementCounter {
return { type: '@counter/decrement' }
},
set(value): SetCounter {
return { type: '@counter/set', value }
},
}
function counter(state = 0, action: CounterAction) {
switch (action.type) {
case '@counter/increment':
return state + 1
case '@counter/decrement':
return state - 1
case '@counter/set':
return action.value
default:
return state
}
}
Reducers are both declarative and simple. This makes them desireable for state management purposes. However, no pattern is perfect. Reducers have a few pain points- particularly when used with certain modern tools. create-reducer
attempts to address three of these common problems in particular:
type
properties. Naively, this means remaking the same logic in each reducer. There are several approaches to combat this duplication. Higher order reducers have been introduced for remaking similar logic with factory methods, though higher order reducers do not allow the resulting reducer logic to be extended in any way. Reducer composition is another potential workaround, but it is unclear how actions for composed reducers could be made reusable without introducing pitfalls and gotchas.createReducer()
The core of create-reducer
is a function called createReducer()
. createReducer()
takes three parameters: a prefix for generated actions, an initial state, and an object with handlers describing all the behaviors of the desired reducer. Notice that the first parameter of each handler is the current state and that the remaining parameters correspond to the parameters of the associated action creator. createReducer()
returns a pair containing the reducer function and a collection of action creators in the same shape as the handler object passed in.
Here's an example of declaring the same reducer from the above example using createReducer()
in typescript
:
import { createReducer } from '@dwalter/create-reducer'
const [counter, actions] = createReducer('counter', 0, {
increment(state) {
return state + 1
},
decrement(state) {
return state - 1
},
set(_, value: number) {
return value
},
})
Note that event though no actions are explicitly typed, typescript
knows the call signatures of all action creator props on the actions
variable. Also, each handler has its own function scope, so variable hoisting is no longer an issue.
When passed a string in the first parameter, createReducer()
generates action types using the property names of the handlers as shown in the first example (`@${prefix}/${handlerName}`
). It is also possible to pass a function which accepts a handler name and returns an action type string if you need more control over the generated action types.
So how does create-reducer
help with reusing reducer logic? Let's imagine we need several pieces of state to be tracked which are arrays. We'll want actions for common array operations in all of them, but we don't want to redeclare the same actions and reducer logic multiple times. So, we declare a reducer mixin as follows:
const arraylike = {
add(state, item) {
/* Reducer Logic */
},
remove(state, item) {
/* Reducer Logic */
},
set(state, index, item) {
/* Reducer Logic */
},
}
There are several ways to then use a reducer mixin. Most of the time, you will probably want to add more functionality to the reducer in addition to the utility provided by the mixin. Sometimes it may even be useful to withhold or rename handlers from the mixin:
const [fooReducer, fooActions] = createReducer('foo', [], arraylike)
const [barReducer, barActions] = createReducer('bar', [], {
...arraylike,
someOtherAction(state, ...payload) {
/* Reducer Logic */
},
})
const [bazReducer, bazActions] = createReducer('baz', [], {
setBaz: arraylike.set,
yetAnotherAction(state, ...payload) {
/* Reducer Logic */
},
})
Notice that these mixin examples are done in javascript without type safety. To use mixins in typescript
effectively, the mixins must include some notion of generics. This is why the mixins provided by create-reducer
are all functions, not objects:
import { createReducer, arraylike } from '@dwalter/create-reducer'
createReducer('foobar', [], {
...arraylike<number>(),
fooTheBar(state, ...payload) {
/* Reducer Logic */
},
})
create-reducer
ships with a few basic mixins out of the box. They are not special or unique mixins in any way, so they can be treeshaken from your bundle if not used.
settable<T>()
Adds a single action creator set()
which accepts a new value for that piece of state.
const [name, nameActions] = createReducer('name', 'Orolo', {
...settable<string>()
})
let state = name(undefined, {})
function dispatch(action: Action){
state = name(state, action)
}
dispatch(nameActions.set('Erasmus'))
arraylike<T>()
Adds several array operations.
const [numbers, numberActions] = createReducer('numbers', [], {
...arraylike<number>()
})
let state = numbers(undefined, {})
function dispatch(action: Action){
state = numbers(state, action)
}
dispatch(numberActions.add(1))
dispatch(numberActions.add(2))
dispatch(numberActions.set(0, 4))
// supports negative indices
dispatch(numberActions.set(-1, 4))
// removes the first occurrence
// of an element
dispatch(numberActions.remove(4))
// removes the element at a
// certain index (also supports
// negative indices)
dispatch(numberActions.delete(-1))
entityTable<T>()
Made for storing entity objects which have some sort of id (and typically shouldn't be represented twice in state). This type of normalization logic is handy both for making caches and for managing memory usage.
interface Foo {
id: number
name: string
}
const [foos, fooActions] = createReducer('foo', {}, {
// the table can be configured to hash
// however you need it to
...entityTable<Foo>(foo => foo.id)
})
let state = foos(undefined, {})
function dispatch(action: Action){
state = foos(state, action)
}
// add is the only way to add an entity to
// the table
dispatch(fooActions.add({ id: 2, name: 'bill' }))
// increments the reference count to the
// entity with id 2
dispatch(fooActions.add({ id: 2, name: 'bill' }))
// entity's reference count is decremented,
// but it was referenced twice so it is not
// removed
dispatch(fooActions.remove({ id: 2, name: 'bill' }))
// updates the entity with id 2
dispatch(fooActions.update({ id: 2, name: 'ted' }))
// entity's reference count is decremented,
// and this time it is removed
dispatch(fooActions.remove({ id: 2, name: 'ted' }))
// does nothing because the entity is not
// being tracked anymore
dispatch(fooActions.update({ id: 2, name: 'ted' }))
create-reducer
come with support for thunk and/or async actions?Yes and no? This library only deals with plain object actions. However, these actions can be dispatched freely from within other actionlike abstractions. these features have more to do with the dispatch function than with the reducers, so create-reducer
opts not to deal with them internally.
FAQs
reusable reducer patterns and helpers
We found that @dwalter/create-reducer 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.