Security News
The Risks of Misguided Research in Supply Chain Security
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
mobx-state-tree
Advanced tools
Opinionated, transactional, MobX powered state container combining the best features of the immutable and mutable world for an optimal DX
DISCLAIMER: Docs are still being worked on, so if you are confused at any point; we hope to see you back soon, or feel free to open issue or join the gitter channel
npm install mobx-state-tree --save
yarn add mobx-state-tree
window.mobxStateTree
)Typescript typings are included in the packages. Use module: "commonjs"
or moduleResolution: "node"
to make sure they are picked up automatically in any consuming project.
mobx-state-tree
is a state container that combines the simplicity and ease of mutable data with the traceability of immutable data and the reactiveness and performance of observable data.
Simply put, mobx-state-tree tries to combine the best features of both immutability (transactionality, traceability and composition) and mutability (discoverability, co-location and encapsulation) based approaches to state management; everything to provide the best developer experience possible. Unlike MobX itself, mobx-state-tree is very opinionated on how data should be structured and updated. This makes it possible to solve many common problems out of the box.
Central in MST (mobx-state-tree) is the concept of a living tree. The tree consists of mutable, but strictly protected objects enriched with runtime type information. In other words; each tree has a shape (type information) and state (data). From this living tree, immutable, structurally shared, snapshots are generated automatically.
import { types, onSnapshot } from "mobx-state-tree"
const Todo = types.model("Todo", {
title: types.string,
done: false
}, {
toggle() {
this.done = !this.done
}
})
const Store = types.model("Store", {
todos: types.array(Todo)
})
// create an instance from a snapshot
const store = Store.create({ todos: [{
title: "Get coffee"
}]})
// listen to new snapshots
onSnapshot(store, (snapshot) => {
console.dir(snapshot)
})
// invoke action that modifies the tree
store.todos[0].toggle()
// prints: `{ todos: [{ title: "Get coffee", done: true }]}`
By using the type information available; snapshots can be converted to living trees and vice versa with zero effort. Because of this, time travelling is supported out of the box, and tools like HMR are trivial to support example.
The type information is designed in such a way that it is used both at design- and run-time to verify type correctness (Design time type checking is TypeScript only atm, Flow PR's are welcome!)
[mobx-state-tree] Value '{\"todos\":[{\"turtle\":\"Get tea\"}]}' is not assignable to type: Store, expected an instance of Store or a snapshot like '{ todos: { title: string; done: boolean }[] }' instead.
Runtime type error
Designtime type error
Because state trees are living, mutable models actions are straight-forward to write; just modify local instance properties where appropiate. See toggleTodo()
above or the examples below. It is not needed to produce a new state tree yourself, MST's snapshot functionality will derive one for you automatically.
Although mutable sounds scary to some, fear not; actions have many interesting properties. By default trees can only be modified by using an action that belongs to the same subtree. Furthermore, actions are replayable and can be used as means to distribute changes (example).
Moreover; since changes can be detected on a fine grained level. JSON patches are supported out of the box. Simply subscribing to the patch stream of a tree is another way to sync diffs with for example back-end servers or other clients (example).
(screenshot of patches being emitted)
Since MST uses MobX behind the scenes, it integrates seamlessly with mobx and mobx-react. But even cooler; because it supports snapshots, middleware and replayable actions out of the box, it is even possible to replace a Redux store and reducer with a MobX state tree. This makes it even possible to connect the Redux devtools to MST. See the Redux / MST TodoMVC example.
(screenshot)
Finally, MST has built-in support for references, identifiers, dependency injection, change recording and circular type definitions (even across files). Even fancier; it analyses liveleness of objects, failing early when you try to access accidentally cached information! (More on that later)
A pretty unique feature of MST is that it offers livelyness guarantees; it will throw when reading or writing from objects that are for whatever reason no longer part of a state tree. This protects you against accidental stale reads of objects still referred by, for example, a closure.
const oldTodo = store.todos[0]
store.removeTodo(0)
function logTodo(todo) {
setTimeout(
() => console.log(todo.title),
1000
)
)
logTodo(store.todos[0])
store.removeTodo(0)
// throws exception in one second for using an stale object!
Despite all that, you will see that the API is pretty straight forward!
Another way to look at mobx-state-tree is to consider it, as argued by Daniel Earwicker, to be "React, but for data". Like React, MST consists of composable components, called models, which capture a small piece of state. They are instantiated from props (snapshots) and after that manage and protect their own internal state (using actions). Moreover, when applying snapshots, tree nodes are reconciled as much as possible. There is even a context-like mechanism, called environments, to pass information to deep descendants.
An introduction to the philosophy can be watched here. Slides. Or, as markdown to read it quickly.
mobx-state-tree "immutable trees" and "graph model" features talk, "Next Generation State Management" at React Europe 2017. Slides.
With MobX state tree, you build, as the name suggests, trees of models.
Each node in the tree is described by two things: Its type (the shape of the thing) and its data (the state it is currently in).
The simplest tree possible:
import {types} from "mobx-state-tree"
// declaring the shape of a node with the type `Todo`
const Todo = types.model({
title: types.string
})
// creating a tree based on the "Todo" type, with initial data:
const coffeeTodo = Todo.create({
title: "Get coffee"
})
The types.model
type declaration is used to describe the shape of an object.
Other built-in types include arrays, maps, primitives etc. See the types overview.
The type information will be used for both
The most important type in MST is types.model
, which can be used to describe the shape of an object.
An example:
const TodoStore = types.model("TodoStore", { // 1
loaded: types.boolean // 2
endpoint: "http://localhost", // 3
todos: types.array(Todo), // 4
selectedTodo: types.reference(Todo), // 5
get completedTodos() { // 6
return this.todos.filter(t => t.done)
},
findTodosByUser(user) { // 7
return this.todos.filter(t => t.assignee = user)
}
}, {
addTodo(title) {
this.todos.push({
id: Math.random(),
title
})
}
})
When defining a model, it is advised to give the model a name for debugging purposes (see // 1
).
A model takes two objects arguments, first all the properties, then the actions.
The properties argument is a key-value set where each key indicates the introduction of a property, and the value it's type. The following types are acceptable as type:
types.boolean
, see // 2
, or a complex, possible earlier defined type (// 4
)// 3
, endpoint: "http://localhost"
is the same as endpoint: types.optional(types.string, "http://localhost")
. The primitive type is inferred from the default value. Properties with a default value can be omitted in snapshots.// 6
. Computed properties are tracked and memoized by MobX. Computed properties will not be stored in snapshots or emit patch events. It is allowed to provid a setter for a computed property as well. A setter should always invoke an action.// 7
). A view function can, unlike computed properties, take arbitrary arguments. It won't be memoized, but it's value can be tracked by Mobx nonetheless. View functions are not allowed to change the model, but should rather be used to retrieve information from the model.The actions argument is a key-value set with actions that are available to manage the model. Only actions are allowed to manage models (including any contained objects).
It is also possible to define lifecycle hooks in the actions object, these are actions with a predefined name that are run at a specific moment. See Lifecycle hooks.
Tip: Note that { action1() { }, action2() { }}
is ES6 syntax for { action1: function() { }, action2: function() { } }
, in other words; it's just an object literal.
For that reason a comma between each member of a model is mandatory, unlike classes which are syntactically a totally different concept.
MST trees have very specific semantics. These semantics purposefully constraint what you can do with MST. The reward for that are all kind of generic features out of the box like snapshots, replayability etc. If these constraints don't suit your app, you are probably better of using plain mobx with your own model classes. Which is perfectly fine as well.
detach
that node, or clone
it.In MST every node in the tree is a tree in itself. Trees can be composed by composing their types:
const TodoStore = types.model({
todos: types.array(Todo)
})
const storeInstance = TodoStore.create({
todos: [{
title: "Get biscuit"
}]
})
The snapshot passed to the create
method of a type will recursively be turned in MST nodes. So you can safely call:
storeInstance.todos[0].setTitle("Chocolate instead plz")
Since any node in a tree, is an tree in itself, any built-in method in MST can be invoked on any node in the tree, not just the root. This makes it possible to get a patch stream of a certain subtree, or to apply middleware to a certain subtree only.
By default, nodes can only be modified by one of their actions, or actions higher up in the tree.
Actions can be defined by passing a second object to to types.model
:
const Todo = types.model({
title: types.string
}, {
setTitle(newTitle) {
this.title = newTitle
}
})
Actions are replayable and are therefore constrained in several ways:
Useful methods:
onAction(model, listener)
listens to any action that is invoked on the model or any of it's descendants. See onAction
for more details.addMiddleware(model, middleware)
listens to any action that is invoked on the model or any of it's descendants. See addMiddleware
for more details.applyAction(model, action)
invokes an action on the model according to the given action descriptionThe difference between action listeners and middlewares is: Middleware can intercept the action that is about to be invoked, modify arguments, return types etc. Action listeners cannot intercept, but are only notified. Action listeners receive the action arguments in a serializable format, while middleware receive the raw arguments. (onAction
is actually just a built-in middleware)
If the default protection of mobx-state-tree doesn't fit your use case. For example if you are not interested in replayable actions or hate the effort of writing actions to modify any field; unprotect(tree)
will disable the protected mode of a tree, allowing anyone to directly modify the tree.
Snapshots are the immutable serialization in plain objects of a tree at a specific point in time.
Snapshots can be inspected through getSnapshot(node)
.
Snapshots don't contain any type information and are stripped from all actions etc, so they are perfectly suitable for tranportation.
Requesting a snapshot is cheap, as MST always maintains a snapshot of each node in the background, and uses structural sharing
coffeeTodo.setTitle("Tea instead plz")
console.dir(getSnapshot(coffeeTodo))
// prints `{ title: "Tea instead plz" }`
Some interesting properties of snapshots:
Snapshots are immutable
Snapshots can be transported
Snapshots can be used to update / restore models to a certain state
Snapshots are automatically converted to models when needed. So the two following statements are equivalent: store.todos.push(Todo.create({ title: "test" }))
and store.todos.push({ title: "test" })
.
Useful methods:
getSnapshot(model)
: returns a snapshot representing the current state of the model
onSnapshot(model, callback)
: creates a listener that fires whenever a new snapshot is available (but only one per MobX transaction).
applySnapshot(model, snapshot)
: updates the state of the model and all its descendants to the state represented by the snapshot
Modifying a model does not only result in a new snapshot, but also in a stream of JSON-patches describing which modifications are made. Patches have the following signature:
export interface IJsonPatch {
op: "replace" | "add" | "remove"
path: string
value?: any
}
path
attribute of a patch considers the relative path of the event from the place where the event listener is attachedUseful methods:
onPatch(model, listener)
attaches a patch listener to the provided model, which will be invoked whenever the model or any of it's descendants is mutatedapplyPatch(model, patch)
applies a patch to the provided modelReferences and identifiers are a first class concept in MST. This makes it possible to declare references and keeping the data normalized in the background, while you interect with it in a denormalized manner.
Example:
const Todo = types.model({
id: types.identifier(),
title: types.string
})
const TodoStore = types.model({
todos: types.array(Todo),
selectedTodo: types.reference(Todo)
})
// create a store with a normalized snapshot
const storeInstance = TodoStore.create({
todos: [{
id: "47",
title: "Get coffee"
}],
selectedTodo: "47"
})
// because `selectedTodo` is declared to be a reference, it returns the actual Todo node with the matching identifier
console.log(storeInstance.selectedTodo.title)
// prints "Get coffee"
identifier()
propertiesmap.put()
method can be used to simplify adding objects that have identifiers to mapstypes.refinement(types.string, v => v.match(/someregex/))
References are defined by mentioning the type they should resolve to. The targetted type should have exactly one attribute of the type identifier()
.
References are looked up through the entire tree, but per type. So identifiers need to be unique in the entire tree.
MST is powered by MobX. This means that it is immediately compatible with observer
components, or reactions like autorun
:
import { autorun } from "mobx"
autorun(() => {
console.log(storeInstance.selectedTodo.title)
})
But since MST keeps immutable snapshots in te background, it is also possible to be notified when a new snapshot of the tree is available, similar to .subscribe
on a redux store:
onSnapshot(storeInstance, newSnapshot => {
console.dir("Got new state: ", newSnapshot)
})
However, sometimes it is more useful to precisely know what has changed rather than just receiving a complete new snapshot. For that, MST supports json-patches out of the box
onPatch(storeInstance, patch => {
console.dir("Got change: ", patch)
})
storeInstance.todos[0].setTitle("Add milk")
// prints:
{
path: "/todos/0",
op: "replace",
value: "Add milk"
}
Similarly, you can be notified whenever an action is invoked by using onAction
onAction(storeInstance, call => {
console.dir("Action was called: ", call)
})
storeInstance.todos[0].setTitle("Add milk")
// prints:
{
path: "/todos/0",
name: "setTitle",
args: ["Add milk"]
}
It is even possible to intercept actions before they are applied by adding middleware using addMiddleware
:
addMiddleware(storeInstance, (call, next) => {
call.args[0] = call.args[0].replace(/tea/gi, "Coffee")
return next(call)
})
Finally, it is not only possible to be notified about snapshots, patches or actions.
It is also possible to re-apply them by respectively applySnapshot
, applyPatch
or applyAction
!
These are the types available in MST. All types can be found in the types
namespace, e.g. types.string
. See Api Docs for examples.
types.model(properties, actions)
Defines a "class like" type, with properties and actions to operate on the object.types.array(type)
Declares an array of the specified typetypes.map(type)
Declares an map of the specified typetypes.string
types.number
types.boolean
types.Date
types.union(dispatcher?, types...)
create a union of multiple types. If the correct type cannot be inferred unambigously from a snapshot, provide a dispatcher function.types.optional(type, defaultValue)
marks an value as being optional (in e.g. a model). If a value is not provided the defaultValue
will be used instead. If defaultValue
is a function, it will be evaluated. This can be used to generate for example id's or timestamps upon creation.types.literal(value)
can be used to create a literal type, a type which only possible value is specifically that value, very powerful in combination with union
s. E.g. temperature: types.union(types.literal("hot"), types.literal("cold"))
.types.refinement(baseType, (snapshot) => boolean)
creates a type that is more specific then the base type, e.g. types.refinement(types.string, value => value.length > 5)
to create a type of strings that can only be longer then 5.types.maybe(type)
makes a type optional and nullable, shorthand for types.optional(types.union(type, types.literal(null)), null)
.types.late(() => type)
can be used to create recursive or circular types, or types that are spread over files in such a way that circular dependencies between files would be an issue otherwise.types.frozen
Accepts any kind of serializable value (both primitive and complex), but assumes that the value itself is immutable.Property types can only be used as direct member of a types.model
type and not further composed (for now).
types.identifier(subType?)
Only one such member can exist in a types.model
and should uniquely identify the object. See identifiers for more details. subType
should be either types.string
or types.number
, defaulting to the first if not specified.types.reference(targetType)
creates a property that is a reference to another item of the given targetType
somewhere in the same tree. See references for more details.types.model
Hook | Meaning |
---|---|
afterCreate | Immediately after an instance is created and initial values are applied. Children will fire this event before parents |
afterAttach | As soon as the direct parent is assigned (this node is attached to an other node) |
beforeDetach | As soon as the node is removed from the direct parent, but only if the node is not destroyed. In other words, when detach(node) is used |
beforeDestroy | Before the node is destroyed as a result of calling destroy or removing or replacing the node from the tree. Child destructors will fire before parents |
signature | |
---|---|
addDisposer(node, () => void) | Function to be invoked whenever the target node is to be destroyed |
addMiddleware(node, middleware: (actionDescription, next) => any) | Attaches middleware to a node. See actions. Returns disposer. |
applyAction(node, actionDescription) | Replays an action on the targeted node |
applyPatch(node, jsonPatch) | Applies a JSON patch to a node in the tree |
applySnapshot(node, snapshot) | Updates a node with the given snapshot |
asReduxStore(node) | Wraps a node in a Redux store compatible api |
clone(node, keepEnvironment?: true | false | newEnvironment) | Creates a full clone of a certain node. By default preserves the same environment |
connectReduxDevtools(removeDevModule, node) | Connects a node to the redux development tools example |
destroy(node) | Kills a node, making it unusable. Removes it from any parent in the process |
detach(node) | Removes a node from it's current parent, and let's it live on as stand alone tree |
getChildType(node, property?) | Returns the declared type of the given property of a node. For arrays and maps property can be omitted as they all have the same type |
getEnv(node) | Returns the environment of the given node, see environments |
getParent(node, depth=1) | Returns the intermediate parent of the given node, or a higher one if depth > 1 |
getPath(node) | Returns the path of a certain node in the tree |
getPathParts(node) | Returns the path of a certain node in the tree, unescaped as separate parts |
getRelativePath(base, target) | Returns the short path which one could use to walk from node base to node target , assuming they are in the same tree. Up is represented as ../ |
getRoot(node) | Returns the root element of the tree containing node |
getSnapshot(node) | Returns the snapshot of provided node. See snapshots |
getType(node) | Returns the type of the given node |
hasParent(node, depth=1) | Returns true if the node has a parent at the given depth |
isAlive(node) | Returns true if the node hasn't died yet |
isMST(value) | Returns true if the value is a node of a mobx-state-tree |
isProtected(value) | Returns true if the given node is protected, see actions |
isRoot(node) | Returns true if the has no parents |
joinJsonPath(parts) | Joins and escapes the given path parts into a json path |
onAction(node, (actionDescription) => void | A built-in middleware that calls the provided callback with an action description upon each invocation. Returns disposer |
onPatch(node, (patch) => void) | Attach a JSONPatch listener, that is invoked for each change in the tree. Returns disposer |
onSnapshot | Attach a snapshot listener, that is invoked for each change in the tree. Returns disposer |
protect | Protects an unprotected tree against modifications from outside actions |
recordActions(node) | Creates a recorder that listens to all actions in the node. Call .stop() on the recorder to stop this, and .replay(target) to replay the recorded actions on another tree |
recordPatches | Creates a recorder that listens to all patches emitted by the node. Call .stop() on the recorder to stop this, and .replay(target) to replay the recorded patches on another tree |
resolve(node, path) | Resolves a path (json path) relatively to the given node |
splitJsonPath(path) | Splits and unescapes the given json path into path parts |
tryResolve(node, path) | Like resolve , but just returns null if resolving fails at any point in the path |
unprotect(node) | Unprotects a node, making it possible to directly modify any value in the subtree, without actions |
walk(startNode, (node) => void) | Performs a depth-first walk through a tree |
A disposer is a function that cancels the effect it was created for.
optionals
and default value functionstypes.optional
can takes as default function also a function, which will be invoked each time the default value is needed. This is useful to generate timestamps, identifiers or even complex objects:
createdDate: types.optional(types.date, () => new Date())
toJSON()
for debuggingFor debugging you might want to use getSnapshot(model)
to print the state of a model. But if you didn't import getSnapshot
while debugging in some debugger; don't worry, model.toJSON()
will produce the same snapshot. (For api consistency, this feature is not part of the typed api)
late
On the exporting file:
export function LateStore() {
return types.model({
title: types.string
})
}
In the importing file
import { LateStore } from "./circular-dep"
const Store = types.late(() => LateStore)
Thanks to function hoisting in combination with types.late
, this makes sure you can have circular dependencies between types accross files.
There is no notion of inheritance in MST. The recommended approach is to keep an references to the original configuration of a model to compose it into a new one. (types.extend
achieves this as well, but it might change or even be removed). So a classical animal inheritance could be expressed using composition as follows:
const animalProperties: {
age: types.number,
sound: types.string
}
const animalActions = {
makeSound() {
console.log(this.sound)
}
}
const Dog = types.model(
{ ...animalProperties, sound: "woof" },
animalActions
)
const Cat = types.model(
{ ...animalProperties, sound: "meaow" },
animalActions
)
const Animal = types.union(Dog, Cat)
There is no built-in type for enumerations, but enumarations can simply be constructed by combining unions and literals:
const Temperature = types.union(types.literal("Hot"), types.literal("Cold"))
Or, fancier:
const Temperature = types.union(...["Hot", "Cold"].map(types.literal))
TODO types.localState
types.identifier()
).If an object is reconciled, the consequence is that localState is preserved and postCreate
/ attach
life-cycle hooks are not fired because applying a snapshot results just in an existing tree node being updated.
For asynchronous processes, for each step that intends to modify a model you need a separate action. So for example one to kick off the process. And one to update the model. In a multi stage async process, consider postponing all updates until the last step is completed.
Yep, perfectly fine. No problem. Go on. observer
, autorun
etc will work as expected.
mobx-state-tree
?No, or, not necessarily. An application can use both state trees and vanilla MobX observables at the same time. State trees are primarily designed to store your domain data, as this kind of state is often distributed and not very local. For, for example, local component state, vanilla MobX observables might often be simpler to use.
TypeScript support is best effort, as not all patterns can be expressed in TypeScript. But except for assigning snapshots to properties we got pretty close! As MST uses the latest fancy typescript features it is recommended to use TypeScript 2.3 or higher, with noImplicitThis
and strictNullChecks
enabled.
When using models, you write interface along with it's property types that will be used to perform type checks at runtime. What about compile time? You can use TypeScript interfaces indeed to perform those checks, but that would require writing again all the properties and their actions!
Good news? You don't need to write it twice! Using the typeof
operator of TypeScript over the .Type
property of a MST Type, will result in a valid TypeScript Type!
const Todo = types.model({
title: types.string
}, {
setTitle(v: string) {
this.title = v
}
})
type ITodo = typeof Todo.Type // => ITodo is now a valid TypeScript type with { title: string; setTitle: (v: string) => void }
So far this might look a lot like an immutable state tree as found for example in Redux apps, but there are a few differences:
0.7.1
array.remove
not workingFAQs
Opinionated, transactional, MobX powered state container
The npm package mobx-state-tree receives a total of 70,255 weekly downloads. As such, mobx-state-tree popularity was classified as popular.
We found that mobx-state-tree demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 8 open source maintainers 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
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
Research
Security News
Socket researchers found several malicious npm packages typosquatting Chalk and Chokidar, targeting Node.js developers with kill switches and data theft.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.