@wonderlandlabs/mirror v0.1.11
Mirror
Mirror is a transactional state system for client side applications. It is designed to produce scalable, testable and clean data change management for complex applications. And let's be honest, all applications end up pretty complex and require testing.
Mirror is for anyone who has been:
- Added Redux to a React app and found the boilerplate unnerving
- Tired of using one state system for local changes and another one for global state
- Attempted to test a Redux driven app and found important functionality buried in closure
- Needs to do validation checks that go beyond simple type
Mirrors
A Mirror is a subscribable change engine; it has value(s), built-in and custom change actions, pre-change sanitizers and validation hooks, and leverages Immer for immutability.
A Mirror is designed to allow validation -- that is -- invalid formats are banned from submission and do not trigger subscriber notifications (or crash the subject).
It also allows for sanitizing input -- you can write input filters that clean input to prevent simple mistakes or enforce things like rounding and string formatting.
Also, Mirrors have a transactional update cycle, in which a variety of child values and sub-values can be updated in a single update cycle. This allows the mirrors and any child mirrors to be validated before subscription notifications occur.
Mirrors follow the following mandates; they are:
- synchronous -- all updates happen inline, allowing for immediate inspection of the curent value, even during an action
- structured -- you can define strict schema for mirrors and mirror fields, rejecting bad values.
- testable -- as an object with testable properties, they can easily be tested free from any view scaffolding.
- independent -- Mirrors are not bound to any particular rendering framework (React, Vue, Angular) and can be used with all of them.
- transactional -- changes that trigger validation errors redact themselves and throw, without notifying subscribers of bad data.
- self-contained -- unlike Redux and all similar patterns, the definition of a Mirror is contained entirely inside its definition.
These traits allow for a strong, managed state experience in any context from React to Angular and even node, bare-metal design or any other context.
Mirror Values
Mirror values are not specified; however, they are intended to be either:
- scalar values (strings, numbers, null)
- Maps (see Javascript Maps)
- Arrays
- Objects -- immerable or POJO objects.
Object's/Map's fields will automatically generate setter actions; i.e., a Mirror with a value {x: 0, y: 0}
will have
an action myMirror.$do.setX(10)
and myMirror.$do.setY(20)
.
Complex objects will be expressed as Immerable instances and therefore are immutable.
Mutability
Mirror uses Immer to keep its state immutable -- by default. This means its best to have basic tress of data in mirror: as above -- scalars, Maps, Sets, Arrays, Objects, and Date instances.
However you can configure mirrors to ignore this mandate and open your state to anything else. (typical anything else's -- DOM nodes, database connectors, streams, video data, buffers...)
A mutable mirror cannot have immutable children, but not vice versa; if a given Mirror is mutable, its child mirrors must be as well. (see below)
Assets
One work-around for mutability is a Mirror's Assets. Mirrors' assets can be of any type and nature, stored in its $assets property. A typical $asset would be a database connection, DOM node, or whatever problematic sorts of stuff that you don't want to follow the mutability rules of state. Updating an asset doesn't send out any event or update a state. Assets are loosely defined by design.
While assets were conceived of as being defined in the constructor,
you can set an asset at any point using myMirror.$setAsset(name, value)
.
Subscriptions
Getting updates from a Mirror follows the standard observable pattern:
const myMirror = new Mirror(3);
const sub = myMirror.subscribe((value) => {
console.log('update: ', value);
});
// 'update: 3'
myMiror.next(4);
// 'update: 4';
sub.unsubscribe();
myMirror.next(5);
// [no update]
more formally you can subscribe({next(value) {...}})
.
note, subscriptions are cancellable; this is important for situations where the mirrors; lifespan is longer than the lifespan of the observation.
Updating State
There are a few ways to update a Mirror's state.
If you want to change a subset of fields and ensure all the changes are valid (all or nothing)
- call
instance.$update({changes})
to update a subset of an mirror values. (like React's setState). - call
instance.$mutate(fn)
to call a function to change the mirror's value. Inside the function you can change the target like a normal POJO object/map. This is best, like update (see notes on immer produce)
If you want to change a single field
- call
instance.$do.setFieldName(newValue)
to change a single field's value, for objects or Maps.
If the mirror is a simple value (string, number) or you want to change all of the mirror's value at once
- call
instance.next({new value})
to replace its current value with another value.
const point = new Mirror({x: 10, y: 20, z: 30});
point.subscribe((value) => console.log('point is ', value));
// point is {x: 10, y: 20, z: 30)};
point.$update({x: 30});
// point is {x: 30, y: 20, z: 30});
point.$do.setX(50);
// point is {x: 50, y: 20, z: 30});
point.$do.setY(60);
// point is {x: 50, y: 60, z: 30});
point.$do.setZ(70);
// point is {x: 50, y: 60, z: 70});
point.next({x: 1, y: 2, z: 3});
// point is {x: 1, y: 2, z: 3}
point.$mutate((point) => {
point.x *= 2;
point.y *= 2;
point.z *= 2;
});
// point is {x: 2, y: 4, z: 6}
Actions
Actions are functions that you can run to enact change on Mirror instances.
They are for the most part identical to running an external function to change the mirrors' state; except, when an action is running all subscription updates are suspended until the action completes (or errors out). This means you can change the mirrors' value several times, and even change child mirror values, and only on the completion of the action are updates broadcast.
Actions do not return values. If you want to capture values generated inside an action you need to pass a callback to the action and send it information from inside the action.
Actions are defined by the actions property of the configuration, and/or calls to $addAction(name, fn)
.
They are accessible from the myInstance.$do.actionName(...args)
property of the mirror.
Asynchronicity in actions
Actions are synchronous by nature -- from the point of view of the mechanics of the Mirror system. If you trigger any async/promise behavior inside an action it will complete -- but it will do so after the actions changes have been flushed/committed into the Mirror and any changed children.
The safest way to manage async in actions is to encase the async resolution (i.e., the then handler) in a single separate action call for maximum containment: i.e.,
let changeId = 0;
const myMirror = new Mirror(
{
current: 0,
sentError: null,
},
{
actions: {
sendToServer(me, change) {
axios.put('http://server.com/change', {change})
.then(me.$do.onSentChange)
.catch(me.$do.setSentError)
},
onSentChange(me, response){
me.$do.setCurrent(response.data);
},
}
});
If you want to keep track of the number of pending actions in the air, you can use a more complex form, and track each pending action with an ID.
const myMirror = new Mirror(
{
current: 0,
nextChange: 0,
pending: new Set(),
sendError: null
},
{
actions: {
newPending(me, change, callback) {
me.$do.setNextChange(me.value.nextChange + 1);
let id = me.value.nextChange;
const pending = {change, id};
const nextSubmitted = new Set(my.value.pending);
nextSubmitted.add(pending);
me.$do.setPending(nextSubmitted);
callback(id);
},
sendToServer(me, change) {
let id = null;
me.$do.newPending(change, (nextId) => {
id = nextId;
});
axios.put('http://server.com/change', {change})
.then((response) => me.$do.onSentChange(response, id))
.catch((err) => me.$do.onError(err, id));
},
onError(me, err, id) {
me.$do.removePending(id);
me.$do.setSendError(err);
},
onSentChange(me, response, id){
me.$do.removePending(id);
me.$do.setCurrent(response.data);
},
removePending(me, id) {
const pendingChanges = Array.from(me.value.pending.values()).filter((value) => value !== id);
me.$do.setPending(new Set(pendingChanges));
}
}
});
Nesting Actions
Actions can even call other actions. So you can create a complex nested series of updates, and their effects are buffered internally until all actions complete.
Errors in actions
An un-trapped error in an action will flush all pending changes that were made after it started, and will be thrown. this means, for instance, you can call an action from another action, catch any error around that call, and be confident that the state of the mirror is unchanged; i.e., all buffered changes to date from other activity in the action is the same, but all changes from the subaction have been cleared.
Validation tests
Validation tests are designed to reject bad candidates for change. "Bad" can mean wrong type, but can also relate to other properties of the mirror or external factors.
const numeric = new Mirror(
{
number: 0,
min: -5,
max: 5,
},
{
name: 'safe-number',
test(next, mirror) {
const { number } = next;
if (typeof number !== 'number') return 'not a number';
if (number > mirror.value.max) return `must be <= ${mirror.value.max}`;
if (number < mirror.value.min) return `must be >= ${mirror.value.min}`;
},
},
);
numeric.$do.setNumber(4);
console.log('value: ', numeric.value.number); // 'value: 4'
let e = null;
try {
numeric.$do.setNumber(6);
} catch (err) {
console.log('error: ', err); // 'error: { target: 'safe-number', error: 'must be <= 5' }';
e = err;
}
numeric.$do.setNumber(2);
console.log('value: ', numeric.value.number); // 'value: 2'
If your mirror has one (or more) tests, it will reject values that emit/return errors from the test function, and throw when they occur.
Every time a Mirror's value is submitted (via next(nextValue)), the update event queue polls any errors
returned/thrown from any test present in the mirror.
If that function (exists and) returns / throws errors, then the submitted value is silently rejected
and no observed notification occurs.
If the next value contains keys of managed children that differs from their current values, those children are passed trial values and tested. Any errors in a pending child value are treated as errors in the root and both the parent and child values are flushed silently.
If the child values are good, they are committed, notifying any direct subscribers; then the parent is committed, notifying its subscribers of their updates.
This careful cycle allows for fine-grained control of values and minimizes the possibility of bad values leaking through the system; "bad" is about more than base type (string, array, etc); it is also about any business logic (range, character count) you wish to enforce.
Validation can be used to enforce type on field values; it can also be used to enforce larger business logic concerns.
Validation errors throw; however they do not terminate the Mirror (an RxJS-ism) or trigger any notification to any subscribers.
Selectors
Selectors are functions that run every time a mirror's value is updated and whose return value is appended to the value prior to return. Selectors must be synchronous.
Each selector is passed (value, mirror) as arguments. Selectors should NOT call any methods of the mirror that changes the mirrors' structure or triggers a value update.
selectors are only relevant for Object and Map - type values.
Selectors must be immerable, unless the state itself is mutable.
const m = new Mirror({ x: 1, y: 2 }, {
name: 'point',
selectors: {
mag(value) {
return Math.sqrt(value.x ** 2 + value.y ** 2);
},
},
});
console.log('mirror value:', m.value); // {x: 1, y: 2, mag: 2.23606797749979}
m.$do.setX(2);
console.log('mirror value:', m.value); // {x: 2, y: 2, mag: 2.8284271247461903}
Selectors should be relatively light-weight; if you have a process-intensive calculus to run, consider using a web worker or a discrete debounced RxJS stream.
If a selector function throws an uncaught error, it will return an object {$error: err}
in the place
of the selector value.
Cleaners
Cleaners are pre-change transforming functions intended to remove simple input errors from change values; for instance, rounding numbers down, trimming whitespace from strings, capitalization, string length enforcement.
- Cleaners execute before validators. which means that you shouldn't assume the value input to them has passed validation.
- Cleaners execute before values are passed upwards to any children, and don't have access to the Mirror, by intent
- Cleaner functions should always return a value; if the input is not valid, return the original value.
- Let upcoming validation inform the user of the validation errors.
- Thrown errors in cleaners will stop the update and be passed on to the calling context. (this should not be done intentionally; use validators for this purpose)
const mir = new Mirror(4, {
cleaner(value) {
if (typeof value === 'number') {
return Math.floor(value);
}
return value;
},
});
mir.subscribe((v) => console.log(v));
mir.next(8.2);
Configuration
The second value of a Mirror's constructor are its configuration.
This is a POJO object; it, and all of its keys, are optional.
Here are the possible values of the configuration object:
actions
Object: a collection of callable actions (see above).mutable
boolean: default false whether the Immer library should be used to freeze the Mirrors' state.children
Object or Map:<value|Mirror instance> predefined children (see below).cleaner
function(): ): default null a function that cleans up input (see above).test
function(value):: default null a function that will block any submitted value if it returns any truthy result (such as an error or string).name
string an identifier for the mirror; used in debugging - has no effect on eecution.debug
boolean default false used to turn off various internal messages for debugging Mirror itself. No guarantee of functionality is assigned to its use.cleaners
- distributes cleaners across a set of children. Note - this method will force-create a child for every named cleaner if necessary.assets
Object a set of resources stored apart from state - see Assets
Children
fields of Mirror values can be delegated to child Mirrors. Mirror children can have their own validators, clean functions,
actions and other configuration. in fact, using tests
or cleaners
in configuration will automatically spawn child mirrors.
Child mirrors are locked in the transactionality of their parents; if they are (directly or indirectly) changed in the context of a parent action, their values will be buffered (see below) until the completion of the action, or will flush if an un-trapped error occurs in the action.
Children can be added via the 'children' configuration field, or the instance.$addChild(name, child)
method.
The buffer (a very technical detail)
During actions and updates from next, changes and actions are registered in a buffer array. During this transition all mirrors will reflect any buffered values as if these changes had been committed.
as long as there are actions in the buffer, the buffer remains active. After an error free action, or during next after validating that the mirrors all are error free, the buffer is flushed.
In flushing, if the buffer is error free and action free, buffered changes are committed to the mirror and broadcast.
Untrapped errors in sub-actions will cause an error in the calling action, causing its effects to be removed from the buffer, eventually clearing all the actions and their changes from the buffers, resulting in a no-op action with no effects on the broadcasting of updates, and the mirror is reset to its previous state.
Binding to React components
Hooks are the easiest way to bind Mirrors to components. Local state can be bound and created in effects; global state can be imported directly or pulled from context.
export default function BoundLogin () {
const [values, setValues] = useState(false);
const [fieldMirror, setFieldMirror] = useState(false);
useEffect(() => {
const mirror = new Mirror({userName: '', password: ''}, {
subjects: {
complete({userName, password}) {
return userName && password;
},
paswordValid({password}) {
return /^[a-zA-Z0-9_-]{8,}$/.test(password);
}
}
});
setFieldMirror(mirror);
const sub = mirror.subscribe(setValues);
return () => sub.unsubscribe();
},
[]);
if (values && fieldMirror) {
// don't directly bind $do
return <Login {...values} setUserName={fieldMirror.$do.setUserName} setPassword={fieldMirror.$do.setPassword} />;
}
}
In a class component you can use a local variable in a similar manner.
class BoundLogin extends Component {
constructor(props) {
super(props);
this._fieldMirror = new Mirror({ // same as above
});
this.state = this._fieldMirror.value;
}
componentDidMount() {
this._sub = mirror.subscribe((values) => {
this.setState(values);
});
}
componentWillUnmount() {
if (this._sub) this._sub.unsubscribe();
}
render() {
const {setUserName, setPassword} = fieldMirror.$do; // don't directly bind $do
return <Login {...values} setUserName={setUserName} setPassword={setPassword} />;
}
}
If you have a global Mirror, you can export it as a resource and bind it in the same way .
Mirrors can also be passed down as context; the whole mirror can be a context's value, or,
if you want context subscribers to update with the mirror, by putting the mirror in local state (as above)
and sharing the context and the value (as above, via {fieldMirror, values}
) as the contexts' value.
Related Libraries
Mirrors and RxJS
Mirrors inherit from RxJS' BehaviorSubject. That means it interoperates with all RxJS functionality.
Immer and Mirror
The value of a TYPE_OBJECT or TYPE_MAP is made immutable via Immer. Immer's produce is bundled with Mirror.
That means if you want to update keys you must either:
- call
myMirror.mutate((draft) => draft.foo = 3)
; // operates in the immer context; - call
myMirror.next(produce(myMirror.value, (draft) => {draft.foo = 3})
; // explicitly acts in the immer context - call
myMirror.update({foo: 3})
; // will internally use immer to update your value, leaving other ones unchanged. - call set(s) as in
myMirror.$do.setFoo(3)
// generally easier unless you need to validate a set of changes in parallel
Disabling Immer
There are use cases for removing the immer integration; You are storing DOM or other complex objects in Mirrors, or instances from libraries that are not easily Immerable.
To disable Immer serialization, pass "mutable" as an option. Note, parent Mirrors will serialize all their children, but child mirrors can be Immered even if their parent is mutable.
And lastly -- what is with all the dollar signs?
As Mirrors extend BehaviorSubjects, in order to minimize the possibility of overlap between Mirror code and anything up the inheritance chain from RXJS, all properties and methods of mirrors are prefixed with dollar signs.