Reimagine state management with CRDTs. Make your app collaborative effortlessly.
Find a file
Zixuan Chen df81aece51
Fix-better-event-order (#595)
The event will now be ordered by (depth, container counter). Therefore, two container creation events within the same layer will be sorted based on the containers’ counter values. This approach can prevent the issue where child tree node events are received before the parent tree node events
2024-12-31 13:26:52 +08:00
.changeset Fix-better-event-order (#595) 2024-12-31 13:26:52 +08:00
.devcontainer chore: add dev container 2022-11-18 00:48:18 +08:00
.github/workflows chore: fix ci release 2024-12-17 13:09:15 +08:00
.vscode Fix warnings (#484) 2024-09-29 21:15:19 +08:00
crates Fix-better-event-order (#595) 2024-12-31 13:26:52 +08:00
docs feat: Stable JSON representation for history (#368) 2024-06-07 13:18:30 +08:00
examples/loro-quill chore: add wasm-base64 build target 2024-11-16 19:18:29 +08:00
scripts feat: add getShallowValue for each container (#581) 2024-12-15 13:34:07 +08:00
supply-chain chore: update crdt-list dep 2022-10-26 23:35:21 +08:00
.editorconfig feat: rle 2022-07-13 00:47:41 +08:00
.gitignore chore: update config 2024-10-10 17:47:41 +08:00
Cargo.lock feat: UndoManager's onPush now can access the change event (#588) 2024-12-23 11:49:35 +08:00
Cargo.toml feat: add FFI for Loro (#420) 2024-09-29 07:41:59 +08:00
cliff.toml chore: bump loro-crdt 2024-04-05 02:25:25 +08:00
CONTRIBUTING.md fix: upgrade wasm-bindgen to fix str free err (#353) 2024-05-09 15:22:34 +08:00
deno.lock chore: update config 2024-10-10 17:47:41 +08:00
deny.toml chore: add cargo deny config 2022-07-17 16:33:16 +08:00
LICENSE Create LICENSE 2023-11-12 23:11:25 +08:00
package.json Refactor: merge two js packages (#532) 2024-10-29 21:46:56 +08:00
pnpm-lock.yaml chore: add wasm-base64 build target 2024-11-16 19:18:29 +08:00
pnpm-workspace.yaml Add richtext example using Quill (#145) 2023-11-03 16:59:27 +08:00
README.md docs: update 2024-12-27 10:32:16 +08:00
rust-toolchain chore: use stable rust as GAT become stable 2022-11-06 23:18:15 +08:00

Loro

Make your JSON data collaborative and version-controlled 🦜

loro-dev%2Floro | Trendshift

Documentation | Getting Started | Rust Doc

https://github.com/loro-dev/loro/assets/18425020/fe246c47-a120-44b3-91d4-1e7232a5b4ac

Loro 1.0 is out! Read the announcement.

Loro is a CRDTs(Conflict-free Replicated Data Types) library that makes building local-first and collaborative apps easier. You can now use it in Rust, JS (via WASM), and Swift.

Features

Features Provided by CRDTs

  • P2P Synchronization
  • Automatic Merging
  • Local Availability
  • Scalability
  • Delta Updates

Supported CRDT Algorithms

Advanced Features in Loro

https://github.com/user-attachments/assets/68e0017a-4987-4f71-b2cf-4ed28a210987

In this example, we demonstrate importing an entire Loro codebase into a Loro-powered version controller, preserving the complete Git DAG history while enabling fast version switching.

Example

Open in StackBlitz

import { expect, test } from 'vitest';
import { LoroDoc, LoroList } from 'loro-crdt';

test('sync example', () => {
  // Sync two docs with two rounds of exchanges

  // Initialize document A
  const docA = new LoroDoc();
  const listA: LoroList = docA.getList('list');
  listA.insert(0, 'A');
  listA.insert(1, 'B');
  listA.insert(2, 'C');

  // Export all updates from docA
  const bytes: Uint8Array = docA.export({ mode: 'update' });

  // Simulate sending `bytes` across the network to another peer, B

  const docB = new LoroDoc();
  // Peer B imports the updates from A
  docB.import(bytes);

  // B's state matches A's state
  expect(docB.toJSON()).toStrictEqual({
    list: ['A', 'B', 'C'],
  });

  // Get the current version of docB
  const version = docB.oplogVersion();

  // Simulate editing at B: delete item 'B'
  const listB: LoroList = docB.getList('list');
  listB.delete(1, 1);

  // Export the updates from B since the last sync point
  const bytesB: Uint8Array = docB.export({ mode: 'update', from: version });

  // Simulate sending `bytesB` back across the network to A

  // A imports the updates from B
  docA.import(bytesB);

  // A has the same state as B
  expect(docA.toJSON()).toStrictEqual({
    list: ['A', 'C'],
  });
});

Blog

Credits

Loro draws inspiration from the innovative work of the following projects and individuals:

  • Ink & Switch: The principles of Local-first Software have greatly influenced this project. The Peritext project has also shaped our approach to rich text CRDTs.
  • Diamond-types: The Event Graph Walker (Eg-walker) algorithm from @josephg has been adapted to reduce the computation and space usage of CRDTs.
  • Automerge: Their use of columnar encoding for CRDTs has informed our strategies for efficient data encoding.
  • Yjs: We have incorporated a similar algorithm for effectively merging collaborative editing operations, thanks to their pioneering work.
  • Matthew Weidner: His work on the Fugue algorithm has been invaluable, enhancing our text editing capabilities.
  • Martin Kleppmann: His work on CRDTs has significantly influenced our comprehension of the field.