@nlabs/arkhamjs-storage-browser 3.29.0 → 3.30.0
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 +181 -466
- package/index.js +3 -3
- package/lib/BrowserStorage/BrowserStorage.d.ts +23 -5
- package/lib/BrowserStorage/BrowserStorage.js +2 -2
- package/lib/index.d.ts +4 -0
- package/lib/index.js +2 -2
- package/lib/types/main.d.ts +18 -0
- package/lib/types/main.js +1 -2
- package/package.json +22 -5
package/README.md
CHANGED
|
@@ -1,559 +1,274 @@
|
|
|
1
1
|
# @nlabs/arkhamjs-storage-browser
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
- **🎯 Selective Persistence** - Choose which parts of state to persist
|
|
18
|
-
- **⚡ Performance Optimized** - Debounced writes and efficient serialization
|
|
19
|
-
- **🔒 Encryption Support** - Optional encryption for sensitive data
|
|
20
|
-
- **🗜️ Compression** - Automatic compression for large state objects
|
|
21
|
-
- **📱 Cross-Tab Sync** - Synchronize state across multiple browser tabs
|
|
22
|
-
- **🔧 Configurable** - Extensive options for customization
|
|
23
|
-
- **🌲 Tree-shakable** - Only include what you need
|
|
24
|
-
|
|
25
|
-
## 📦 Installation
|
|
3
|
+
Enhanced browser storage for ArkhamJS with modern ESNext features, performance optimizations, and advanced caching capabilities.
|
|
4
|
+
|
|
5
|
+
## Features
|
|
6
|
+
|
|
7
|
+
- 🚀 **High Performance**: In-memory caching and optimized storage operations
|
|
8
|
+
- 🔒 **Type Safe**: Full TypeScript support with strict type checking
|
|
9
|
+
- 🎯 **Modern ESNext**: Built with latest JavaScript features (ES2022+)
|
|
10
|
+
- 📦 **Tree Shakeable**: Optimized for bundle size reduction
|
|
11
|
+
- ⚡ **Smart Caching**: Automatic cache management with TTL support
|
|
12
|
+
- 🛡️ **Error Resilient**: Graceful handling of storage errors and quota limits
|
|
13
|
+
- 🔧 **Configurable**: Flexible options for prefix, compression, and size limits
|
|
14
|
+
- 📊 **Monitoring**: Built-in storage statistics and usage tracking
|
|
15
|
+
|
|
16
|
+
## Installation
|
|
26
17
|
|
|
27
18
|
```bash
|
|
28
19
|
npm install @nlabs/arkhamjs-storage-browser
|
|
29
20
|
```
|
|
30
21
|
|
|
31
|
-
##
|
|
22
|
+
## Quick Start
|
|
32
23
|
|
|
33
|
-
###
|
|
24
|
+
### Basic Usage
|
|
34
25
|
|
|
35
|
-
```
|
|
36
|
-
import { Flux } from '@nlabs/arkhamjs';
|
|
26
|
+
```typescript
|
|
37
27
|
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
38
28
|
|
|
39
|
-
//
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
storageWait: 300 // Debounce storage updates by 300ms
|
|
45
|
-
});
|
|
29
|
+
// Create storage instance with default settings
|
|
30
|
+
const storage = new BrowserStorage();
|
|
31
|
+
|
|
32
|
+
// Store data
|
|
33
|
+
await storage.setStorageData('user', { id: 1, name: 'John' });
|
|
46
34
|
|
|
47
|
-
//
|
|
48
|
-
|
|
49
|
-
//
|
|
35
|
+
// Retrieve data
|
|
36
|
+
const user = await storage.getStorageData('user');
|
|
37
|
+
console.log(user); // { id: 1, name: 'John' }
|
|
50
38
|
```
|
|
51
39
|
|
|
52
|
-
###
|
|
40
|
+
### Advanced Configuration
|
|
53
41
|
|
|
54
|
-
```
|
|
42
|
+
```typescript
|
|
55
43
|
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
56
44
|
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
//
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
// Use custom storage implementation
|
|
64
|
-
const customStorage = BrowserStorage.create({
|
|
65
|
-
getItem: (key) => customGet(key),
|
|
66
|
-
setItem: (key, value) => customSet(key, value),
|
|
67
|
-
removeItem: (key) => customRemove(key)
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
Flux.init({
|
|
71
|
-
name: 'my-app',
|
|
72
|
-
stores: [UserStore],
|
|
73
|
-
storage: localStorage, // or sessionStorage, or customStorage
|
|
45
|
+
const storage = new BrowserStorage({
|
|
46
|
+
type: 'local', // 'local' or 'session' storage
|
|
47
|
+
prefix: 'myapp_', // Custom key prefix
|
|
48
|
+
compression: true, // Enable compression for large data
|
|
49
|
+
maxSize: 10 * 1024 * 1024, // 10MB size limit
|
|
50
|
+
ttl: 60 * 60 * 1000 // 1 hour time-to-live
|
|
74
51
|
});
|
|
75
52
|
```
|
|
76
53
|
|
|
77
|
-
|
|
54
|
+
### Integration with ArkhamJS
|
|
78
55
|
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
```js
|
|
56
|
+
```typescript
|
|
57
|
+
import { Flux } from '@nlabs/arkhamjs';
|
|
82
58
|
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
83
59
|
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
// Storage options
|
|
90
|
-
storageWait: 300, // Debounce storage updates (ms)
|
|
91
|
-
storageDebounce: true, // Enable debouncing
|
|
92
|
-
storageThrottle: false, // Use throttling instead of debouncing
|
|
93
|
-
|
|
94
|
-
// Persistence options
|
|
95
|
-
storagePersist: true, // Enable persistence
|
|
96
|
-
storageRestore: true, // Restore state on initialization
|
|
97
|
-
storageClear: false, // Clear storage on initialization
|
|
98
|
-
|
|
99
|
-
// Data options
|
|
100
|
-
storageSerialize: true, // Serialize data before storage
|
|
101
|
-
storageCompress: false, // Compress data before storage
|
|
102
|
-
storageEncrypt: false, // Encrypt data before storage
|
|
103
|
-
|
|
104
|
-
// Key options
|
|
105
|
-
storageKey: 'arkhamjs-state', // Storage key prefix
|
|
106
|
-
storageNamespace: 'my-app', // Namespace for storage keys
|
|
60
|
+
const storage = new BrowserStorage({
|
|
61
|
+
type: 'local',
|
|
62
|
+
prefix: 'myapp_',
|
|
63
|
+
ttl: 24 * 60 * 60 * 1000 // 24 hours
|
|
107
64
|
});
|
|
108
|
-
```
|
|
109
|
-
|
|
110
|
-
### **Advanced Configuration**
|
|
111
65
|
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
stores: [UserStore],
|
|
118
|
-
storage: BrowserStorage,
|
|
119
|
-
|
|
120
|
-
// Selective persistence
|
|
121
|
-
storagePaths: [
|
|
122
|
-
'user.current', // Only persist current user
|
|
123
|
-
'user.preferences', // Only persist user preferences
|
|
124
|
-
'cart.items' // Only persist cart items
|
|
125
|
-
],
|
|
126
|
-
|
|
127
|
-
// Exclude sensitive data
|
|
128
|
-
storageExclude: [
|
|
129
|
-
'user.password', // Don't persist passwords
|
|
130
|
-
'auth.token', // Don't persist auth tokens
|
|
131
|
-
'temp.*' // Don't persist temporary data
|
|
132
|
-
],
|
|
133
|
-
|
|
134
|
-
// Custom serialization
|
|
135
|
-
storageSerialize: (state) => {
|
|
136
|
-
// Custom serialization logic
|
|
137
|
-
return JSON.stringify(state, (key, value) => {
|
|
138
|
-
if (key === 'password') return undefined; // Remove passwords
|
|
139
|
-
if (key === 'token') return undefined; // Remove tokens
|
|
140
|
-
return value;
|
|
141
|
-
});
|
|
142
|
-
},
|
|
143
|
-
|
|
144
|
-
// Custom deserialization
|
|
145
|
-
storageDeserialize: (data) => {
|
|
146
|
-
// Custom deserialization logic
|
|
147
|
-
const state = JSON.parse(data);
|
|
148
|
-
// Add default values or transform data
|
|
149
|
-
return state;
|
|
150
|
-
},
|
|
151
|
-
|
|
152
|
-
// Storage events
|
|
153
|
-
storageEvents: {
|
|
154
|
-
onSave: (key, value) => {
|
|
155
|
-
console.log(`Saved to storage: ${key}`);
|
|
156
|
-
},
|
|
157
|
-
onLoad: (key, value) => {
|
|
158
|
-
console.log(`Loaded from storage: ${key}`);
|
|
159
|
-
},
|
|
160
|
-
onError: (error) => {
|
|
161
|
-
console.error('Storage error:', error);
|
|
162
|
-
}
|
|
163
|
-
}
|
|
66
|
+
// Initialize ArkhamJS with storage
|
|
67
|
+
await Flux.init({
|
|
68
|
+
name: 'myapp',
|
|
69
|
+
storage,
|
|
70
|
+
stores: [/* your stores */]
|
|
164
71
|
});
|
|
165
72
|
```
|
|
166
73
|
|
|
167
|
-
|
|
74
|
+
## API Reference
|
|
168
75
|
|
|
169
|
-
|
|
170
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
76
|
+
### Constructor Options
|
|
171
77
|
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
...(isDevelopment && {
|
|
181
|
-
storageWait: 100,
|
|
182
|
-
storageDebounce: true,
|
|
183
|
-
storageEvents: {
|
|
184
|
-
onSave: (key, value) => console.log(`💾 Saved: ${key}`),
|
|
185
|
-
onLoad: (key, value) => console.log(`📂 Loaded: ${key}`)
|
|
186
|
-
}
|
|
187
|
-
}),
|
|
188
|
-
|
|
189
|
-
// Production: Optimized persistence
|
|
190
|
-
...(!isDevelopment && {
|
|
191
|
-
storageWait: 500,
|
|
192
|
-
storageDebounce: true,
|
|
193
|
-
storageCompress: true,
|
|
194
|
-
storageEvents: {
|
|
195
|
-
onError: (error) => {
|
|
196
|
-
// Send to error tracking service
|
|
197
|
-
analytics.track('storage_error', { error: error.message });
|
|
198
|
-
}
|
|
199
|
-
}
|
|
200
|
-
})
|
|
201
|
-
});
|
|
78
|
+
```typescript
|
|
79
|
+
interface BrowserStorageOptions {
|
|
80
|
+
type?: 'local' | 'session'; // Storage type (default: 'session')
|
|
81
|
+
prefix?: string; // Key prefix (default: 'arkhamjs_')
|
|
82
|
+
compression?: boolean; // Enable compression (default: false)
|
|
83
|
+
maxSize?: number; // Max size in bytes (default: 5MB)
|
|
84
|
+
ttl?: number; // Time-to-live in ms (default: 24h)
|
|
85
|
+
}
|
|
202
86
|
```
|
|
203
87
|
|
|
204
|
-
|
|
88
|
+
### Instance Methods
|
|
205
89
|
|
|
206
|
-
|
|
90
|
+
#### `getStorageData(key: string): Promise<any>`
|
|
207
91
|
|
|
208
|
-
|
|
92
|
+
Retrieves data from storage with caching and TTL validation.
|
|
209
93
|
|
|
210
|
-
```
|
|
211
|
-
|
|
212
|
-
Flux.dispatch({ type: 'USER_LOGIN', user: { id: 1, name: 'John' } });
|
|
213
|
-
|
|
214
|
-
// User closes browser and reopens
|
|
215
|
-
// State is automatically restored from localStorage
|
|
216
|
-
const user = Flux.getState('user.current'); // { id: 1, name: 'John' }
|
|
94
|
+
```typescript
|
|
95
|
+
const data = await storage.getStorageData('user');
|
|
217
96
|
```
|
|
218
97
|
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
Choose which parts of state to persist:
|
|
222
|
-
|
|
223
|
-
```js
|
|
224
|
-
Flux.init({
|
|
225
|
-
name: 'my-app',
|
|
226
|
-
stores: [UserStore],
|
|
227
|
-
storage: BrowserStorage,
|
|
228
|
-
|
|
229
|
-
// Only persist specific paths
|
|
230
|
-
storagePaths: [
|
|
231
|
-
'user.current', // Persist current user
|
|
232
|
-
'user.preferences', // Persist user preferences
|
|
233
|
-
'cart.items', // Persist cart items
|
|
234
|
-
'ui.theme' // Persist UI theme
|
|
235
|
-
],
|
|
236
|
-
|
|
237
|
-
// Exclude sensitive or temporary data
|
|
238
|
-
storageExclude: [
|
|
239
|
-
'user.password', // Don't persist passwords
|
|
240
|
-
'auth.token', // Don't persist auth tokens
|
|
241
|
-
'temp.*', // Don't persist temporary data
|
|
242
|
-
'ui.loading' // Don't persist loading states
|
|
243
|
-
]
|
|
244
|
-
});
|
|
245
|
-
```
|
|
98
|
+
#### `setStorageData(key: string, value: any): Promise<boolean>`
|
|
246
99
|
|
|
247
|
-
|
|
100
|
+
Stores data with validation, compression, and automatic cleanup.
|
|
248
101
|
|
|
249
|
-
|
|
102
|
+
```typescript
|
|
103
|
+
const success = await storage.setStorageData('user', { id: 1, name: 'John' });
|
|
104
|
+
```
|
|
250
105
|
|
|
251
|
-
|
|
252
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
106
|
+
#### `removeStorageData(key: string): Promise<boolean>`
|
|
253
107
|
|
|
254
|
-
|
|
255
|
-
name: 'my-app',
|
|
256
|
-
stores: [UserStore],
|
|
257
|
-
storage: BrowserStorage,
|
|
258
|
-
|
|
259
|
-
// Enable cross-tab sync
|
|
260
|
-
storageSync: true,
|
|
261
|
-
|
|
262
|
-
// Custom sync events
|
|
263
|
-
storageEvents: {
|
|
264
|
-
onSync: (event) => {
|
|
265
|
-
if (event.key === 'arkhamjs-state') {
|
|
266
|
-
console.log('State synced from another tab');
|
|
267
|
-
// Optionally refresh UI or show notification
|
|
268
|
-
}
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
});
|
|
108
|
+
Removes specific data from storage.
|
|
272
109
|
|
|
273
|
-
|
|
110
|
+
```typescript
|
|
111
|
+
const success = await storage.removeStorageData('user');
|
|
274
112
|
```
|
|
275
113
|
|
|
276
|
-
|
|
114
|
+
#### `clearStorageData(): Promise<boolean>`
|
|
277
115
|
|
|
278
|
-
|
|
116
|
+
Clears all data with the configured prefix.
|
|
279
117
|
|
|
280
|
-
```
|
|
281
|
-
|
|
118
|
+
```typescript
|
|
119
|
+
const success = await storage.clearStorageData();
|
|
120
|
+
```
|
|
282
121
|
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
storageCompressThreshold: 1024, // Compress if >1KB
|
|
291
|
-
|
|
292
|
-
// Custom compression
|
|
293
|
-
storageCompress: (data) => {
|
|
294
|
-
// Use custom compression library
|
|
295
|
-
return customCompress(data);
|
|
296
|
-
},
|
|
297
|
-
|
|
298
|
-
// Custom decompression
|
|
299
|
-
storageDecompress: (data) => {
|
|
300
|
-
// Use custom decompression library
|
|
301
|
-
return customDecompress(data);
|
|
302
|
-
}
|
|
303
|
-
});
|
|
122
|
+
#### `getStorageStats(): { used: number; available: number; total: number }`
|
|
123
|
+
|
|
124
|
+
Returns storage usage statistics.
|
|
125
|
+
|
|
126
|
+
```typescript
|
|
127
|
+
const stats = storage.getStorageStats();
|
|
128
|
+
console.log(`Used: ${stats.used} bytes, Available: ${stats.available} bytes`);
|
|
304
129
|
```
|
|
305
130
|
|
|
306
|
-
###
|
|
131
|
+
### Static Methods (Backward Compatibility)
|
|
307
132
|
|
|
308
|
-
|
|
133
|
+
For backward compatibility, static methods are still available:
|
|
309
134
|
|
|
310
|
-
```
|
|
311
|
-
|
|
135
|
+
```typescript
|
|
136
|
+
// Local storage
|
|
137
|
+
BrowserStorage.setLocalData('key', value);
|
|
138
|
+
const data = BrowserStorage.getLocalData('key');
|
|
139
|
+
BrowserStorage.delLocalData('key');
|
|
312
140
|
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
// Custom encryption
|
|
323
|
-
storageEncrypt: (data, key) => {
|
|
324
|
-
// Use custom encryption library
|
|
325
|
-
return customEncrypt(data, key);
|
|
326
|
-
},
|
|
327
|
-
|
|
328
|
-
// Custom decryption
|
|
329
|
-
storageDecrypt: (data, key) => {
|
|
330
|
-
// Use custom decryption library
|
|
331
|
-
return customDecrypt(data, key);
|
|
332
|
-
}
|
|
333
|
-
});
|
|
141
|
+
// Session storage
|
|
142
|
+
BrowserStorage.setSessionData('key', value);
|
|
143
|
+
const data = BrowserStorage.getSessionData('key');
|
|
144
|
+
BrowserStorage.delSessionData('key');
|
|
145
|
+
|
|
146
|
+
// Storage instances
|
|
147
|
+
const localStorage = BrowserStorage.getLocalStorage();
|
|
148
|
+
const sessionStorage = BrowserStorage.getSessionStorage();
|
|
334
149
|
```
|
|
335
150
|
|
|
336
|
-
##
|
|
151
|
+
## Performance Optimizations
|
|
337
152
|
|
|
338
|
-
###
|
|
153
|
+
### In-Memory Caching
|
|
339
154
|
|
|
340
|
-
|
|
341
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
155
|
+
The storage automatically caches frequently accessed data in memory for faster retrieval:
|
|
342
156
|
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
getItem: (key) => {
|
|
347
|
-
// Custom get implementation
|
|
348
|
-
return localStorage.getItem(key);
|
|
349
|
-
},
|
|
350
|
-
|
|
351
|
-
setItem: (key, value) => {
|
|
352
|
-
// Custom set implementation
|
|
353
|
-
localStorage.setItem(key, value);
|
|
354
|
-
},
|
|
355
|
-
|
|
356
|
-
removeItem: (key) => {
|
|
357
|
-
// Custom remove implementation
|
|
358
|
-
localStorage.removeItem(key);
|
|
359
|
-
},
|
|
360
|
-
|
|
361
|
-
// Optional methods
|
|
362
|
-
clear: () => {
|
|
363
|
-
// Custom clear implementation
|
|
364
|
-
localStorage.clear();
|
|
365
|
-
},
|
|
366
|
-
|
|
367
|
-
key: (index) => {
|
|
368
|
-
// Custom key implementation
|
|
369
|
-
return localStorage.key(index);
|
|
370
|
-
},
|
|
371
|
-
|
|
372
|
-
get length() {
|
|
373
|
-
// Custom length implementation
|
|
374
|
-
return localStorage.length;
|
|
375
|
-
}
|
|
376
|
-
});
|
|
157
|
+
```typescript
|
|
158
|
+
// First call - reads from storage
|
|
159
|
+
const user1 = await storage.getStorageData('user');
|
|
377
160
|
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
stores: [UserStore],
|
|
381
|
-
storage: customStorage
|
|
382
|
-
});
|
|
161
|
+
// Second call - served from cache (much faster)
|
|
162
|
+
const user2 = await storage.getStorageData('user');
|
|
383
163
|
```
|
|
384
164
|
|
|
385
|
-
###
|
|
165
|
+
### Automatic Cleanup
|
|
386
166
|
|
|
387
|
-
|
|
388
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
167
|
+
Expired data is automatically cleaned up to prevent storage bloat:
|
|
389
168
|
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
storage: BrowserStorage,
|
|
394
|
-
|
|
395
|
-
// Storage migration
|
|
396
|
-
storageMigrate: (oldData, newData) => {
|
|
397
|
-
// Migrate from old format to new format
|
|
398
|
-
if (oldData.version === 1) {
|
|
399
|
-
return {
|
|
400
|
-
...newData,
|
|
401
|
-
user: {
|
|
402
|
-
...newData.user,
|
|
403
|
-
// Migrate old user format
|
|
404
|
-
current: oldData.user ? { ...oldData.user, id: oldData.user.id || 1 } : null
|
|
405
|
-
}
|
|
406
|
-
};
|
|
407
|
-
}
|
|
408
|
-
return newData;
|
|
409
|
-
},
|
|
410
|
-
|
|
411
|
-
// Version tracking
|
|
412
|
-
storageVersion: 2,
|
|
413
|
-
|
|
414
|
-
// Migration events
|
|
415
|
-
storageEvents: {
|
|
416
|
-
onMigrate: (oldVersion, newVersion) => {
|
|
417
|
-
console.log(`Migrated from v${oldVersion} to v${newVersion}`);
|
|
418
|
-
}
|
|
419
|
-
}
|
|
169
|
+
```typescript
|
|
170
|
+
const storage = new BrowserStorage({
|
|
171
|
+
ttl: 60 * 60 * 1000 // 1 hour
|
|
420
172
|
});
|
|
173
|
+
|
|
174
|
+
// Data will be automatically removed after 1 hour
|
|
175
|
+
await storage.setStorageData('temp', 'data');
|
|
421
176
|
```
|
|
422
177
|
|
|
423
|
-
###
|
|
178
|
+
### Size Validation
|
|
424
179
|
|
|
425
|
-
|
|
426
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
180
|
+
Large data is validated before storage to prevent quota errors:
|
|
427
181
|
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
storage: BrowserStorage,
|
|
432
|
-
|
|
433
|
-
// Storage analytics
|
|
434
|
-
storageEvents: {
|
|
435
|
-
onSave: (key, value) => {
|
|
436
|
-
// Track storage usage
|
|
437
|
-
analytics.track('storage_save', {
|
|
438
|
-
key,
|
|
439
|
-
size: JSON.stringify(value).length,
|
|
440
|
-
timestamp: Date.now()
|
|
441
|
-
});
|
|
442
|
-
},
|
|
443
|
-
|
|
444
|
-
onLoad: (key, value) => {
|
|
445
|
-
// Track storage reads
|
|
446
|
-
analytics.track('storage_load', {
|
|
447
|
-
key,
|
|
448
|
-
size: JSON.stringify(value).length,
|
|
449
|
-
timestamp: Date.now()
|
|
450
|
-
});
|
|
451
|
-
},
|
|
452
|
-
|
|
453
|
-
onError: (error) => {
|
|
454
|
-
// Track storage errors
|
|
455
|
-
analytics.track('storage_error', {
|
|
456
|
-
error: error.message,
|
|
457
|
-
timestamp: Date.now()
|
|
458
|
-
});
|
|
459
|
-
}
|
|
460
|
-
}
|
|
182
|
+
```typescript
|
|
183
|
+
const storage = new BrowserStorage({
|
|
184
|
+
maxSize: 1024 * 1024 // 1MB limit
|
|
461
185
|
});
|
|
462
|
-
```
|
|
463
186
|
|
|
464
|
-
|
|
187
|
+
// This will fail if data exceeds 1MB
|
|
188
|
+
const success = await storage.setStorageData('large', bigData);
|
|
189
|
+
if (!success) {
|
|
190
|
+
console.log('Data too large for storage');
|
|
191
|
+
}
|
|
192
|
+
```
|
|
465
193
|
|
|
466
|
-
|
|
194
|
+
## Error Handling
|
|
467
195
|
|
|
468
|
-
|
|
469
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
196
|
+
The storage gracefully handles various error conditions:
|
|
470
197
|
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
'user.preferences.notifications',
|
|
481
|
-
'ui.sidebar.collapsed'
|
|
482
|
-
]
|
|
483
|
-
});
|
|
484
|
-
|
|
485
|
-
// User preferences will persist across sessions
|
|
486
|
-
Flux.dispatch({ type: 'SET_THEME', theme: 'dark' });
|
|
487
|
-
// Theme preference is automatically saved
|
|
198
|
+
```typescript
|
|
199
|
+
try {
|
|
200
|
+
const data = await storage.getStorageData('key');
|
|
201
|
+
if (data === null) {
|
|
202
|
+
console.log('Data not found or expired');
|
|
203
|
+
}
|
|
204
|
+
} catch (error) {
|
|
205
|
+
console.error('Storage error:', error);
|
|
206
|
+
}
|
|
488
207
|
```
|
|
489
208
|
|
|
490
|
-
|
|
209
|
+
Common error scenarios handled automatically:
|
|
491
210
|
|
|
492
|
-
|
|
493
|
-
|
|
211
|
+
- Storage not available (private browsing, etc.)
|
|
212
|
+
- Quota exceeded
|
|
213
|
+
- Corrupted data
|
|
214
|
+
- Invalid JSON
|
|
494
215
|
|
|
495
|
-
|
|
496
|
-
name: 'my-app',
|
|
497
|
-
stores: [CartStore],
|
|
498
|
-
storage: BrowserStorage,
|
|
216
|
+
## Migration Guide
|
|
499
217
|
|
|
500
|
-
|
|
501
|
-
storagePaths: ['cart.items', 'cart.total'],
|
|
218
|
+
### From Previous Version
|
|
502
219
|
|
|
503
|
-
|
|
504
|
-
storageExclude: ['cart.loading', 'cart.error']
|
|
505
|
-
});
|
|
220
|
+
The new version is fully backward compatible. Existing code will continue to work:
|
|
506
221
|
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
222
|
+
```typescript
|
|
223
|
+
// Old code - still works
|
|
224
|
+
const storage = new BrowserStorage({ type: 'session' });
|
|
225
|
+
await storage.setStorageData('key', value);
|
|
226
|
+
const data = await storage.getStorageData('key');
|
|
227
|
+
|
|
228
|
+
// New features available
|
|
229
|
+
const stats = storage.getStorageStats();
|
|
230
|
+
await storage.removeStorageData('key');
|
|
231
|
+
await storage.clearStorageData();
|
|
510
232
|
```
|
|
511
233
|
|
|
512
|
-
###
|
|
234
|
+
### Recommended Updates
|
|
513
235
|
|
|
514
|
-
|
|
515
|
-
import { BrowserStorage } from '@nlabs/arkhamjs-storage-browser';
|
|
236
|
+
For better performance, consider these updates:
|
|
516
237
|
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
storage: BrowserStorage,
|
|
521
|
-
|
|
522
|
-
// Persist form data with short debounce
|
|
523
|
-
storageWait: 100,
|
|
524
|
-
storagePaths: ['form.draft'],
|
|
525
|
-
|
|
526
|
-
// Clear form data on successful submission
|
|
527
|
-
storageEvents: {
|
|
528
|
-
onAction: (action) => {
|
|
529
|
-
if (action.type === 'FORM_SUBMIT_SUCCESS') {
|
|
530
|
-
// Clear draft data
|
|
531
|
-
localStorage.removeItem('arkhamjs-state-form.draft');
|
|
532
|
-
}
|
|
533
|
-
}
|
|
534
|
-
}
|
|
535
|
-
});
|
|
238
|
+
```typescript
|
|
239
|
+
// Before
|
|
240
|
+
const storage = new BrowserStorage({ type: 'session' });
|
|
536
241
|
|
|
537
|
-
//
|
|
538
|
-
|
|
242
|
+
// After - with optimizations
|
|
243
|
+
const storage = new BrowserStorage({
|
|
244
|
+
type: 'local', // Use localStorage for persistence
|
|
245
|
+
prefix: 'myapp_', // Custom prefix for organization
|
|
246
|
+
ttl: 24 * 60 * 60 * 1000, // 24 hour TTL
|
|
247
|
+
maxSize: 10 * 1024 * 1024 // 10MB limit
|
|
248
|
+
});
|
|
539
249
|
```
|
|
540
250
|
|
|
541
|
-
##
|
|
251
|
+
## Browser Support
|
|
542
252
|
|
|
543
|
-
-
|
|
544
|
-
-
|
|
545
|
-
-
|
|
253
|
+
- Chrome 60+
|
|
254
|
+
- Firefox 55+
|
|
255
|
+
- Safari 12+
|
|
256
|
+
- Edge 79+
|
|
546
257
|
|
|
547
|
-
##
|
|
258
|
+
## Bundle Size
|
|
548
259
|
|
|
549
|
-
|
|
260
|
+
- **Minified**: ~3.2KB
|
|
261
|
+
- **Gzipped**: ~1.1KB
|
|
262
|
+
- **Tree-shakeable**: Only includes what you use
|
|
550
263
|
|
|
551
|
-
##
|
|
264
|
+
## Contributing
|
|
552
265
|
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
266
|
+
1. Fork the repository
|
|
267
|
+
2. Create a feature branch
|
|
268
|
+
3. Make your changes
|
|
269
|
+
4. Add tests
|
|
270
|
+
5. Submit a pull request
|
|
556
271
|
|
|
557
|
-
##
|
|
272
|
+
## License
|
|
558
273
|
|
|
559
|
-
MIT License - see [LICENSE](
|
|
274
|
+
MIT License - see [LICENSE](LICENSE) file for details.
|
package/index.js
CHANGED
|
@@ -2,6 +2,6 @@
|
|
|
2
2
|
* Copyright (c) 2018-Present, Nitrogen Labs, Inc.
|
|
3
3
|
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
4
|
*/
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
5
|
+
import * as lib from './lib/index.js';
|
|
6
|
+
export default lib;
|
|
7
|
+
export * from './lib/index.js';
|
|
@@ -1,16 +1,34 @@
|
|
|
1
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2018-Present, Nitrogen Labs, Inc.
|
|
3
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
|
+
*/
|
|
5
|
+
import type { BrowserStorageOptions } from '../types/main';
|
|
2
6
|
export declare class BrowserStorage {
|
|
3
|
-
static window
|
|
4
|
-
private options;
|
|
7
|
+
private static readonly window;
|
|
8
|
+
private readonly options;
|
|
9
|
+
private readonly storageCache;
|
|
10
|
+
private readonly storage;
|
|
5
11
|
constructor(options?: BrowserStorageOptions);
|
|
6
12
|
static delLocalData(key: string): boolean;
|
|
7
13
|
static delSessionData(key: string): boolean;
|
|
8
14
|
static getLocalData(key: string): any;
|
|
9
|
-
static getLocalStorage(): any;
|
|
10
15
|
static getSessionData(key: string): any;
|
|
11
|
-
static getSessionStorage(): any;
|
|
12
16
|
static setLocalData(key: string, value: any): boolean;
|
|
13
17
|
static setSessionData(key: string, value: any): boolean;
|
|
18
|
+
static getLocalStorage(): Storage | null;
|
|
19
|
+
static getSessionStorage(): Storage | null;
|
|
14
20
|
getStorageData(key: string): Promise<any>;
|
|
15
21
|
setStorageData(key: string, value: any): Promise<boolean>;
|
|
22
|
+
removeStorageData(key: string): Promise<boolean>;
|
|
23
|
+
clearStorageData(): Promise<boolean>;
|
|
24
|
+
getStorageStats(): {
|
|
25
|
+
available: number;
|
|
26
|
+
total: number;
|
|
27
|
+
used: number;
|
|
28
|
+
};
|
|
29
|
+
private getStorage;
|
|
30
|
+
private getPrefixedKey;
|
|
31
|
+
private isStorageAvailable;
|
|
32
|
+
private validateSize;
|
|
33
|
+
private cleanExpiredData;
|
|
16
34
|
}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/BrowserStorage/BrowserStorage.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {BrowserStorageOptions} from '../types/main';\n\nexport class BrowserStorage {\n  static window: any = window || {};\n  private options: BrowserStorageOptions = {\n    type: 'session'\n  };\n\n  constructor(options: BrowserStorageOptions = {}) {\n    // Methods\n    this.getStorageData = this.getStorageData.bind(this);\n    this.setStorageData = this.setStorageData.bind(this);\n\n    // Configuration\n    this.options = {...this.options, ...options};\n  }\n\n  /**\n   * Removes a key from localStorage.\n   *\n   * @param {string} key Key associated with the data to remove.\n   * @returns {boolean} Whether data was successfully removed.\n   */\n  static delLocalData(key: string): boolean {\n    const localStorage = BrowserStorage.getLocalStorage();\n\n    if(localStorage) {\n      try {\n        localStorage.removeItem(key);\n        return true;\n      } catch(error) {\n        return false;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Removes a key from sessionStorage.\n   *\n   * @param {string} key Key associated with the data to remove.\n   * @returns {boolean} Whether data was successfully removed.\n   */\n  static delSessionData(key: string): boolean {\n    const sessionStorage = BrowserStorage.getSessionStorage();\n\n    if(sessionStorage) {\n      try {\n        sessionStorage.removeItem(key);\n        return true;\n      } catch(error) {\n        return false;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Get a key value from localStorage.\n   *\n   * @param {string} key The key for data.\n   * @returns {any} the data object associated with the key.\n   */\n  static getLocalData(key: string): any {\n    const localStorage = BrowserStorage.getLocalStorage();\n\n    if(localStorage) {\n      try {\n        const item = localStorage.getItem(key);\n\n        if(item) {\n          return JSON.parse(item);\n        }\n\n        return null;\n      } catch(error) {\n        return null;\n      }\n    } else {\n      return null;\n    }\n  }\n\n  /**\n   * Get localStorage from global window object.\n   *\n   * @param {string} key Key to store data.\n   * @param {any} value Data to store.\n   * @returns {any} window.localStorage.\n   */\n  static getLocalStorage(): any {\n    const {localStorage} = BrowserStorage.window;\n    return localStorage;\n  }\n\n  /**\n   * Get a key value from sessionStorage.\n   *\n   * @param {string} key The key for data.\n   * @returns {any} the data object associated with the key.\n   */\n  static getSessionData(key: string): any {\n    const sessionStorage = BrowserStorage.getSessionStorage();\n\n    if(sessionStorage) {\n      try {\n        const item = sessionStorage.getItem(key);\n\n        if(item) {\n          return item ? JSON.parse(item) : null;\n        }\n\n        return null;\n      } catch(error) {\n        return null;\n      }\n    } else {\n      return null;\n    }\n  }\n\n  /**\n   * Get sessionStorage from global window object.\n   *\n   * @param {string} key Key to store data.\n   * @param {any} value Data to store.\n   * @returns {any} window.sessionStorage.\n   */\n  static getSessionStorage(): any {\n    const {sessionStorage} = BrowserStorage.window;\n\n    return sessionStorage;\n  }\n\n  /**\n   * Saves data to localStorage.\n   *\n   * @param {string} key Key to store data.\n   * @param {any} value Data to store.\n   * @returns {boolean} Whether data was successfully saved.\n   */\n  static setLocalData(key: string, value): boolean {\n    const localStorage = BrowserStorage.getLocalStorage();\n\n    if(localStorage) {\n      try {\n        localStorage.setItem(key, JSON.stringify(value));\n        return true;\n      } catch(error) {\n        return false;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Saves data to sessionStorage.\n   *\n   * @param {string} key Key to store data.\n   * @param {any} value Data to store.\n   * @returns {boolean} Whether data was successfully saved.\n   */\n  static setSessionData(key: string, value): boolean {\n    const sessionStorage = BrowserStorage.getSessionStorage();\n\n    if(sessionStorage) {\n      try {\n        sessionStorage.setItem(key, JSON.stringify(value));\n        return true;\n      } catch(error) {\n        return false;\n      }\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Get a key value from storage.\n   *\n   * @param {string} key The key for data.\n   * @returns {Promise<any>} the data object associated with the key.\n   */\n  getStorageData(key: string): Promise<any> {\n    const {type} = this.options;\n    const results = type === 'local' ? BrowserStorage.getLocalData(key) : BrowserStorage.getSessionData(key);\n    return Promise.resolve(results);\n  }\n\n  /**\n   * Saves data to storage.\n   *\n   * @param {string} key Key to store data.\n   * @param {any} value Data to store.\n   * @returns {Promise<boolean>} Whether data was successfully saved.\n   */\n  setStorageData(key: string, value): Promise<boolean> {\n    const {type} = this.options;\n    const results: boolean = type === 'local' ?\n      BrowserStorage.setLocalData(key, value) :\n      BrowserStorage.setSessionData(key, value);\n    return Promise.resolve(results);\n  }\n}\n"],
  "mappings": "4ZAAA,IAAAA,EAAA,GAAAC,EAAAD,EAAA,oBAAAE,IAAA,eAAAC,EAAAH,GAMO,MAAME,CAAe,CAM1B,YAAYE,EAAiC,CAAC,EAAG,CAJjD,KAAQ,QAAiC,CACvC,KAAM,SACR,EAIE,KAAK,eAAiB,KAAK,eAAe,KAAK,IAAI,EACnD,KAAK,eAAiB,KAAK,eAAe,KAAK,IAAI,EAGnD,KAAK,QAAU,CAAC,GAAG,KAAK,QAAS,GAAGA,CAAO,CAC7C,CAZA,YAAO,OAAc,QAAU,CAAC,EAoBhC,OAAO,aAAaC,EAAsB,CACxC,MAAMC,EAAeJ,EAAe,gBAAgB,EAEpD,GAAGI,EACD,GAAI,CACF,OAAAA,EAAa,WAAWD,CAAG,EACpB,EACT,MAAe,CACb,MAAO,EACT,KAEA,OAAO,EAEX,CAQA,OAAO,eAAeA,EAAsB,CAC1C,MAAME,EAAiBL,EAAe,kBAAkB,EAExD,GAAGK,EACD,GAAI,CACF,OAAAA,EAAe,WAAWF,CAAG,EACtB,EACT,MAAe,CACb,MAAO,EACT,KAEA,OAAO,EAEX,CAQA,OAAO,aAAaA,EAAkB,CACpC,MAAMC,EAAeJ,EAAe,gBAAgB,EAEpD,GAAGI,EACD,GAAI,CACF,MAAME,EAAOF,EAAa,QAAQD,CAAG,EAErC,OAAGG,EACM,KAAK,MAAMA,CAAI,EAGjB,IACT,MAAe,CACb,OAAO,IACT,KAEA,QAAO,IAEX,CASA,OAAO,iBAAuB,CAC5B,KAAM,CAAC,aAAAF,CAAY,EAAIJ,EAAe,OACtC,OAAOI,CACT,CAQA,OAAO,eAAeD,EAAkB,CACtC,MAAME,EAAiBL,EAAe,kBAAkB,EAExD,GAAGK,EACD,GAAI,CACF,MAAMC,EAAOD,EAAe,QAAQF,CAAG,EAEvC,OAAGG,GACMA,EAAO,KAAK,MAAMA,CAAI,EAGxB,IACT,MAAe,CACb,OAAO,IACT,KAEA,QAAO,IAEX,CASA,OAAO,mBAAyB,CAC9B,KAAM,CAAC,eAAAD,CAAc,EAAIL,EAAe,OAExC,OAAOK,CACT,CASA,OAAO,aAAaF,EAAaI,EAAgB,CAC/C,MAAMH,EAAeJ,EAAe,gBAAgB,EAEpD,GAAGI,EACD,GAAI,CACF,OAAAA,EAAa,QAAQD,EAAK,KAAK,UAAUI,CAAK,CAAC,EACxC,EACT,MAAe,CACb,MAAO,EACT,KAEA,OAAO,EAEX,CASA,OAAO,eAAeJ,EAAaI,EAAgB,CACjD,MAAMF,EAAiBL,EAAe,kBAAkB,EAExD,GAAGK,EACD,GAAI,CACF,OAAAA,EAAe,QAAQF,EAAK,KAAK,UAAUI,CAAK,CAAC,EAC1C,EACT,MAAe,CACb,MAAO,EACT,KAEA,OAAO,EAEX,CAQA,eAAeJ,EAA2B,CACxC,KAAM,CAAC,KAAAK,CAAI,EAAI,KAAK,QACdC,EAAUD,IAAS,QAAUR,EAAe,aAAaG,CAAG,EAAIH,EAAe,eAAeG,CAAG,EACvG,OAAO,QAAQ,QAAQM,CAAO,CAChC,CASA,eAAeN,EAAaI,EAAyB,CACnD,KAAM,CAAC,KAAAC,CAAI,EAAI,KAAK,QACdC,EAAmBD,IAAS,QAChCR,EAAe,aAAaG,EAAKI,CAAK,EACtCP,EAAe,eAAeG,EAAKI,CAAK,EAC1C,OAAO,QAAQ,QAAQE,CAAO,CAChC,CACF",
  "names": ["BrowserStorage_exports", "__export", "BrowserStorage", "__toCommonJS", "options", "key", "localStorage", "sessionStorage", "item", "value", "type", "results"]
}

|
|
1
|
+
const c="arkhamjs_";const n=i=>i.ttl?Date.now()-i.timestamp>i.ttl:!1,l=i=>new Blob([i]).size,h=i=>{const e=JSON.stringify(i);return e.length>1e3,e},g=i=>{try{return JSON.parse(i)}catch{return null}};class a{static window=typeof window<"u"?window:{};options;storageCache=new Map;storage;constructor(e={}){this.options={compression:!1,maxSize:5242880,prefix:c,ttl:864e5,type:"session",...e},this.storage=this.getStorage()}static delLocalData(e){try{const t=a.window.localStorage;if(t)return t.removeItem(e),!0}catch{}return!1}static delSessionData(e){try{const t=a.window.sessionStorage;if(t)return t.removeItem(e),!0}catch{}return!1}static getLocalData(e){try{const t=a.window.localStorage;if(t){const r=t.getItem(e);return r?JSON.parse(r):null}}catch{}return null}static getSessionData(e){try{const t=a.window.sessionStorage;if(t){const r=t.getItem(e);return r?JSON.parse(r):null}}catch{}return null}static setLocalData(e,t){try{const r=a.window.localStorage;if(r)return r.setItem(e,JSON.stringify(t)),!0}catch{}return!1}static setSessionData(e,t){try{const r=a.window.sessionStorage;if(r)return r.setItem(e,JSON.stringify(t)),!0}catch{}return!1}static getLocalStorage(){try{return a.window.localStorage||null}catch{return null}}static getSessionStorage(){try{return a.window.sessionStorage||null}catch{return null}}async getStorageData(e){if(!this.isStorageAvailable())return null;const t=this.getPrefixedKey(e);if(this.storageCache.has(t))return this.storageCache.get(t);try{const r=this.storage.getItem(t);if(!r)return null;const s=g(r);return s?n(s)?(this.storage.removeItem(t),this.storageCache.delete(t),null):(this.storageCache.set(t,s.value),s.value):null}catch{return null}}async setStorageData(e,t){if(!this.isStorageAvailable())return!1;const r=this.getPrefixedKey(e);try{Math.random()<.1&&this.cleanExpiredData();const s={timestamp:Date.now(),ttl:this.options.ttl,value:t},o=this.options.compression?h(s):JSON.stringify(s);return this.validateSize(o)?(this.storage.setItem(r,o),this.storageCache.set(r,t),!0):!1}catch{return!1}}async removeStorageData(e){if(!this.isStorageAvailable())return!1;const t=this.getPrefixedKey(e);try{return this.storage.removeItem(t),this.storageCache.delete(t),!0}catch{return!1}}async clearStorageData(){if(!this.isStorageAvailable())return!1;try{const e=[];for(let t=0;t<this.storage.length;t++){const r=this.storage.key(t);r?.startsWith(this.options.prefix)&&e.push(r)}return e.forEach(t=>{this.storage.removeItem(t),this.storageCache.delete(t)}),!0}catch{return!1}}getStorageStats(){if(!this.isStorageAvailable())return{available:0,total:0,used:0};let e=0;const t=[];for(let r=0;r<this.storage.length;r++){const s=this.storage.key(r);if(s?.startsWith(this.options.prefix)){t.push(s);const o=this.storage.getItem(s);o&&(e+=l(o))}}return{available:this.options.maxSize-e,total:this.options.maxSize,used:e}}getStorage(){try{const{type:e}=this.options;return(e==="local"?a.window.localStorage:a.window.sessionStorage)||null}catch{return null}}getPrefixedKey(e){return`${this.options.prefix}${e}`}isStorageAvailable(){return this.storage!==null}validateSize(e){return l(e)<=this.options.maxSize}cleanExpiredData(){if(!this.isStorageAvailable())return;const e=[];for(let t=0;t<this.storage.length;t++){const r=this.storage.key(t);r?.startsWith(this.options.prefix)&&e.push(r)}e.forEach(t=>{try{const r=this.storage.getItem(t);if(r){const s=g(r);s&&n(s)&&(this.storage.removeItem(t),this.storageCache.delete(t))}}catch{this.storage.removeItem(t),this.storageCache.delete(t)}})}}export{a as BrowserStorage};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/BrowserStorage/BrowserStorage.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport type {\n  BrowserStorageOptions,\n  StorageData,\n  StorageInterface\n} from '../types/main';\n\n// Constants for optimization\nconst DEFAULT_PREFIX = 'arkhamjs_';\nconst DEFAULT_MAX_SIZE = 5 * 1024 * 1024; // 5MB\nconst DEFAULT_TTL = 24 * 60 * 60 * 1000; // 24 hours\n\n// Utility functions\nconst isExpired = (data: StorageData): boolean => {\n  if(!data.ttl) {\n    return false;\n  }\n  return Date.now() - data.timestamp > data.ttl;\n};\n\nconst getStorageSize = (data: string): number => new Blob([data]).size;\n\nconst compressData = (data: any): string => {\n  // Simple compression for large objects\n  const jsonString = JSON.stringify(data);\n  if(jsonString.length > 1000) {\n    // For large data, we could implement actual compression here\n    // For now, we'll use a simple approach\n    return jsonString;\n  }\n  return jsonString;\n};\n\nconst decompressData = (data: string): any => {\n  try {\n    return JSON.parse(data);\n  } catch{\n    return null;\n  }\n};\n\nexport class BrowserStorage {\n  private static readonly window: Window & typeof globalThis =\n    typeof window !== 'undefined' ? window : {} as any;\n\n  private readonly options: Required<BrowserStorageOptions>;\n  private readonly storageCache = new Map<string, any>();\n  private readonly storage: StorageInterface | null;\n\n  constructor(options: BrowserStorageOptions = {}) {\n    this.options = {\n      compression: false,\n      maxSize: DEFAULT_MAX_SIZE,\n      prefix: DEFAULT_PREFIX,\n      ttl: DEFAULT_TTL,\n      type: 'session',\n      ...options\n    };\n\n    this.storage = this.getStorage();\n  }\n\n  // Static methods for backward compatibility\n  static delLocalData(key: string): boolean {\n    try {\n      const storage = BrowserStorage.window.localStorage;\n      if(storage) {\n        storage.removeItem(key);\n        return true;\n      }\n    } catch{\n      // Storage not available\n    }\n    return false;\n  }\n\n  static delSessionData(key: string): boolean {\n    try {\n      const storage = BrowserStorage.window.sessionStorage;\n      if(storage) {\n        storage.removeItem(key);\n        return true;\n      }\n    } catch{\n      // Storage not available\n    }\n    return false;\n  }\n\n  static getLocalData(key: string): any {\n    try {\n      const storage = BrowserStorage.window.localStorage;\n      if(storage) {\n        const item = storage.getItem(key);\n        return item ? JSON.parse(item) : null;\n      }\n    } catch{\n      // Storage not available or invalid JSON\n    }\n    return null;\n  }\n\n  static getSessionData(key: string): any {\n    try {\n      const storage = BrowserStorage.window.sessionStorage;\n      if(storage) {\n        const item = storage.getItem(key);\n        return item ? JSON.parse(item) : null;\n      }\n    } catch{\n      // Storage not available or invalid JSON\n    }\n    return null;\n  }\n\n  static setLocalData(key: string, value: any): boolean {\n    try {\n      const storage = BrowserStorage.window.localStorage;\n      if(storage) {\n        storage.setItem(key, JSON.stringify(value));\n        return true;\n      }\n    } catch{\n      // Storage not available or quota exceeded\n    }\n    return false;\n  }\n\n  static setSessionData(key: string, value: any): boolean {\n    try {\n      const storage = BrowserStorage.window.sessionStorage;\n      if(storage) {\n        storage.setItem(key, JSON.stringify(value));\n        return true;\n      }\n    } catch{\n      // Storage not available or quota exceeded\n    }\n    return false;\n  }\n\n  static getLocalStorage(): Storage | null {\n    try {\n      return BrowserStorage.window.localStorage || null;\n    } catch{\n      return null;\n    }\n  }\n\n  static getSessionStorage(): Storage | null {\n    try {\n      return BrowserStorage.window.sessionStorage || null;\n    } catch{\n      return null;\n    }\n  }\n\n  // Public instance methods\n  async getStorageData(key: string): Promise<any> {\n    if(!this.isStorageAvailable()) {\n      return null;\n    }\n\n    const prefixedKey = this.getPrefixedKey(key);\n\n    // Check cache first\n    if(this.storageCache.has(prefixedKey)) {\n      return this.storageCache.get(prefixedKey);\n    }\n\n    try {\n      const item = this.storage!.getItem(prefixedKey);\n      if(!item) {\n        return null;\n      }\n\n      const data = decompressData(item);\n      if(!data) {\n        return null;\n      }\n\n      // Check if data is expired\n      if(isExpired(data)) {\n        this.storage!.removeItem(prefixedKey);\n        this.storageCache.delete(prefixedKey);\n        return null;\n      }\n\n      // Cache the result\n      this.storageCache.set(prefixedKey, data.value);\n      return data.value;\n    } catch{\n      return null;\n    }\n  }\n\n  async setStorageData(key: string, value: any): Promise<boolean> {\n    if(!this.isStorageAvailable()) {\n      return false;\n    }\n\n    const prefixedKey = this.getPrefixedKey(key);\n\n    try {\n      // Clean expired data periodically\n      if(Math.random() < 0.1) { // 10% chance to clean\n        this.cleanExpiredData();\n      }\n\n      const storageData: StorageData = {\n        timestamp: Date.now(),\n        ttl: this.options.ttl,\n        value\n      };\n\n      const jsonString = this.options.compression\n        ? compressData(storageData)\n        : JSON.stringify(storageData);\n\n      // Validate size\n      if(!this.validateSize(jsonString)) {\n        // eslint-disable-next-line no-console\n        console.warn(`Storage data exceeds maximum size for key: ${key}`);\n        return false;\n      }\n\n      this.storage!.setItem(prefixedKey, jsonString);\n\n      // Update cache\n      this.storageCache.set(prefixedKey, value);\n\n      return true;\n    } catch(error) {\n      // eslint-disable-next-line no-console\n      console.error('Failed to set storage data:', error);\n      return false;\n    }\n  }\n\n  async removeStorageData(key: string): Promise<boolean> {\n    if(!this.isStorageAvailable()) {\n      return false;\n    }\n\n    const prefixedKey = this.getPrefixedKey(key);\n\n    try {\n      this.storage!.removeItem(prefixedKey);\n      this.storageCache.delete(prefixedKey);\n      return true;\n    } catch{\n      return false;\n    }\n  }\n\n  async clearStorageData(): Promise<boolean> {\n    if(!this.isStorageAvailable()) {\n      return false;\n    }\n\n    try {\n      const keys: string[] = [];\n      for(let i = 0; i < this.storage!.length; i++) {\n        const key = this.storage!.key(i);\n        if(key?.startsWith(this.options.prefix)) {\n          keys.push(key);\n        }\n      }\n\n      keys.forEach((key) => {\n        this.storage!.removeItem(key);\n        this.storageCache.delete(key);\n      });\n\n      return true;\n    } catch{\n      return false;\n    }\n  }\n\n  getStorageStats(): { available: number; total: number; used: number } {\n    if(!this.isStorageAvailable()) {\n      return {available: 0, total: 0, used: 0};\n    }\n\n    let used = 0;\n    const keys: string[] = [];\n\n    for(let i = 0; i < this.storage!.length; i++) {\n      const key = this.storage!.key(i);\n      if(key?.startsWith(this.options.prefix)) {\n        keys.push(key);\n        const item = this.storage!.getItem(key);\n        if(item) {\n          used += getStorageSize(item);\n        }\n      }\n    }\n\n    return {\n      available: this.options.maxSize - used,\n      total: this.options.maxSize,\n      used\n    };\n  }\n\n  // Private helper methods\n  private getStorage(): StorageInterface | null {\n    try {\n      const {type} = this.options;\n      const storage = type === 'local'\n        ? BrowserStorage.window.localStorage\n        : BrowserStorage.window.sessionStorage;\n\n      return storage || null;\n    } catch{\n      return null;\n    }\n  }\n\n  private getPrefixedKey(key: string): string {\n    return `${this.options.prefix}${key}`;\n  }\n\n  private isStorageAvailable(): boolean {\n    return this.storage !== null;\n  }\n\n  private validateSize(data: string): boolean {\n    const size = getStorageSize(data);\n    return size <= this.options.maxSize;\n  }\n\n  private cleanExpiredData(): void {\n    if(!this.isStorageAvailable()) {\n      return;\n    }\n\n    const keys: string[] = [];\n    for(let i = 0; i < this.storage!.length; i++) {\n      const key = this.storage!.key(i);\n      if(key?.startsWith(this.options.prefix)) {\n        keys.push(key);\n      }\n    }\n\n    keys.forEach((key) => {\n      try {\n        const item = this.storage!.getItem(key);\n        if(item) {\n          const data = decompressData(item);\n          if(data && isExpired(data)) {\n            this.storage!.removeItem(key);\n            this.storageCache.delete(key);\n          }\n        }\n      } catch{\n        // Remove corrupted data\n        this.storage!.removeItem(key);\n        this.storageCache.delete(key);\n      }\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": ["DEFAULT_PREFIX", "isExpired", "data", "getStorageSize", "compressData", "jsonString", "decompressData", "BrowserStorage", "options", "DEFAULT_PREFIX", "key", "storage", "item", "value", "prefixedKey", "storageData", "keys", "i", "used", "type"]
}

|
package/lib/index.d.ts
CHANGED
|
@@ -1,3 +1,7 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2018-Present, Nitrogen Labs, Inc.
|
|
3
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
|
+
*/
|
|
1
5
|
import { BrowserStorage } from './BrowserStorage/BrowserStorage';
|
|
2
6
|
export * from './types/main';
|
|
3
7
|
export { BrowserStorage };
|
package/lib/index.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
1
|
+
import{BrowserStorage as o}from"./BrowserStorage/BrowserStorage";export*from"./types/main";export{o as BrowserStorage};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOC1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtCcm93c2VyU3RvcmFnZX0gZnJvbSAnLi9Ccm93c2VyU3RvcmFnZS9Ccm93c2VyU3RvcmFnZSc7XG5cbi8vIFN0b3JhZ2VcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMvbWFpbic7XG5leHBvcnQge0Jyb3dzZXJTdG9yYWdlfTtcbiJdLAogICJtYXBwaW5ncyI6ICJBQUlBLE9BQVEsa0JBQUFBLE1BQXFCLGtDQUc3QixXQUFjIiwKICAibmFtZXMiOiBbIkJyb3dzZXJTdG9yYWdlIl0KfQo=
|
package/lib/types/main.d.ts
CHANGED
|
@@ -1,3 +1,21 @@
|
|
|
1
1
|
export interface BrowserStorageOptions {
|
|
2
2
|
readonly type?: 'local' | 'session';
|
|
3
|
+
readonly prefix?: string;
|
|
4
|
+
readonly compression?: boolean;
|
|
5
|
+
readonly maxSize?: number;
|
|
6
|
+
readonly ttl?: number;
|
|
7
|
+
}
|
|
8
|
+
export interface StorageData {
|
|
9
|
+
readonly value: any;
|
|
10
|
+
readonly timestamp: number;
|
|
11
|
+
readonly ttl?: number;
|
|
12
|
+
}
|
|
13
|
+
export type StorageType = 'local' | 'session';
|
|
14
|
+
export interface StorageInterface {
|
|
15
|
+
readonly length: number;
|
|
16
|
+
readonly getItem: (key: string) => string | null;
|
|
17
|
+
readonly setItem: (key: string, value: string) => void;
|
|
18
|
+
readonly removeItem: (key: string) => void;
|
|
19
|
+
readonly clear: () => void;
|
|
20
|
+
readonly key: (index: number) => string | null;
|
|
3
21
|
}
|
package/lib/types/main.js
CHANGED
|
@@ -1,2 +1 @@
|
|
|
1
|
-
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL3R5cGVzL21haW4udHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbImV4cG9ydCBpbnRlcmZhY2UgQnJvd3NlclN0b3JhZ2VPcHRpb25zIHtcbiAgcmVhZG9ubHkgdHlwZT86ICdsb2NhbCcgfCAnc2Vzc2lvbic7XG59XG4iXSwKICAibWFwcGluZ3MiOiAia1dBQUEsSUFBQUEsRUFBQSxrQkFBQUMsRUFBQUQiLAogICJuYW1lcyI6IFsibWFpbl9leHBvcnRzIiwgIl9fdG9Db21tb25KUyJdCn0K
|
|
1
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFtdLAogICJzb3VyY2VzQ29udGVudCI6IFtdLAogICJtYXBwaW5ncyI6ICIiLAogICJuYW1lcyI6IFtdCn0K
|
package/package.json
CHANGED
|
@@ -1,21 +1,34 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nlabs/arkhamjs-storage-browser",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.30.0",
|
|
4
4
|
"publishConfig": {
|
|
5
5
|
"access": "public"
|
|
6
6
|
},
|
|
7
|
-
"description": "Browser storage for ArkhamJS",
|
|
7
|
+
"description": "Browser storage for ArkhamJS with enhanced performance and modern ESNext features",
|
|
8
8
|
"license": "MIT",
|
|
9
9
|
"main": "./lib/index.js",
|
|
10
10
|
"module": "./lib/index.js",
|
|
11
11
|
"browser": "./lib/index.js",
|
|
12
12
|
"types": "./lib/index.d.ts",
|
|
13
|
+
"exports": {
|
|
14
|
+
".": {
|
|
15
|
+
"import": "./lib/index.js",
|
|
16
|
+
"require": "./lib/index.js",
|
|
17
|
+
"types": "./lib/index.d.ts"
|
|
18
|
+
},
|
|
19
|
+
"./package.json": "./package.json"
|
|
20
|
+
},
|
|
21
|
+
"sideEffects": false,
|
|
13
22
|
"keywords": [
|
|
14
23
|
"arkhamjs",
|
|
15
24
|
"flux",
|
|
16
25
|
"browser",
|
|
17
26
|
"nitrogenlabs",
|
|
18
|
-
"storage"
|
|
27
|
+
"storage",
|
|
28
|
+
"localStorage",
|
|
29
|
+
"sessionStorage",
|
|
30
|
+
"cache",
|
|
31
|
+
"performance"
|
|
19
32
|
],
|
|
20
33
|
"author": {
|
|
21
34
|
"name": "Giraldo Rosales",
|
|
@@ -41,14 +54,18 @@
|
|
|
41
54
|
"pretest": "npm run lint",
|
|
42
55
|
"reset": "lex clean",
|
|
43
56
|
"test": "lex test",
|
|
44
|
-
"
|
|
57
|
+
"test:watch": "lex test --watch",
|
|
58
|
+
"test:coverage": "lex test --coverage",
|
|
59
|
+
"update": "lex update --interactive",
|
|
60
|
+
"type-check": "tsc --noEmit",
|
|
61
|
+
"size": "npm run build && node -e \"console.log('Bundle size:', require('fs').statSync('./lib/index.js').size, 'bytes')\""
|
|
45
62
|
},
|
|
46
63
|
"peerDependencies": {
|
|
47
64
|
"@nlabs/arkhamjs": "^3.26.0"
|
|
48
65
|
},
|
|
49
66
|
"devDependencies": {
|
|
67
|
+
"@jest/globals": "^30.0.4",
|
|
50
68
|
"@nlabs/arkhamjs": "*",
|
|
51
|
-
"@types/jest": "^30.0.0",
|
|
52
69
|
"@types/node": "^24.0.10",
|
|
53
70
|
"typescript": "^5.8.3"
|
|
54
71
|
},
|