Docs
Introduction to Loro

Introduction to Loro

It is well-known that syncing data/building realtime collaborative apps is challenging, especially when devices can be offline or part of a peer-to-peer network. Loro simplifies this process for you.

We want to provide better DevTools to make building local-first apps (opens in a new tab) easy and enjoyable. You can read our blog post to learn more about our vision.

Loro uses Conflict-free Replicated Data Types (CRDTs) to resolve parallel edits. By utilizing Loro's data types, your applications can be made collaborative and keep the editing history with low overhead.

After you model your app state by Loro, syncing is simple:

// Assume docA and docB are two Loro documents in two different devices
const bytesA = docA.exportFrom();
// send bytes to docB by any method
docB.import(bytes);
// docB is now updated with all the changes from docA
 
const bytesB = docB.exportFrom();
// send bytes to docA by any method
docA.import(bytesB);
// docA and docB are now in sync, they have the same state

Saving your app state is also straightforward:

const doc = new Loro();
doc.getText("text").insert(0, "Hello world!");
const bytes = doc.exportSnapshot();
// Bytes can be saved to local storage, database, or sent over the network

Loading your app state:

const newDoc = new Loro();
newDoc.import(bytes);

Loro also makes it easy for you to time travel the history and add version control to your app. Learn more about time travel.

doc.checkout(version); // Checkout the doc to the given version

Loro is compatible with the JSON schema. If you can model your app state with JSON, you probably can sync your app with Loro. Because we need to adhere to the JSON schema, using a number as a key in a Map is not permitted, and cyclic links should be avoided.

doc.toJSON(); // Get the JSON representation of the doc

Differences from other CRDT libraries

The table below summarizes Loro's features, which may not be present in other CRDT libraries.

Features/Important design decisionsLoroDiamond-typesYjsAutomerge
Replayable Event Graph (opens in a new tab)✅ Inventor
Peritext (opens in a new tab) rich text CRDTs✅[1]✅ Inventor
Movable Tree (opens in a new tab)❌ Inventor
Movable List (opens in a new tab)❌ Inventor
Time travel✅[2]
Fugue (opens in a new tab) / Maximal non-interleaving
JSON types
Merging elements in memory by run length encoding✅ Inventor
Byzantine-fault-tolerance
  • [1] Loro uses a Peritext-like algorithm to merge the concurrent edits to rich text styles. It's different from the original design but it passes all the criteria listed in the paper.
  • [2] Unlike others, Yjs requires users to store a version vector and a delete set, enabling time travel back to a specific point.
  • Fugue (opens in a new tab) is a text/list CRDTs that can minimize the chance of the interleaving anomalies.