zod-browser-storage
Type-safe Web Storage wrapper with Zod runtime validation for React, Vue, Angular, and vanilla JavaScript.
Disclaimer: This library is not affiliated with, endorsed by, or sponsored by the Zod project or its creators. It is an independent utility that leverages Zod for runtime validation.
Features
- Type-safe: Full TypeScript support with automatic type inference
- Runtime validation: Powered by Zod schema validation
- Framework agnostic: Works with React, Vue, Angular, or vanilla JS
- Dual storage: Supports both localStorage and sessionStorage
- Lightweight: Minimal bundle size with tree-shaking support
- Simple API: Intuitive methods with flexible error handling
Installation
npm install zod-browser-storage zod
# or
pnpm add zod-browser-storage zod
# or
yarn add zod-browser-storage zodQuick Start
import { z } from 'zod';
import { zs, zodStorage } from 'zod-browser-storage';
// Define your storage with schema
const userStorage = zs({
key: 'user',
schema: z.object({
name: z.string(),
age: z.number(),
email: z.string().email(),
}),
defaultValue: { name: '', age: 0, email: '' }
});
// Set data (automatically validated)
zodStorage.set(userStorage, {
name: 'John Doe',
age: 30,
email: 'john@example.com',
});
// Get data (returns typed data or null)
const user = zodStorage.get(userStorage);
console.log(user); // { name: 'John Doe', age: 30, email: 'john@example.com' }
// Clear data
zodStorage.clear(userStorage);API Reference
zs(config)
Creates a storage configuration object.
Parameters:
config.key(string): The storage keyconfig.schema(ZodType): Zod schema for validationconfig.defaultValue(T): Default value for initializationconfig.storage('local' | 'session', optional): Storage type (default: 'local')
Returns: SafeStorage<T> configuration object
zodStorage.get(storage, options?)
Retrieves and validates data from storage.
Parameters:
storage(SafeStorage): Storage configurationoptions.onFailure('null' | 'default' | 'throw', optional): Error handling behavior'null': Returnsnullon failure (default)'default': ReturnsdefaultValueon failure'throw': Throws on failure — the originalZodError(with.issues/.flatten()) for schema validation failures, or aSafeStorageErrorfor JSON parse failures
Returns: The validated value (schema output type), or null
SSR / non-browser / absent key: When there is no value to validate — Web Storage is unavailable (server-side rendering, non-browser runtimes, or storage blocked), or the key is simply absent —
get()does not throw. It returnsnull, ordefaultValuewithonFailure: 'default'. So it is safe to call during the first/server render.
Detecting the failure cause: prefer
err instanceof SafeStorageErrorto identify parse/serialization/write failures; treat everything else as a Zod validation error. Relying onerr instanceof z.ZodErrorrequires the consumer and this library to share a singlezodinstance (a duplicatezodin the bundle breaks cross-realminstanceof).
Nullable schemas:
get()returnsnullfor three distinct states — key absent, a validly-storednull, and a failed read withonFailure: 'null'. With a nullable schema,get(s) ?? s.defaultValuewill replace a deliberately-storednullwith the default; branch on the value explicitly if you need to tell these apart.
Example:
// Return null on validation failure (default)
const data = zodStorage.get(userStorage);
// Return default value on validation failure
const data = zodStorage.get(userStorage, { onFailure: 'default' });
// Throw error on validation failure
const data = zodStorage.get(userStorage, { onFailure: 'throw' });zodStorage.set(storage, data)
Validates data against the schema, then stores it. Validation happens at runtime, not just
compile time — so untyped (any/JS) callers and structural constraints (min/max/regex/
refine) are enforced on write too.
Parameters:
storage(SafeStorage): Storage configurationdata: Data to store (schema input type)
Returns: void
Throws:
- The original
ZodErrorwhendatafails schema validation (nothing is written). - A
SafeStorageErrorwhen serialization fails (circular reference, BigInt) or the storage write fails (e.g.QuotaExceededError, Safari private mode).
SSR / non-browser:
set()(andclear()/init()) is a silent no-op when Web Storage is unavailable.
zodStorage.clear(storage)
Clears data from storage.
Parameters:
storage(SafeStorage): Storage configuration
Returns: void
zodStorage.init(storage)
Initializes storage with the default value.
Parameters:
storage(SafeStorage): Storage configuration
Returns: void
Usage Examples
Basic Types
// Number array
const numbersStorage = zs({
key: 'numbers',
schema: z.array(z.number()),
defaultValue: []
});
// String
const nameStorage = zs({
key: 'name',
schema: z.string(),
defaultValue: ''
});
// Boolean
const flagStorage = zs({
key: 'flag',
schema: z.boolean(),
defaultValue: false
});Enum Types
const themeStorage = zs({
key: 'theme',
schema: z.enum(['light', 'dark', 'auto']),
defaultValue: 'light'
});
zodStorage.set(themeStorage, 'dark');Complex Objects
const profileSchema = z.object({
user: z.object({
id: z.number(),
name: z.string(),
}),
settings: z.object({
theme: z.string(),
notifications: z.boolean(),
}),
});
const profileStorage = zs({
key: 'profile',
schema: profileSchema,
defaultValue: {
user: { id: 0, name: '' },
settings: { theme: 'light', notifications: true }
}
});SessionStorage
const sessionData = zs({
key: 'tempData',
schema: z.string(),
defaultValue: '',
storage: 'session' // Use sessionStorage instead of localStorage
});
zodStorage.set(sessionData, 'temporary value');React Integration
import { useState, useEffect } from 'react';
import { z } from 'zod';
import { zs, zodStorage } from 'zod-browser-storage';
const settingsSchema = z.object({
notifications: z.boolean(),
theme: z.enum(['light', 'dark']),
});
const settingsStorage = zs({
key: 'settings',
schema: settingsSchema,
defaultValue: { notifications: true, theme: 'light' }
});
function useSettings() {
const [settings, setSettings] = useState(() =>
zodStorage.get(settingsStorage) ?? settingsStorage.defaultValue
);
useEffect(() => {
zodStorage.set(settingsStorage, settings);
}, [settings]);
return [settings, setSettings] as const;
}Vue Integration
<script setup lang="ts">
import { ref, watch } from 'vue';
import { z } from 'zod';
import { zs, zodStorage } from 'zod-browser-storage';
const userSchema = z.object({
name: z.string(),
age: z.number(),
});
const userStorage = zs({
key: 'user',
schema: userSchema,
defaultValue: { name: '', age: 0 }
});
const user = ref(zodStorage.get(userStorage) ?? userStorage.defaultValue);
watch(user, (newUser) => {
zodStorage.set(userStorage, newUser);
}, { deep: true });
</script>Error Handling
const dataStorage = zs({
key: 'data',
schema: z.array(z.number()),
defaultValue: [1, 2, 3]
});
// Scenario 1: Invalid data in storage
localStorage.setItem('data', 'invalid json');
// Returns null (default behavior)
const result1 = zodStorage.get(dataStorage);
console.log(result1); // null
// Returns default value
const result2 = zodStorage.get(dataStorage, { onFailure: 'default' });
console.log(result2); // [1, 2, 3]
// Throws error
try {
const result3 = zodStorage.get(dataStorage, { onFailure: 'throw' });
} catch (error) {
console.error('Validation failed:', error);
}Advanced Validation
// Email validation
const emailStorage = zs({
key: 'email',
schema: z.string().email(),
defaultValue: ''
});
// Number constraints
const ageStorage = zs({
key: 'age',
schema: z.number().min(0).max(120),
defaultValue: 0
});
// Pattern matching
const codeStorage = zs({
key: 'code',
schema: z.string().regex(/^[A-Z]{3}-\d{3}$/),
defaultValue: ''
});
// Transformed values
const upperCaseStorage = zs({
key: 'name',
schema: z.string().transform(val => val.toUpperCase()),
defaultValue: ''
});Transform / coerce schemas: For schemas whose input differs from their output (
.transform(),z.coerce.*,.pipe()),set()accepts the input type andget()returns the output type. The raw input is stored and re-parsed on read, so the round-trip is correct:const lenStorage = zs({ key: 'word', schema: z.string().transform((s) => s.length), // input: string, output: number defaultValue: 'abc', }); zodStorage.set(lenStorage, 'hello'); // accepts string (input) zodStorage.get(lenStorage); // returns 5 (output)Note: with
onFailure: 'default',get()returns the rawdefaultValue(the input type), which is not re-parsed — so for transform schemas the'default'result is the input type, not the output type.set(storage, undefined)stores the resolved value for.default()schemas and removes the key for.optional()schemas.
TypeScript
The library is written in TypeScript and provides full type safety:
import { SafeStorage, SafeStorageGetOptions, StorageType } from 'zod-browser-storage';
// All types are automatically inferred
const userStorage = zs({
key: 'data',
schema: z.object({ id: z.number(), name: z.string() }),
defaultValue: { id: 0, name: '' }
});
// TypeScript knows the exact type
const data = zodStorage.get(userStorage); // { id: number, name: string } | nullWhy zod-browser-storage?
Data Integrity
Without validation, localStorage can contain corrupted or invalid data:
// Without zod-browser-storage - No type safety
localStorage.setItem('user', JSON.stringify({ id: '123' })); // Wrong type!
const user = JSON.parse(localStorage.getItem('user')!);
console.log(user.id + 1); // "1231" - String concatenation bug!
// With zod-browser-storage - Runtime validation catches errors
const userStorage = zs({
key: 'user',
schema: z.object({ id: z.number() }),
defaultValue: { id: 0 }
});
const user = zodStorage.get(userStorage); // null (validation failed)
const safeUser = zodStorage.get(userStorage, { onFailure: 'default' }); // { id: 0 }Type Safety
const countStorage = zs({
key: 'count',
schema: z.number(),
defaultValue: 0
});
zodStorage.set(countStorage, 'invalid'); // TypeScript error!
zodStorage.set(countStorage, 42); // OKFramework Agnostic
Works seamlessly with any JavaScript framework or vanilla JS. No framework-specific dependencies.
License
MIT YESHYUNGSEOK
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.