npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

soulstate

v1.0.2

Published

The Zero-Overhead State of Mind. A minimalist, high-performance state management library.

Readme


SoulState is a minimalist, high-performance state management library for React and vanilla JS. It provides a simple, unopinionated API inspired by Zustand, but with a re-engineered core focused on extreme performance and zero overhead.

Core Features

  • 🔥 Blazing Fast: Re-engineered core with automatic microtask batching, linked-list subscribers, and minimal structural sharing to avoid unnecessary re-renders.
  • ⚛️ React 18 Ready: First-class support for concurrent rendering via useSyncExternalStore.
  • 🤏 Tiny Footprint: Under 1KB, with zero dependencies.
  • 🧘‍♀️ Simple API: If you know Zustand, you know SoulState. createStore, set, subscribe.
  • 🔒 Mutation Guard: In development mode, state is frozen outside of set calls to prevent accidental mutations.
  • 🌲 Tree-shakeable: Fully modular, ensuring you only bundle what you use.

Installation

npm install soulstate
# or
yarn add soulstate

Basic Usage with React

Create a store, bind it to a component, and you're done.

store.js

import { createStore } from 'soulstate';

export const useCounterStore = createStore({ count: 0 });

// Actions can be defined anywhere, even outside the store
export const increment = () => {
  useCounterStore.set(state => ({ count: state.count + 1 }));
};

Counter.jsx

import { useStore } from 'soulstate/react';
import { useCounterStore, increment } from './store';

function Counter() {
  const count = useStore(useCounterStore, state => state.count);

  return (
    <div>
      <h1>{count}</h1>
      <button onClick={increment}>+1</button>
    </div>
  );
}

TypeScript Usage

SoulState is written in TypeScript and provides excellent type inference.

import { createStore } from 'soulstate';

interface BearState {
  bears: number;
  increase: (by: number) => void;
}

export const useBearStore = createStore<BearState>((set) => ({
  bears: 0,
  increase: (by) => set((state) => ({ bears: state.bears + by })),
}));

Advanced Usage: Managing Complex State

SoulState is not just for simple counters. Its performance and flexibility make it ideal for managing complex, nested state structures, such as the state of an AI agent or a multi-step form.

Let's model a KnowledgeState for a hypothetical research agent:

knowledgeStore.ts

import { createStore } from 'soulstate';

interface KnowledgeState {
    originalQuery: string;
    keyFacts: string[];
    uncertainties: string[];
    searchHistory: Record<string, any>[];
    candidateAnswers: Record<string, any>[];
    confidence: number;
    iteration: number;
}

export const useKnowledgeStore = createStore<KnowledgeState>((set) => ({
    originalQuery: '',
    keyFacts: [],
    uncertainties: [],
    searchHistory: [],
    candidateAnswers: [],
    confidence: 0.0,
    iteration: 0,
}));

// Actions can be defined to manipulate this complex state
export const addFact = (fact: string) => {
    useKnowledgeStore.set(state => ({
        keyFacts: [...state.keyFacts, fact]
    }));
};

export const addSearchHistory = (search: Record<string, any>) => {
    useKnowledgeStore.set(state => ({
        searchHistory: [...state.searchHistory, search]
    }));
};

export const setConfidence = (confidence: number) => {
    useKnowledgeStore.set({ confidence });
}

With this setup, you can subscribe to any part of the KnowledgeState from your React components with minimal overhead, ensuring your UI stays in sync with the agent's "mind" without performance bottlenecks.

function AgentStatus() {
  const { iteration, confidence } = useStoreShallow(useKnowledgeStore, state => ({
    iteration: state.iteration,
    confidence: state.confidence,
  }));

  return (
    <p>Iteration: {iteration}, Confidence: {(confidence * 100).toFixed(1)}%</p>
  );
}

This demonstrates how SoulState can be the backbone for sophisticated applications while maintaining a simple and predictable API.

SSR and Hydration Guide

For Server-Side Rendering (SSR) with frameworks like Next.js, it's crucial to handle state correctly to avoid mismatches between the server and client. The recommended approach is to create a new store instance for each request on the server, and then hydrate it on the client.

While SoulState can be used with a singleton store on the client, for SSR you should use a React Context Provider to isolate state per request.

store.js

// No changes needed here
import { createStore } from 'soulstate';
export const createMyStore = (initialState) => createStore(initialState);

_app.js (Next.js example)

import { Provider, useStoreContext } from 'soulstate/react';
import { createMyStore } from './store';

function App({ Component, pageProps }) {
  const store = createMyStore(pageProps.initialState);
  return (
    <Provider store={store}>
      <Component {...pageProps} />
    </Provider>
  );
}

Subscription Model

At its core, SoulState uses a highly optimized pub/sub model built on a doubly linked list. This provides O(1) complexity for unsubscribe operations and avoids memory allocation for iterators during notifications, making it extremely fast in dynamic environments.

You can subscribe to state changes in vanilla JS:

const unsubscribe = useCounterStore.subscribe(
  state => state.count,
  (count, prevCount) => {
    console.log(`Count changed from ${prevCount} to ${count}`);
  }
);

// To stop listening
unsubscribe();

Selector Model

Selectors are the primary way to consume state. They allow components to subscribe to only the data they need, preventing unnecessary re-renders.

  • Minimal Re-renders: A component will only re-render if the value returned by its selector changes.
  • Memoization: For selectors that return new objects/arrays, use useStoreShallow or provide a custom equality function to prevent re-renders when the underlying data is the same.
import { useStoreShallow } from 'soulstate/react';

// This component only re-renders if user.id or user.name changes.
function User() {
  const { id, name } = useStoreShallow(useUserStore, state => ({
    id: state.user.id,
    name: state.user.name,
  }));
  return <p>{id}: {name}</p>;
}

Best Practices

  1. Keep Selectors Small: Select only the state your component needs.
  2. Use useStoreShallow for Objects: When selecting multiple values into a new object, use useStoreShallow to avoid unnecessary re-renders.
  3. Actions Outside the Store: Define actions as separate functions. This makes them easier to test and ensures they have stable references.
  4. Keep State Flat: A flatter state structure is often easier to reason about and update.

API Reference

  • createStore<T>(initialState: T): Store<T>
  • store.get(): T
  • store.set(updater: Partial<T> | (state: T) => Partial<T> | T)
  • store.subscribe(selector, listener, options?)
  • useStore<T, S>(store, selector, equalityFn?)
  • useStoreShallow<T, S>(store, selector)

Benchmark & Comparison

SoulState is built for performance in complex, highly-interactive applications. Its core architecture makes a deliberate trade-off: it optimizes for scenarios where state updates need to be efficiently propagated to many subscribers (e.g., components), which is often the bottleneck in real-world React applications.

The benchmarks below compare SoulState against other popular state management libraries.


Key Benchmark: Updates with Many Subscribers

This is the most critical benchmark for UI performance. It simulates a state update that triggers re-renders in 100 subscribed components. SoulState's linked-list subscription model is designed specifically for this scenario, resulting in a significant performance advantage.

10k Updates w/ 100 Subscribers | Library | Mean Time (Lower is Better) | Relative Performance | | ---------------- | --------------------------- | -------------------- | | 🚀 SoulState | ~1.4 ms | 19.1x Faster | | Signals (Mock) | ~26.7 ms | (Baseline) | | Zustand (Mock) | ~26.8 ms | 0.99x | | Valtio (Mock) | ~27.0 ms | 0.99x | | MobX (Mock) | ~27.4 ms | 0.97x | | Redux (Mock) | ~27.6 ms | 0.97x |

Conclusion: In high-subscription environments, SoulState is an order of magnitude faster than the alternatives.


Secondary Benchmarks

These benchmarks measure performance in other areas. While SoulState is not the fastest in every category, it remains competitive and its architecture is a conscious choice to prioritize the "many subscribers" use case above all else.

100k Sequential Updates (Raw update throughput) | Library | Mean Time (Lower is Better) | | ------------------ | --------------------------- | | Nano Stores (Mock) | ~3.7 ms | | Jotai (Mock) | ~3.7 ms | | Zustand (Mock) | ~5.9 ms | | Valtio (Mock) | ~6.1 ms | | MobX (Mock) | ~10.1 ms | | Redux (Mock) | ~11.9 ms | | SoulState | ~13.7 ms |

100k Selector Reads (Raw read throughput) | Library | Mean Time (Lower is Better) | | --------------- | --------------------------- | | Valtio (Mock) | ~0.14 ms | | Signals (Mock) | ~0.41 ms | | Redux (Mock) | ~0.53 ms | | MobX (Mock) | ~0.53 ms | | SoulState | ~0.68 ms |

Benchmarks executed with vitest bench. Results are indicative. Run npm run test:bench on your machine for precise results.