@biglogic/rgs 3.9.1 โ 3.9.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +479 -0
- package/index.cjs +1386 -1
- package/index.js +1390 -1
- package/package.json +1 -1
- package/tsup.setup.d.ts +9 -0
- package/core/advanced.cjs +0 -1
- package/core/advanced.js +0 -1
- package/core/minimal.cjs +0 -1
- package/core/minimal.js +0 -1
package/README.md
ADDED
|
@@ -0,0 +1,479 @@
|
|
|
1
|
+
# ๐ ARGIS (RGS) - Reactive Global State
|
|
2
|
+
|
|
3
|
+
> **"Atomic Precision. Immutable Safety. Zen Simplicity."**
|
|
4
|
+
> The state management engine that **won't let you fail**.
|
|
5
|
+
|
|
6
|
+
[](https://npmjs.org/package/@biglogic/rgs)
|
|
7
|
+
[](https://npmjs.org/package/@biglogic/rgs)
|
|
8
|
+
[](https://opensource.org/licenses/MIT)
|
|
9
|
+
[](https://www.typescriptlang.org/)
|
|
10
|
+
[](https://react.dev/)
|
|
11
|
+
|
|
12
|
+
> **๐ Security Compliance**: This project is fully compliant with **NIST SP 800-132** standards for password-based key derivation, featuring AES-256-GCM encryption, PBKDF2 with 600k iterations, and 32-byte salts.
|
|
13
|
+
|
|
14
|
+
---
|
|
15
|
+
|
|
16
|
+
## โก TL;DR
|
|
17
|
+
|
|
18
|
+
>Why ARGIS (RGS) Will Change How You Code Forever
|
|
19
|
+
|
|
20
|
+
```tsx
|
|
21
|
+
// One line. Zero boilerplate. Enterprise-grade power.
|
|
22
|
+
const useUser = gstate({
|
|
23
|
+
name: 'Alice',
|
|
24
|
+
cart: [],
|
|
25
|
+
preferences: { theme: 'dark' }
|
|
26
|
+
})
|
|
27
|
+
|
|
28
|
+
// That's it. Use it anywhere:
|
|
29
|
+
const name = useUser(s => s.name)
|
|
30
|
+
const theme = useUser(s => s.preferences.theme)
|
|
31
|
+
|
|
32
|
+
// Mutations? Just write it. Immer handles immutability automatically.
|
|
33
|
+
useUser(s => {
|
|
34
|
+
s.cart.push({ id: 1, item: 'Coffee Machine', price: 99 })
|
|
35
|
+
})
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
**Stop fighting your state management. Start building.**
|
|
39
|
+
|
|
40
|
+
---
|
|
41
|
+
|
|
42
|
+
## ๐ฏ Why Developers Are **Obsessed** With ARGIS (RGS)
|
|
43
|
+
|
|
44
|
+
| Feature | Other Libraries | ARGIS (RGS) |
|
|
45
|
+
|---------|----------------|--------------|
|
|
46
|
+
| **API Complexity** | 10+ functions, providers, reducers | **1 function** - `gstate()` |
|
|
47
|
+
| **Immutability** | Manual spreads, Redux boilerplate | **Automatic** - Powered by Immer |
|
|
48
|
+
| **Security** | None | **AES-256 + RBAC + GDPR** built-in |
|
|
49
|
+
| **Persistence** | Manual localStorage/sessionStorage | **First-class** - Auto-save anywhere |
|
|
50
|
+
| **Offline/Cloud Sync** | Non-existent | **Local-First + Cloud Sync** included |
|
|
51
|
+
| **Bundle Size** | 10-50KB+ | **~2KB** minimal / **~32KB** full |
|
|
52
|
+
| **Type Safety** | Partial | **100% TypeScript** out of the box |
|
|
53
|
+
|
|
54
|
+
### ๐ฅ The Truth About State Management
|
|
55
|
+
|
|
56
|
+
Most libraries make you **choose** between:
|
|
57
|
+
- โ Simplicity vs Power
|
|
58
|
+
- โ Security vs Speed
|
|
59
|
+
- โ Offline vs Cloud
|
|
60
|
+
|
|
61
|
+
**ARGIS (RGS) gives you ALL of it.**
|
|
62
|
+
|
|
63
|
+
---
|
|
64
|
+
|
|
65
|
+
## ๐ RGS vs The Competition
|
|
66
|
+
|
|
67
|
+
| Feature | **RGS (Argis)** | Zustand | Redux Toolkit | Recoil | Jotai |
|
|
68
|
+
|---------|:---:|:---:|:---:|:---:|:---:|
|
|
69
|
+
| **Philosophy** | Zen State | Minimalist | Enterprise Flux | Atomic | Atomic |
|
|
70
|
+
| **API Surface** | **1 Function** | Simple | Complex | Complex | Simple |
|
|
71
|
+
| **Mutations** | **Magic (Immer)** | Manual | Magic | Manual | Manual |
|
|
72
|
+
| **Security** | ๐ก๏ธ **AES-256 + RBAC** | โ | โ | โ | โ |
|
|
73
|
+
| **Persistence** | ๐พ **First-class** | ๐ | ๐ | ๐ | ๐ |
|
|
74
|
+
| **Local-First Sync** | โ
**Built-in** | โ | โ | โ | โ |
|
|
75
|
+
| **Bundle Size** | **~2KB/32KB** | ~1KB | >10KB | >20KB | ~3KB |
|
|
76
|
+
|
|
77
|
+
> **RGS is the ONLY library treating Security and Persistence as first-class citizens.**
|
|
78
|
+
|
|
79
|
+
---
|
|
80
|
+
|
|
81
|
+
## ๐ Installation
|
|
82
|
+
|
|
83
|
+
```bash
|
|
84
|
+
# The fastest way to upgrade your React app
|
|
85
|
+
npm install @biglogic/rgs
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
```bash
|
|
89
|
+
# Or with pnpm
|
|
90
|
+
pnpm add @biglogic/rgs
|
|
91
|
+
|
|
92
|
+
# Or with yarn
|
|
93
|
+
yarn add @biglogic/rgs
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
---
|
|
97
|
+
|
|
98
|
+
## ๐ฎ Quick Start - 30 Seconds to Glory
|
|
99
|
+
|
|
100
|
+
### The Zen Way (Recommended)
|
|
101
|
+
|
|
102
|
+
```tsx
|
|
103
|
+
import { gstate } from '@biglogic/rgs'
|
|
104
|
+
|
|
105
|
+
// ONE line creates a typed store hook
|
|
106
|
+
const useStore = gstate({
|
|
107
|
+
count: 0,
|
|
108
|
+
user: { name: 'Alice', email: 'alice@example.com' }
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
function Counter() {
|
|
112
|
+
// Type-safe selectors - the React way
|
|
113
|
+
const count = useStore(s => s.count)
|
|
114
|
+
const userName = useStore(s => s.user.name)
|
|
115
|
+
|
|
116
|
+
// Direct mutations - just write JavaScript
|
|
117
|
+
const increment = () => useStore(s => { s.count++ })
|
|
118
|
+
|
|
119
|
+
return (
|
|
120
|
+
<div>
|
|
121
|
+
<h1>Hello, {userName}!</h1>
|
|
122
|
+
<p>Count: {count}</p>
|
|
123
|
+
<button onClick={increment}>+1</button>
|
|
124
|
+
</div>
|
|
125
|
+
)
|
|
126
|
+
}
|
|
127
|
+
```
|
|
128
|
+
|
|
129
|
+
### The Classic Way (Global Store)
|
|
130
|
+
|
|
131
|
+
```tsx
|
|
132
|
+
import { initState, useStore } from '@biglogic/rgs'
|
|
133
|
+
|
|
134
|
+
// Initialize once at app root
|
|
135
|
+
initState({ namespace: 'myapp' })
|
|
136
|
+
|
|
137
|
+
// Use anywhere in your app
|
|
138
|
+
const [user, setUser] = useStore('user')
|
|
139
|
+
const [theme, setTheme] = useStore('theme')
|
|
140
|
+
```
|
|
141
|
+
|
|
142
|
+
---
|
|
143
|
+
|
|
144
|
+
## ๐ก๏ธ Enterprise-Grade Security (No Extra Code)
|
|
145
|
+
|
|
146
|
+
### AES-256-GCM Encryption
|
|
147
|
+
|
|
148
|
+
```tsx
|
|
149
|
+
// Your sensitive data is encrypted automatically
|
|
150
|
+
const useAuth = gstate({
|
|
151
|
+
token: 'jwt-token-here',
|
|
152
|
+
refreshToken: 'refresh-token'
|
|
153
|
+
}, { encoded: true }) // ๐ AES-256-GCM encryption enabled
|
|
154
|
+
```
|
|
155
|
+
|
|
156
|
+
### RBAC (Role-Based Access Control)
|
|
157
|
+
|
|
158
|
+
```tsx
|
|
159
|
+
const store = gstate({
|
|
160
|
+
adminPanel: { users: [], settings: {} },
|
|
161
|
+
userProfile: {}
|
|
162
|
+
}, {
|
|
163
|
+
rbac: {
|
|
164
|
+
admin: ['adminPanel', 'userProfile'],
|
|
165
|
+
user: ['userProfile'],
|
|
166
|
+
guest: []
|
|
167
|
+
}
|
|
168
|
+
})
|
|
169
|
+
|
|
170
|
+
// Only admins can touch adminPanel
|
|
171
|
+
store.set('adminPanel', { users: ['new'] }) // โ
Works for admins
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
### GDPR Compliance
|
|
175
|
+
|
|
176
|
+
```tsx
|
|
177
|
+
// Export all user data (GDPR requirement)
|
|
178
|
+
store.exportData() // Returns JSON of all stored data
|
|
179
|
+
|
|
180
|
+
// Delete all user data (GDPR "right to be forgotten")
|
|
181
|
+
store.deleteData()
|
|
182
|
+
```
|
|
183
|
+
|
|
184
|
+
---
|
|
185
|
+
|
|
186
|
+
## ๐พ Persistence That Just Works
|
|
187
|
+
|
|
188
|
+
### Built-in Storage Adapters
|
|
189
|
+
|
|
190
|
+
```tsx
|
|
191
|
+
// localStorage (default) - survives browser restart
|
|
192
|
+
const useSettings = gstate({ theme: 'dark' }, { persist: true })
|
|
193
|
+
|
|
194
|
+
// sessionStorage - survives page refresh but not tab close
|
|
195
|
+
const useSession = gstate({ temporary: 'data' }, { storage: 'session' })
|
|
196
|
+
|
|
197
|
+
// Memory only - for ephemeral data
|
|
198
|
+
const useMemory = gstate({ cache: [] }, { storage: 'memory' })
|
|
199
|
+
|
|
200
|
+
// IndexedDB - for GB-scale data
|
|
201
|
+
const useBigData = gstate({ dataset: [] })
|
|
202
|
+
store._addPlugin(indexedDBPlugin({ dbName: 'my-app-db' }))
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
---
|
|
206
|
+
|
|
207
|
+
## ๐ Plugins Ecosystem - Extend Without Limits
|
|
208
|
+
|
|
209
|
+
RGS comes with **11+ official plugins** ready to supercharge your app:
|
|
210
|
+
|
|
211
|
+
| Plugin | Purpose | One-Liner |
|
|
212
|
+
|--------|---------|-----------|
|
|
213
|
+
| `undoRedoPlugin` | Time travel through state | `store.undo()` / `store.redo()` |
|
|
214
|
+
| `syncPlugin` | Cross-tab sync (no server) | `store._addPlugin(syncPlugin(...))` |
|
|
215
|
+
| `indexedDBPlugin` | GB-scale storage | `store._addPlugin(indexedDBPlugin(...))` |
|
|
216
|
+
| `cloudSyncPlugin` | Cloud backup & sync | `store.plugins.cloudSync.sync()` |
|
|
217
|
+
| `devToolsPlugin` | Redux DevTools | `store._addPlugin(devToolsPlugin(...))` |
|
|
218
|
+
| `immerPlugin` | Mutable-style updates | `store.setWithProduce('key', fn)` |
|
|
219
|
+
| `snapshotPlugin` | Save/restore checkpoints | `store.takeSnapshot('backup')` |
|
|
220
|
+
| `schemaPlugin` | Runtime validation | `store._addPlugin(schemaPlugin(...))` |
|
|
221
|
+
| `guardPlugin` | Transform on set | `store._addPlugin(guardPlugin(...))` |
|
|
222
|
+
| `analyticsPlugin` | Track changes | `store._addPlugin(analyticsPlugin(...))` |
|
|
223
|
+
| `debugPlugin` | Console access (DEV) | `window.gstate.list()` |
|
|
224
|
+
|
|
225
|
+
### Undo/Redo - Never Lose Work
|
|
226
|
+
|
|
227
|
+
```tsx
|
|
228
|
+
import { undoRedoPlugin } from '@biglogic/rgs'
|
|
229
|
+
|
|
230
|
+
const store = gstate({ text: '' })
|
|
231
|
+
store._addPlugin(undoRedoPlugin({ limit: 50 }))
|
|
232
|
+
|
|
233
|
+
// User makes changes...
|
|
234
|
+
store.set('text', 'Hello World')
|
|
235
|
+
store.set('text', 'Hello Universe')
|
|
236
|
+
|
|
237
|
+
// Oops! Let's go back
|
|
238
|
+
store.undo() // Returns to 'Hello World'
|
|
239
|
+
store.redo() // Returns to 'Hello Universe'
|
|
240
|
+
```
|
|
241
|
+
|
|
242
|
+
### Cross-Tab Sync - Real-Time Everywhere
|
|
243
|
+
|
|
244
|
+
```tsx
|
|
245
|
+
import { syncPlugin } from '@biglogic/rgs/advanced'
|
|
246
|
+
|
|
247
|
+
const store = gstate({ theme: 'light' })
|
|
248
|
+
store._addPlugin(syncPlugin({ channelName: 'my-app' }))
|
|
249
|
+
|
|
250
|
+
// Change in Tab 1 โ Instantly updates Tab 2, Tab 3, etc.
|
|
251
|
+
// ZERO server calls. Pure browser magic.
|
|
252
|
+
```
|
|
253
|
+
|
|
254
|
+
### Cloud Sync - Your Data, Everywhere
|
|
255
|
+
|
|
256
|
+
```tsx
|
|
257
|
+
import { cloudSyncPlugin, createMongoAdapter } from '@biglogic/rgs/advanced'
|
|
258
|
+
|
|
259
|
+
const adapter = createMongoAdapter(
|
|
260
|
+
'https://data.mongodb-api.com/...',
|
|
261
|
+
'your-api-key'
|
|
262
|
+
)
|
|
263
|
+
|
|
264
|
+
const store = gstate({ todos: [] })
|
|
265
|
+
store._addPlugin(cloudSyncPlugin({
|
|
266
|
+
adapter,
|
|
267
|
+
autoSyncInterval: 30000 // Every 30 seconds
|
|
268
|
+
}))
|
|
269
|
+
|
|
270
|
+
// Manual sync when needed
|
|
271
|
+
await store.plugins.cloudSync.sync()
|
|
272
|
+
```
|
|
273
|
+
|
|
274
|
+
---
|
|
275
|
+
|
|
276
|
+
## โ๏ธ Local-First Sync - Offline by Default
|
|
277
|
+
|
|
278
|
+
The **killer feature** that makes RGS unique:
|
|
279
|
+
|
|
280
|
+
```tsx
|
|
281
|
+
import { gstate, useSyncedState } from '@biglogic/rgs'
|
|
282
|
+
|
|
283
|
+
// Create store with automatic offline-first sync
|
|
284
|
+
const store = gstate({
|
|
285
|
+
todos: [],
|
|
286
|
+
user: null
|
|
287
|
+
}, {
|
|
288
|
+
namespace: 'myapp',
|
|
289
|
+
sync: {
|
|
290
|
+
endpoint: 'https://api.example.com/sync',
|
|
291
|
+
authToken: () => localStorage.getItem('auth_token'),
|
|
292
|
+
autoSyncInterval: 30000,
|
|
293
|
+
syncOnReconnect: true,
|
|
294
|
+
strategy: 'last-write-wins'
|
|
295
|
+
}
|
|
296
|
+
})
|
|
297
|
+
|
|
298
|
+
// Works offline. Automatically syncs when online.
|
|
299
|
+
function TodoList() {
|
|
300
|
+
const [todos, setTodos] = useSyncedState('todos')
|
|
301
|
+
|
|
302
|
+
const addTodo = (text) => {
|
|
303
|
+
setTodos([...todos, { id: Date.now(), text }])
|
|
304
|
+
// Synced automatically! โจ
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
return <ul>{todos.map(t => <li>{t.text}</li>)}</ul>
|
|
308
|
+
}
|
|
309
|
+
```
|
|
310
|
+
|
|
311
|
+
---
|
|
312
|
+
|
|
313
|
+
## โก Advanced Superpowers
|
|
314
|
+
|
|
315
|
+
### Computed Values (Derived State)
|
|
316
|
+
|
|
317
|
+
```tsx
|
|
318
|
+
const store = gstate({
|
|
319
|
+
firstName: 'John',
|
|
320
|
+
lastName: 'Doe'
|
|
321
|
+
})
|
|
322
|
+
|
|
323
|
+
// Auto-calculated derived state
|
|
324
|
+
store.compute('fullName', (get) => `${get('firstName')} ${get('lastName')}`)
|
|
325
|
+
|
|
326
|
+
const fullName = store.get('fullName') // "John Doe"
|
|
327
|
+
```
|
|
328
|
+
|
|
329
|
+
### Error Handling
|
|
330
|
+
|
|
331
|
+
```tsx
|
|
332
|
+
const store = gstate({ data: null }, {
|
|
333
|
+
onError: (error, context) => {
|
|
334
|
+
console.error(`Error in ${context.operation}:`, error.message)
|
|
335
|
+
// Send to Sentry, Datadog, etc.
|
|
336
|
+
}
|
|
337
|
+
})
|
|
338
|
+
```
|
|
339
|
+
|
|
340
|
+
### Size Limits (Memory Protection)
|
|
341
|
+
|
|
342
|
+
```tsx
|
|
343
|
+
const store = gstate({ data: {} }, {
|
|
344
|
+
maxObjectSize: 5 * 1024 * 1024, // Warn if single value > 5MB
|
|
345
|
+
maxTotalSize: 50 * 1024 * 1024 // Warn if total > 50MB
|
|
346
|
+
})
|
|
347
|
+
```
|
|
348
|
+
|
|
349
|
+
---
|
|
350
|
+
|
|
351
|
+
## ๐ฆ Build Sizes - Choose Your Weapon
|
|
352
|
+
|
|
353
|
+
### Minimal Version (~0.16 KB)
|
|
354
|
+
For embedded systems, widgets, IoT:
|
|
355
|
+
|
|
356
|
+
```javascript
|
|
357
|
+
import { createStore } from '@biglogic/rgs/core/minimal'
|
|
358
|
+
|
|
359
|
+
const store = createStore({ count: 0 })
|
|
360
|
+
store.get('count') // โ 0
|
|
361
|
+
store.set('count', 5) // โ true
|
|
362
|
+
```
|
|
363
|
+
|
|
364
|
+
### Full Version (~32 KB)
|
|
365
|
+
For production React apps with all features:
|
|
366
|
+
|
|
367
|
+
```javascript
|
|
368
|
+
import { gstate, createStore } from '@biglogic/rgs'
|
|
369
|
+
|
|
370
|
+
const useCounter = gstate({ count: 0 })
|
|
371
|
+
const count = useCounter(s => s.count)
|
|
372
|
+
```
|
|
373
|
+
|
|
374
|
+
| Version | Size | Use Case |
|
|
375
|
+
|---------|------|----------|
|
|
376
|
+
| **Minimal** | 0.16 KB | Embedded, IoT, Widgets |
|
|
377
|
+
| **Full** | ~32 KB | React Apps, Enterprise |
|
|
378
|
+
|
|
379
|
+
---
|
|
380
|
+
|
|
381
|
+
## ๐งช Testing - Rock-Solid Reliability
|
|
382
|
+
|
|
383
|
+
```bash
|
|
384
|
+
# Run unit tests
|
|
385
|
+
npm run test
|
|
386
|
+
|
|
387
|
+
# Run E2E tests (Playwright)
|
|
388
|
+
npm run test:e2e
|
|
389
|
+
```
|
|
390
|
+
|
|
391
|
+
- โ
**100+ Unit Tests** (Jest) - Core logic, stores, hooks
|
|
392
|
+
- โ
**E2E Tests** (Playwright) - Real browser, cross-tab sync
|
|
393
|
+
- โ
**Concurrency Testing** - Race condition verification
|
|
394
|
+
- โ
**Security Tests** - AES-256, RBAC, GDPR
|
|
395
|
+
|
|
396
|
+
---
|
|
397
|
+
|
|
398
|
+
## ๐๏ธ Architecture
|
|
399
|
+
|
|
400
|
+
```mermaid
|
|
401
|
+
graph TB
|
|
402
|
+
subgraph API
|
|
403
|
+
A[gstate] --> D[IStore]
|
|
404
|
+
B[useStore] --> D
|
|
405
|
+
C[createStore] --> D
|
|
406
|
+
end
|
|
407
|
+
|
|
408
|
+
D --> E[Storage Adapter]
|
|
409
|
+
D --> F[Immer Proxy]
|
|
410
|
+
D --> G[Plugins]
|
|
411
|
+
D --> H[Security]
|
|
412
|
+
D --> I[Async Store]
|
|
413
|
+
|
|
414
|
+
E --> J[local]
|
|
415
|
+
E --> K[session]
|
|
416
|
+
E --> L[memory]
|
|
417
|
+
|
|
418
|
+
G --> M[UndoRedo]
|
|
419
|
+
G --> N[Sync]
|
|
420
|
+
G --> O[Schema]
|
|
421
|
+
G --> P[Analytics]
|
|
422
|
+
```
|
|
423
|
+
|
|
424
|
+
### Core Components
|
|
425
|
+
|
|
426
|
+
| Component | Description |
|
|
427
|
+
|-----------|-------------|
|
|
428
|
+
| **gstate()** | Creates store + hook in one line |
|
|
429
|
+
| **useStore()** | React hook for subscribing to state |
|
|
430
|
+
| **createStore()** | Classic store factory |
|
|
431
|
+
| **IStore** | Core interface with get/set/subscribe |
|
|
432
|
+
| **StorageAdapters** | local, session, memory persistence |
|
|
433
|
+
| **Plugins** | Immer, Undo/Redo, Sync, Schema, etc. |
|
|
434
|
+
| **Security** | Encryption, RBAC, GDPR consent |
|
|
435
|
+
|
|
436
|
+
---
|
|
437
|
+
|
|
438
|
+
## ๐ Documentation
|
|
439
|
+
|
|
440
|
+
- [Getting Started](docs/markdown/getting-started.md)
|
|
441
|
+
- [Plugin SDK](docs/markdown/plugin-sdk.md)
|
|
442
|
+
- [Security Architecture](docs/markdown/security-architecture.md)
|
|
443
|
+
- [Migration Guide](docs/markdown/migration-guide.md)
|
|
444
|
+
- [API Reference](docs/markdown/api.md)
|
|
445
|
+
|
|
446
|
+
---
|
|
447
|
+
|
|
448
|
+
## ๐ค Contributing
|
|
449
|
+
|
|
450
|
+
Contributions are welcome! Please read our [contributing guidelines](CONTRIBUTING.md) first.
|
|
451
|
+
|
|
452
|
+
```bash
|
|
453
|
+
# Clone and setup
|
|
454
|
+
git clone https://github.com/BigLogic-ca/rgs.git
|
|
455
|
+
cd rgs
|
|
456
|
+
npm install
|
|
457
|
+
|
|
458
|
+
# Run tests
|
|
459
|
+
npm run test
|
|
460
|
+
```
|
|
461
|
+
|
|
462
|
+
---
|
|
463
|
+
|
|
464
|
+
## ๐ License
|
|
465
|
+
|
|
466
|
+
**MIT** ยฉ [Dario Passariello](https://github.com/passariello)
|
|
467
|
+
|
|
468
|
+
---
|
|
469
|
+
|
|
470
|
+
## ๐ฅ Built for Those Who Demand **Excellence**
|
|
471
|
+
|
|
472
|
+
[](https://github.com/immerjs/immer)
|
|
473
|
+
[](https://www.typescriptlang.org/)
|
|
474
|
+
[](https://jestjs.io/)
|
|
475
|
+
[](https://playwright.dev/)
|
|
476
|
+
|
|
477
|
+
**Made with โค๏ธ and a lot of caffรจ espresso!**
|
|
478
|
+
|
|
479
|
+
[โฌ Back to top](#-argis-rgs---reactive-global-state)
|