Pure-Go port of Yjs, the CRDT framework for collaborative applications.
Ygo speaks the Yjs V1 and V2 wire formats byte-for-byte. JavaScript clients running yjs@13.x synchronize directly with Go servers and vice versa, with both directions verified through 158 cross-language fixture scenarios generated from yjs@13.6.31. The bundled WebSocket server is Hocuspocus-compatible. No CGO; gomobile bind produces verified iOS xcframework and Android AAR.
- Byte-for-byte wire compatibility, verified in both directions. 158 cross-language fixture scenarios (generated from
yjs@13.6.31) cover the V1 and V2 update formats, snapshots, subdocuments, undo, relative positions, GC, awareness, and the sync protocol, JS to Go and Go to JS, plus 56 lib0 primitive vectors. The suite runs in CI on every push, so a regression in either direction fails the build. - Pure Go, no CGO. Builds for any Go target, compiles to WASM, and cross-compiles freely.
gomobile bindproduces a verified iOS xcframework and Android AAR. No V8, no embedded JavaScript engine, no Rust FFI bridge. - Complete CRDT type set. Map, Array, Text (rich-text formatting, Quill deltas, embeds), XML types, Awareness, UndoManager, Snapshots / time-travel, and Subdocuments.
- Compact encoding. Commit-time block squash collapses per-character edits into single items (about 1 byte per character in V1), and garbage collection frees deleted content at commit. On a real-world editing trace V1 document size drops from ~1.97 MB to ~223 KB, competitive with V2.
- Forward-looking wire handling. ygo already handles both confirmed wire-level changes in the
yjs@14release candidate: 53-bit client IDs throughout (byte-verified above 2^32) and Skip structs in the update stream (decoded as no-op gaps). Full attribution / IdMap support waits for the v14 format to stabilize. - Ready-to-run server. A Hocuspocus-compatible WebSocket server with optional sqlite persistence ships in
cmd/ygo-server. - EU-sovereign mirror on codeberg.org/Deln0r/ygo, auto-synced from GitHub on every push for adopters who prefer or require EU-hosted code infrastructure.
Live demo: open ygo.deln0r.com in two browser tabs and start typing. Same protocol any standard Yjs ecosystem client speaks, with a pure-Go server behind it.
package main
import (
"fmt"
"github.com/Deln0r/ygo"
)
func main() {
src := ygo.NewDoc()
m := ygo.NewMap(src, "settings")
txn := src.WriteTxn()
m.Set(txn, "theme", "dark")
m.Set(txn, "lang", "go")
txn.Commit()
// Encode the source doc's full state as wire bytes.
update := ygo.EncodeStateAsUpdate(src)
// Apply to a fresh peer doc — same bytes JS Yjs's Y.applyUpdate consumes.
dst := ygo.NewDoc()
if err := ygo.ApplyUpdate(dst, update); err != nil {
panic(err)
}
dstMap := ygo.NewMap(dst, "settings")
fmt.Println(dstMap.Get("theme")) // dark
}For a collaborative server backend, see cmd/ygo-server — a stand-alone Hocuspocus-compatible WebSocket server with optional sqlite persistence:
go run ./cmd/ygo-server -addr :1234 -store data.dbWrap any shared types in an UndoManager to get scoped, grouped Undo / Redo:
d := ygo.NewDoc()
m := ygo.NewMap(d, "settings")
um := ygo.NewUndoManager(d, m) // watch m; defaults to local edits, 500ms grouping
defer um.Close()
txn := d.WriteTxn()
m.Set(txn, "theme", "dark")
txn.Commit()
um.Undo() // m no longer has "theme"
um.Redo() // "theme" == "dark" againOnly local edits under the watched types are captured; remote updates applied via ApplyUpdate are not. Rapid edits inside the capture-timeout window collapse into one undo step; call um.StopCapturing() to force a boundary. The semantics match yjs@13.6.31's UndoManager, checked by cross-language conformance fixtures.
Capture a point in a document's history and reconstruct it later. The source doc must have GC disabled so deleted content is retained:
d := ygo.NewDocWithOptions(ygo.Options{DisableGC: true})
txt := ygo.NewText(d, "t")
txn := d.WriteTxn()
txt.Insert(txn, 0, "world!")
txn.Commit()
snap := ygo.CreateSnapshot(d) // mark this moment
saved := ygo.EncodeSnapshot(snap) // persist it (byte-compatible with Y.encodeSnapshot)
txn = d.WriteTxn()
txt.Insert(txn, 0, "hello ") // doc moves on
txn.Commit()
restored, _ := ygo.RestoreSnapshot(d, snap) // reconstruct the marked state
ygo.NewText(restored, "t").String() // "world!"The snapshot wire format (EncodeSnapshot / DecodeSnapshot) is byte-compatible with yjs@13.6.31's Y.encodeSnapshot, verified by cross-language fixtures including multi-client delete-set ordering. RestoreSnapshot mirrors Y.createDocFromSnapshot.
Nest a Y.Doc inside a Map. The parent stores a reference (the subdoc's GUID); the subdocument's own content syncs as a separate update stream:
d := ygo.NewDoc()
m := ygo.NewMap(d, "m")
txn := d.WriteTxn()
sub := m.SetDoc(txn, "child") // nest a new subdocument
txn.Commit()
// after syncing the parent to another replica:
got, ok := m.GetDoc(d, "child") // got.GUID() == sub.GUID()The ContentDoc wire format (GUID + options) is byte-compatible with yjs@13.6.31, verified by cross-language fixtures. Lifecycle events are observable via d.OnSubdocs (added / removed / loaded GUIDs per transaction); SetDocWithOptions(..., autoLoad) and subdoc.Load() drive the loaded set, so a sync provider knows which nested documents to fetch.
Feature-complete and stable. The CRDT engine, the V1 and V2 wire formats, and the full type set above are validated bidirectionally against yjs@13.6.31 and exercised in CI on every push. The public API is considered stable for the v1.x line; changes follow semantic versioning, with new functionality as minor releases and breaking changes deferred to a future major.
| Layer | Status |
|---|---|
internal/lib0 varint + RLE encoding |
done; verified byte-equivalent vs JS lib0@0.2.117 (40 + 16 fixtures) |
internal/block (Item, Content, Branch, Splice, Integrate-YATA, TrySquash, Repair, search markers) |
done; full YATA conflict resolution + per-branch LRU position cache |
internal/store (BlockStore, ItemSlice, Materialize) |
done |
internal/doc (Doc, Transaction, TransactionMut) |
done; lock semantics + root-branch registry |
internal/encoding (StateVector, IdSet, Update encode/decode/apply, Pending buffer, V1 + V2 codecs) |
done; JS Yjs → Go proven by 29 V1 + 32 V2 fixture scenarios; Go → JS proven by 48 reverse fixtures (Map / Array / Text / XmlFragment) |
internal/utf16 (UTF-16 length / byte-offset / surrogate-aware split) |
done |
internal/types/Map (Set / Get / Delete / Has / Len / Range / Clear + SetMap / SetArray / SetText) |
done; nested-type construction supported |
internal/types/Array (Insert / InsertRange / Push / Delete / Get / Len / Range / ToSlice + InsertMap / InsertArray / InsertText) |
done; nested-type construction supported |
internal/types/Text (Insert / Delete / String / Length + InsertWithAttributes / Format / InsertEmbed / Range / ToDelta / ApplyDelta) |
done; full rich-text + Quill delta batch API |
| Nested-type construction (Map-in-Map, Array-in-Map, etc., to arbitrary depth) | done; ContentType wire format + Repair ParentID resolution + pending-queue retry |
internal/types/Xml* (XmlFragment, XmlElement, XmlText) |
done; ProseMirror/Tiptap/BlockNote unblocked. XmlHook (legacy) deferred. |
Persistence (Store interface + modernc.org/sqlite reference impl) |
done; append-only update log, Flush compaction, LoadDoc / GetStateVector / GetDiff helpers; pure-Go (no CGO) |
y-sync protocol (internal/sync) |
done; full Hocuspocus message subset (Sync + Awareness + QueryAwareness + Auth + Stateless + BroadcastStateless + Close + SyncStatus); per-document Auth permission scoping deferred (tech-debt) |
Awareness (internal/awareness) |
done; LWW presence map, JSON wire payload per y-protocols, self-eviction defense, SweepOutdated |
server/ (WebSocket sync server) |
done; http.Handler mount-anywhere shape, per-doc broadcaster, persists every applied update to optional persist.Store, awareness disconnect tombstones |
cmd/ygo-server (Hocuspocus-compat binary) |
done; stand-alone WS server with optional sqlite persistence via -store flag |
gomobile/ (bytes-only subset for iOS/Android) |
done; bindable Doc + Awareness wrappers with bytes-in/bytes-out methods only; pure-Go (no CGO). Both targets verified end-to-end on Xcode 16 + NDK 27 + Go 1.26: produces a valid Ygo.xcframework (real-device arm64 + simulator universal, 6.6 + 13 MB) and a valid Android .aar (4 archs incl. arm64-v8a / armeabi-v7a / x86 / x86_64, 8.4 MB), each drop-in for the respective IDE. See gomobile/README.md for the exact commands. |
| V2 update encoding | done; lib0 RLE primitives + column encoder/decoder + Update.{EncodeV2,DecodeV2} + public ygo.{EncodeStateAsUpdateV2,EncodeDiffV2,ApplyUpdateV2}; bidirectional cross-language fixtures vs yjs@13.6.31 |
| dmonad/crdt-benchmarks B1-B4 port | done; B1.1-B1.11 / B2.1-B2.4 / B3.1+3+4 / B4 (260k-edit real-world LaTeX trace). Baseline in BENCHMARKS.md. |
UndoManager (internal/undo) |
done; scoped Undo / Redo over Map / Array / Text with capture-timeout grouping, tracked-origin filtering, and a Redone chain for deletion restore. Cross-language conformance vs yjs@13.6.31 (7 scenarios) |
Snapshots (CreateSnapshot / EncodeSnapshot / RestoreSnapshot) |
done; V1 wire format byte-compatible with yjs@13.6.31 (cross-language fixtures incl. multi-client), RestoreSnapshot mirrors Y.createDocFromSnapshot |
Subdocuments (Map.SetDoc / Map.GetDoc) |
done; ContentDoc wire format (GUID + options) byte-compatible with yjs@13.6.31, cross-language fixtures. Lifecycle events via OnSubdocs / autoLoad / Load |
| Wire client-ID width | 53-bit client IDs throughout (uint64 + varint), byte-verified against yjs@13.6.31 for IDs above 2^32. Forward-compatible with the wider client-ID space yjs@14 introduces |
| Commit-time block squash | done; merges same-client adjacent-clock items at commit (~1 byte/char V1), paired with Apply-side partial-overlap slicing for correct remote integration of merged blocks |
| GC merging | done; deleted content is freed at commit (ContentDeleted, byte-aligned with yjs) and adjacent deleted runs are merged. Deleting a nested shared type recursively collapses its whole subtree into garbage-collected runs (cross-language fixtures), matching yjs. Skipped when GC is disabled or for items an UndoManager keeps |
- Binary protocol compatibility with Yjs v13.x in both V1 and V2 wire formats. Byte-for-byte. JS clients sync with Go servers and vice versa, bidirectionally verified.
- Idiomatic Go API. Channels for events, explicit transactions,
errorreturns. - Pure Go. No CGO.
gomobile bindworks for iOS/Android. - Pluggable persistence with
modernc.org/sqlitereference implementation. - Performance within 2× of yrs on
dmonad/crdt-benchmarksB1-B4. See BENCHMARKS.md.
- C-FFI surface. Yrs already provides this; Ygo's unique value is pure-Go native binaries.
- Drop-in replacement for the Node.js Yjs runtime. Ygo is the Go port; use
yjsitself if you want a JavaScript runtime. - Loro, Automerge, RGA, or other CRDT designs. Ygo implements the Yjs wire format, period.
The single most-important guarantee of this project is byte-level wire compatibility with yjs@13.x. This is enforced by 158 cross-language fixture scenarios (plus 56 lib0 primitive vectors):
- 29 V1 forward fixtures (
testdata/yjs-updates.json) — JS Yjs encodes viaY.encodeStateAsUpdate, Go decodes and applies, state matches. - 32 V2 forward fixtures (
testdata/yjs-update-v2-fixtures.json) — same withY.encodeStateAsUpdateV2. - 48 reverse fixtures (
testdata/go-updates.json+go-update-v2-fixtures.json) — Go encodes viaEncodeStateAsUpdate/EncodeStateAsUpdateV2, JS Yjs decodes viaY.applyUpdate/Y.applyUpdateV2, state matches. - 49 feature fixtures — XML (5), awareness (6), sync protocol (6), undo (7), snapshots (4), subdocuments (3), wire edge cases incl. 53-bit client IDs (3), nested-type GC (4), relative positions (11), all captured from the pinned JS reference and byte-compared in both directions where the feature has a Go encoder.
The fixtures regenerate from pinned yjs@13.6.31 + lib0@0.2.117 + y-protocols@1.0.7 on every CI run; git diff --exit-code testdata/ catches byte-level regressions.
| Project | Runtime | What it provides | Relationship to Ygo |
|---|---|---|---|
yjs (npm) |
Node / browser | The reference CRDT implementation | Ygo's wire-format target |
y-websocket |
Node | Reference WebSocket server | Ygo's cmd/ygo-server is a Go-native equivalent |
Hocuspocus |
Node | Production WebSocket server with auth, persistence, extensions | Ygo's cmd/ygo-server speaks the same 8-message envelope (Sync / Awareness / QueryAwareness / Auth / Stateless / BroadcastStateless / Close / SyncStatus) |
yrs |
Rust | Reference Rust port | Ygo's executable spec for porting decisions |
y-leveldb, y-indexeddb |
Node / browser | Persistence backends | Ygo's persist/sqlite is a Go-native equivalent |
| Ygo | Go | CRDT engine + WS server + persistence in one monorepo, pure-Go for native mobile | This project |
If you have an existing Yjs deployment and want to move the server side to Go (no Node runtime, single static binary, native iOS / Android via gomobile) — Ygo is the path. If you're starting fresh and your team is comfortable with Node, Hocuspocus is the mature choice.
See BENCHMARKS.md for the full table. Highlights from B4 (259,778-edit real-world LaTeX paper trace) on Apple M3, Go 1.26:
| Metric | Ygo V1 | Ygo V2 | yjs (Node, Intel i5-8400) | ywasm (Intel i5-8400) |
|---|---|---|---|---|
| Apply all edits | 20.3 s | 20.3 s | 5.7 s | 28.7 s |
| Encoded doc size | 223 KB | 160 KB | 160 KB | 160 KB |
| Encode time | 0.6 ms | 10.5 ms | 11 ms | 3 ms |
| Parse time | 4.4 ms | 4.5 ms | 39 ms | 16 ms |
How to read this:
- Doc sizes are now at or near yjs parity. V1 lands within 1.4× of yjs (223 KB vs 160 KB; it was 1.97 MB, ~12× bloat, before commit-time block squash + GC shipped). V2 matches yjs byte-for-byte-scale at 160 KB. A 2,000-character sequential insert encodes to ~1.0 byte/char in V1. Squash ships with the paired Apply-side partial-overlap handling that keeps remote integration correct when a peer sends a merged block overlapping the receiver's state.
- Apply throughput is the trade. ~3.5× yjs wall-clock on different hardware (Apple M3 vs the slower i5-8400, so the normalized gap is wider): the per-commit squash + GC scans that buy the 8.8× smaller documents roughly doubled apply time versus the pre-squash build. Against yrs's published sub-10-s B4 numbers this sits at about 2×, at the DESIGN.md target boundary, with known commit-pipeline optimization headroom. (ywasm is yrs compiled to WebAssembly and is not representative of native yrs — wasm overhead inflates it ~5×.)
- Encode and parse are fast once the doc is compact: V1 encode 0.6 ms and parse 4.4 ms on the 223 KB document, both ahead of yjs's published numbers on its hardware.
A direct head-to-head harness against native yrs under identical hardware is on the roadmap but not yet run; the numbers above are honest absolute figures with hardware caveats.
Towards v1.0: benchmarks refresh · documentation site · external security audit. (Undo manager, Snapshots, Subdocuments, commit-time block squash, GC merging: done.)
Per-layer port notes live in docs/yrs-port-notes/. Items intentionally deferred or partial are tracked in docs/tech-debt.md. Detailed design decisions in DESIGN.md.
- DESIGN.md — project design document
- BENCHMARKS.md — performance baseline + B1-B4 methodology
- gomobile/README.md — iOS xcframework + Android AAR build instructions
- docs/yrs-port-notes/ — per-layer port notes describing how each yrs subsystem maps to Go
- docs/tech-debt.md — deferred work and known limitations
- CONTRIBUTING.md — DCO sign-off, no CLA
MIT. See LICENSE.
- Kevin Jahns (dmonad) for Yjs and the YATA algorithm.
- Bartosz Sypytkowski for yrs and the architecture deep dive.