You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
This is a proposal for a pattern for creating Zustand stores. The goals of this proposed pattern are:
Avoid creating global singletons of store instances, which results in complex test setup to clear the store between each test (see Zustand testing docs)
Expose store instances to enable integrations outside of the react render cycle
Keep actions out of the Zustand store, to avoid possible mutations of actions and challenges when persisting.
Each "store" consists of a Zustand store API, which only contains the state, and "actions", which is a map of functions to mutate the state.
Better naming would help distinguish between a "Zustand store" which we will use only for state, and our "store" which contains both the Zustand store and actions.
Each "store" should export:
A store creation function
A provider, which takes a store instance as a value prop (could rename this default)
A useStoreState(selector) hook which returns the state of the store
A useStoreActions() hook which returns actions to mutate the store
E.g.
import{createContext,useContext}from'react';import{createStore,useStore,typeStoreApi}from'zustand';import{persistascreatePersistedState}from'zustand/middleware';typeBearState={bears: number;};functioncreateInitialState(): BearState{return{bears: 0,};}constBEAR_STORE_PERSIST_KEY='bearStore';exportfunctioncreateBearStore({persist =false}: {persist: boolean}){letstore: StoreApi<BearState>;if(persist){store=createStore(createPersistedState(createInitialState,{name: BEAR_STORE_PERSIST_KEY}),);}else{store=createStore(createInitialState);}constactions={increasePopulation(){store.setState(state=>({bears: state.bears+1}));},removeAllBears: ()=>store.setState({bears: 0}),updateBears: (newBears: number)=>store.setState({bears: newBears}),};return{store, actions};}constBearStoreContext=createContext<ReturnType<typeofcreateBearStore>|null>(null);exportconstBearStoreProvider=BearStoreContext.Provider;// Shouldn't need to export this, just a helper for DRYfunctionuseBearContext(){constvalue=useContext(BearStoreContext);if(!value){thrownewError('Must set up the BearStoreProvider first');}returnvalue;}exportfunctionuseBearState(): BearState;exportfunctionuseBearState<T>(selector: (state: BearState)=>T): T;exportfunctionuseBearState<T>(selector?: (state: BearState)=>T){const{store}=useBearContext();returnuseStore(store,selector!);}// No need for a selector because this is stable across rendersexportfunctionuseBearActions(){const{actions}=useBearContext();returnactions;}
describe('test 1',()=>{render(MyComponent,{wrapper: TestProvider})// test something})describe('test 2',()=>{render(MyComponent,{wrapper: TestProvier})// test something else})
This way each test is run with a new instance of the (non-persisted) store.
If we want to expose the actual store, in order to read or mutate its value during tests, then we could do:
exportfunctioncreateTestContext(){constbearStore=createBearStore({persist: false})constWrapper=({ children })=><BearStoreProvidervalue={bearStore}>{children}</BearStoreProvider>;return{ bearStore, Wrapper }}
Then in tests:
describe('test 1',()=>{const{ Wrapper, bearStore }=createTestContext()bearStore.store.setState({bears: 100})render(MyComponent,{wrapper: Wrapper})// test something})
However we should probably avoid interacting with the actual store directly in tests, because it leaks an implementation detail into the tests (if we change the store structure, we want tests to still pass if they still do the same thing), so I do not think we should do this. If all we want to do is set an initial state for the store, and it really is too complicated to create that state in the test via userEvents, then we could expose an "initialState" argument for createBearStore, however I think it's much better if we create the state that we need in each test. Running tests in JSDOM should be fast, and we can create test helpers that can create the initial states we might need (like selecting a project Id). Doing things this way will result in more robust tests that allow us to change the implementation without breaking tests, which is important because we want to test the experience of the user using the app.
We might want to end up creating our own persist middleware to wrap zustand one with some defaults, however it might be better to keep things explicit and just use the persist from zustand/middleware each time, and if we need a custom storage, abstract just that instead e.g.
This is a proposal for a pattern for creating Zustand stores. The goals of this proposed pattern are:
Some ideas for this pattern come from https://tkdodo.eu/blog/zustand-and-react-context
Each "store" consists of a Zustand store API, which only contains the state, and "actions", which is a map of functions to mutate the state.
Each "store" should export:
value
prop (could rename this default)E.g.
Provider setup could be done as:
And then the store can be used in components:
For testing we want a different instance of the store for each test, which we can do with something like:
Then when running tests:
This way each test is run with a new instance of the (non-persisted) store.
If we want to expose the actual store, in order to read or mutate its value during tests, then we could do:
Then in tests:
However we should probably avoid interacting with the actual store directly in tests, because it leaks an implementation detail into the tests (if we change the store structure, we want tests to still pass if they still do the same thing), so I do not think we should do this. If all we want to do is set an initial state for the store, and it really is too complicated to create that state in the test via userEvents, then we could expose an "initialState" argument for
createBearStore
, however I think it's much better if we create the state that we need in each test. Running tests in JSDOM should be fast, and we can create test helpers that can create the initial states we might need (like selecting a project Id). Doing things this way will result in more robust tests that allow us to change the implementation without breaking tests, which is important because we want to test the experience of the user using the app.We might want to end up creating our own persist middleware to wrap zustand one with some defaults, however it might be better to keep things explicit and just use the
persist
fromzustand/middleware
each time, and if we need a custom storage, abstract just that instead e.g.The text was updated successfully, but these errors were encountered: