Storex - Frontend store with the state on the backend

Storex is a frontend store with state management handled on the backend. It allows you to update the store state both from the frontend and backend, with all communication occurring over WebSocket.

More info available on Github.

Why Storex?

Features

  • Efficient state management: Only the differences (diffs) in the store state are sent with each mutation, minimizing data transfer.

  • Real-time updates: State changes are immediately reflected across all connected clients via WebSocket communication.

  • Backend-driven state: Storex allows both the frontend and backend to update the store state seamlessly.

  • Lightweight and fast: Designed for minimal overhead, ensuring rapid state updates and communication.

Key Differences from Phoenix LiveView

Phoenix LiveView is a powerful tool for building rich, interactive web applications without writing custom JavaScript. However, as your application grows, managing complex client-side state across multiple LiveViews or components can become challenging. This is where Storex comes in.

  • Client-Side State Management: While Phoenix LiveView handles server-side rendering and event handling, Storex focuses on managing state on the client side. It allows you to keep your client-side state in sync with the server, but with more flexibility in how that state is stored, updated, and accessed.

  • Decoupled State Logic: Storex decouples state management from the LiveView itself, enabling you to manage state across multiple components or even across the entire application. This contrasts with LiveView, where state is typically tied to a specific LiveView process.

  • Predictable State Updates: Storex follows a predictable, unidirectional data flow similar to Redux. This makes it easier to reason about state changes and debug issues, especially in complex applications.

  • Extensibility: Storex is designed to be highly extensible, allowing you to integrate it with other tools and libraries in the Elixir ecosystem. You can also define custom middleware to handle side effects, logging, or other tasks.

For an overview of Storex in action, check out the example provided here.

13 Likes

Hello,

I want to announce that stex was renamed to storex and published to hex.pm with version 0.1.0.

Changelog:

  • The only diff of the store state is being sent on each mutation.
  • Subscriber of connection status
  • Fixes in library

Links:

4 Likes

Version 0.2.0 released

Changelog:

  • Dynamic registry declaration
    • Default registry on ETS
  • Fix issue with a restart of Store when stopped on disconnect
  • Update dependencies
1 Like

@drozdzynski FYI, I noticed the title of the thread was out-of-sync with the new name and adjusted it.

4 Likes

Version 0.2.2 released

Changelog:

  • Typescript/Javascript improvements
  • Fix reconnect of WebSocket on connection close

Version 0.2.3 released

Changelog:

  • Fix reconnect of WebSocket on connection close

That’s interesting.
Do you think it would be feasible to add fw-specific wrappers around Storex, so that we could (for example) use a SolidJS store transparently?

So (in this case) Solid’s reactive system could create an event, which is (maybe) processed in the backend (without any further ado) which then modifies the store which then leads to a reaction (without further ado) on the client side…?

1 Like

I designed it as a replacement for frontend stores.
If you want to use it with SolidJS you can use from function to display the current state.

Example:

import { render } from "solid-js/web";
import { from } from "solid-js";
import Storex from "storex";

const store = new Storex({
  store: "App.RandomNumber",
  params: {},
});

function RandomNumber() {
  const state = from(store);

  return (
    <button onClick={() => store.commit("change", Math.random())}>
      {state()}
    </button>
  );
}

render(() => <RandomNumber />, document.getElementById("app")!);
2 Likes

Thats very nice. So Storex on the client side acts as an observable (implementing subscribe)…?

I’ve only played a little with SolidJS, so I can’t say if there are any drawbacks (besides having to “learn” the Storex API for writing). If it works as I think (have to try it out) this just rocks. Liveview is really great, but I’ve a project now, that is so dynamic that I need a frontend framework. This could bring one of the biggest benefits of LV - removing all the layers between backend and frontend - to any (reactive?) framework (implementing sth like from)

Yes, storex has the subscribe method in Store to work with reactivity.
I’m using storex with svelte frontend framework which works on subscribe method on reactivity.

Example:

<script lang="ts">
  import Storex from 'storex'
  
  const store = new Storex<RandomNumber>({
    store: 'App.RandomNumber',
    params: {},
  })
</script>
{#if $store}
  Random number: {$store}
{/if}
2 Likes

I finally had the time to check this out. Looks very promising.

Some things I noticed (I have very limited JS-skills, so may be bullshit)

  • In your “connect to store” example, I replaced const with var here: subscribe: (state) => {var state = state} (does not work otherwise)
  • your example uses an integer as state, which did not work for me. Changed to a map.
  • storex depends on Jason, which is not installed with deps.

I created a sample app. Especially the websocket handling is not obvious, I used https://github.com/kevinlang/plug_socket which makes that easy.

Sample app: GitHub - georgfaust/storex_example_app

2 Likes

In your “connect to store” example, I replaced const with var here: subscribe: (state) => {var state = state} (does not work otherwise)

You are right, it’s my mistake, I will update README

your example uses an integer as state, which did not work for me. Changed to a map.

I tested that as well and it’s a bug, thanks for catching that

storex depends on Jason, which is not installed with deps.

I will fix it with this integer as a state

1 Like

Version 0.2.4 released

Changelog:

  • Fix root state update
  • Remove optional from jason dependency
3 Likes

Version 0.2.5 released

Changelog:

  • Fix diff of Date struct
  • Rewrite tests from Hound to Wallaby
1 Like

Version 0.3.0

Changelog:

  • [BREAKING] Rename Cowbow handler module from Storex.Socket.Handler to Storex.Handler.Cowboy
  • Add support for Plug based apps plug Storex.Plug
  • Update Storex application supervisor children spec
2 Likes

Bumped my examples repo.

3 Likes

Version 0.4.0

  • [BREAKING] Storex.mutate/3 is no longer based on session_id
  • [BREAKING] Store.init/2 callback now need to return {:ok, state} | {:ok, state, key} | {:error, reason}
  • [BREAKING] Remove custom Registry logic
  • [BREAKING] Remove connection callback from javascript client
  • New registry mechanism provides distributed mutations across the cluster
  • Fix terminate callback in Storex.Handler.Plug
  • Added three callbacks to frontend client onConnected, onError and onDisconnected
1 Like

Version 0.5.0

  • [BREAKING] Frontend client fully rewritten
  • [BREAKING] Updated the minimum required version of Elixir to 1.10
  • Added support for non browser environment
1 Like

Version 0.5.1

  • Fix frontend client type for commit
1 Like

Version 0.6.0

  • [BREAKING] Keys of params in Store are now type binary instead of atom
  • Added message validation with structured casting
1 Like

Version 0.6.1

  • Add missing cast to Storex.Message