Logo

dev-resources.site

for different kinds of informations.

How to configure DevTools for your Zustand store?

Published at
12/19/2024
Categories
nextjs
opensource
zustand
devtools
Author
ramunarasinga-11
Author
16 person written this
ramunarasinga-11
open
How to configure DevTools for your Zustand store?

In this article, you will learn how to configure DevTools for your Zustand store. We will use the Lobechat source code and Zustand documentation as our reference.

Image description

Debugging a store

In the docs, Debugging a store provides this below code as an example:

import { create, StateCreator } from 'zustand'
import { devtools } from 'zustand/middleware'

type JungleStore = {
  bears: number
  addBear: () => void
  fishes: number
  addFish: () => void
}

const useJungleStore = create<JungleStore>()(
 devtools((…args) => ({
   bears: 0,
   addBear: () =>
     set((state) => ({ bears: state.bears + 1 }), undefined, 'jungle/addBear'),
   fishes: 0,
   addFish: () => set(
     (state) => ({ fishes: state.fishes + 1 }),
     undefined,
     'jungle/addFish',
     ),
   })),
)
Enter fullscreen mode Exit fullscreen mode

The question to ask here is how is this different from the simple createStore API without DevTools. In the Create a store documentation,

you will see this below code snippet:

import { create } from 'zustand'
const useStore = create((set) => ({
 bears: 0,
 increasePopulation: () => set((state) => ({ bears: state.bears + 1 })),
 removeAllBears: () => set({ bears: 0 }),
 updateBears: (newBears) => set({ bears: newBears }),
}))
Enter fullscreen mode Exit fullscreen mode

Well, can you tell the difference between these two?

1. Debug example has better types.

type JungleStore = {
 bears: number
 addBear: () => void
 fishes: number
 addFish: () => void
}
const useJungleStore = create<JungleStore>()(
Enter fullscreen mode Exit fullscreen mode

2. The function passed to create is wrapped over by devTools function in the Debug example.

// Debugging enabled
const useJungleStore = create<JungleStore>()(
 devtools((…args) => ({
// Without debugging
const useStore = create((set) => ({
 bears: 0,
Enter fullscreen mode Exit fullscreen mode

3. There are three parameters passed into the set function when the debug is enabled.

// Debugging enabled
addBear: () =>
 set((state) => ({ bears: state.bears + 1 }), undefined, 'jungle/addBear'),
// Without debugging
increasePopulation: () => set((state) => ({ bears: state.bears + 1 })),
Enter fullscreen mode Exit fullscreen mode

It is easy to understand that the third parameter is the action name, but why is the second parameter above is defined as undefined. I found the below statements about this undefined parameter in the documentation.

Additionally, to preserve the default behavior of the replacement logic, the second parameter should be set to undefined.

Do not set the second parameter to true or false unless you want to override the

default replacement logic

Now that we understand the basics of configuring DevTools for a Zustand store. Let’s review the code in Lobechat Zustand store.

DevTools configuration in Lobechat

Lobechat’s state management is complex, but for simplicity purposes, let’s choose store/session/store.ts

devTools

The way devTools is implemented is different in LobeChat.

const devtools = createDevtools('session');

export const useSessionStore = createWithEqualityFn<SessionStore>()(
 subscribeWithSelector(
 devtools(createStore, {
   name: 'LobeChat_Session' + (isDev ? '_DEV' : ''),
 }),
 ),
 shallow,
);
Enter fullscreen mode Exit fullscreen mode

Instead of directly importing devtools from zustand/middleware. createDevTools is a function imported from createDevTools.ts and has the below code:

import { optionalDevtools } from 'zustand-utils';
import { devtools as _devtools } from 'zustand/middleware';

import { isDev } from '@/utils/env';

export const createDevtools =
  (name: string): typeof _devtools =>
  (initializer) => {
    let showDevtools = false;

    // check url to show devtools
    if (typeof window !== 'undefined') {
      const url = new URL(window.location.href);
      const debug = url.searchParams.get('debug');
      if (debug?.includes(name)) {
        showDevtools = true;
      }
    }

    return optionalDevtools(showDevtools)(initializer, {
      name: `LobeChat_${name}` + (isDev ? '_DEV' : ''),
    });
  };
Enter fullscreen mode Exit fullscreen mode

This code tells us that if the url contains a query param named showDevtools and is true, only then show DevTools and uses optionalDevTools imported from zustand-utils.

Often times, concepts found in documentation are not implemented as is in the open-source, this above way of configuring debugging demonstrates that developers at LobeChat are extremely good at what they do.

Action labels in the DevTools

Even the action labels are done differently. At line 167 in session slice action, you will find this below code snippet:

switchSession: (sessionId) => {
 if (get().activeId === sessionId) return;
set({ activeId: sessionId }, false, n(`activeSession/${sessionId}`));
},
Enter fullscreen mode Exit fullscreen mode

What’s the ’n’ here? Line 31 indicates its got something to do with Namespace.

const n = setNamespace('session');
Enter fullscreen mode Exit fullscreen mode

This namespace concept deserves its own article and you can find it on our blog.

About me:

Hey, my name is Ramu Narasinga. I study large open-source projects and create content about their codebase architecture and best practices, sharing it through articles, videos.

I am open to work on an interesting project. Send me an email at [email protected]

My Github - https://github.com/ramu-narasinga
My website - https://ramunarasinga.com
My Youtube channel - https://www.youtube.com/@thinkthroo
Learning platform - https://thinkthroo.com
Codebase Architecture - https://app.thinkthroo.com/architecture
Best practices - https://app.thinkthroo.com/best-practices
Production-grade projects - https://app.thinkthroo.com/production-grade-projects

References:

  1. https://github.com/lobehub/lobe-chat/blob/main/src/store/session/store.ts

  2. https://zustand.docs.pmnd.rs/middlewares/devtools

  3. https://redux.js.org/style-guide/#use-the-redux-devtools-extension-for-debugging

  4. https://zustand.docs.pmnd.rs/getting-started/introduction#first-create-a-store

  5. https://github.com/lobehub/lobe-chat/blob/main/src/store/middleware/createDevtools.ts#L6

devtools Article's
30 articles in total
Favicon
Simplify Email Testing with a Local Papercut SMTP Server Using Docker
Favicon
Performance Audit: Analyzing Namshi’s Mobile Website with Live Core Web Vitals
Favicon
How Daytona Helped Me Streamline My Development Workflow
Favicon
Live core web vitals (local metrics) in browser devtools
Favicon
15 Best Chrome Extensions for Devs 🧑‍💻
Favicon
My 2025 Tech Stack: Tools & Tech I'm Using This Year
Favicon
Chrome DevTools: Everything You Need to Know
Favicon
Latest DocWire SDK Release: Modern Features for C++ Developers
Favicon
LobeChat uses Namespace for action labels in DevTools configuration
Favicon
How to configure DevTools for your Zustand store?
Favicon
Interceptando Requisições com DevTools
Favicon
Validate Your FreeBSD rc.conf
Favicon
Proyect Fugu
Favicon
Introduction to Helm for Kubernetes
Favicon
It's 2AM. Your coffee's cold. The code is flowing.
Favicon
Manual Coding vs Auto-Generated Code: Which One Improves Code Quality?
Favicon
Setting Up a WordPress Development Environment with DDEV
Favicon
Automatizando Formulários com DevTools
Favicon
Full Page Screenshots in Chrome
Favicon
Browser Developer Tools: Essential Tips for Debugging and Optimizing Code
Favicon
A New Reliable AI Tool for Developers
Favicon
Push Express
Favicon
Best Open-Source React Dashboards on GitHub
Favicon
🚀 New open-source alert!
Favicon
Log Streaming - what we got wrong and how we fixed it
Favicon
Buildstash joins Techstars NYC
Favicon
Building a Developer-First SaaS
Favicon
The Changes tab in Google Chrome DevTools
Favicon
Why we're making Buildstash - build-to-release management for app and game devs
Favicon
Let me automate your Github project to showcase my platform!

Featured ones: