True offline-first sync for modern appsβwithout vendor lock-in
Getting Started β’ Documentation β’ Examples β’ Discussions β’ Roadmap
Build collaborative apps in hours, not months.
SyncKit is a production-ready sync engine that gives you everything for local-first collaboration:
"Add
sync.document()to your app, get real-time sync automatically."
The reality: Building sync from scratch takes months. SyncKit gives you production-ready collaboration in 3 lines of code.
const sync = new SyncKit()
await sync.init()
const doc = sync.document<Todo>('todo-123')
await doc.update({ completed: true })
// β¨ Works offline, syncs automatically, resolves conflicts
β‘ Try Live Demo - Open in two tabs to see real-time sync!
1. Complex State (Kanban) SyncKit handles structural data like lists and nested objects with automatic conflict resolution.
2. Collaborative Text (New in v0.2.0) Add Google Docs-style collaboration to your app with a single hook.
// It's this simple:
import { useSyncText } from '@synckit-js/sdk/react'
function Editor() {
// β¨ Automatic conflict resolution & real-time sync
const [text, { insert, delete: del }] = useSyncText('doc-1')
return <textarea value={text} onChange={e => insert(0, e.target.value)} />
}
True offline-first architectureβnot just caching. Your app works perfectly on planes, trains, tunnels, and coffee shops with spotty WiFi.
154KB gzipped - Complete local-first sync solution with everything you need.
What you get:
Size-critical apps? Use Lite variant (46KB gzipped, basic sync only)
Every byte is justified. We chose completeness over minimal sizeβrich text, undo/redo, cursors, and framework adapters all work together out of the box.
Open source and self-hostable. No vendor lock-in, no surprise $2,000/month bills, complete data sovereignty.
npm install @synckit-js/sdk
import { SyncKit } from '@synckit-js/sdk'
import { SyncProvider, useSyncDocument } from '@synckit-js/sdk/react'
// Initialize (works offline-only, no server needed!)
const sync = new SyncKit()
await sync.init()
function App() {
return (
<SyncProvider synckit={sync}>
<TodoApp />
</SyncProvider>
)
}
function TodoApp() {
const [todo, { update }] = useSyncDocument<Todo>('todo-1')
if (!todo || !todo.text) return <div>Loading...</div>
return (
<div>
<input
type="checkbox"
checked={todo.completed}
onChange={(e) => update({ completed: e.target.checked })}
/>
<span>{todo.text}</span>
</div>
)
}
That's it! Your app now:
Bundle: SyncKit (154KB gzipped) + React (156KB) = ~310KB total
Size-critical? import { SyncKit } from '@synckit-js/sdk/lite' (46KB gzipped, local-only)
graph TD
A[Your Application<br/>React/Vue/Svelte] --> B[SyncKit SDK<br/>TypeScript]
B -->|Simple API| B1[document, text, counter]
B -->|Framework adapters| B2[React/Vue/Svelte hooks]
B -->|Offline queue| B3[Storage adapters]
B --> C[Rust Core Engine<br/>WASM + Native]
C -->|80% of use cases| C1[LWW Sync]
C -->|Collaborative editing| C2[Text CRDTs]
C -->|Advanced features| C3[Custom CRDTs<br/>counters, sets]
C --> D[IndexedDB Storage<br/>Your local source of truth]
D -.->|Optional| E[SyncKit Server<br/>TypeScript/Python/Go/Rust]
E -->|Real-time sync| E1[WebSocket]
E -->|Persistence| E2[PostgreSQL/MongoDB]
E -->|Security| E3[JWT auth + RBAC]
style A fill:#e1f5ff,stroke:#333,stroke-width:2px,color:#1a1a1a
style B fill:#fff4e1,stroke:#333,stroke-width:2px,color:#1a1a1a
style C fill:#ffe1e1,stroke:#333,stroke-width:2px,color:#1a1a1a
style D fill:#e1ffe1,stroke:#333,stroke-width:2px,color:#1a1a1a
style E fill:#f0e1ff,stroke:#333,stroke-width:2px,color:#1a1a1a
Detailed architecture docs β
Perfect for: Task apps, CRMs, project management, note apps (80% of applications)
import { SyncKit } from '@synckit-js/sdk'
import { useSyncDocument } from '@synckit-js/sdk/react'
// Initialize once
const sync = new SyncKit()
await sync.init()
// Use anywhere
const doc = sync.document<Project>('project-123')
await doc.update({ status: 'completed' })
// Conflicts resolved automatically with Last-Write-Wins
Perfect for: Collaborative editors, documentation, notes
import { useSyncText } from '@synckit-js/sdk/react'
const [text, { insert, delete: del }] = useSyncText('document-456')
await insert(0, 'Hello ')
// Character-level sync, conflict-free convergence
Perfect for: Likes, votes, tags, participants
import { useCounter, useSet } from '@synckit-js/sdk/react'
const [count, { increment, decrement }] = useCounter('likes-789')
await increment() // Conflict-free counter
const [tags, { add, remove }] = useSet<string>('post-tags')
await add('typescript') // Observed-remove set
Different libraries make different trade-offs. Here's how SyncKit compares:
| Feature | SyncKit | Firebase | Supabase | Yjs | Automerge |
|---|---|---|---|---|---|
| Bundle Size (gzipped) | 154KB (46KB lite) |
~150β200KB (typical client) |
~45KB (JS client) |
65KB (core) |
300KB+ (JS/WASM) |
| Text CRDT | β Fugue | β No | β No | β Y.Text | β Yes |
| Rich Text | β Peritext | β No | β No | β οΈ Limited | β Yes |
| Undo/Redo | β Cross-tab | β No | β No | β οΈ Basic | β Yes |
| Awareness/Cursors | β Built-in | β No | β No | β οΈ Extension | β No |
| Framework Adapters | β React/Vue/Svelte | β No | β No | β οΈ Community | β No |
| True Offline-First | β Native | β οΈ Limited (cache + persistence) | β No native support | β Full | β Full |
| Works Without Server | β Yes | β No | β No | β Yes | β Yes |
| Self-Hosted | β Yes | β No | β Yes | β Yes | β Yes |
| TypeScript Support | β Native | β Good | β Good | β οΈ Issues | β Good |
| Production Status | β v0.2.0 | β Mature | β Mature | β Mature | β οΈ Stable core, evolving ecosystem |
Choose SyncKit if:
Choose alternatives if:
See detailed migration guides β
@synckit-js/sdk - Core SDK (TypeScript) + WASM engine@synckit-js/sdk/react - React hooks and components (export from SDK)@synckit-js/sdk/vue - Vue 3 composables (export from SDK)@synckit-js/sdk/svelte - Svelte 5 stores with runes (export from SDK)@synckit-js/sdk/lite - Lightweight version (local-only, 46KB gzipped)@synckit-js/server - Bun + Hono reference server (production-ready)Current Version: v0.2.0
The core sync engine is battle-tested and ready for production:
New features we're testing with the community - stable but gathering feedback:
We welcome contributions from the community!
Ways to contribute:
Need enterprise support?
Contact: [email protected]
SyncKit (lite): 46 KB ββββββββ
Yjs (assembled): 65 KB βββββββββββ
SyncKit (default): 154 KB ββββββββββββββββββββββββββββ
Firebase: 150 KB ββββββββββββββββββββββββββββ
Automerge: 300 KB ββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
Local update: <1 ms ββββ
Cross-tab sync: <1 ms ββββ
Network sync: 10-50 ms ββββββββ
Firebase (cold): 2-30 s ββββββββββββββββββββββββββββββββ
SyncKit: 3 MB ββββ
Yjs: 8 MB βββββββββ
Automerge: 180 MB ββββββββββββββββββββββββββββββββββββββββ
SyncKit's core algorithms are based on published research:
Built with inspiration from:
Special thanks to the local-first community for pioneering this movement.
MIT License - see LICENSE for details.
Copyright (c) 2025 Daniel Bitengo
Built with β€οΈ for the local-first future
β Star us on GitHub β’ π Read the docs β’ π Get started