Rust implementation of NextGraph, a Decentralized and local-first web 3.0 ecosystem
https://nextgraph.org
byzantine-fault-tolerancecrdtsdappsdecentralizede2eeeventual-consistencyjson-ldlocal-firstmarkdownocapoffline-firstp2pp2p-networkprivacy-protectionrdfrich-text-editorself-hostedsemantic-websparqlweb3collaboration
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
527 lines
16 KiB
527 lines
16 KiB
12 months ago
|
// Copyright (c) 2022-2024 Niko Bonnieure, Par le Peuple, NextGraph.org developers
|
||
2 years ago
|
// All rights reserved.
|
||
|
// Licensed under the Apache License, Version 2.0
|
||
2 years ago
|
// <LICENSE-APACHE2 or http://www.apache.org/licenses/LICENSE-2.0>
|
||
2 years ago
|
// or the MIT license <LICENSE-MIT or http://opensource.org/licenses/MIT>,
|
||
|
// at your option. All files in the project carrying such
|
||
|
// notice may not be copied, modified, or distributed except
|
||
|
// according to those terms.
|
||
|
|
||
|
//! Branch of a Repository
|
||
|
|
||
11 months ago
|
use std::collections::HashSet;
|
||
10 months ago
|
use zeroize::{Zeroize, ZeroizeOnDrop};
|
||
2 years ago
|
|
||
10 months ago
|
// use fastbloom_rs::{BloomFilter as Filter, Membership};
|
||
2 years ago
|
|
||
10 months ago
|
use crate::errors::*;
|
||
2 years ago
|
use crate::object::*;
|
||
|
use crate::store::*;
|
||
|
use crate::types::*;
|
||
10 months ago
|
use crate::utils::encrypt_in_place;
|
||
2 years ago
|
|
||
|
impl BranchV0 {
|
||
|
pub fn new(
|
||
|
id: PubKey,
|
||
11 months ago
|
repo: ObjectRef,
|
||
10 months ago
|
root_branch_readcap_id: ObjectId,
|
||
11 months ago
|
topic_priv: PrivKey,
|
||
2 years ago
|
metadata: Vec<u8>,
|
||
|
) -> BranchV0 {
|
||
11 months ago
|
let topic_privkey: Vec<u8> = vec![];
|
||
10 months ago
|
//TODO: topic_privkey is topic_priv encrypted with RepoWriteCapSecret, TopicId, BranchId
|
||
11 months ago
|
let topic = topic_priv.to_pub();
|
||
2 years ago
|
BranchV0 {
|
||
|
id,
|
||
11 months ago
|
repo,
|
||
10 months ago
|
root_branch_readcap_id,
|
||
2 years ago
|
topic,
|
||
11 months ago
|
topic_privkey,
|
||
2 years ago
|
metadata,
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl Branch {
|
||
10 months ago
|
/// topic private key (a BranchWriteCapSecret), encrypted with a key derived as follow
|
||
|
/// BLAKE3 derive_key ("NextGraph Branch WriteCap Secret BLAKE3 key",
|
||
|
/// RepoWriteCapSecret, TopicId, BranchId )
|
||
|
/// so that only editors of the repo can decrypt the privkey
|
||
|
/// nonce = 0
|
||
|
pub fn encrypt_topic_priv_key(
|
||
|
privkey: &BranchWriteCapSecret,
|
||
|
topic_id: TopicId,
|
||
|
branch_id: BranchId,
|
||
|
repo_write_cap_secret: &RepoWriteCapSecret,
|
||
|
) -> Vec<u8> {
|
||
|
let mut plaintext = serde_bare::to_vec(privkey).unwrap();
|
||
|
let repo_write_cap_secret = serde_bare::to_vec(repo_write_cap_secret).unwrap();
|
||
|
let topic_id = serde_bare::to_vec(&topic_id).unwrap();
|
||
|
let branch_id = serde_bare::to_vec(&branch_id).unwrap();
|
||
|
let mut key_material = [repo_write_cap_secret, topic_id, branch_id].concat();
|
||
|
let mut key: [u8; 32] = blake3::derive_key(
|
||
|
"NextGraph Branch WriteCap Secret BLAKE3 key",
|
||
|
key_material.as_slice(),
|
||
|
);
|
||
|
encrypt_in_place(&mut plaintext, key, [0; 12]);
|
||
|
key.zeroize();
|
||
|
key_material.zeroize();
|
||
|
plaintext
|
||
|
}
|
||
|
|
||
2 years ago
|
pub fn new(
|
||
|
id: PubKey,
|
||
11 months ago
|
repo: ObjectRef,
|
||
10 months ago
|
root_branch_readcap_id: ObjectId,
|
||
11 months ago
|
topic_priv: PrivKey,
|
||
2 years ago
|
metadata: Vec<u8>,
|
||
|
) -> Branch {
|
||
|
Branch::V0(BranchV0::new(
|
||
11 months ago
|
id,
|
||
|
repo,
|
||
10 months ago
|
root_branch_readcap_id,
|
||
11 months ago
|
topic_priv,
|
||
|
metadata,
|
||
2 years ago
|
))
|
||
|
}
|
||
|
|
||
|
/// Branch sync request from another peer
|
||
10 months ago
|
/// `target_heads` represents the list of heads the requester would like to reach. this list should not be empty.
|
||
|
/// if the requester doesn't know what to reach, the responder should fill this list with their own current local head.
|
||
|
/// `known_heads` represents the list of current heads at the requester replica at the moment of request.
|
||
|
/// an empty list means the requester has an empty branch locally
|
||
2 years ago
|
///
|
||
|
/// Return ObjectIds to send
|
||
|
pub fn sync_req(
|
||
10 months ago
|
target_heads: &[ObjectId],
|
||
|
known_heads: &[ObjectId],
|
||
|
//their_filter: &BloomFilter,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
2 years ago
|
) -> Result<Vec<ObjectId>, ObjectParseError> {
|
||
2 years ago
|
//log_debug!(">> sync_req");
|
||
10 months ago
|
//log_debug!(" target_heads: {:?}", target_heads);
|
||
|
//log_debug!(" known_heads: {:?}", known_heads);
|
||
2 years ago
|
|
||
10 months ago
|
/// Load causal past of a Commit `cobj` in a `Branch` from the `RepoStore`,
|
||
|
/// and collect in `visited` the ObjectIds encountered on the way, stopping at any commit already belonging to `theirs` or the root of DAG.
|
||
|
/// optionally collecting the missing objects/blocks that couldn't be found locally on the way
|
||
|
fn load_causal_past(
|
||
2 years ago
|
cobj: &Object,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
10 months ago
|
theirs: &HashSet<ObjectId>,
|
||
2 years ago
|
visited: &mut HashSet<ObjectId>,
|
||
10 months ago
|
missing: &mut Option<&mut HashSet<ObjectId>>,
|
||
|
) -> Result<(), ObjectParseError> {
|
||
2 years ago
|
let id = cobj.id();
|
||
|
|
||
10 months ago
|
// check if this commit object is present in theirs or has already been visited in the current walk
|
||
|
// load deps, stop at the root(including it in visited) or if this is a commit object from known_heads
|
||
|
if !theirs.contains(&id) && !visited.contains(&id) {
|
||
2 years ago
|
visited.insert(id);
|
||
10 months ago
|
for id in cobj.acks_and_nacks() {
|
||
11 months ago
|
match Object::load(id, None, store) {
|
||
2 years ago
|
Ok(o) => {
|
||
10 months ago
|
load_causal_past(&o, store, theirs, visited, missing)?;
|
||
2 years ago
|
}
|
||
10 months ago
|
Err(ObjectParseError::MissingBlocks(blocks)) => {
|
||
|
missing.as_mut().map(|m| m.extend(blocks));
|
||
2 years ago
|
}
|
||
|
Err(e) => return Err(e),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
10 months ago
|
Ok(())
|
||
2 years ago
|
}
|
||
|
|
||
|
// their commits
|
||
|
let mut theirs = HashSet::new();
|
||
|
|
||
10 months ago
|
// collect causal past of known_heads
|
||
|
for id in known_heads {
|
||
|
if let Ok(cobj) = Object::load(*id, None, store) {
|
||
|
load_causal_past(&cobj, store, &HashSet::new(), &mut theirs, &mut None)?;
|
||
|
}
|
||
|
// we silently discard any load error on the known_heads as the responder might not know them (yet).
|
||
2 years ago
|
}
|
||
|
|
||
10 months ago
|
let mut visited = HashSet::new();
|
||
|
// collect all commits reachable from target_heads
|
||
|
// up to the root or until encountering a commit from theirs
|
||
|
for id in target_heads {
|
||
|
if let Ok(cobj) = Object::load(*id, None, store) {
|
||
|
load_causal_past(&cobj, store, &theirs, &mut visited, &mut None)?;
|
||
|
}
|
||
|
// we silently discard any load error on the target_heads as they can be wrong if the requester is confused about what the responder has locally.
|
||
2 years ago
|
}
|
||
|
|
||
2 years ago
|
//log_debug!("!! ours: {:?}", ours);
|
||
|
//log_debug!("!! theirs: {:?}", theirs);
|
||
2 years ago
|
|
||
|
// remove their_commits from result
|
||
10 months ago
|
// let filter = Filter::from_u8_array(their_filter.f.as_slice(), their_filter.k.into());
|
||
|
// for id in result.clone() {
|
||
|
// match id {
|
||
|
// Digest::Blake3Digest32(d) => {
|
||
|
// if filter.contains(&d) {
|
||
|
// result.remove(&id);
|
||
|
// }
|
||
|
// }
|
||
|
// }
|
||
|
// }
|
||
2 years ago
|
//log_debug!("!! result filtered: {:?}", result);
|
||
10 months ago
|
Ok(Vec::from_iter(visited))
|
||
2 years ago
|
}
|
||
|
}
|
||
|
|
||
10 months ago
|
#[cfg(test)]
|
||
2 years ago
|
mod test {
|
||
|
|
||
10 months ago
|
//use fastbloom_rs::{BloomFilter as Filter, FilterBuilder, Membership};
|
||
2 years ago
|
|
||
10 months ago
|
struct Test<'a> {
|
||
|
storage: Box<dyn RepoStore + Send + Sync + 'a>,
|
||
|
}
|
||
|
|
||
|
impl<'a> Test<'a> {
|
||
|
fn storage(s: impl RepoStore + 'a) -> Self {
|
||
|
Test {
|
||
|
storage: Box::new(s),
|
||
|
}
|
||
|
}
|
||
|
fn s(&self) -> &Box<dyn RepoStore + Send + Sync + 'a> {
|
||
|
&self.storage
|
||
|
}
|
||
|
}
|
||
|
|
||
2 years ago
|
use crate::branch::*;
|
||
10 months ago
|
|
||
11 months ago
|
use crate::repo::Repo;
|
||
10 months ago
|
|
||
|
use crate::log::*;
|
||
10 months ago
|
use crate::utils::*;
|
||
2 years ago
|
|
||
|
#[test]
|
||
|
pub fn test_branch() {
|
||
|
fn add_obj(
|
||
11 months ago
|
content: ObjectContentV0,
|
||
10 months ago
|
header: Option<CommitHeader>,
|
||
|
store_pubkey: &StoreRepo,
|
||
|
store_secret: &ReadCapSecret,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
2 years ago
|
) -> ObjectRef {
|
||
|
let max_object_size = 4000;
|
||
10 months ago
|
let mut obj = Object::new(
|
||
|
ObjectContent::V0(content),
|
||
|
header,
|
||
|
max_object_size,
|
||
|
store_pubkey,
|
||
|
store_secret,
|
||
|
);
|
||
2 years ago
|
log_debug!(">>> add_obj");
|
||
|
log_debug!(" id: {:?}", obj.id());
|
||
11 months ago
|
log_debug!(" header: {:?}", obj.header());
|
||
10 months ago
|
obj.save_in_test(store).unwrap();
|
||
2 years ago
|
obj.reference().unwrap()
|
||
|
}
|
||
|
|
||
|
fn add_commit(
|
||
11 months ago
|
branch: BranchId,
|
||
2 years ago
|
author_privkey: PrivKey,
|
||
|
author_pubkey: PubKey,
|
||
11 months ago
|
seq: u64,
|
||
2 years ago
|
deps: Vec<ObjectRef>,
|
||
|
acks: Vec<ObjectRef>,
|
||
|
body_ref: ObjectRef,
|
||
10 months ago
|
store_pubkey: &StoreRepo,
|
||
|
store_secret: &ReadCapSecret,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
2 years ago
|
) -> ObjectRef {
|
||
10 months ago
|
let header = CommitHeader::new_with_deps_and_acks(
|
||
11 months ago
|
deps.iter().map(|r| r.id).collect(),
|
||
|
acks.iter().map(|r| r.id).collect(),
|
||
|
);
|
||
2 years ago
|
|
||
10 months ago
|
let overlay = store_pubkey.overlay_id_for_read_purpose();
|
||
|
|
||
2 years ago
|
let obj_ref = ObjectRef {
|
||
|
id: ObjectId::Blake3Digest32([1; 32]),
|
||
|
key: SymKey::ChaCha20Key([2; 32]),
|
||
|
};
|
||
|
let refs = vec![obj_ref];
|
||
|
let metadata = vec![5u8; 55];
|
||
|
|
||
11 months ago
|
let commit = CommitV0::new(
|
||
10 months ago
|
&author_privkey,
|
||
|
&author_pubkey,
|
||
|
overlay,
|
||
2 years ago
|
branch,
|
||
11 months ago
|
QuorumType::NoSigning,
|
||
2 years ago
|
deps,
|
||
11 months ago
|
vec![],
|
||
2 years ago
|
acks,
|
||
11 months ago
|
vec![],
|
||
2 years ago
|
refs,
|
||
11 months ago
|
vec![],
|
||
2 years ago
|
metadata,
|
||
|
body_ref,
|
||
|
)
|
||
|
.unwrap();
|
||
2 years ago
|
//log_debug!("commit: {:?}", commit);
|
||
2 years ago
|
add_obj(
|
||
10 months ago
|
ObjectContentV0::Commit(Commit::V0(commit)),
|
||
11 months ago
|
header,
|
||
10 months ago
|
store_pubkey,
|
||
|
store_secret,
|
||
2 years ago
|
store,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
fn add_body_branch(
|
||
11 months ago
|
branch: BranchV0,
|
||
10 months ago
|
store_pubkey: &StoreRepo,
|
||
|
store_secret: &ReadCapSecret,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
2 years ago
|
) -> ObjectRef {
|
||
10 months ago
|
let body: CommitBodyV0 = CommitBodyV0::Branch(Branch::V0(branch));
|
||
2 years ago
|
//log_debug!("body: {:?}", body);
|
||
2 years ago
|
add_obj(
|
||
10 months ago
|
ObjectContentV0::CommitBody(CommitBody::V0(body)),
|
||
11 months ago
|
None,
|
||
10 months ago
|
store_pubkey,
|
||
|
store_secret,
|
||
2 years ago
|
store,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
fn add_body_trans(
|
||
10 months ago
|
header: Option<CommitHeader>,
|
||
|
store_pubkey: &StoreRepo,
|
||
|
store_secret: &ReadCapSecret,
|
||
11 months ago
|
store: &Box<impl RepoStore + ?Sized>,
|
||
2 years ago
|
) -> ObjectRef {
|
||
|
let content = [7u8; 777].to_vec();
|
||
10 months ago
|
let body = CommitBodyV0::AsyncTransaction(Transaction::V0(content));
|
||
2 years ago
|
//log_debug!("body: {:?}", body);
|
||
2 years ago
|
add_obj(
|
||
10 months ago
|
ObjectContentV0::CommitBody(CommitBody::V0(body)),
|
||
11 months ago
|
header,
|
||
10 months ago
|
store_pubkey,
|
||
|
store_secret,
|
||
2 years ago
|
store,
|
||
|
)
|
||
|
}
|
||
|
|
||
10 months ago
|
let hashmap_storage = HashMapRepoStore::new();
|
||
|
let t = Test::storage(hashmap_storage);
|
||
2 years ago
|
|
||
|
// repo
|
||
|
|
||
10 months ago
|
let (repo_privkey, repo_pubkey) = generate_keypair();
|
||
|
let (store_repo, repo_secret) = StoreRepo::dummy_public_v0();
|
||
2 years ago
|
|
||
|
// branch
|
||
|
|
||
10 months ago
|
let (branch_privkey, branch_pubkey) = generate_keypair();
|
||
2 years ago
|
|
||
10 months ago
|
let (member_privkey, member_pubkey) = generate_keypair();
|
||
2 years ago
|
|
||
|
let metadata = [66u8; 64].to_vec();
|
||
11 months ago
|
|
||
|
let repo = Repo::new_with_member(
|
||
|
&repo_pubkey,
|
||
10 months ago
|
&member_pubkey,
|
||
|
&[PermissionV0::WriteAsync],
|
||
10 months ago
|
store_repo.overlay_id_for_read_purpose(),
|
||
10 months ago
|
t.s(),
|
||
11 months ago
|
);
|
||
|
|
||
|
let repo_ref = ObjectRef {
|
||
|
id: ObjectId::Blake3Digest32([1; 32]),
|
||
|
key: SymKey::ChaCha20Key([2; 32]),
|
||
|
};
|
||
|
|
||
|
let root_branch_def_id = ObjectId::Blake3Digest32([1; 32]);
|
||
|
|
||
|
let branch = BranchV0::new(
|
||
2 years ago
|
branch_pubkey,
|
||
11 months ago
|
repo_ref,
|
||
|
root_branch_def_id,
|
||
|
repo_privkey,
|
||
2 years ago
|
metadata,
|
||
|
);
|
||
2 years ago
|
//log_debug!("branch: {:?}", branch);
|
||
2 years ago
|
|
||
|
fn print_branch() {
|
||
2 years ago
|
log_debug!("branch deps/acks:");
|
||
|
log_debug!("");
|
||
|
log_debug!(" br");
|
||
|
log_debug!(" / \\");
|
||
|
log_debug!(" t1 t2");
|
||
|
log_debug!(" / \\ / \\");
|
||
|
log_debug!(" a3 t4<--t5-->(t1)");
|
||
|
log_debug!(" / \\");
|
||
|
log_debug!(" a6 a7");
|
||
|
log_debug!("");
|
||
2 years ago
|
}
|
||
|
|
||
|
print_branch();
|
||
|
|
||
|
// commit bodies
|
||
|
|
||
10 months ago
|
let branch_body = add_body_branch(
|
||
|
branch.clone(),
|
||
|
&store_repo,
|
||
|
&repo_secret,
|
||
|
repo.get_storage(),
|
||
|
);
|
||
11 months ago
|
|
||
10 months ago
|
let trans_body = add_body_trans(None, &store_repo, &repo_secret, repo.get_storage());
|
||
2 years ago
|
|
||
|
// create & add commits to store
|
||
|
|
||
2 years ago
|
log_debug!(">> br");
|
||
2 years ago
|
let br = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
0,
|
||
|
vec![],
|
||
|
vec![],
|
||
2 years ago
|
branch_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
2 years ago
|
log_debug!(">> t1");
|
||
2 years ago
|
let t1 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
1,
|
||
2 years ago
|
vec![br.clone()],
|
||
2 years ago
|
vec![],
|
||
2 years ago
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
2 years ago
|
log_debug!(">> t2");
|
||
2 years ago
|
let t2 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
2,
|
||
2 years ago
|
vec![br.clone()],
|
||
2 years ago
|
vec![],
|
||
2 years ago
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
11 months ago
|
// log_debug!(">> a3");
|
||
|
// let a3 = add_commit(
|
||
|
// branch_pubkey,
|
||
|
// member_privkey.clone(),
|
||
|
// member_pubkey,
|
||
|
// 3,
|
||
|
// vec![t1.clone()],
|
||
|
// vec![],
|
||
|
// ack_body.clone(),
|
||
|
// repo_pubkey,
|
||
|
// repo_secret.clone(),
|
||
|
// &mut store,
|
||
|
// );
|
||
2 years ago
|
|
||
2 years ago
|
log_debug!(">> t4");
|
||
2 years ago
|
let t4 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
4,
|
||
2 years ago
|
vec![t2.clone()],
|
||
|
vec![t1.clone()],
|
||
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
2 years ago
|
log_debug!(">> t5");
|
||
2 years ago
|
let t5 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
5,
|
||
2 years ago
|
vec![t1.clone(), t2.clone()],
|
||
|
vec![t4.clone()],
|
||
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
2 years ago
|
log_debug!(">> a6");
|
||
2 years ago
|
let a6 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
6,
|
||
2 years ago
|
vec![t4.clone()],
|
||
2 years ago
|
vec![],
|
||
11 months ago
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
2 years ago
|
log_debug!(">> a7");
|
||
2 years ago
|
let a7 = add_commit(
|
||
11 months ago
|
branch_pubkey,
|
||
2 years ago
|
member_privkey.clone(),
|
||
2 years ago
|
member_pubkey,
|
||
|
7,
|
||
2 years ago
|
vec![t4.clone()],
|
||
2 years ago
|
vec![],
|
||
11 months ago
|
trans_body.clone(),
|
||
10 months ago
|
&store_repo,
|
||
|
&repo_secret,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
);
|
||
|
|
||
10 months ago
|
let c7 = Commit::load(a7.clone(), repo.get_storage(), true).unwrap();
|
||
10 months ago
|
c7.verify(&repo).unwrap();
|
||
2 years ago
|
|
||
10 months ago
|
// let mut filter = Filter::new(FilterBuilder::new(10, 0.01));
|
||
|
// for commit_ref in [br, t1, t2, t5.clone(), a6.clone()] {
|
||
|
// match commit_ref.id {
|
||
|
// ObjectId::Blake3Digest32(d) => filter.add(&d),
|
||
|
// }
|
||
|
// }
|
||
|
// let cfg = filter.config();
|
||
|
// let their_commits = BloomFilter {
|
||
|
// k: cfg.hashes,
|
||
|
// f: filter.get_u8_array().to_vec(),
|
||
|
// };
|
||
2 years ago
|
|
||
|
print_branch();
|
||
2 years ago
|
log_debug!(">> sync_req");
|
||
|
log_debug!(" our_heads: [a3, t5, a6, a7]");
|
||
10 months ago
|
log_debug!(" known_heads: [a3, t5]");
|
||
2 years ago
|
log_debug!(" their_commits: [br, t1, t2, a3, t5, a6]");
|
||
2 years ago
|
|
||
|
let ids = Branch::sync_req(
|
||
11 months ago
|
&[t5.id, a6.id, a7.id],
|
||
|
&[t5.id],
|
||
10 months ago
|
//&their_commits,
|
||
10 months ago
|
repo.get_storage(),
|
||
2 years ago
|
)
|
||
|
.unwrap();
|
||
|
|
||
|
assert_eq!(ids.len(), 1);
|
||
|
assert!(ids.contains(&a7.id));
|
||
|
}
|
||
|
}
|