> tinybase
You are an expert in TinyBase, the reactive data store for local-first applications. You help developers build offline-capable apps with structured tables, automatic reactivity, CRDT-based sync, persistence to IndexedDB/SQLite/Postgres, and relationship modeling — providing a complete client-side database that syncs across devices without a custom backend.
curl "https://skillshub.wtf/TerminalSkills/skills/tinybase?format=md"TinyBase — Reactive Data Store for Local-First Apps
You are an expert in TinyBase, the reactive data store for local-first applications. You help developers build offline-capable apps with structured tables, automatic reactivity, CRDT-based sync, persistence to IndexedDB/SQLite/Postgres, and relationship modeling — providing a complete client-side database that syncs across devices without a custom backend.
Core Capabilities
Store
import { createStore, createQueries, createRelationships } from "tinybase";
import { createLocalPersister } from "tinybase/persisters/persister-browser";
// Create store with tables
const store = createStore()
.setTablesSchema({
todos: {
text: { type: "string" },
done: { type: "boolean", default: false },
priority: { type: "number", default: 0 },
categoryId: { type: "string" },
},
categories: {
name: { type: "string" },
color: { type: "string" },
},
});
// Add data
store.setRow("todos", "t1", { text: "Build app", done: false, priority: 1, categoryId: "c1" });
store.setRow("todos", "t2", { text: "Ship it", done: false, priority: 2, categoryId: "c1" });
store.setRow("categories", "c1", { name: "Work", color: "#3b82f6" });
// Queries
const queries = createQueries(store);
queries.setQueryDefinition("activeTodos", "todos",
({ select, where, order }) => {
select("text");
select("priority");
where("done", false);
order("priority", true); // Descending
},
);
// Relationships
const relationships = createRelationships(store);
relationships.setRelationshipDefinition("todoCategory", "todos", "categories", "categoryId");
// Reactivity — listener fires on any change
store.addRowListener("todos", null, (store, tableId, rowId) => {
console.log(`Todo ${rowId} changed:`, store.getRow(tableId, rowId!));
});
// Persist to IndexedDB
const persister = createLocalPersister(store, "my-app");
await persister.startAutoLoad(); // Load on startup
await persister.startAutoSave(); // Save on every change
React Integration
import { Provider, useRow, useTable, useResultTable, useCell } from "tinybase/ui-react";
function App() {
return (
<Provider store={store} queries={queries}>
<TodoList />
</Provider>
);
}
function TodoList() {
const activeTodos = useResultTable("activeTodos"); // Auto-updates on data change
return (
<ul>
{Object.entries(activeTodos).map(([id, todo]) => (
<TodoItem key={id} id={id} />
))}
</ul>
);
}
function TodoItem({ id }: { id: string }) {
const done = useCell("todos", id, "done"); // Subscribes to single cell
const text = useCell("todos", id, "text");
return (
<li onClick={() => store.setCell("todos", id, "done", !done)}
style={{ textDecoration: done ? "line-through" : "none" }}>
{text}
</li>
);
}
CRDT Sync
import { createWsSynchronizer } from "tinybase/synchronizers/synchronizer-ws-client";
import { createMergeableStore } from "tinybase";
// Mergeable store (CRDT-based, conflict-free)
const store = createMergeableStore();
// Sync via WebSocket
const synchronizer = createWsSynchronizer(store, new WebSocket("wss://sync.example.com"));
await synchronizer.startSync();
// Now any device with the same store syncs automatically
// Offline changes merge cleanly when reconnected
Installation
npm install tinybase
Best Practices
- Local-first — Data lives on the client; works offline, syncs when online; instant UI responses
- Fine-grained reactivity — Use
useCellfor single value subscriptions; only re-renders what changed - CRDT sync — Use MergeableStore for multi-device sync; conflicts resolve automatically
- Queries — Define queries with select/where/order; reactive, auto-update when underlying data changes
- Relationships — Model foreign keys with
setRelationshipDefinition; navigate between tables - Persistence — Auto-save to IndexedDB, SQLite (via OPFS), or remote Postgres; seamless persistence
- Tiny bundle — Core is ~5KB gzipped; add only the modules you need (queries, relationships, sync)
- Schema validation — Define table schemas for type safety; rejects invalid data at write time
> related_skills --same-repo
> zustand
You are an expert in Zustand, the small, fast, and scalable state management library for React. You help developers manage global state without boilerplate using Zustand's hook-based stores, selectors for performance, middleware (persist, devtools, immer), computed values, and async actions — replacing Redux complexity with a simple, un-opinionated API in under 1KB.
> zod
You are an expert in Zod, the TypeScript-first schema declaration and validation library. You help developers define schemas that validate data at runtime AND infer TypeScript types at compile time — eliminating the need to write types and validators separately. Used for API input validation, form validation, environment variables, config files, and any data boundary.
> xero-accounting
Integrate with the Xero accounting API to sync invoices, expenses, bank transactions, and contacts — and generate financial reports like P&L and balance sheet. Use when: connecting apps to Xero, automating bookkeeping workflows, syncing accounting data, or pulling financial reports programmatically.
> windsurf-rules
Configure Windsurf AI coding assistant with .windsurfrules and workspace rules. Use when: customizing Windsurf for a project, setting AI coding standards, creating team-shared Windsurf configurations, or tuning Cascade AI behavior.