2个版本

0.1.1 2023年9月2日
0.1.0 2023年9月2日

文本编辑器中排名493

MIT许可证

255KB
5.5K SLoC

🥤 cola

CI Latest version Docs badge

cola是一个针对实时协同编辑纯文本文档的冲突免费复制数据类型。

它允许分布式网络上的多个对等体同时编辑相同的文本文档,确保它们都收敛到相同的最终状态,而无需依赖中央服务器来协调编辑。

查看文档以了解cola的API,或这篇博客文章深入了解其设计和实现。

示例用法

use std::ops::Range;

use cola::{Deletion, Replica, ReplicaId};

struct Document {
    buffer: String,
    crdt: Replica,
}

struct Insertion {
    text: String,
    crdt: cola::Insertion,
}

impl Document {
    fn new<S: Into<String>>(text: S, replica_id: ReplicaId) -> Self {
        let buffer = text.into();
        let crdt = Replica::new(replica_id, buffer.len());
        Document { buffer, crdt }
    }

    fn fork(&self, new_replica_id: ReplicaId) -> Self {
        let crdt = self.crdt.fork(new_replica_id);
        Document { buffer: self.buffer.clone(), crdt }
    }

    fn insert<S: Into<String>>(
        &mut self,
        insert_at: usize,
        text: S,
    ) -> Insertion {
        let text = text.into();
        self.buffer.insert_str(insert_at, &text);
        let insertion = self.crdt.inserted(insert_at, text.len());
        Insertion { text, crdt: insertion }
    }

    fn delete(&mut self, range: Range<usize>) -> Deletion {
        self.buffer.replace_range(range.clone(), "");
        self.crdt.deleted(range)
    }

    fn integrate_insertion(&mut self, insertion: Insertion) {
        if let Some(offset) = self.crdt.integrate_insertion(&insertion.crdt) {
            self.buffer.insert_str(offset, &insertion.text);
        }
    }

    fn integrate_deletion(&mut self, deletion: Deletion) {
        let ranges = self.crdt.integrate_deletion(&deletion);
        for range in ranges.into_iter().rev() {
            self.buffer.replace_range(range, "");
        }
    }
}

fn main() {
    let mut peer_1 = Document::new("Hello, world", 1);
    let mut peer_2 = peer_1.fork(2);

    let delete_comma = peer_1.delete(5..6);
    let insert_exclamation = peer_2.insert(12, "!");

    peer_1.integrate_insertion(insert_exclamation);
    peer_2.integrate_deletion(delete_comma);

    assert_eq!(peer_1.buffer, "Hello world!");
    assert_eq!(peer_2.buffer, "Hello world!");
}

关于crate命名方案的说明

cola的package.namecola-crdt,而其lib.name只是cola。这是因为包名必须唯一才能发布到crates.io,但遗憾的是cola已被crate占位者占用。

这对于您,即库的用户,实际上意味着您应该在您的Cargo.toml中将cola导入为cola-crdt,并在源代码中use它作为cola

例如

# Cargo.toml
[dependencies]
cola-crdt = "0.1"
// main.rs
use cola::Replica;

fn main() {
    println!("{:?}", Replica::new(1, 42));
}

依赖项

~0–315KB