rad:z3gqcJUoA1n9HaHKufZs5FCSGazv5 heartwood48551cde934370c0cf8a4127385d4a4a41c5ba4d
{
"request": "trigger",
"version": 1,
"event_type": "push",
"repository": {
"id": "rad:z3gqcJUoA1n9HaHKufZs5FCSGazv5",
"name": "heartwood",
"description": "Radicle Heartwood Protocol & Stack",
"private": false,
"default_branch": "master",
"delegates": [
"did:key:z6MksFqXN3Yhqk8pTJdUGLwATkRfQvwZXPqR2qMEhbS9wzpT",
"did:key:z6MktaNvN1KVFMkSRAiN4qK5yvX1zuEEaseeX5sffhzPZRZW",
"did:key:z6MkireRatUThvd3qzfKht1S44wpm4FEWSSa4PRMTSQZ3voM",
"did:key:z6MkgFq6z5fkF2hioLLSNu1zP2qEL1aHXHZzGH1FLFGAnBGz",
"did:key:z6MkkPvBfjP4bQmco5Dm7UGsX2ruDBieEHi8n9DVJWX5sTEz"
]
},
"pusher": {
"id": "did:key:z6MkireRatUThvd3qzfKht1S44wpm4FEWSSa4PRMTSQZ3voM",
"alias": "fintohaps"
},
"before": "48551cde934370c0cf8a4127385d4a4a41c5ba4d",
"after": "48551cde934370c0cf8a4127385d4a4a41c5ba4d",
"branch": "master",
"commits": [
"48551cde934370c0cf8a4127385d4a4a41c5ba4d"
]
}
{
"response": "triggered",
"run_id": {
"id": "4f2a868f-8682-4b20-bcf0-15e16e8940fe"
},
"info_url": "https://cci.rad.levitte.org//4f2a868f-8682-4b20-bcf0-15e16e8940fe.html"
}
Started at: 2026-04-02 17:10:39.651359+02:00
Commands:
$ rad clone rad:z3gqcJUoA1n9HaHKufZs5FCSGazv5 .
✓ Creating checkout in ./...
✓ Remote cloudhead@z6MksFqXN3Yhqk8pTJdUGLwATkRfQvwZXPqR2qMEhbS9wzpT added
✓ Remote-tracking branch cloudhead@z6MksFqXN3Yhqk8pTJdUGLwATkRfQvwZXPqR2qMEhbS9wzpT/master created for z6MksFqXN3Yhqk8pTJdUGLwATkRfQvwZXPqR2qMEhbS9wzpT
✓ Remote cloudhead@z6MktaNvN1KVFMkSRAiN4qK5yvX1zuEEaseeX5sffhzPZRZW added
✓ Remote-tracking branch cloudhead@z6MktaNvN1KVFMkSRAiN4qK5yvX1zuEEaseeX5sffhzPZRZW/master created for z6MktaNvN1KVFMkSRAiN4qK5yvX1zuEEaseeX5sffhzPZRZW
✓ Remote fintohaps@z6MkireRatUThvd3qzfKht1S44wpm4FEWSSa4PRMTSQZ3voM added
✓ Remote-tracking branch fintohaps@z6MkireRatUThvd3qzfKht1S44wpm4FEWSSa4PRMTSQZ3voM/master created for z6MkireRatUThvd3qzfKht1S44wpm4FEWSSa4PRMTSQZ3voM
✓ Remote erikli@z6MkgFq6z5fkF2hioLLSNu1zP2qEL1aHXHZzGH1FLFGAnBGz added
✓ Remote-tracking branch erikli@z6MkgFq6z5fkF2hioLLSNu1zP2qEL1aHXHZzGH1FLFGAnBGz/master created for z6MkgFq6z5fkF2hioLLSNu1zP2qEL1aHXHZzGH1FLFGAnBGz
✓ Remote lorenz@z6MkkPvBfjP4bQmco5Dm7UGsX2ruDBieEHi8n9DVJWX5sTEz added
✓ Remote-tracking branch lorenz@z6MkkPvBfjP4bQmco5Dm7UGsX2ruDBieEHi8n9DVJWX5sTEz/master created for z6MkkPvBfjP4bQmco5Dm7UGsX2ruDBieEHi8n9DVJWX5sTEz
✓ Repository successfully cloned under /opt/radcis/ci.rad.levitte.org/cci/state/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/
╭────────────────────────────────────╮
│ heartwood │
│ Radicle Heartwood Protocol & Stack │
│ 154 issues · 42 patches │
╰────────────────────────────────────╯
Run `cd ./.` to go to the repository directory.
Exit code: 0
$ git config advice.detachedHead false
Exit code: 0
$ git checkout 48551cde934370c0cf8a4127385d4a4a41c5ba4d
HEAD is now at 48551cde crypto: Use `ssh-agent-lib` for SSH Agent
Exit code: 0
$ git show 48551cde934370c0cf8a4127385d4a4a41c5ba4d
commit 48551cde934370c0cf8a4127385d4a4a41c5ba4d
Author: Wiktor Kwapisiewicz <wiktor@metacode.biz>
Date: Wed Jun 12 14:12:24 2024 +0200
crypto: Use `ssh-agent-lib` for SSH Agent
This makes the `radicle-ssh` crate obsolete.
Signed-off-by: Wiktor Kwapisiewicz <wiktor@metacode.biz>
Co-authored-by: Lorenz Leutgeb <lorenz.leutgeb@radicle.xyz>
diff --git a/Cargo.lock b/Cargo.lock
index f979b7aa..b8451af1 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -686,6 +686,32 @@ dependencies = [
"cipher",
]
+[[package]]
+name = "curve25519-dalek"
+version = "4.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97fb8b7c4503de7d6ae7b42ab72a5a59857b4c937ec27a3d4539dba95b5ab2be"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "curve25519-dalek-derive",
+ "digest",
+ "fiat-crypto",
+ "rustc_version",
+ "subtle",
+]
+
+[[package]]
+name = "curve25519-dalek-derive"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f46882e17999c6cc590af592290432be3bce0428cb0d5f8b6715e4dc7b383eb3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
[[package]]
name = "cypheraddr"
version = "0.4.0"
@@ -843,7 +869,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bdfd533a2fc01178c738c99412ae1f7e1ad2cb37c2e14bfd87e9d4618171c825"
dependencies = [
"ct-codecs",
- "ed25519",
+ "ed25519 1.5.3",
"getrandom 0.2.17",
]
@@ -870,6 +896,27 @@ dependencies = [
"signature 1.6.4",
]
+[[package]]
+name = "ed25519"
+version = "2.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "115531babc129696a58c64a4fef0a8bf9e9698629fb97e9e40767d235cfbcd53"
+dependencies = [
+ "signature 2.2.0",
+]
+
+[[package]]
+name = "ed25519-dalek"
+version = "2.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "70e796c081cee67dc755e1a36a0a172b897fab85fc3f6bc48307991f64e4eca9"
+dependencies = [
+ "curve25519-dalek",
+ "ed25519 2.2.3",
+ "sha2",
+ "subtle",
+]
+
[[package]]
name = "either"
version = "1.15.0"
@@ -1021,6 +1068,12 @@ dependencies = [
"subtle",
]
+[[package]]
+name = "fiat-crypto"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "28dea519a9695b9977216879a3ebfddf92f1c08c05d984f8996aecd6ecdc811d"
+
[[package]]
name = "filetime"
version = "0.2.27"
@@ -2953,7 +3006,6 @@ dependencies = [
"radicle-git-ref-format",
"radicle-localtime",
"radicle-oid",
- "radicle-ssh",
"schemars",
"serde",
"serde-untagged",
@@ -3083,14 +3135,15 @@ dependencies = [
"multibase",
"qcheck",
"qcheck-macros",
- "radicle-ssh",
"schemars",
"serde",
"signature 2.2.0",
"sqlite",
+ "ssh-agent-lib",
"ssh-key",
"tempfile",
"thiserror 2.0.18",
+ "winpipe",
"zeroize",
]
@@ -3270,15 +3323,6 @@ dependencies = [
"windows 0.62.2",
]
-[[package]]
-name = "radicle-ssh"
-version = "0.10.0"
-dependencies = [
- "thiserror 2.0.18",
- "winpipe",
- "zeroize",
-]
-
[[package]]
name = "radicle-std-ext"
version = "0.2.0"
@@ -3648,6 +3692,15 @@ dependencies = [
"zeroize",
]
+[[package]]
+name = "secrecy"
+version = "0.10.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e891af845473308773346dc847b2c23ee78fe442e0472ac50e22a18a93d3ae5a"
+dependencies = [
+ "zeroize",
+]
+
[[package]]
name = "sem_safe"
version = "0.2.1"
@@ -3989,6 +4042,22 @@ dependencies = [
"sqlite3-src",
]
+[[package]]
+name = "ssh-agent-lib"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3d0582e6e5724c4a5d038472976e8fad64d824b3de37210aa60c35883e3e766"
+dependencies = [
+ "byteorder",
+ "log",
+ "secrecy",
+ "signature 2.2.0",
+ "ssh-encoding",
+ "ssh-key",
+ "subtle",
+ "thiserror 2.0.18",
+]
+
[[package]]
name = "ssh-cipher"
version = "0.2.0"
@@ -4024,6 +4093,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3b86f5297f0f04d08cabaa0f6bff7cb6aec4d9c3b49d87990d63da9d9156a8c3"
dependencies = [
"bcrypt-pbkdf",
+ "ed25519-dalek",
+ "num-bigint-dig",
"p256",
"p384",
"p521",
diff --git a/Cargo.toml b/Cargo.toml
index 418d26fe..16e3773e 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -59,7 +59,6 @@ radicle-node = { version = "0.19", path = "crates/radicle-node" }
radicle-oid = { version = "0.1.0", path = "crates/radicle-oid", default-features = false }
radicle-protocol = { version = "0.7", path = "crates/radicle-protocol" }
radicle-signals = { version = "0.11", path = "crates/radicle-signals" }
-radicle-ssh = { version = "0.10", path = "crates/radicle-ssh", default-features = false }
radicle-systemd = { version = "0.12", path = "crates/radicle-systemd" }
radicle-term = { version = "0.17", path = "crates/radicle-term" }
radicle-windows = { version = "0.1", path = "crates/radicle-windows" }
diff --git a/HACKING.md b/HACKING.md
index e900fcf9..a587c0c4 100644
--- a/HACKING.md
+++ b/HACKING.md
@@ -26,7 +26,6 @@ The repository is structured in *crates*, as follows:
* `radicle-dag`: A simple directed acyclic graph implementation used by `radicle-cob`.
* `radicle-node`: The radicle peer-to-peer daemon that enables users to connect to the network and share code.
* `radicle-remote-helper`: A Git remote helper for `rad://` remotes.
-* `radicle-ssh`: OpenSSH functionality, including a library used to interface with `ssh-agent`.
* `radicle-term`: A generic terminal library used by the Radicle CLI.
* `radicle-tools`: Tools used to aid in the development of Radicle.
diff --git a/crates/radicle-cli/src/commands/self.rs b/crates/radicle-cli/src/commands/self.rs
index 1feb12f5..ec3d3c2a 100644
--- a/crates/radicle-cli/src/commands/self.rs
+++ b/crates/radicle-cli/src/commands/self.rs
@@ -63,12 +63,7 @@ fn all(profile: &Profile) -> anyhow::Result<()> {
table.push([term::format::style("Node").into(), node.to_string().into()]);
let ssh_agent = match ssh::agent::Agent::connect() {
- Ok(c) => term::format::positive(format!(
- "running ({})",
- c.path()
- .map(|p| p.display().to_string())
- .unwrap_or(String::from("?"))
- )),
+ Ok(c) => term::format::positive(format!("running ({})", c.path().display())),
Err(e) if e.is_not_running() => term::format::yellow(String::from("not running")),
Err(e) => term::format::negative(format!("error: {e}")),
};
diff --git a/crates/radicle-crypto/Cargo.toml b/crates/radicle-crypto/Cargo.toml
index 1450f488..0dff49ee 100644
--- a/crates/radicle-crypto/Cargo.toml
+++ b/crates/radicle-crypto/Cargo.toml
@@ -14,7 +14,7 @@ rust-version.workspace = true
[features]
test = ["fastrand", "qcheck"]
-ssh = ["radicle-ssh", "ssh-key"]
+ssh = ["ssh-agent-lib", "ssh-key"]
[dependencies]
amplify = { workspace = true }
@@ -24,15 +24,18 @@ fastrand = { workspace = true, optional = true }
multibase = { workspace = true }
qcheck = { workspace = true, optional = true }
git-ref-format-core = { workspace = true, optional = true }
-radicle-ssh = { workspace = true, optional = true }
schemars = { workspace = true, optional = true, features = ["derive", "std"] }
serde = { workspace = true, features = ["derive", "std"] }
signature = { workspace = true, features = ["std"] }
sqlite = { workspace = true, features = ["bundled"], optional = true }
+ssh-agent-lib = { version = "0.5.2", optional = true, default-features = false, features = ["log"] }
ssh-key = { version = "0.6.3", default-features = false, features = ["std", "encryption", "getrandom"], optional = true }
thiserror = { workspace = true, default-features = true }
zeroize = { workspace = true }
+[target.'cfg(windows)'.dependencies]
+winpipe = { workspace = true }
+
[dev-dependencies]
fastrand = { workspace = true }
qcheck = { workspace = true }
diff --git a/crates/radicle-crypto/src/ssh.rs b/crates/radicle-crypto/src/ssh.rs
index 45259ca5..9c1516bd 100644
--- a/crates/radicle-crypto/src/ssh.rs
+++ b/crates/radicle-crypto/src/ssh.rs
@@ -1,17 +1,9 @@
pub mod agent;
pub mod keystore;
-use std::io;
-
use thiserror::Error;
-use radicle_ssh::encoding;
-use radicle_ssh::encoding::Encodable;
-use radicle_ssh::encoding::Encoding;
-use radicle_ssh::encoding::Reader;
-
use crate as crypto;
-use crate::PublicKey;
pub use keystore::{Keystore, Passphrase};
@@ -127,218 +119,3 @@ pub mod fmt {
}
}
}
-
-#[derive(Debug, Error)]
-#[non_exhaustive]
-pub enum SignatureError {
- #[error(transparent)]
- Invalid(#[from] crypto::Error),
- #[error(transparent)]
- Encoding(#[from] encoding::Error),
- #[error("unknown algorithm '{0}'")]
- UnknownAlgorithm(String),
-}
-
-impl Encodable for crypto::Signature {
- type Error = SignatureError;
-
- fn read(r: &mut encoding::Cursor) -> Result<Self, Self::Error> {
- let buf = r.read_string()?;
- let mut inner_strs = buf.reader(0);
-
- let sig_type = inner_strs.read_string()?;
- if sig_type != b"ssh-ed25519" {
- return Err(SignatureError::UnknownAlgorithm(
- String::from_utf8_lossy(sig_type).to_string(),
- ));
- }
- let sig = crypto::Signature::try_from(inner_strs.read_string()?)?;
-
- Ok(sig)
- }
-
- fn write<E: Encoding>(&self, buf: &mut E) {
- let mut inner_strs = Vec::new();
- inner_strs.extend_ssh_string(b"ssh-ed25519");
- inner_strs.extend_ssh_string(self.as_ref());
- buf.extend_ssh_string(&inner_strs);
- }
-}
-
-#[derive(Debug, Error)]
-#[non_exhaustive]
-pub enum PublicKeyError {
- #[error(transparent)]
- Invalid(#[from] crypto::Error),
- #[error(transparent)]
- Encoding(#[from] encoding::Error),
- #[error("unknown algorithm '{0}'")]
- UnknownAlgorithm(String),
-}
-
-impl Encodable for PublicKey {
- type Error = PublicKeyError;
-
- fn read(r: &mut encoding::Cursor) -> Result<Self, Self::Error> {
- match r.read_string()? {
- b"ssh-ed25519" => {
- let s = r.read_string()?;
- let p = PublicKey::try_from(s)?;
-
- Ok(p)
- }
- v => Err(PublicKeyError::UnknownAlgorithm(
- String::from_utf8_lossy(v).to_string(),
- )),
- }
- }
-
- fn write<E: Encoding>(&self, w: &mut E) {
- let mut str_w: Vec<u8> = Vec::<u8>::new();
- str_w.extend_ssh_string(b"ssh-ed25519");
- str_w.extend_ssh_string(&self[..]);
- w.extend_ssh_string(&str_w)
- }
-}
-
-#[derive(Debug, Error)]
-#[non_exhaustive]
-pub enum SecretKeyError {
- #[error(transparent)]
- Encoding(#[from] encoding::Error),
- #[error(transparent)]
- Crypto(#[from] crypto::Error),
- #[error(transparent)]
- Io(#[from] io::Error),
- #[error("unknown algorithm '{0}'")]
- UnknownAlgorithm(String),
- #[error("public key does not match secret key")]
- Mismatch,
-}
-
-impl Encodable for crypto::SecretKey {
- type Error = SecretKeyError;
-
- fn read(r: &mut encoding::Cursor) -> Result<Self, Self::Error> {
- match r.read_string()? {
- b"ssh-ed25519" => {
- let public = r.read_string()?;
- let pair = r.read_string()?;
- let _comment = r.read_string()?;
- let key = crypto::SecretKey::try_from(pair)?;
-
- if public != key.public_key().as_ref() {
- return Err(SecretKeyError::Mismatch);
- }
- Ok(key)
- }
- s => Err(SecretKeyError::UnknownAlgorithm(
- String::from_utf8_lossy(s).to_string(),
- )),
- }
- }
-
- fn write<E: Encoding>(&self, buf: &mut E) {
- let public = self.0.public_key();
-
- buf.extend_ssh_string(b"ssh-ed25519");
- buf.extend_ssh_string(public.as_ref());
- buf.extend_ssh_string(self.0.as_ref());
- buf.extend_ssh_string(b"radicle");
- }
-}
-
-#[cfg(test)]
-mod test {
- use std::sync::{Arc, Mutex};
-
- use qcheck_macros::quickcheck;
-
- use crate as crypto;
- use crate::{PublicKey, SecretKey};
- use radicle_ssh::agent::client::{AgentClient, ClientStream, Error};
- use radicle_ssh::encoding::*;
-
- #[derive(Clone, Default)]
- struct DummyStream {
- incoming: Arc<Mutex<Vec<u8>>>,
- }
-
- impl ClientStream for DummyStream {
- fn request(&mut self, buf: &[u8]) -> Result<Buffer, Error> {
- *self.incoming.lock().unwrap() = buf.to_vec();
-
- Ok(Buffer::default())
- }
- }
-
- #[quickcheck]
- fn prop_encode_decode_sk(input: [u8; 64]) {
- let mut buf = Buffer::default();
- let sk = crypto::SecretKey::from(input);
- sk.write(&mut buf);
-
- let mut cursor = buf.reader(0);
- let output = SecretKey::read(&mut cursor).unwrap();
-
- assert_eq!(sk, output);
- }
-
- #[test]
- fn test_agent_encoding_remove() {
- use std::str::FromStr;
-
- let pk = PublicKey::from_str("z6MktWkM9vcfysWFq1c2aaLjJ6j4PYYg93TLPswR4qtuoAeT").unwrap();
- let expected = [
- 0, 0, 0, 56, // Message length
- 18, // Message type (remove identity)
- 0, 0, 0, 51, // Key blob length
- 0, 0, 0, 11, // Key type length
- 115, 115, 104, 45, 101, 100, 50, 53, 53, 49, 57, // Key type
- 0, 0, 0, 32, // Key length
- 208, 232, 92, 138, 225, 114, 116, 99, 156, 177, 148, 93, 65, 93, 198, 25, 46, 203, 79,
- 37, 145, 51, 176, 174, 61, 136, 160, 107, 4, 95, 175, 144, // Key
- ];
-
- let stream = DummyStream::default();
- let mut agent = AgentClient::new(None, stream.clone());
-
- agent.remove_identity(&pk).unwrap();
-
- assert_eq!(
- stream.incoming.lock().unwrap().as_slice(),
- expected.as_slice()
- );
- }
-
- #[test]
- fn test_agent_encoding_sign() {
- use std::str::FromStr;
-
- let pk = PublicKey::from_str("z6MktWkM9vcfysWFq1c2aaLjJ6j4PYYg93TLPswR4qtuoAeT").unwrap();
- let expected = [
- 0, 0, 0, 73, // Message length
- 13, // Message type (sign request)
- 0, 0, 0, 51, // Key blob length
- 0, 0, 0, 11, // Key type length
- 115, 115, 104, 45, 101, 100, 50, 53, 53, 49, 57, // Key type
- 0, 0, 0, 32, // Public key
- 208, 232, 92, 138, 225, 114, 116, 99, 156, 177, 148, 93, 65, 93, 198, 25, 46, 203, 79,
- 37, 145, 51, 176, 174, 61, 136, 160, 107, 4, 95, 175, 144, // Key
- 0, 0, 0, 9, // Length of data to sign
- 1, 2, 3, 4, 5, 6, 7, 8, 9, // Data to sign
- 0, 0, 0, 0, // Signature flags
- ];
-
- let stream = DummyStream::default();
- let mut agent = AgentClient::new(None, stream.clone());
- let data: Vec<u8> = vec![1, 2, 3, 4, 5, 6, 7, 8, 9];
-
- agent.sign(&pk, &data).ok();
-
- assert_eq!(
- stream.incoming.lock().unwrap().as_slice(),
- expected.as_slice()
- );
- }
-}
diff --git a/crates/radicle-crypto/src/ssh/agent.rs b/crates/radicle-crypto/src/ssh/agent.rs
index 864d53b9..93ea00b8 100644
--- a/crates/radicle-crypto/src/ssh/agent.rs
+++ b/crates/radicle-crypto/src/ssh/agent.rs
@@ -1,40 +1,109 @@
use std::cell::RefCell;
+use std::env::VarError;
use std::path::Path;
+use std::path::PathBuf;
-pub use radicle_ssh as ssh;
-pub use ssh::agent::client::{AgentClient, Error};
+use proto::Credential;
+use ssh_agent_lib::blocking::Client;
+pub use ssh_agent_lib::error::AgentError;
+use ssh_agent_lib::proto;
+use ssh_key::public::{Ed25519PublicKey, KeyData};
+use thiserror::Error;
use crate::{PublicKey, SecretKey, Signature, Signer};
use super::ExtendedSignature;
+#[cfg(unix)]
+use std::os::unix::net::UnixStream as Stream;
+
+#[cfg(windows)]
+use winpipe::WinStream as Stream;
+
+#[derive(Debug, Error)]
+pub enum ConnectError {
+ #[error(transparent)]
+ Agent(#[from] AgentError),
+ #[error("Unable to read environment variable '{var}': {source}")]
+ EnvVar { var: String, source: VarError },
+}
+
+impl ConnectError {
+ pub fn is_not_running(&self) -> bool {
+ use std::io::ErrorKind::*;
+ match self {
+ Self::EnvVar {
+ source: VarError::NotPresent,
+ ..
+ } => true,
+ Self::Agent(AgentError::IO(source)) if source.kind() == NotFound => true,
+ #[cfg(windows)]
+ Self::Agent(AgentError::IO(source)) if source.kind() == ConnectionRefused => {
+ // On Windows, a named pipe might be used, and if no
+ // agent is running, we might get a "connection refused"
+ // error, even though the `SSH_AUTH_SOCK` environment variable
+ // is set and a named pipe exists.
+ true
+ }
+ _ => false,
+ }
+ }
+}
+
pub struct Agent {
- client: AgentClient,
+ path: PathBuf,
+ client: Client<Stream>,
}
impl Agent {
/// Connect to a running SSH agent.
- pub fn connect() -> Result<Self, Error> {
- Ok(Self {
- client: AgentClient::connect_env()?,
- })
+ pub fn connect() -> Result<Self, ConnectError> {
+ const SSH_AUTH_SOCK: &str = "SSH_AUTH_SOCK";
+
+ let path =
+ PathBuf::from(
+ std::env::var(SSH_AUTH_SOCK).map_err(|err| ConnectError::EnvVar {
+ var: SSH_AUTH_SOCK.to_string(),
+ source: err,
+ })?,
+ );
+
+ let client = Client::new(
+ Stream::connect(&path).map_err(|err| ConnectError::Agent(AgentError::IO(err)))?,
+ );
+
+ Ok(Self { path, client })
}
/// Register a key with the agent.
- pub fn register(&mut self, key: &SecretKey) -> Result<(), ssh::Error> {
- self.client.add_identity(key, &[])
+ pub fn register(&mut self, key: &SecretKey) -> Result<(), AgentError> {
+ use ssh_key::private::{Ed25519Keypair, KeypairData};
+ self.client.add_identity(proto::AddIdentity {
+ credential: Credential::Key {
+ privkey: KeypairData::Ed25519(Ed25519Keypair::from_bytes(key).unwrap()),
+ comment: "".into(),
+ },
+ })
}
- pub fn unregister(&mut self, key: &PublicKey) -> Result<(), ssh::Error> {
- self.client.remove_identity(key)
+ pub fn unregister(&mut self, key: &PublicKey) -> Result<(), AgentError> {
+ self.client.remove_identity(proto::RemoveIdentity {
+ pubkey: Self::key_data(key),
+ })
}
- pub fn unregister_all(&mut self) -> Result<(), ssh::Error> {
+ pub fn unregister_all(&mut self) -> Result<(), AgentError> {
self.client.remove_all_identities()
}
- pub fn sign(&mut self, key: &PublicKey, data: &[u8]) -> Result<[u8; 64], ssh::Error> {
- self.client.sign(key, data)
+ pub fn sign(&mut self, key: &PublicKey, data: &[u8]) -> Result<[u8; 64], AgentError> {
+ let sig = self.client.sign(proto::SignRequest {
+ pubkey: Self::key_data(key),
+ data: data.to_vec(),
+ flags: 0,
+ })?;
+
+ Ok(sig.as_bytes().to_owned().try_into().unwrap())
}
/// Get a signer from this agent, given the public key.
@@ -42,12 +111,21 @@ impl Agent {
AgentSigner::new(self, key)
}
- pub fn path(&self) -> Option<&Path> {
- self.client.path()
+ pub fn path(&self) -> &Path {
+ self.path.as_ref()
+ }
+
+ pub fn request_identities(&mut self) -> Result<Vec<PublicKey>, AgentError> {
+ Ok(self
+ .client
+ .request_identities()?
+ .into_iter()
+ .filter_map(|identity| identity.pubkey.ed25519().map(|key| PublicKey::from(key.0)))
+ .collect())
}
- pub fn request_identities(&mut self) -> Result<Vec<PublicKey>, ssh::agent::client::Error> {
- self.client.request_identities()
+ fn key_data(key: &PublicKey) -> KeyData {
+ KeyData::Ed25519(Ed25519PublicKey(***key))
}
}
@@ -95,7 +173,7 @@ impl AgentSigner {
Self { agent, public }
}
- pub fn is_ready(&self) -> Result<bool, Error> {
+ pub fn is_ready(&self) -> Result<bool, AgentError> {
let ids = self.agent.borrow_mut().request_identities()?;
Ok(ids.contains(&self.public))
@@ -106,3 +184,79 @@ impl AgentSigner {
Box::new(self)
}
}
+
+#[cfg(test)]
+mod test {
+ use crate::PublicKey;
+ use ssh_agent_lib::blocking::Client;
+ use ssh_agent_lib::proto::SignRequest;
+ use ssh_agent_lib::ssh_key::public::{Ed25519PublicKey, KeyData};
+ use std::ops::Deref;
+
+ #[test]
+ fn test_agent_encoding_remove() {
+ use std::str::FromStr;
+
+ let pk = PublicKey::from_str("z6MktWkM9vcfysWFq1c2aaLjJ6j4PYYg93TLPswR4qtuoAeT").unwrap();
+ let expected = [
+ 0, 0, 0, 56, // Message length
+ 18, // Message type (remove identity)
+ 0, 0, 0, 51, // Key blob length
+ 0, 0, 0, 11, // Key type length
+ 115, 115, 104, 45, 101, 100, 50, 53, 53, 49, 57, // Key type
+ 0, 0, 0, 32, // Key length
+ 208, 232, 92, 138, 225, 114, 116, 99, 156, 177, 148, 93, 65, 93, 198, 25, 46, 203, 79,
+ 37, 145, 51, 176, 174, 61, 136, 160, 107, 4, 95, 175, 144, // Key
+ ];
+
+ let mut client = Client::new(std::io::Cursor::new(vec![]));
+
+ // We expect this to fail with an unexpected EOF, since the client will
+ // attempt to read a response from the stream, but the stream is empty,
+ // since we are not actually connected to SSH agent.
+ assert!(
+ matches!(client.remove_identity(ssh_agent_lib::proto::RemoveIdentity {
+ pubkey: KeyData::Ed25519(Ed25519PublicKey(**pk.deref())),
+ }),
+ Err(
+ super::AgentError::Proto(ssh_agent_lib::proto::ProtoError::IO(err)),
+ ) if err.kind() == std::io::ErrorKind::UnexpectedEof
+ )
+ );
+
+ assert_eq!(client.into_inner().into_inner(), expected.as_slice());
+ }
+
+ #[test]
+ fn test_agent_encoding_sign() {
+ use std::str::FromStr;
+
+ let pk = PublicKey::from_str("z6MktWkM9vcfysWFq1c2aaLjJ6j4PYYg93TLPswR4qtuoAeT").unwrap();
+ let expected = [
+ 0, 0, 0, 73, // Message length
+ 13, // Message type (sign request)
+ 0, 0, 0, 51, // Key blob length
+ 0, 0, 0, 11, // Key type length
+ 115, 115, 104, 45, 101, 100, 50, 53, 53, 49, 57, // Key type
+ 0, 0, 0, 32, // Public key
+ 208, 232, 92, 138, 225, 114, 116, 99, 156, 177, 148, 93, 65, 93, 198, 25, 46, 203, 79,
+ 37, 145, 51, 176, 174, 61, 136, 160, 107, 4, 95, 175, 144, // Key
+ 0, 0, 0, 9, // Length of data to sign
+ 1, 2, 3, 4, 5, 6, 7, 8, 9, // Data to sign
+ 0, 0, 0, 0, // Signature flags
+ ];
+
+ let mut client = Client::new(std::io::Cursor::new(vec![]));
+ let data: Vec<u8> = vec![1, 2, 3, 4, 5, 6, 7, 8, 9];
+
+ client
+ .sign(SignRequest {
+ pubkey: KeyData::Ed25519(Ed25519PublicKey(**pk.deref())),
+ data,
+ flags: 0,
+ })
+ .ok();
+
+ assert_eq!(client.into_inner().into_inner(), expected);
+ }
+}
diff --git a/crates/radicle-crypto/src/test/signer.rs b/crates/radicle-crypto/src/test/signer.rs
index b879fee0..59507bfb 100644
--- a/crates/radicle-crypto/src/test/signer.rs
+++ b/crates/radicle-crypto/src/test/signer.rs
@@ -1,4 +1,6 @@
-use crate::{KeyPair, PublicKey, SecretKey, Seed, Signature, ssh::ExtendedSignature};
+#[cfg(feature = "ssh")]
+use crate::ssh::ExtendedSignature;
+use crate::{KeyPair, PublicKey, SecretKey, Seed, Signature};
#[derive(Debug, Clone)]
pub struct MockSigner {
@@ -6,6 +8,7 @@ pub struct MockSigner {
sk: SecretKey,
}
+#[cfg(feature = "ssh")]
impl signature::Signer<ExtendedSignature> for MockSigner {
fn try_sign(&self, msg: &[u8]) -> Result<ExtendedSignature, signature::Error> {
use signature::Keypair as _;
diff --git a/crates/radicle-ssh/Cargo.toml b/crates/radicle-ssh/Cargo.toml
deleted file mode 100644
index 17c32abd..00000000
--- a/crates/radicle-ssh/Cargo.toml
+++ /dev/null
@@ -1,21 +0,0 @@
-[package]
-name = "radicle-ssh"
-description = "Radicle SSH library"
-homepage.workspace = true
-repository.workspace = true
-license = "Apache-2.0"
-version = "0.10.0"
-authors = [
- "Fintan Halpenny <fintan.halpenny@gmail.com>",
- "Pierre-Étienne Meunier <pe@pijul.org>",
- "cloudhead <cloudhead@radicle.xyz>"
-]
-edition.workspace = true
-rust-version.workspace = true
-
-[dependencies]
-thiserror = { workspace = true, default-features = true }
-zeroize = { workspace = true }
-
-[target.'cfg(windows)'.dependencies]
-winpipe = { workspace = true }
diff --git a/crates/radicle-ssh/LICENSE b/crates/radicle-ssh/LICENSE
deleted file mode 100644
index f433b1a5..00000000
--- a/crates/radicle-ssh/LICENSE
+++ /dev/null
@@ -1,177 +0,0 @@
-
- Apache License
- Version 2.0, January 2004
- http://www.apache.org/licenses/
-
- TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
-
- 1. Definitions.
-
- "License" shall mean the terms and conditions for use, reproduction,
- and distribution as defined by Sections 1 through 9 of this document.
-
- "Licensor" shall mean the copyright owner or entity authorized by
- the copyright owner that is granting the License.
-
- "Legal Entity" shall mean the union of the acting entity and all
- other entities that control, are controlled by, or are under common
- control with that entity. For the purposes of this definition,
- "control" means (i) the power, direct or indirect, to cause the
- direction or management of such entity, whether by contract or
- otherwise, or (ii) ownership of fifty percent (50%) or more of the
- outstanding shares, or (iii) beneficial ownership of such entity.
-
- "You" (or "Your") shall mean an individual or Legal Entity
- exercising permissions granted by this License.
-
- "Source" form shall mean the preferred form for making modifications,
- including but not limited to software source code, documentation
- source, and configuration files.
-
- "Object" form shall mean any form resulting from mechanical
- transformation or translation of a Source form, including but
- not limited to compiled object code, generated documentation,
- and conversions to other media types.
-
- "Work" shall mean the work of authorship, whether in Source or
- Object form, made available under the License, as indicated by a
- copyright notice that is included in or attached to the work
- (an example is provided in the Appendix below).
-
- "Derivative Works" shall mean any work, whether in Source or Object
- form, that is based on (or derived from) the Work and for which the
- editorial revisions, annotations, elaborations, or other modifications
- represent, as a whole, an original work of authorship. For the purposes
- of this License, Derivative Works shall not include works that remain
- separable from, or merely link (or bind by name) to the interfaces of,
- the Work and Derivative Works thereof.
-
- "Contribution" shall mean any work of authorship, including
- the original version of the Work and any modifications or additions
- to that Work or Derivative Works thereof, that is intentionally
- submitted to Licensor for inclusion in the Work by the copyright owner
- or by an individual or Legal Entity authorized to submit on behalf of
- the copyright owner. For the purposes of this definition, "submitted"
- means any form of electronic, verbal, or written communication sent
- to the Licensor or its representatives, including but not limited to
- communication on electronic mailing lists, source code control systems,
- and issue tracking systems that are managed by, or on behalf of, the
- Licensor for the purpose of discussing and improving the Work, but
- excluding communication that is conspicuously marked or otherwise
- designated in writing by the copyright owner as "Not a Contribution."
-
- "Contributor" shall mean Licensor and any individual or Legal Entity
- on behalf of whom a Contribution has been received by Licensor and
- subsequently incorporated within the Work.
-
- 2. Grant of Copyright License. Subject to the terms and conditions of
- this License, each Contributor hereby grants to You a perpetual,
- worldwide, non-exclusive, no-charge, royalty-free, irrevocable
- copyright license to reproduce, prepare Derivative Works of,
- publicly display, publicly perform, sublicense, and distribute the
- Work and such Derivative Works in Source or Object form.
-
- 3. Grant of Patent License. Subject to the terms and conditions of
- this License, each Contributor hereby grants to You a perpetual,
- worldwide, non-exclusive, no-charge, royalty-free, irrevocable
- (except as stated in this section) patent license to make, have made,
- use, offer to sell, sell, import, and otherwise transfer the Work,
- where such license applies only to those patent claims licensable
- by such Contributor that are necessarily infringed by their
- Contribution(s) alone or by combination of their Contribution(s)
- with the Work to which such Contribution(s) was submitted. If You
- institute patent litigation against any entity (including a
- cross-claim or counterclaim in a lawsuit) alleging that the Work
- or a Contribution incorporated within the Work constitutes direct
- or contributory patent infringement, then any patent licenses
- granted to You under this License for that Work shall terminate
- as of the date such litigation is filed.
-
- 4. Redistribution. You may reproduce and distribute copies of the
- Work or Derivative Works thereof in any medium, with or without
- modifications, and in Source or Object form, provided that You
- meet the following conditions:
-
- (a) You must give any other recipients of the Work or
- Derivative Works a copy of this License; and
-
- (b) You must cause any modified files to carry prominent notices
- stating that You changed the files; and
-
- (c) You must retain, in the Source form of any Derivative Works
- that You distribute, all copyright, patent, trademark, and
- attribution notices from the Source form of the Work,
- excluding those notices that do not pertain to any part of
- the Derivative Works; and
-
- (d) If the Work includes a "NOTICE" text file as part of its
- distribution, then any Derivative Works that You distribute must
- include a readable copy of the attribution notices contained
- within such NOTICE file, excluding those notices that do not
- pertain to any part of the Derivative Works, in at least one
- of the following places: within a NOTICE text file distributed
- as part of the Derivative Works; within the Source form or
- documentation, if provided along with the Derivative Works; or,
- within a display generated by the Derivative Works, if and
- wherever such third-party notices normally appear. The contents
- of the NOTICE file are for informational purposes only and
- do not modify the License. You may add Your own attribution
- notices within Derivative Works that You distribute, alongside
- or as an addendum to the NOTICE text from the Work, provided
- that such additional attribution notices cannot be construed
- as modifying the License.
-
- You may add Your own copyright statement to Your modifications and
- may provide additional or different license terms and conditions
- for use, reproduction, or distribution of Your modifications, or
- for any such Derivative Works as a whole, provided Your use,
- reproduction, and distribution of the Work otherwise complies with
- the conditions stated in this License.
-
- 5. Submission of Contributions. Unless You explicitly state otherwise,
- any Contribution intentionally submitted for inclusion in the Work
- by You to the Licensor shall be under the terms and conditions of
- this License, without any additional terms or conditions.
- Notwithstanding the above, nothing herein shall supersede or modify
- the terms of any separate license agreement you may have executed
- with Licensor regarding such Contributions.
-
- 6. Trademarks. This License does not grant permission to use the trade
- names, trademarks, service marks, or product names of the Licensor,
- except as required for reasonable and customary use in describing the
- origin of the Work and reproducing the content of the NOTICE file.
-
- 7. Disclaimer of Warranty. Unless required by applicable law or
- agreed to in writing, Licensor provides the Work (and each
- Contributor provides its Contributions) on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
- implied, including, without limitation, any warranties or conditions
- of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
- PARTICULAR PURPOSE. You are solely responsible for determining the
- appropriateness of using or redistributing the Work and assume any
- risks associated with Your exercise of permissions under this License.
-
- 8. Limitation of Liability. In no event and under no legal theory,
- whether in tort (including negligence), contract, or otherwise,
- unless required by applicable law (such as deliberate and grossly
- negligent acts) or agreed to in writing, shall any Contributor be
- liable to You for damages, including any direct, indirect, special,
- incidental, or consequential damages of any character arising as a
- result of this License or out of the use or inability to use the
- Work (including but not limited to damages for loss of goodwill,
- work stoppage, computer failure or malfunction, or any and all
- other commercial damages or losses), even if such Contributor
- has been advised of the possibility of such damages.
-
- 9. Accepting Warranty or Additional Liability. While redistributing
- the Work or Derivative Works thereof, You may choose to offer,
- and charge a fee for, acceptance of support, warranty, indemnity,
- or other liability obligations and/or rights consistent with this
- License. However, in accepting such obligations, You may act only
- on Your own behalf and on Your sole responsibility, not on behalf
- of any other Contributor, and only if You agree to indemnify,
- defend, and hold each Contributor harmless for any liability
- incurred by, or claims asserted against, such Contributor by reason
- of your accepting any such warranty or additional liability.
-
- END OF TERMS AND CONDITIONS
diff --git a/crates/radicle-ssh/src/agent.rs b/crates/radicle-ssh/src/agent.rs
deleted file mode 100644
index fd54bf85..00000000
--- a/crates/radicle-ssh/src/agent.rs
+++ /dev/null
@@ -1,15 +0,0 @@
-/// Write clients for SSH agents.
-pub mod client;
-
-mod msg;
-
-/// Constraints on how keys can be used.
-#[derive(Debug, PartialEq, Eq)]
-pub enum Constraint {
- /// The key shall disappear from the agent's memory after that many seconds.
- KeyLifetime { seconds: u32 },
- /// Signatures need to be confirmed by the agent (for instance using a dialog).
- Confirm,
- /// Custom constraints
- Extensions { name: Vec<u8>, details: Vec<u8> },
-}
diff --git a/crates/radicle-ssh/src/agent/client.rs b/crates/radicle-ssh/src/agent/client.rs
deleted file mode 100644
index 510b5f88..00000000
--- a/crates/radicle-ssh/src/agent/client.rs
+++ /dev/null
@@ -1,448 +0,0 @@
-use std::fmt;
-use std::io::{Read, Write};
-use std::path::{Path, PathBuf};
-
-#[cfg(unix)]
-pub use std::os::unix::net::UnixStream as Stream;
-
-#[cfg(windows)]
-pub use winpipe::WinStream as Stream;
-
-use thiserror::Error;
-use zeroize::Zeroize as _;
-
-use crate::agent::Constraint;
-use crate::agent::msg;
-use crate::encoding::{self, Encodable};
-use crate::encoding::{Buffer, Encoding, Reader};
-
-/// An ed25519 Signature.
-pub type Signature = [u8; 64];
-
-#[derive(Debug, Error)]
-pub enum Error {
- /// Agent protocol error.
- #[error("SSH agent replied with unexpected data, violating the SSH agent protocol.")]
- AgentProtocolError,
- #[error(
- "SSH agent replied with failure (protocol message number 5), which could not be handled."
- )]
- AgentFailure,
- #[error("Unable to connect to SSH agent because '{path}' was not found: {source}")]
- BadAuthSock {
- path: String,
- source: std::io::Error,
- },
- #[error("Encoding error while communicating with SSH agent: {0}")]
- Encoding(#[from] encoding::Error),
- #[error("Unable to read environment variable '{var}': {source}")]
- EnvVar {
- var: String,
- source: std::env::VarError,
- },
- #[error("Unable to connect SSH agent using the path '{path}': {source}")]
- Connect {
- path: String,
- #[source]
- source: std::io::Error,
- },
- #[error("I/O error while communicating with SSH agent: {0}")]
- Io(#[from] std::io::Error),
-}
-
-impl Error {
- pub fn is_not_running(&self) -> bool {
- match self {
- Self::EnvVar { .. } | Self::BadAuthSock { .. } => true,
- #[cfg(windows)]
- Self::Connect { source, .. }
- if source.kind() == std::io::ErrorKind::ConnectionRefused =>
- {
- // On Windows, a named pipe might be used, and if no
- // agent is running, we might get a "connection refused"
- // error, even though the `SSH_AUTH_SOCK` environment
- // variable is set and the named pipe exists.
- true
- }
- _ => false,
- }
- }
-}
-
-/// SSH agent client.
-pub struct AgentClient<S = Stream> {
- /// The path that was originally used to connect to the agent.
- path: Option<PathBuf>,
-
- /// The underlying stream to the SSH agent.
- stream: S,
-}
-
-impl<S> AgentClient<S> {
- pub fn path(&self) -> Option<&Path> {
- self.path.as_deref()
- }
-}
-
-impl AgentClient<Stream> {
- /// Connect to an SSH agent at the provided path.
- pub fn connect<P>(path: P) -> Result<Self, Error>
- where
- P: AsRef<Path>,
- {
- let path = path.as_ref().to_owned();
-
- let stream = match Stream::connect(&path) {
- Err(err) if err.kind() == std::io::ErrorKind::NotFound => {
- return Err(Error::BadAuthSock {
- path: path.display().to_string(),
- source: err,
- });
- }
- Err(err) => {
- return Err(Error::Connect {
- path: path.display().to_string(),
- source: err,
- });
- }
- Ok(stream) => stream,
- };
-
- Ok(Self {
- path: Some(path),
- stream,
- })
- }
-
- pub fn connect_env() -> Result<Self, Error> {
- const SSH_AUTH_SOCK: &str = "SSH_AUTH_SOCK";
-
- let var = std::env::var(SSH_AUTH_SOCK);
-
- #[cfg(windows)]
- let var = var.or({
- // Windows uses a named pipe for the SSH agent, which
- // we fall back to in case reading the environment
- // variable fails.
- Ok(r"\\.\pipe\openssh-ssh-agent".to_string())
- });
-
- Self::connect(var.map_err(|err| Error::EnvVar {
- var: SSH_AUTH_SOCK.to_string(),
- source: err,
- })?)
- }
-}
-
-impl<Stream: ClientStream> AgentClient<Stream> {
- pub fn new(path: Option<PathBuf>, stream: Stream) -> Self {
- Self { path, stream }
- }
-
- /// Send a key to the agent, with a (possibly empty) slice of constraints
- /// to apply when using the key to sign.
- pub fn add_identity<K>(&mut self, key: &K, constraints: &[Constraint]) -> Result<(), Error>
- where
- K: Encodable,
- K::Error: std::error::Error + Send + Sync + 'static,
- {
- let mut buf = Buffer::default();
-
- buf.resize(4, 0);
-
- if constraints.is_empty() {
- buf.push(msg::ADD_IDENTITY)
- } else {
- buf.push(msg::ADD_ID_CONSTRAINED)
- }
- key.write(&mut buf);
-
- if !constraints.is_empty() {
- for cons in constraints {
- match *cons {
- Constraint::KeyLifetime { seconds } => {
- buf.push(msg::CONSTRAIN_LIFETIME);
- buf.extend_u32(seconds);
- }
- Constraint::Confirm => buf.push(msg::CONSTRAIN_CONFIRM),
- Constraint::Extensions {
- ref name,
- ref details,
- } => {
- buf.push(msg::CONSTRAIN_EXTENSION);
- buf.extend_ssh_string(name);
- buf.extend_ssh_string(details);
- }
- }
- }
- }
- buf.write_len();
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Add a smart card to the agent, with a (possibly empty) set of
- /// constraints to apply when signing.
- pub fn add_smartcard_key(
- &mut self,
- id: &str,
- pin: &[u8],
- constraints: &[Constraint],
- ) -> Result<(), Error> {
- let mut buf = Buffer::default();
-
- buf.resize(4, 0);
-
- if constraints.is_empty() {
- buf.push(msg::ADD_SMARTCARD_KEY)
- } else {
- buf.push(msg::ADD_SMARTCARD_KEY_CONSTRAINED)
- }
- buf.extend_ssh_string(id.as_bytes());
- buf.extend_ssh_string(pin);
-
- if !constraints.is_empty() {
- buf.extend_usize(constraints.len());
- for cons in constraints {
- match *cons {
- Constraint::KeyLifetime { seconds } => {
- buf.push(msg::CONSTRAIN_LIFETIME);
- buf.extend_u32(seconds);
- }
- Constraint::Confirm => buf.push(msg::CONSTRAIN_CONFIRM),
- Constraint::Extensions {
- ref name,
- ref details,
- } => {
- buf.push(msg::CONSTRAIN_EXTENSION);
- buf.extend_ssh_string(name);
- buf.extend_ssh_string(details);
- }
- }
- }
- }
- buf.write_len();
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Lock the agent, making it refuse to sign until unlocked.
- pub fn lock(&mut self, passphrase: &[u8]) -> Result<(), Error> {
- let mut buf = Buffer::default();
-
- buf.resize(4, 0);
- buf.push(msg::LOCK);
- buf.extend_ssh_string(passphrase);
- buf.write_len();
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Unlock the agent, allowing it to sign again.
- pub fn unlock(&mut self, passphrase: &[u8]) -> Result<(), Error> {
- let mut buf = Buffer::default();
- buf.resize(4, 0);
- buf.push(msg::UNLOCK);
- buf.extend_ssh_string(passphrase);
- buf.write_len();
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Ask the agent for a list of the currently registered secret
- /// keys.
- pub fn request_identities<K>(&mut self) -> Result<Vec<K>, Error>
- where
- K: Encodable,
- K::Error: std::error::Error + Send + Sync + 'static,
- {
- let mut buf = Buffer::default();
- buf.resize(4, 0);
- buf.push(msg::REQUEST_IDENTITIES);
- buf.write_len();
-
- let mut keys = Vec::new();
- let resp = self.stream.request(&buf)?;
-
- if resp[0] == msg::IDENTITIES_ANSWER {
- let mut r = resp.reader(1);
- let n = r.read_u32()?;
-
- for _ in 0..n {
- let key = r.read_string()?;
- let _ = r.read_string()?;
- let mut r = key.reader(0);
-
- if let Ok(pk) = K::read(&mut r) {
- keys.push(pk);
- }
- }
- }
-
- Ok(keys)
- }
-
- /// Ask the agent to sign the supplied piece of data.
- pub fn sign<K>(&mut self, public: &K, data: &[u8]) -> Result<Signature, Error>
- where
- K: Encodable + fmt::Debug,
- {
- let req = self.prepare_sign_request(public, data);
- let resp = self.stream.request(&req)?;
-
- if !resp.is_empty() && resp[0] == msg::SIGN_RESPONSE {
- self.read_signature(&resp)
- } else if !resp.is_empty() && resp[0] == msg::FAILURE {
- Err(Error::AgentFailure)
- } else {
- Err(Error::AgentProtocolError)
- }
- }
-
- fn prepare_sign_request<K>(&self, public: &K, data: &[u8]) -> Buffer
- where
- K: Encodable + fmt::Debug,
- {
- // byte SSH_AGENTC_SIGN_REQUEST
- // string key blob
- // string data
- // uint32 flags
-
- let mut pk = Buffer::default();
- public.write(&mut pk);
-
- let total = 1 + pk.len() + 4 + data.len() + 4;
-
- let mut buf = Buffer::default();
- buf.extend_usize(total);
- buf.push(msg::SIGN_REQUEST);
- buf.extend_from_slice(&pk);
- buf.extend_ssh_string(data);
-
- // Signature flags should be zero for ed25519.
- buf.extend_u32(0);
- buf
- }
-
- fn read_signature(&self, sig: &[u8]) -> Result<Signature, Error> {
- let mut r = sig.reader(1);
- let mut resp = r.read_string()?.reader(0);
- let _t = resp.read_string()?;
- let sig = resp.read_string()?;
-
- let mut out = [0; 64];
- out.copy_from_slice(sig);
-
- Ok(out)
- }
-
- /// Ask the agent to remove a key from its memory.
- pub fn remove_identity<K>(&mut self, public: &K) -> Result<(), Error>
- where
- K: Encodable,
- {
- let mut pk: Buffer = Vec::new().into();
- public.write(&mut pk);
-
- let total = 1 + pk.len();
-
- let mut buf = Buffer::default();
- buf.extend_usize(total);
- buf.push(msg::REMOVE_IDENTITY);
- buf.extend_from_slice(&pk);
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Ask the agent to remove a smartcard from its memory.
- pub fn remove_smartcard_key(&mut self, id: &str, pin: &[u8]) -> Result<(), Error> {
- let mut buf = Buffer::default();
- buf.resize(4, 0);
- buf.push(msg::REMOVE_SMARTCARD_KEY);
- buf.extend_ssh_string(id.as_bytes());
- buf.extend_ssh_string(pin);
- buf.write_len();
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Ask the agent to forget all known keys.
- pub fn remove_all_identities(&mut self) -> Result<(), Error> {
- let mut buf = Buffer::default();
- buf.resize(4, 0);
- buf.push(msg::REMOVE_ALL_IDENTITIES);
- buf.write_len();
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Send a custom message to the agent.
- pub fn extension(&mut self, typ: &[u8], ext: &[u8]) -> Result<(), Error> {
- let mut buf = Buffer::default();
-
- buf.resize(4, 0);
- buf.push(msg::EXTENSION);
- buf.extend_ssh_string(typ);
- buf.extend_ssh_string(ext);
- buf.write_len();
-
- self.stream.request(&buf)?;
-
- Ok(())
- }
-
- /// Ask the agent about supported extensions.
- pub fn query_extension(&mut self, typ: &[u8], mut ext: Buffer) -> Result<bool, Error> {
- let mut req = Buffer::default();
-
- req.resize(4, 0);
- req.push(msg::EXTENSION);
- req.extend_ssh_string(typ);
- req.write_len();
-
- let resp = self.stream.request(&req)?;
- let mut r = resp.reader(1);
- ext.extend(r.read_string()?);
-
- Ok(!resp.is_empty() && resp[0] == msg::SUCCESS)
- }
-}
-
-pub trait ClientStream: Sized + Send + Sync {
- fn request(&mut self, msg: &[u8]) -> Result<Buffer, Error>;
-}
-
-impl<S: Read + Write + Sized + Send + Sync> ClientStream for S {
- fn request(&mut self, msg: &[u8]) -> Result<Buffer, Error> {
- let mut resp = Buffer::default();
-
- // Write the message
- self.write_all(msg)?;
- self.flush()?;
-
- // Read the length
- resp.resize(4, 0);
- self.read_exact(&mut resp)?;
-
- // Read the rest of the buffer
- let len = u32::from_be_bytes(resp.as_slice().try_into().unwrap()) as usize;
-
- resp.zeroize();
- resp.resize(len, 0);
- self.read_exact(&mut resp)?;
-
- Ok(resp)
- }
-}
diff --git a/crates/radicle-ssh/src/agent/msg.rs b/crates/radicle-ssh/src/agent/msg.rs
deleted file mode 100644
index d0786fd4..00000000
--- a/crates/radicle-ssh/src/agent/msg.rs
+++ /dev/null
@@ -1,23 +0,0 @@
-pub const FAILURE: u8 = 5;
-pub const SUCCESS: u8 = 6;
-pub const IDENTITIES_ANSWER: u8 = 12;
-pub const SIGN_RESPONSE: u8 = 14;
-#[allow(dead_code)]
-pub const EXTENSION_FAILURE: u8 = 28;
-
-pub const REQUEST_IDENTITIES: u8 = 11;
-pub const SIGN_REQUEST: u8 = 13;
-pub const ADD_IDENTITY: u8 = 17;
-pub const REMOVE_IDENTITY: u8 = 18;
-pub const REMOVE_ALL_IDENTITIES: u8 = 19;
-pub const ADD_ID_CONSTRAINED: u8 = 25;
-pub const ADD_SMARTCARD_KEY: u8 = 20;
-pub const REMOVE_SMARTCARD_KEY: u8 = 21;
-pub const LOCK: u8 = 22;
-pub const UNLOCK: u8 = 23;
-pub const ADD_SMARTCARD_KEY_CONSTRAINED: u8 = 26;
-pub const EXTENSION: u8 = 27;
-
-pub const CONSTRAIN_LIFETIME: u8 = 1;
-pub const CONSTRAIN_CONFIRM: u8 = 2;
-pub const CONSTRAIN_EXTENSION: u8 = 3;
diff --git a/crates/radicle-ssh/src/encoding.rs b/crates/radicle-ssh/src/encoding.rs
deleted file mode 100644
index db3df0ac..00000000
--- a/crates/radicle-ssh/src/encoding.rs
+++ /dev/null
@@ -1,254 +0,0 @@
-// Copyright 2016 Pierre-Étienne Meunier
-//
-// Licensed under the Apache License, Version 2.0 (the "License");
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-//
-// http://www.apache.org/licenses/LICENSE-2.0
-//
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an "AS IS" BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-//
-use std::ops::DerefMut as _;
-
-use thiserror::Error;
-use zeroize::Zeroizing;
-
-/// General purpose writable byte buffer we use everywhere.
-pub type Buffer = Zeroizing<Vec<u8>>;
-
-#[derive(Debug, Error)]
-pub enum Error {
- /// Index out of bounds
- #[error("Index out of bounds")]
- IndexOutOfBounds,
-}
-
-pub trait Encodable: Sized {
- type Error: std::error::Error + Send + Sync + 'static;
-
- /// Read from the SSH format.
- fn read(reader: &mut Cursor) -> Result<Self, Self::Error>;
- /// Write to the SSH format.
- fn write<E: Encoding>(&self, buf: &mut E);
-}
-
-/// Encode in the SSH format.
-pub trait Encoding {
- /// Push an SSH-encoded string to `self`.
- fn extend_ssh_string(&mut self, s: &[u8]);
- /// Push an SSH-encoded blank string of length `s` to `self`.
- fn extend_ssh_string_blank(&mut self, s: usize) -> &mut [u8];
- /// Push an SSH-encoded multiple-precision integer.
- fn extend_ssh_mpint(&mut self, s: &[u8]);
- /// Push an SSH-encoded list.
- fn extend_list<'a, I: Iterator<Item = &'a [u8]>>(&mut self, list: I);
- /// Push an SSH-encoded unsigned 32-bit integer.
- fn extend_u32(&mut self, u: u32);
- /// Push an SSH-encoded empty list.
- fn write_empty_list(&mut self);
- /// Write the buffer length at the beginning of the buffer.
- fn write_len(&mut self);
- /// Push a [`usize`] as an SSH-encoded unsigned 32-bit integer.
- /// May panic if the argument is greater than [`u32::MAX`].
- /// This is a convenience method, to spare callers casting or converting
- /// [`usize`] to [`u32`]. If callers end up in a situation where they
- /// need to push a 32-bit unsigned integer, but the value they would
- /// like to push does not fit 32 bits, then the implementation will not
- /// comply with the SSH format anyway.
- fn extend_usize(&mut self, u: usize) {
- self.extend_u32(u.try_into().unwrap())
- }
-}
-
-/// Encoding length of the given mpint.
-pub fn mpint_len(s: &[u8]) -> usize {
- let mut i = 0;
- while i < s.len() && s[i] == 0 {
- i += 1
- }
- (if s[i] & 0x80 != 0 { 5 } else { 4 }) + s.len() - i
-}
-
-impl Encoding for Vec<u8> {
- fn extend_ssh_string(&mut self, s: &[u8]) {
- self.extend_usize(s.len());
- self.extend(s);
- }
-
- fn extend_ssh_string_blank(&mut self, len: usize) -> &mut [u8] {
- self.extend_usize(len);
- let current = self.len();
- self.resize(current + len, 0u8);
-
- &mut self[current..]
- }
-
- fn extend_ssh_mpint(&mut self, s: &[u8]) {
- // Skip initial 0s.
- let mut i = 0;
- while i < s.len() && s[i] == 0 {
- i += 1
- }
- // If the first non-zero is >= 128, write its length (u32, BE), followed by 0.
- if s[i] & 0x80 != 0 {
- self.extend_usize(s.len() - i + 1);
- self.push(0)
- } else {
- self.extend_usize(s.len() - i);
- }
- self.extend(&s[i..]);
- }
-
- fn extend_u32(&mut self, s: u32) {
- self.extend(s.to_be_bytes());
- }
-
- fn extend_list<'a, I: Iterator<Item = &'a [u8]>>(&mut self, list: I) {
- let len0 = self.len();
-
- let mut first = true;
- for i in list {
- if !first {
- self.push(b',')
- } else {
- first = false;
- }
- self.extend(i)
- }
- let len = (self.len() - len0 - 4) as u32;
-
- self.splice(len0..len0, len.to_be_bytes());
- }
-
- fn write_empty_list(&mut self) {
- self.extend([0, 0, 0, 0]);
- }
-
- fn write_len(&mut self) {
- let len = self.len() - 4;
- self[..4].copy_from_slice((len as u32).to_be_bytes().as_slice());
- }
-}
-
-impl Encoding for Buffer {
- fn extend_ssh_string(&mut self, s: &[u8]) {
- self.deref_mut().extend_ssh_string(s)
- }
-
- fn extend_ssh_string_blank(&mut self, len: usize) -> &mut [u8] {
- self.deref_mut().extend_ssh_string_blank(len)
- }
-
- fn extend_ssh_mpint(&mut self, s: &[u8]) {
- self.deref_mut().extend_ssh_mpint(s)
- }
-
- fn extend_list<'a, I: Iterator<Item = &'a [u8]>>(&mut self, list: I) {
- self.deref_mut().extend_list(list)
- }
-
- fn write_empty_list(&mut self) {
- self.deref_mut().write_empty_list()
- }
-
- fn extend_u32(&mut self, s: u32) {
- self.deref_mut().extend_u32(s);
- }
-
- fn write_len(&mut self) {
- self.deref_mut().write_len()
- }
-}
-
-/// A cursor-like trait to read SSH-encoded things.
-pub trait Reader {
- /// Create an SSH reader for `self`.
- fn reader(&self, starting_at: usize) -> Cursor<'_>;
-}
-
-impl Reader for Buffer {
- fn reader(&self, starting_at: usize) -> Cursor<'_> {
- Cursor {
- s: self,
- position: starting_at,
- }
- }
-}
-
-impl Reader for [u8] {
- fn reader(&self, starting_at: usize) -> Cursor<'_> {
- Cursor {
- s: self,
- position: starting_at,
- }
- }
-}
-
-/// A cursor-like type to read SSH-encoded values.
-#[derive(Debug)]
-pub struct Cursor<'a> {
- s: &'a [u8],
- #[doc(hidden)]
- pub position: usize,
-}
-
-impl<'a> Cursor<'a> {
- /// Read one string from this reader.
- pub fn read_string(&mut self) -> Result<&'a [u8], Error> {
- let len = self.read_u32()? as usize;
- if self.position + len <= self.s.len() {
- let result = &self.s[self.position..(self.position + len)];
- self.position += len;
- Ok(result)
- } else {
- Err(Error::IndexOutOfBounds)
- }
- }
-
- /// Read a `u32` from this reader.
- pub fn read_u32(&mut self) -> Result<u32, Error> {
- if self.position + 4 <= self.s.len() {
- let u =
- u32::from_be_bytes(self.s[self.position..self.position + 4].try_into().unwrap());
- self.position += 4;
- Ok(u)
- } else {
- Err(Error::IndexOutOfBounds)
- }
- }
-
- /// Read one byte from this reader.
- pub fn read_byte(&mut self) -> Result<u8, Error> {
- if self.position < self.s.len() {
- let u = self.s[self.position];
- self.position += 1;
- Ok(u)
- } else {
- Err(Error::IndexOutOfBounds)
- }
- }
-
- pub fn read_bytes<const S: usize>(&mut self) -> Result<[u8; S], Error> {
- let mut buf = [0; S];
- for b in buf.iter_mut() {
- *b = self.read_byte()?;
- }
- Ok(buf)
- }
-
- /// Read one byte from this reader.
- pub fn read_mpint(&mut self) -> Result<&'a [u8], Error> {
- let len = self.read_u32()? as usize;
- if self.position + len <= self.s.len() {
- let result = &self.s[self.position..(self.position + len)];
- self.position += len;
- Ok(result)
- } else {
- Err(Error::IndexOutOfBounds)
- }
- }
-}
diff --git a/crates/radicle-ssh/src/lib.rs b/crates/radicle-ssh/src/lib.rs
deleted file mode 100644
index bd07d900..00000000
--- a/crates/radicle-ssh/src/lib.rs
+++ /dev/null
@@ -1,4 +0,0 @@
-pub mod agent;
-pub mod encoding;
-
-pub use agent::client::Error;
diff --git a/crates/radicle/Cargo.toml b/crates/radicle/Cargo.toml
index 850adb80..67e7dc88 100644
--- a/crates/radicle/Cargo.toml
+++ b/crates/radicle/Cargo.toml
@@ -50,7 +50,6 @@ radicle-git-ref-format = { workspace = true, features = ["macro", "serde"] }
radicle-git-metadata = { workspace = true }
radicle-localtime = { workspace = true, features = ["serde"] }
radicle-oid = { workspace = true, features = ["git2", "serde", "std", "sha1"] }
-radicle-ssh = { workspace = true }
schemars = { workspace = true, optional = true, features = ["derive", "std"] }
serde = { workspace = true, features = ["derive"] }
serde_json = { workspace = true, features = ["preserve_order"] }
diff --git a/crates/radicle/src/profile.rs b/crates/radicle/src/profile.rs
index bf029b1d..fc557910 100644
--- a/crates/radicle/src/profile.rs
+++ b/crates/radicle/src/profile.rs
@@ -199,7 +199,7 @@ pub enum SignerError {
MemorySigner(#[from] keystore::MemorySignerError),
#[error(transparent)]
- Agent(#[from] crate::crypto::ssh::agent::Error),
+ Agent(#[from] crate::crypto::ssh::agent::AgentError),
#[error("radicle key `{0}` is not registered; run `rad auth` to register it with ssh-agent")]
KeyNotRegistered(PublicKey),
@@ -209,7 +209,7 @@ pub enum SignerError {
#[error("error connecting to ssh-agent: {source}")]
AgentConnection {
- source: crate::crypto::ssh::agent::Error,
+ source: crate::crypto::ssh::agent::ConnectError,
},
}
Exit code: 0
shell: 'export RUSTDOCFLAGS=''-D warnings'' cargo --version rustc --version cargo fmt --check cargo clippy --all-targets --workspace -- --deny warnings cargo build --all-targets --workspace cargo doc --workspace --no-deps --all-features cargo test --workspace --no-fail-fast '
Commands:
$ podman run --name 4f2a868f-8682-4b20-bcf0-15e16e8940fe -v /opt/radcis/ci.rad.levitte.org/cci/state/4f2a868f-8682-4b20-bcf0-15e16e8940fe/s:/4f2a868f-8682-4b20-bcf0-15e16e8940fe/s:ro -v /opt/radcis/ci.rad.levitte.org/cci/state/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w:/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w -w /4f2a868f-8682-4b20-bcf0-15e16e8940fe/w -v /opt/radcis/ci.rad.levitte.org/.radicle:/${id}/.radicle:ro -e RAD_HOME=/${id}/.radicle rust:trixie bash /4f2a868f-8682-4b20-bcf0-15e16e8940fe/s/script.sh
+ export 'RUSTDOCFLAGS=-D warnings'
+ RUSTDOCFLAGS='-D warnings'
+ cargo --version
info: syncing channel updates for '1.90-x86_64-unknown-linux-gnu'
info: latest update on 2025-09-18, rust version 1.90.0 (1159e78c4 2025-09-14)
info: downloading component 'cargo'
info: downloading component 'clippy'
info: downloading component 'rust-docs'
info: downloading component 'rust-src'
info: downloading component 'rust-std'
info: downloading component 'rustc'
info: downloading component 'rustfmt'
info: installing component 'cargo'
info: installing component 'clippy'
info: installing component 'rust-docs'
info: installing component 'rust-src'
info: installing component 'rust-std'
info: installing component 'rustc'
info: installing component 'rustfmt'
cargo 1.90.0 (840b83a10 2025-07-30)
+ rustc --version
rustc 1.90.0 (1159e78c4 2025-09-14)
+ cargo fmt --check
+ cargo clippy --all-targets --workspace -- --deny warnings
Updating crates.io index
Downloading crates ...
Downloaded anstyle-query v1.1.5
Downloaded cypheraddr v0.4.0
Downloaded ctr v0.9.2
Downloaded crypto-bigint v0.5.5
Downloaded gix-revwalk v0.26.0
Downloaded errno v0.3.14
Downloaded find-msvc-tools v0.1.9
Downloaded cyphergraphy v0.3.0
Downloaded fast-glob v0.3.3
Downloaded clap_lex v1.1.0
Downloaded ghash v0.5.1
Downloaded git-ref-format-macro v0.6.0
Downloaded gix-glob v0.24.0
Downloaded gix-prompt v0.13.1
Downloaded gix-refspec v0.37.0
Downloaded gix-negotiate v0.27.0
Downloaded gix-shallow v0.8.1
Downloaded gix-hashtable v0.12.0
Downloaded gix-error v0.0.0
Downloaded gix-tempfile v21.0.1
Downloaded gix-ref v0.59.0
Downloaded inout v0.1.4
Downloaded lazy_static v1.5.0
Downloaded idna_adapter v1.2.1
Downloaded itoa v1.0.17
Downloaded lexopt v0.3.2
Downloaded heck v0.5.0
Downloaded percent-encoding v2.3.2
Downloaded noise-framework v0.4.0
Downloaded proc-macro-error2 v2.0.1
Downloaded hash32 v0.3.1
Downloaded normalize-line-endings v0.3.0
Downloaded gix-url v0.35.2
Downloaded nonempty v0.9.0
Downloaded phf_shared v0.11.3
Downloaded pastey v0.2.1
Downloaded hmac v0.12.1
Downloaded scopeguard v1.2.0
Downloaded erased-serde v0.4.10
Downloaded qcheck-macros v1.0.0
Downloaded sha1 v0.10.6
Downloaded miniz_oxide v0.8.9
Downloaded indicatif v0.18.4
Downloaded rand_chacha v0.3.1
Downloaded quick-error v1.2.3
Downloaded shell-words v1.1.1
Downloaded ref-cast-impl v1.0.25
Downloaded poly1305 v0.8.0
Downloaded rustc_version v0.4.1
Downloaded signal-hook-mio v0.2.5
Downloaded scrypt v0.11.0
Downloaded socks5-client v0.4.1
Downloaded secrecy v0.10.3
Downloaded serde_fmt v1.1.0
Downloaded ssh-encoding v0.2.0
Downloaded serde_spanned v1.0.4
Downloaded same-file v1.0.6
Downloaded stable_deref_trait v1.2.1
Downloaded ssh-cipher v0.2.0
Downloaded sval_dynamic v2.17.0
Downloaded signal-hook-registry v1.4.8
Downloaded sval_fmt v2.17.0
Downloaded sval_serde v2.17.0
Downloaded sem_safe v0.2.1
Downloaded signature v1.6.4
Downloaded sval_json v2.17.0
Downloaded spki v0.7.3
Downloaded amplify v4.9.0
Downloaded test-log v0.2.19
Downloaded signals_receipts v0.2.5
Downloaded tinyvec_macros v0.1.1
Downloaded rand v0.8.5
Downloaded tree-sitter-json v0.24.8
Downloaded utf8_iter v1.0.4
Downloaded unit-prefix v0.5.2
Downloaded typeid v1.0.3
Downloaded unarray v0.1.4
Downloaded universal-hash v0.5.1
Downloaded version_check v0.9.5
Downloaded unicode-display-width v0.3.0
Downloaded tinystr v0.8.2
Downloaded zerofrom v0.1.6
Downloaded toml v0.9.12+spec-1.1.0
Downloaded zerofrom-derive v0.1.6
Downloaded similar v2.7.0
Downloaded vsimd v0.8.0
Downloaded zmij v1.0.21
Downloaded thread_local v1.1.9
Downloaded serde v1.0.228
Downloaded yoke v0.8.1
Downloaded zeroize v1.8.2
Downloaded tracing-core v0.1.36
Downloaded prodash v31.0.0
Downloaded unicode-ident v1.0.24
Downloaded tar v0.4.45
Downloaded aes-gcm v0.10.3
Downloaded sval v2.17.0
Downloaded url v2.5.8
Downloaded uuid v1.22.0
Downloaded zerotrie v0.2.3
Downloaded yansi v1.0.1
Downloaded unicode-normalization v0.1.25
Downloaded radicle-surf v0.27.1
Downloaded clap_builder v4.6.0
Downloaded libm v0.2.16
Downloaded zerovec v0.11.5
Downloaded winnow v0.7.15
Downloaded proptest v1.10.0
Downloaded zlib-rs v0.6.3
Downloaded tree-sitter-c v0.23.4
Downloaded tracing-subscriber v0.3.23
Downloaded object v0.37.3
Downloaded zerocopy v0.8.42
Downloaded vcpkg v0.2.15
Downloaded syn v2.0.117
Downloaded unicode-width v0.2.2
Downloaded tree-sitter-rust v0.23.3
Downloaded rustix v1.1.4
Downloaded regex-syntax v0.8.10
Downloaded tree-sitter-bash v0.23.3
Downloaded tracing v0.1.44
Downloaded tree-sitter-md v0.3.2
Downloaded bstr v1.12.1
Downloaded tree-sitter-ruby v0.23.1
Downloaded bloomy v1.2.0
Downloaded syn v1.0.109
Downloaded curve25519-dalek v4.1.3
Downloaded sysinfo v0.37.2
Downloaded tree-sitter v0.24.7
Downloaded regex-automata v0.4.14
Downloaded jiff v0.2.23
Downloaded chrono v0.4.44
Downloaded regex v1.12.3
Downloaded p384 v0.13.1
Downloaded unicode-segmentation v1.12.0
Downloaded tree-sitter-python v0.23.6
Downloaded libc v0.2.183
Downloaded tree-sitter-go v0.23.4
Downloaded gimli v0.32.3
Downloaded tree-sitter-typescript v0.23.2
Downloaded libz-sys v1.1.25
Downloaded sha1-checked v0.10.0
Downloaded tokio v1.50.0
Downloaded sha3 v0.10.8
Downloaded typenum v1.19.0
Downloaded serde_json v1.0.149
Downloaded portable-atomic v1.13.1
Downloaded crossterm v0.29.0
Downloaded num-bigint-dig v0.8.6
Downloaded aes v0.8.4
Downloaded p521 v0.13.3
Downloaded tinyvec v1.11.0
Downloaded ssh-key v0.6.7
Downloaded memchr v2.8.0
Downloaded value-bag v1.12.0
Downloaded tree-sitter-css v0.23.2
Downloaded zerovec-derive v0.11.2
Downloaded writeable v0.6.2
Downloaded walkdir v2.5.0
Downloaded ssh-agent-lib v0.5.2
Downloaded rand v0.9.2
Downloaded uuid-simd v0.8.0
Downloaded utf8parse v0.2.2
Downloaded tree-sitter-toml-ng v0.6.0
Downloaded tree-sitter-highlight v0.24.7
Downloaded tempfile v3.27.0
Downloaded ed25519-dalek v2.2.0
Downloaded schemars v1.2.1
Downloaded rsa v0.9.10
Downloaded yoke-derive v0.8.1
Downloaded xattr v1.6.1
Downloaded tree-sitter-html v0.23.2
Downloaded proc-macro2 v1.0.106
Downloaded cyphernet v0.5.2
Downloaded socket2 v0.5.10
Downloaded serde_derive v1.0.228
Downloaded wait-timeout v0.2.1
Downloaded tracing-log v0.2.0
Downloaded toml_writer v1.0.7+spec-1.1.0
Downloaded thiserror-impl v2.0.18
Downloaded signal-hook v0.3.18
Downloaded value-bag-sval2 v1.12.0
Downloaded value-bag-serde1 v1.12.0
Downloaded thiserror v2.0.18
Downloaded thiserror v1.0.69
Downloaded thiserror-impl v1.0.69
Downloaded serde_core v1.0.228
Downloaded tree-sitter-language v0.1.7
Downloaded derive_more v2.1.1
Downloaded der v0.7.10
Downloaded streaming-iterator v0.1.9
Downloaded toml_datetime v0.7.5+spec-1.1.0
Downloaded timeago v0.4.2
Downloaded test-log-macros v0.2.19
Downloaded systemd-journal-logger v2.2.2
Downloaded sqlite v0.37.0
Downloaded libgit2-sys v0.18.3+1.9.2
Downloaded spin v0.9.8
Downloaded itertools v0.14.0
Downloaded sval_nested v2.17.0
Downloaded sval_buffer v2.17.0
Downloaded smallvec v1.15.1
Downloaded sharded-slab v0.1.7
Downloaded emojis v0.6.4
Downloaded backtrace v0.3.76
Downloaded synstructure v0.13.2
Downloaded structured-logger v1.0.5
Downloaded sqlite3-sys v0.18.0
Downloaded simd-adler32 v0.3.8
Downloaded hashbrown v0.16.1
Downloaded snapbox v0.4.17
Downloaded ryu v1.0.23
Downloaded pretty_assertions v1.4.1
Downloaded const-oid v0.9.6
Downloaded chacha20poly1305 v0.10.1
Downloaded snapbox-macros v0.3.10
Downloaded sha2 v0.10.9
Downloaded semver v1.0.27
Downloaded p256 v0.13.2
Downloaded sec1 v0.7.3
Downloaded rusty-fork v0.3.1
Downloaded sval_ref v2.17.0
Downloaded strsim v0.11.1
Downloaded serde-untagged v0.1.9
Downloaded mio v1.1.1
Downloaded rand_core v0.9.5
Downloaded num-bigint v0.4.6
Downloaded shlex v1.3.0
Downloaded num-traits v0.2.19
Downloaded cc v1.2.57
Downloaded base64 v0.21.7
Downloaded subtle v2.6.1
Downloaded signature v2.2.0
Downloaded schemars_derive v1.2.1
Downloaded siphasher v1.0.2
Downloaded rustversion v1.0.22
Downloaded referencing v0.30.0
Downloaded parking_lot_core v0.9.12
Downloaded log v0.4.29
Downloaded crossbeam-utils v0.8.21
Downloaded rustc-demangle v0.1.27
Downloaded quote v1.0.45
Downloaded jsonschema v0.30.0
Downloaded indexmap v2.13.0
Downloaded icu_properties_data v2.1.2
Downloaded console v0.16.3
Downloaded clap_complete v4.6.0
Downloaded siphasher v0.3.11
Downloaded serde_derive_internals v0.29.1
Downloaded rand_core v0.6.4
Downloaded rand_chacha v0.9.0
Downloaded parking_lot v0.12.5
Downloaded gix-pack v0.65.0
Downloaded salsa20 v0.10.2
Downloaded sqlite3-src v0.7.0
Downloaded phf v0.11.3
Downloaded pem-rfc7468 v0.7.0
Downloaded memmap2 v0.9.10
Downloaded qcheck v1.0.0
Downloaded linux-raw-sys v0.12.1
Downloaded primeorder v0.13.6
Downloaded num-integer v0.1.46
Downloaded litrs v1.0.0
Downloaded idna v1.1.0
Downloaded rfc6979 v0.4.0
Downloaded rand_xorshift v0.4.0
Downloaded inquire v0.9.4
Downloaded ppv-lite86 v0.2.21
Downloaded num-rational v0.4.2
Downloaded lock_api v0.4.14
Downloaded amplify_derive v4.0.1
Downloaded radicle-std-ext v0.2.0
Downloaded litemap v0.8.1
Downloaded pin-project-lite v0.2.17
Downloaded ed25519 v2.2.3
Downloaded data-encoding v2.10.0
Downloaded crc32fast v1.5.0
Downloaded ascii v1.1.0
Downloaded ref-cast v1.0.25
Downloaded radicle-git-ext v0.12.0
Downloaded heapless v0.8.0
Downloaded clap_derive v4.6.0
Downloaded icu_locale_core v2.1.1
Downloaded diff v0.1.13
Downloaded proc-macro-error-attr2 v2.0.0
Downloaded potential_utf v0.1.4
Downloaded polyval v0.6.2
Downloaded pkg-config v0.3.32
Downloaded pkcs8 v0.10.2
Downloaded nu-ansi-term v0.50.3
Downloaded icu_collections v2.1.1
Downloaded arc-swap v1.8.2
Downloaded icu_normalizer_data v2.1.1
Downloaded cipher v0.4.4
Downloaded bytecount v0.6.9
Downloaded once_cell v1.21.4
Downloaded num-cmp v0.1.0
Downloaded jiff-static v0.2.23
Downloaded ahash v0.8.12
Downloaded num-complex v0.4.6
Downloaded maybe-async v0.2.10
Downloaded iana-time-zone v0.1.65
Downloaded gix-protocol v0.57.0
Downloaded env_logger v0.11.9
Downloaded ecdsa v0.16.9
Downloaded digest v0.10.7
Downloaded anstream v1.0.0
Downloaded displaydoc v0.2.5
Downloaded anstream v0.6.21
Downloaded jobserver v0.1.34
Downloaded const-str v0.4.3
Downloaded base64ct v1.8.3
Downloaded autocfg v1.5.0
Downloaded ec25519 v0.1.0
Downloaded convert_case v0.10.0
Downloaded num-iter v0.1.45
Downloaded num v0.4.3
Downloaded pbkdf2 v0.12.2
Downloaded outref v0.5.2
Downloaded opaque-debug v0.3.1
Downloaded fancy-regex v0.14.0
Downloaded anstyle-parse v0.2.7
Downloaded icu_provider v2.1.1
Downloaded humantime v2.3.0
Downloaded gix-traverse v0.52.0
Downloaded crossbeam-channel v0.5.15
Downloaded gix-validate v0.11.0
Downloaded gix-revwalk v0.27.0
Downloaded escargot v0.5.15
Downloaded either v1.15.0
Downloaded crypto-common v0.1.7
Downloaded blowfish v0.9.1
Downloaded pkcs1 v0.7.5
Downloaded multibase v0.9.2
Downloaded match-lookup v0.1.2
Downloaded is_terminal_polyfill v1.70.2
Downloaded gix-path v0.11.1
Downloaded bit-set v0.8.0
Downloaded human-panic v2.0.6
Downloaded gix-utils v0.3.1
Downloaded gix-packetline v0.21.1
Downloaded matchers v0.2.0
Downloaded equivalent v1.0.2
Downloaded bcrypt-pbkdf v0.10.0
Downloaded gix-chunk v0.6.0
Downloaded curve25519-dalek-derive v0.1.1
Downloaded env_filter v1.0.0
Downloaded gix-trace v0.1.18
Downloaded gix-odb v0.75.0
Downloaded fastrand v2.3.0
Downloaded base-x v0.2.11
Downloaded gix-object v0.55.0
Downloaded gix-features v0.46.1
Downloaded getrandom v0.3.4
Downloaded byteorder v1.5.0
Downloaded keccak v0.1.6
Downloaded gix-lock v21.0.1
Downloaded gix-fs v0.19.1
Downloaded gix-date v0.13.0
Downloaded gix-commitgraph v0.33.0
Downloaded gix-command v0.7.1
Downloaded gix-actor v0.39.0
Downloaded getrandom v0.2.17
Downloaded icu_properties v2.1.2
Downloaded icu_normalizer v2.1.1
Downloaded group v0.13.0
Downloaded gix-transport v0.54.0
Downloaded gix-sec v0.13.1
Downloaded gix-quote v0.6.2
Downloaded filetime v0.2.27
Downloaded dyn-clone v1.0.20
Downloaded data-encoding-macro v0.1.19
Downloaded cfg-if v1.0.4
Downloaded gix-object v0.56.0
Downloaded gix-diff v0.58.0
Downloaded gix-commitgraph v0.32.0
Downloaded faster-hex v0.10.0
Downloaded email_address v0.2.9
Downloaded bytesize v2.3.1
Downloaded anstyle v1.0.14
Downloaded gix-revision v0.41.0
Downloaded dunce v1.0.5
Downloaded gix-chunk v0.5.0
Downloaded git-ref-format v0.6.0
Downloaded fluent-uri v0.3.2
Downloaded document-features v0.2.12
Downloaded ct-codecs v1.1.6
Downloaded base64 v0.22.1
Downloaded base256emoji v1.0.2
Downloaded anstyle-parse v1.0.0
Downloaded gix-hash v0.22.1
Downloaded gix-error v0.1.0
Downloaded gix-date v0.14.0
Downloaded generic-array v0.14.7
Downloaded gix-credentials v0.36.0
Downloaded git2 v0.20.4
Downloaded gix-config-value v0.17.1
Downloaded getrandom v0.4.2
Downloaded fraction v0.15.3
Downloaded fnv v1.0.7
Downloaded ed25519 v1.5.3
Downloaded data-encoding-macro-internal v0.1.17
Downloaded colored v2.2.0
Downloaded block-buffer v0.10.4
Downloaded base32 v0.4.0
Downloaded amplify_num v0.5.3
Downloaded gix-actor v0.38.0
Downloaded ff v0.13.1
Downloaded elliptic-curve v0.13.8
Downloaded colorchoice v1.0.5
Downloaded borrow-or-share v0.2.4
Downloaded anyhow v1.0.102
Downloaded git-ref-format-core v0.6.0
Downloaded form_urlencoded v1.2.2
Downloaded clap v4.6.0
Downloaded bit-vec v0.8.0
Downloaded flate2 v1.1.9
Downloaded base16ct v0.2.0
Downloaded block-padding v0.3.3
Downloaded bytes v1.11.1
Downloaded derive_more-impl v2.1.1
Downloaded adler2 v2.0.1
Downloaded aho-corasick v1.1.4
Downloaded addr2line v0.25.1
Downloaded bitflags v2.11.0
Downloaded aead v0.5.2
Downloaded cpufeatures v0.2.17
Downloaded chacha20 v0.9.1
Downloaded cbc v0.1.2
Downloaded amplify_syn v2.0.1
Compiling libc v0.2.183
Compiling proc-macro2 v1.0.106
Compiling unicode-ident v1.0.24
Compiling quote v1.0.45
Checking cfg-if v1.0.4
Checking zeroize v1.8.2
Compiling version_check v0.9.5
Compiling typenum v1.19.0
Compiling generic-array v0.14.7
Checking memchr v2.8.0
Compiling syn v2.0.117
Checking getrandom v0.2.17
Checking rand_core v0.6.4
Compiling shlex v1.3.0
Checking crypto-common v0.1.7
Checking subtle v2.6.1
Compiling find-msvc-tools v0.1.9
Checking regex-syntax v0.8.10
Checking aho-corasick v1.1.4
Compiling jobserver v0.1.34
Compiling cc v1.2.57
Checking const-oid v0.9.6
Checking smallvec v1.15.1
Compiling serde_core v1.0.228
Checking regex-automata v0.4.14
Checking block-buffer v0.10.4
Checking digest v0.10.7
Checking cpufeatures v0.2.17
Compiling thiserror v2.0.18
Checking stable_deref_trait v1.2.1
Checking fastrand v2.3.0
Compiling parking_lot_core v0.9.12
Checking scopeguard v1.2.0
Checking lock_api v0.4.14
Checking tinyvec_macros v0.1.1
Checking tinyvec v1.11.0
Checking parking_lot v0.12.5
Compiling crc32fast v1.5.0
Checking bitflags v2.11.0
Checking bstr v1.12.1
Checking gix-trace v0.1.18
Checking unicode-normalization v0.1.25
Checking byteorder v1.5.0
Checking gix-validate v0.11.0
Checking itoa v1.0.17
Checking gix-utils v0.3.1
Compiling typeid v1.0.3
Compiling erased-serde v0.4.10
Compiling serde v1.0.228
Checking same-file v1.0.6
Checking hashbrown v0.16.1
Checking walkdir v2.5.0
Checking prodash v31.0.0
Compiling thiserror-impl v2.0.18
Compiling serde_derive v1.0.228
Checking zlib-rs v0.6.3
Checking serde_fmt v1.1.0
Compiling heapless v0.8.0
Checking value-bag-serde1 v1.12.0
Checking hash32 v0.3.1
Compiling synstructure v0.13.2
Checking gix-path v0.11.1
Checking value-bag v1.12.0
Checking gix-features v0.46.1
Checking log v0.4.29
Checking faster-hex v0.10.0
Compiling zerofrom-derive v0.1.6
Compiling yoke-derive v0.8.1
Checking sha1 v0.10.6
Checking sha1-checked v0.10.0
Compiling pkg-config v0.3.32
Compiling rustix v1.1.4
Checking gix-hash v0.22.1
Checking zerofrom v0.1.6
Checking yoke v0.8.1
Compiling zerovec-derive v0.11.2
Checking block-padding v0.3.3
Compiling libm v0.2.16
Compiling autocfg v1.5.0
Checking linux-raw-sys v0.12.1
Compiling zerocopy v0.8.42
Compiling num-traits v0.2.19
Checking inout v0.1.4
Compiling displaydoc v0.2.5
Checking sha2 v0.10.9
Checking zerovec v0.11.5
Compiling getrandom v0.4.2
Checking cipher v0.4.4
Checking tinystr v0.8.2
Checking der v0.7.10
Checking litemap v0.8.1
Checking writeable v0.6.2
Checking once_cell v1.21.4
Checking percent-encoding v2.3.2
Checking icu_locale_core v2.1.1
Checking zerotrie v0.2.3
Checking potential_utf v0.1.4
Compiling icu_properties_data v2.1.2
Compiling thiserror v1.0.69
Compiling icu_normalizer_data v2.1.1
Compiling zmij v1.0.21
Compiling syn v1.0.109
Checking icu_provider v2.1.1
Checking icu_collections v2.1.1
Compiling thiserror-impl v1.0.69
Checking equivalent v1.0.2
Compiling serde_json v1.0.149
Checking indexmap v2.13.0
Checking ppv-lite86 v0.2.21
Checking num-integer v0.1.46
Checking hmac v0.12.1
Checking universal-hash v0.5.1
Compiling vcpkg v0.2.15
Compiling tree-sitter-language v0.1.7
Checking opaque-debug v0.3.1
Compiling ref-cast v1.0.25
Checking icu_properties v2.1.2
Compiling libz-sys v1.1.25
Checking icu_normalizer v2.1.1
Checking tempfile v3.27.0
Checking spki v0.7.3
Compiling ref-cast-impl v1.0.25
Checking signature v2.2.0
Checking ff v0.13.1
Checking base16ct v0.2.0
Checking spin v0.9.8
Checking group v0.13.0
Checking lazy_static v1.5.0
Checking sec1 v0.7.3
Checking idna_adapter v1.2.1
Checking rand_chacha v0.3.1
Checking crypto-bigint v0.5.5
Checking dyn-clone v1.0.20
Checking utf8_iter v1.0.4
Checking rand v0.8.5
Checking idna v1.1.0
Compiling amplify_syn v2.0.1
Checking num-iter v0.1.45
Compiling libgit2-sys v0.18.3+1.9.2
Checking aead v0.5.2
Compiling semver v1.0.27
Checking signature v1.6.4
Checking ed25519 v1.5.3
Compiling amplify_derive v4.0.1
Checking elliptic-curve v0.13.8
Compiling rustc_version v0.4.1
Checking poly1305 v0.8.0
Checking rfc6979 v0.4.0
Checking form_urlencoded v1.2.2
Checking chacha20 v0.9.1
Compiling serde_derive_internals v0.29.1
Checking ct-codecs v1.1.6
Checking ascii v1.1.0
Checking amplify_num v0.5.3
Compiling schemars_derive v1.2.1
Checking ec25519 v0.1.0
Checking url v2.5.8
Checking amplify v4.9.0
Checking ecdsa v0.16.9
Compiling curve25519-dalek v4.1.3
Checking primeorder v0.13.6
Checking git-ref-format-core v0.6.0
Checking polyval v0.6.2
Compiling num-bigint-dig v0.8.6
Checking base64ct v1.8.3
Checking ghash v0.5.1
Checking schemars v1.2.1
Checking cyphergraphy v0.3.0
Checking pem-rfc7468 v0.7.0
Checking pkcs8 v0.10.2
Checking pbkdf2 v0.12.2
Checking ctr v0.9.2
Checking aes v0.8.4
Compiling sqlite3-src v0.7.0
Compiling curve25519-dalek-derive v0.1.1
Checking keccak v0.1.6
Checking sha3 v0.10.8
Checking aes-gcm v0.10.3
Checking pkcs1 v0.7.5
Checking ssh-encoding v0.2.0
Checking ed25519 v2.2.3
Checking blowfish v0.9.1
Checking cbc v0.1.2
Compiling data-encoding v2.10.0
Checking base32 v0.4.0
Checking cypheraddr v0.4.0
Compiling data-encoding-macro-internal v0.1.17
Checking rsa v0.9.10
Checking ssh-cipher v0.2.0
Checking bcrypt-pbkdf v0.10.0
Checking ed25519-dalek v2.2.0
Checking p384 v0.13.1
Checking p256 v0.13.2
Checking p521 v0.13.3
Checking chacha20poly1305 v0.10.1
Checking qcheck v1.0.0
Compiling match-lookup v0.1.2
Checking const-str v0.4.3
Checking jiff v0.2.23
Checking data-encoding-macro v0.1.19
Checking base256emoji v1.0.2
Checking ssh-key v0.6.7
Checking noise-framework v0.4.0
Checking socks5-client v0.4.1
Checking secrecy v0.10.3
Compiling crossbeam-utils v0.8.21
Checking base-x v0.2.11
Checking multibase v0.9.2
Checking ssh-agent-lib v0.5.2
Checking cyphernet v0.5.2
Checking winnow v0.7.15
Checking crossbeam-channel v0.5.15
Checking anstyle-query v1.1.5
Checking utf8parse v0.2.2
Checking nonempty v0.9.0
Checking gix-hashtable v0.12.0
Checking siphasher v1.0.2
Checking radicle-git-metadata v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-metadata)
Checking gix-error v0.1.0
Checking radicle-dag v0.10.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-dag)
Checking memmap2 v0.9.10
Checking colorchoice v1.0.5
Checking anstyle v1.0.14
Checking is_terminal_polyfill v1.70.2
Checking radicle-git-ref-format v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-ref-format)
Checking iana-time-zone v0.1.65
Checking base64 v0.21.7
Checking chrono v0.4.44
Checking radicle-localtime v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-localtime)
Checking colored v2.2.0
Checking serde-untagged v0.1.9
Checking gix-error v0.0.0
Checking bytesize v2.3.1
Checking errno v0.3.14
Checking dunce v1.0.5
Checking fast-glob v0.3.3
Checking gix-date v0.14.0
Checking anstyle-parse v0.2.7
Checking gix-actor v0.39.0
Checking anstream v0.6.21
Checking gix-date v0.13.0
Checking gix-fs v0.19.1
Checking gix-actor v0.38.0
Checking gix-tempfile v21.0.1
Checking gix-object v0.56.0
Checking gix-chunk v0.6.0
Checking gix-quote v0.6.2
Checking gix-commitgraph v0.33.0
Checking gix-object v0.55.0
Checking gix-chunk v0.5.0
Checking mio v1.1.1
Checking sem_safe v0.2.1
Checking either v1.15.0
Checking shell-words v1.1.1
Checking gix-command v0.7.1
Checking signals_receipts v0.2.5
Checking gix-commitgraph v0.32.0
Checking gix-revwalk v0.27.0
Compiling object v0.37.3
Compiling rustversion v1.0.22
Checking gix-revwalk v0.26.0
Checking gix-lock v21.0.1
Checking gix-url v0.35.2
Checking gix-config-value v0.17.1
Checking gix-sec v0.13.1
Compiling signal-hook v0.3.18
Compiling unicode-segmentation v1.12.0
Checking gimli v0.32.3
Checking adler2 v2.0.1
Checking miniz_oxide v0.8.9
Compiling convert_case v0.10.0
Checking gix-prompt v0.13.1
Checking gix-traverse v0.52.0
Checking addr2line v0.25.1
Checking gix-revision v0.41.0
Checking radicle-signals v0.11.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-signals)
Checking gix-diff v0.58.0
Checking signal-hook-registry v1.4.8
Checking gix-packetline v0.21.1
Checking gix-glob v0.24.0
Compiling tree-sitter v0.24.7
Compiling anyhow v1.0.102
Checking rustc-demangle v0.1.27
Checking backtrace v0.3.76
Checking gix-refspec v0.37.0
Checking gix-transport v0.54.0
Checking gix-pack v0.65.0
Checking arc-swap v1.8.2
Checking gix-credentials v0.36.0
Compiling derive_more-impl v2.1.1
Checking gix-shallow v0.8.1
Checking gix-ref v0.59.0
Checking gix-negotiate v0.27.0
Compiling maybe-async v0.2.10
Checking regex v1.12.3
Compiling proc-macro-error-attr2 v2.0.0
Compiling simd-adler32 v0.3.8
Compiling litrs v1.0.0
Compiling portable-atomic v1.13.1
Checking unicode-width v0.2.2
Compiling getrandom v0.3.4
Compiling document-features v0.2.12
Compiling proc-macro-error2 v2.0.1
Checking gix-protocol v0.57.0
Checking derive_more v2.1.1
Checking gix-odb v0.75.0
Checking signal-hook-mio v0.2.5
Compiling xattr v1.6.1
Compiling filetime v0.2.27
Checking anstyle-parse v1.0.0
Checking uuid v1.22.0
Checking bytes v1.11.1
Checking anstream v1.0.0
Compiling tar v0.4.45
Compiling flate2 v1.1.9
Checking crossterm v0.29.0
Compiling git-ref-format-macro v0.6.0
Checking console v0.16.3
Checking snapbox-macros v0.3.10
Checking salsa20 v0.10.2
Checking clap_lex v1.1.0
Checking streaming-iterator v0.1.9
Checking unit-prefix v0.5.2
Checking strsim v0.11.1
Checking siphasher v0.3.11
Checking similar v2.7.0
Compiling heck v0.5.0
Checking normalize-line-endings v0.3.0
Compiling clap_derive v4.6.0
Checking snapbox v0.4.17
Checking bloomy v1.2.0
Checking clap_builder v4.6.0
Checking indicatif v0.18.4
Compiling radicle-surf v0.27.1
Checking sqlite3-sys v0.18.0
Checking scrypt v0.11.0
Checking git-ref-format v0.6.0
Checking inquire v0.9.4
Checking sqlite v0.37.0
Checking unicode-display-width v0.3.0
Checking systemd-journal-logger v2.2.2
Checking serde_spanned v1.0.4
Checking radicle-crypto v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-crypto)
Checking toml_datetime v0.7.5+spec-1.1.0
Compiling tree-sitter-c v0.23.4
Compiling tree-sitter-typescript v0.23.2
Compiling tree-sitter-ruby v0.23.1
Compiling tree-sitter-toml-ng v0.6.0
Compiling tree-sitter-python v0.23.6
Compiling tree-sitter-html v0.23.2
Compiling tree-sitter-bash v0.23.3
Compiling tree-sitter-css v0.23.2
Compiling tree-sitter-md v0.3.2
Compiling tree-sitter-go v0.23.4
Compiling tree-sitter-rust v0.23.3
Compiling tree-sitter-json v0.24.8
Checking pin-project-lite v0.2.17
Checking toml_writer v1.0.7+spec-1.1.0
Checking radicle-std-ext v0.2.0
Checking tokio v1.50.0
Checking toml v0.9.12+spec-1.1.0
Checking clap v4.6.0
Checking sysinfo v0.37.2
Checking yansi v1.0.1
Compiling radicle-node v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-node)
Compiling radicle-cli v0.20.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli)
Checking diff v0.1.13
Checking pretty_assertions v1.4.1
Checking human-panic v2.0.6
Checking clap_complete v4.6.0
Checking structured-logger v1.0.5
Checking radicle-systemd v0.12.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-systemd)
Checking tree-sitter-highlight v0.24.7
Checking itertools v0.14.0
Compiling qcheck-macros v1.0.0
Checking socket2 v0.5.10
Checking lexopt v0.3.2
Checking humantime v2.3.0
Checking timeago v0.4.2
Compiling escargot v0.5.15
Checking bit-vec v0.8.0
Checking bit-set v0.8.0
Checking rand_core v0.9.5
Checking num-bigint v0.4.6
Compiling ahash v0.8.12
Checking num-complex v0.4.6
Checking env_filter v1.0.0
Checking borrow-or-share v0.2.4
Checking fluent-uri v0.3.2
Checking env_logger v0.11.9
Checking num-rational v0.4.2
Checking phf_shared v0.11.3
Compiling test-log-macros v0.2.19
Checking wait-timeout v0.2.1
Checking fnv v1.0.7
Checking num v0.4.3
Compiling radicle-remote-helper v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-remote-helper)
Checking vsimd v0.8.0
Checking quick-error v1.2.3
Checking outref v0.5.2
Checking rusty-fork v0.3.1
Checking test-log v0.2.19
Checking fraction v0.15.3
Checking phf v0.11.3
Checking uuid-simd v0.8.0
Checking referencing v0.30.0
Checking rand_xorshift v0.4.0
Checking rand_chacha v0.9.0
Checking rand v0.9.2
Checking fancy-regex v0.14.0
Checking email_address v0.2.9
Checking base64 v0.22.1
Checking unarray v0.1.4
Checking num-cmp v0.1.0
Checking bytecount v0.6.9
Checking proptest v1.10.0
Checking emojis v0.6.4
Checking jsonschema v0.30.0
Compiling pastey v0.2.1
Checking radicle-windows v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-windows)
Checking git2 v0.20.4
Checking radicle-oid v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-oid)
Checking radicle-git-ext v0.12.0
Checking radicle-term v0.17.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-term)
Checking radicle-core v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-core)
Checking radicle-cob v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cob)
Checking radicle v0.23.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle)
Checking radicle-fetch v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-fetch)
Checking radicle-cli-test v0.13.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli-test)
Checking radicle-schemars v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-schemars)
Checking radicle-protocol v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-protocol)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 44.42s
+ cargo build --all-targets --workspace
Compiling libc v0.2.183
Compiling cfg-if v1.0.4
Compiling zeroize v1.8.2
Compiling typenum v1.19.0
Compiling memchr v2.8.0
Compiling subtle v2.6.1
Compiling shlex v1.3.0
Compiling regex-syntax v0.8.10
Compiling generic-array v0.14.7
Compiling aho-corasick v1.1.4
Compiling getrandom v0.2.17
Compiling rand_core v0.6.4
Compiling jobserver v0.1.34
Compiling crypto-common v0.1.7
Compiling cc v1.2.57
Compiling const-oid v0.9.6
Compiling smallvec v1.15.1
Compiling serde_core v1.0.228
Compiling regex-automata v0.4.14
Compiling block-buffer v0.10.4
Compiling digest v0.10.7
Compiling cpufeatures v0.2.17
Compiling stable_deref_trait v1.2.1
Compiling thiserror v2.0.18
Compiling fastrand v2.3.0
Compiling scopeguard v1.2.0
Compiling lock_api v0.4.14
Compiling parking_lot_core v0.9.12
Compiling bitflags v2.11.0
Compiling tinyvec_macros v0.1.1
Compiling tinyvec v1.11.0
Compiling parking_lot v0.12.5
Compiling gix-trace v0.1.18
Compiling byteorder v1.5.0
Compiling unicode-normalization v0.1.25
Compiling crc32fast v1.5.0
Compiling itoa v1.0.17
Compiling typeid v1.0.3
Compiling same-file v1.0.6
Compiling hashbrown v0.16.1
Compiling gix-utils v0.3.1
Compiling walkdir v2.5.0
Compiling erased-serde v0.4.10
Compiling prodash v31.0.0
Compiling zlib-rs v0.6.3
Compiling serde v1.0.228
Compiling bstr v1.12.1
Compiling serde_fmt v1.1.0
Compiling value-bag-serde1 v1.12.0
Compiling gix-validate v0.11.0
Compiling gix-path v0.11.1
Compiling hash32 v0.3.1
Compiling heapless v0.8.0
Compiling value-bag v1.12.0
Compiling faster-hex v0.10.0
Compiling log v0.4.29
Compiling zerofrom v0.1.6
Compiling sha1 v0.10.6
Compiling sha1-checked v0.10.0
Compiling yoke v0.8.1
Compiling block-padding v0.3.3
Compiling linux-raw-sys v0.12.1
Compiling zerovec v0.11.5
Compiling inout v0.1.4
Compiling sha2 v0.10.9
Compiling cipher v0.4.4
Compiling zerocopy v0.8.42
Compiling gix-features v0.46.1
Compiling gix-hash v0.22.1
Compiling rustix v1.1.4
Compiling libm v0.2.16
Compiling getrandom v0.4.2
Compiling tinystr v0.8.2
Compiling der v0.7.10
Compiling num-traits v0.2.19
Compiling percent-encoding v2.3.2
Compiling once_cell v1.21.4
Compiling litemap v0.8.1
Compiling writeable v0.6.2
Compiling zerotrie v0.2.3
Compiling icu_locale_core v2.1.1
Compiling potential_utf v0.1.4
Compiling icu_collections v2.1.1
Compiling ppv-lite86 v0.2.21
Compiling icu_provider v2.1.1
Compiling equivalent v1.0.2
Compiling indexmap v2.13.0
Compiling icu_properties_data v2.1.2
Compiling zmij v1.0.21
Compiling icu_normalizer_data v2.1.1
Compiling num-integer v0.1.46
Compiling hmac v0.12.1
Compiling universal-hash v0.5.1
Compiling opaque-debug v0.3.1
Compiling libz-sys v1.1.25
Compiling thiserror v1.0.69
Compiling icu_properties v2.1.2
Compiling icu_normalizer v2.1.1
Compiling tempfile v3.27.0
Compiling serde_json v1.0.149
Compiling spki v0.7.3
Compiling signature v2.2.0
Compiling ff v0.13.1
Compiling spin v0.9.8
Compiling base16ct v0.2.0
Compiling lazy_static v1.5.0
Compiling idna_adapter v1.2.1
Compiling sec1 v0.7.3
Compiling group v0.13.0
Compiling ref-cast v1.0.25
Compiling rand_chacha v0.3.1
Compiling crypto-bigint v0.5.5
Compiling dyn-clone v1.0.20
Compiling utf8_iter v1.0.4
Compiling rand v0.8.5
Compiling idna v1.1.0
Compiling num-iter v0.1.45
Compiling libgit2-sys v0.18.3+1.9.2
Compiling aead v0.5.2
Compiling signature v1.6.4
Compiling ed25519 v1.5.3
Compiling elliptic-curve v0.13.8
Compiling poly1305 v0.8.0
Compiling rfc6979 v0.4.0
Compiling form_urlencoded v1.2.2
Compiling chacha20 v0.9.1
Compiling ascii v1.1.0
Compiling ct-codecs v1.1.6
Compiling amplify_num v0.5.3
Compiling ec25519 v0.1.0
Compiling url v2.5.8
Compiling ecdsa v0.16.9
Compiling primeorder v0.13.6
Compiling git-ref-format-core v0.6.0
Compiling amplify v4.9.0
Compiling polyval v0.6.2
Compiling base64ct v1.8.3
Compiling cyphergraphy v0.3.0
Compiling ghash v0.5.1
Compiling schemars v1.2.1
Compiling pem-rfc7468 v0.7.0
Compiling pkcs8 v0.10.2
Compiling pbkdf2 v0.12.2
Compiling ctr v0.9.2
Compiling aes v0.8.4
Compiling sqlite3-src v0.7.0
Compiling keccak v0.1.6
Compiling aes-gcm v0.10.3
Compiling sha3 v0.10.8
Compiling curve25519-dalek v4.1.3
Compiling pkcs1 v0.7.5
Compiling ssh-encoding v0.2.0
Compiling num-bigint-dig v0.8.6
Compiling ed25519 v2.2.3
Compiling cbc v0.1.2
Compiling blowfish v0.9.1
Compiling base32 v0.4.0
Compiling cypheraddr v0.4.0
Compiling rsa v0.9.10
Compiling bcrypt-pbkdf v0.10.0
Compiling ssh-cipher v0.2.0
Compiling ed25519-dalek v2.2.0
Compiling p256 v0.13.2
Compiling p384 v0.13.1
Compiling p521 v0.13.3
Compiling chacha20poly1305 v0.10.1
Compiling qcheck v1.0.0
Compiling data-encoding v2.10.0
Compiling jiff v0.2.23
Compiling const-str v0.4.3
Compiling base256emoji v1.0.2
Compiling data-encoding-macro v0.1.19
Compiling ssh-key v0.6.7
Compiling noise-framework v0.4.0
Compiling socks5-client v0.4.1
Compiling secrecy v0.10.3
Compiling base-x v0.2.11
Compiling multibase v0.9.2
Compiling ssh-agent-lib v0.5.2
Compiling cyphernet v0.5.2
Compiling crossbeam-utils v0.8.21
Compiling winnow v0.7.15
Compiling crossbeam-channel v0.5.15
Compiling anstyle-query v1.1.5
Compiling utf8parse v0.2.2
Compiling gix-hashtable v0.12.0
Compiling nonempty v0.9.0
Compiling siphasher v1.0.2
Compiling gix-error v0.1.0
Compiling radicle-dag v0.10.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-dag)
Compiling radicle-git-metadata v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-metadata)
Compiling memmap2 v0.9.10
Compiling anstyle v1.0.14
Compiling colorchoice v1.0.5
Compiling is_terminal_polyfill v1.70.2
Compiling radicle-git-ref-format v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-ref-format)
Compiling base64 v0.21.7
Compiling iana-time-zone v0.1.65
Compiling chrono v0.4.44
Compiling radicle-localtime v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-localtime)
Compiling colored v2.2.0
Compiling tree-sitter-language v0.1.7
Compiling gix-error v0.0.0
Compiling serde-untagged v0.1.9
Compiling bytesize v2.3.1
Compiling errno v0.3.14
Compiling dunce v1.0.5
Compiling fast-glob v0.3.3
Compiling gix-date v0.14.0
Compiling anstyle-parse v0.2.7
Compiling adler2 v2.0.1
Compiling gix-actor v0.39.0
Compiling anstream v0.6.21
Compiling gix-date v0.13.0
Compiling gix-fs v0.19.1
Compiling gix-actor v0.38.0
Compiling gix-tempfile v21.0.1
Compiling gix-object v0.56.0
Compiling gix-chunk v0.6.0
Compiling gix-quote v0.6.2
Compiling gix-commitgraph v0.33.0
Compiling gix-object v0.55.0
Compiling gix-chunk v0.5.0
Compiling mio v1.1.1
Compiling sem_safe v0.2.1
Compiling shell-words v1.1.1
Compiling either v1.15.0
Compiling gix-command v0.7.1
Compiling signals_receipts v0.2.5
Compiling gix-commitgraph v0.32.0
Compiling gix-revwalk v0.27.0
Compiling unicode-segmentation v1.12.0
Compiling gix-revwalk v0.26.0
Compiling gix-lock v21.0.1
Compiling gix-url v0.35.2
Compiling gix-config-value v0.17.1
Compiling gix-sec v0.13.1
Compiling gimli v0.32.3
Compiling gix-prompt v0.13.1
Compiling convert_case v0.10.0
Compiling gix-traverse v0.52.0
Compiling object v0.37.3
Compiling addr2line v0.25.1
Compiling gix-revision v0.41.0
Compiling radicle-signals v0.11.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-signals)
Compiling gix-diff v0.58.0
Compiling miniz_oxide v0.8.9
Compiling signal-hook-registry v1.4.8
Compiling gix-glob v0.24.0
Compiling gix-packetline v0.21.1
Compiling tree-sitter v0.24.7
Compiling rustc-demangle v0.1.27
Compiling backtrace v0.3.76
Compiling gix-refspec v0.37.0
Compiling sqlite3-sys v0.18.0
Compiling sqlite v0.37.0
Compiling radicle-crypto v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-crypto)
Compiling gix-transport v0.54.0
Compiling signal-hook v0.3.18
Compiling gix-pack v0.65.0
Compiling arc-swap v1.8.2
Compiling derive_more-impl v2.1.1
Compiling gix-credentials v0.36.0
Compiling gix-ref v0.59.0
Compiling gix-shallow v0.8.1
Compiling gix-negotiate v0.27.0
Compiling regex v1.12.3
Compiling unicode-width v0.2.2
Compiling gix-protocol v0.57.0
Compiling derive_more v2.1.1
Compiling gix-odb v0.75.0
Compiling signal-hook-mio v0.2.5
Compiling xattr v1.6.1
Compiling anstyle-parse v1.0.0
Compiling uuid v1.22.0
Compiling filetime v0.2.27
Compiling bytes v1.11.1
Compiling tar v0.4.45
Compiling git-ref-format-macro v0.6.0
Compiling anstream v1.0.0
Compiling crossterm v0.29.0
Compiling flate2 v1.1.9
Compiling anyhow v1.0.102
Compiling console v0.16.3
Compiling getrandom v0.3.4
Compiling portable-atomic v1.13.1
Compiling snapbox-macros v0.3.10
Compiling salsa20 v0.10.2
Compiling clap_lex v1.1.0
Compiling normalize-line-endings v0.3.0
Compiling siphasher v0.3.11
Compiling streaming-iterator v0.1.9
Compiling strsim v0.11.1
Compiling similar v2.7.0
Compiling unit-prefix v0.5.2
Compiling indicatif v0.18.4
Compiling snapbox v0.4.17
Compiling clap_builder v4.6.0
Compiling bloomy v1.2.0
Compiling scrypt v0.11.0
Compiling inquire v0.9.4
Compiling radicle-surf v0.27.1
Compiling git-ref-format v0.6.0
Compiling unicode-display-width v0.3.0
Compiling systemd-journal-logger v2.2.2
Compiling serde_spanned v1.0.4
Compiling toml_datetime v0.7.5+spec-1.1.0
Compiling tree-sitter-python v0.23.6
Compiling tree-sitter-ruby v0.23.1
Compiling tree-sitter-html v0.23.2
Compiling tree-sitter-toml-ng v0.6.0
Compiling tree-sitter-json v0.24.8
Compiling tree-sitter-css v0.23.2
Compiling tree-sitter-bash v0.23.3
Compiling tree-sitter-md v0.3.2
Compiling tree-sitter-go v0.23.4
Compiling tree-sitter-c v0.23.4
Compiling tree-sitter-rust v0.23.3
Compiling tree-sitter-typescript v0.23.2
Compiling toml_writer v1.0.7+spec-1.1.0
Compiling radicle-std-ext v0.2.0
Compiling pin-project-lite v0.2.17
Compiling toml v0.9.12+spec-1.1.0
Compiling tokio v1.50.0
Compiling clap v4.6.0
Compiling sysinfo v0.37.2
Compiling diff v0.1.13
Compiling radicle-node v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-node)
Compiling yansi v1.0.1
Compiling radicle-cli v0.20.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli)
Compiling pretty_assertions v1.4.1
Compiling human-panic v2.0.6
Compiling clap_complete v4.6.0
Compiling structured-logger v1.0.5
Compiling radicle-systemd v0.12.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-systemd)
Compiling tree-sitter-highlight v0.24.7
Compiling itertools v0.14.0
Compiling socket2 v0.5.10
Compiling humantime v2.3.0
Compiling lexopt v0.3.2
Compiling timeago v0.4.2
Compiling bit-vec v0.8.0
Compiling escargot v0.5.15
Compiling bit-set v0.8.0
Compiling rand_core v0.9.5
Compiling num-bigint v0.4.6
Compiling num-complex v0.4.6
Compiling env_filter v1.0.0
Compiling borrow-or-share v0.2.4
Compiling fluent-uri v0.3.2
Compiling num-rational v0.4.2
Compiling num v0.4.3
Compiling env_logger v0.11.9
Compiling ahash v0.8.12
Compiling phf_shared v0.11.3
Compiling wait-timeout v0.2.1
Compiling quick-error v1.2.3
Compiling fnv v1.0.7
Compiling outref v0.5.2
Compiling vsimd v0.8.0
Compiling radicle-remote-helper v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-remote-helper)
Compiling rusty-fork v0.3.1
Compiling uuid-simd v0.8.0
Compiling test-log v0.2.19
Compiling phf v0.11.3
Compiling referencing v0.30.0
Compiling fraction v0.15.3
Compiling rand_xorshift v0.4.0
Compiling rand_chacha v0.9.0
Compiling rand v0.9.2
Compiling fancy-regex v0.14.0
Compiling git2 v0.20.4
Compiling email_address v0.2.9
Compiling base64 v0.22.1
Compiling unarray v0.1.4
Compiling num-cmp v0.1.0
Compiling bytecount v0.6.9
Compiling proptest v1.10.0
Compiling jsonschema v0.30.0
Compiling emojis v0.6.4
Compiling radicle-oid v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-oid)
Compiling radicle-cob v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cob)
Compiling radicle-core v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-core)
Compiling radicle-git-ext v0.12.0
Compiling radicle-term v0.17.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-term)
Compiling radicle v0.23.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle)
Compiling radicle-windows v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-windows)
Compiling radicle-fetch v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-fetch)
Compiling radicle-protocol v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-protocol)
Compiling radicle-cli-test v0.13.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli-test)
Compiling radicle-schemars v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-schemars)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 59.29s
+ cargo doc --workspace --no-deps --all-features
Checking regex-automata v0.4.14
Compiling num-traits v0.2.19
Compiling syn v1.0.109
Checking once_cell v1.21.4
Checking tempfile v3.27.0
Checking idna v1.1.0
Checking url v2.5.8
Checking num-integer v0.1.46
Checking git2 v0.20.4
Checking num-iter v0.1.45
Checking num-bigint-dig v0.8.6
Compiling amplify_syn v2.0.1
Checking bstr v1.12.1
Checking rsa v0.9.10
Checking gix-validate v0.11.0
Checking gix-path v0.11.1
Checking git-ref-format-core v0.6.0
Compiling amplify_derive v4.0.1
Checking gix-features v0.46.1
Checking ssh-key v0.6.7
Checking gix-hash v0.22.1
Checking radicle-git-ref-format v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-ref-format)
Checking gix-hashtable v0.12.0
Checking radicle-oid v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-oid)
Checking gix-error v0.1.0
Checking rusty-fork v0.3.1
Checking gix-error v0.0.0
Checking proptest v1.10.0
Checking radicle-git-metadata v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-metadata)
Checking radicle-dag v0.10.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-dag)
Checking ssh-agent-lib v0.5.2
Checking gix-date v0.14.0
Checking chrono v0.4.44
Checking radicle-localtime v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-localtime)
Checking gix-actor v0.39.0
Checking gix-date v0.13.0
Checking gix-fs v0.19.1
Checking amplify v4.9.0
Checking gix-tempfile v21.0.1
Checking gix-actor v0.38.0
Checking cyphergraphy v0.3.0
Checking gix-object v0.56.0
Checking gix-chunk v0.6.0
Checking cypheraddr v0.4.0
Checking noise-framework v0.4.0
Checking gix-quote v0.6.2
Checking socks5-client v0.4.1
Checking gix-commitgraph v0.33.0
Checking gix-object v0.55.0
Checking cyphernet v0.5.2
Checking radicle-crypto v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-crypto)
Checking gix-chunk v0.5.0
Checking gix-command v0.7.1
Checking gix-revwalk v0.27.0
Checking gix-commitgraph v0.32.0
Checking radicle-cob v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cob)
Checking radicle-core v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-core)
Checking gix-lock v21.0.1
Checking gix-url v0.35.2
Checking gix-config-value v0.17.1
Checking gix-revwalk v0.26.0
Checking gix-diff v0.58.0
Checking gix-prompt v0.13.1
Checking gix-revision v0.41.0
Checking gix-traverse v0.52.0
Checking radicle v0.23.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle)
Checking radicle-signals v0.11.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-signals)
Checking gix-glob v0.24.0
Checking gix-packetline v0.21.1
Checking regex v1.12.3
Checking gix-transport v0.54.0
Checking gix-refspec v0.37.0
Checking tree-sitter v0.24.7
Checking gix-pack v0.65.0
Checking gix-credentials v0.36.0
Checking git-ref-format v0.6.0
Checking inquire v0.9.4
Checking gix-shallow v0.8.1
Checking gix-ref v0.59.0
Checking gix-negotiate v0.27.0
Checking radicle-git-ext v0.12.0
Checking gix-odb v0.75.0
Checking gix-protocol v0.57.0
Checking uuid v1.22.0
Compiling radicle-cli v0.20.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli)
Checking human-panic v2.0.6
Checking radicle-surf v0.27.1
Checking tree-sitter-toml-ng v0.6.0
Checking radicle-term v0.17.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-term)
Checking tree-sitter-highlight v0.24.7
Checking radicle-systemd v0.12.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-systemd)
Documenting radicle-systemd v0.12.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-systemd)
Documenting radicle-term v0.17.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-term)
Documenting radicle v0.23.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle)
Documenting radicle-core v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-core)
Documenting radicle-cob v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cob)
Documenting radicle-signals v0.11.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-signals)
Documenting radicle-crypto v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-crypto)
Documenting radicle-oid v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-oid)
Documenting radicle-git-ref-format v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-ref-format)
Documenting radicle-localtime v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-localtime)
Documenting radicle-dag v0.10.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-dag)
Documenting radicle-git-metadata v0.2.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-git-metadata)
Documenting radicle-windows v0.1.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-windows)
Checking radicle-fetch v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-fetch)
Documenting radicle-cli v0.20.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli)
Documenting radicle-cli-test v0.13.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-cli-test)
Checking radicle-protocol v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-protocol)
Documenting radicle-protocol v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-protocol)
Documenting radicle-node v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-node)
Documenting radicle-schemars v0.7.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-schemars)
Documenting radicle-fetch v0.19.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-fetch)
Documenting radicle-remote-helper v0.16.0 (/4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/crates/radicle-remote-helper)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 9.83s
Generated /4f2a868f-8682-4b20-bcf0-15e16e8940fe/w/target/doc/radicle/index.html and 20 other files
+ cargo test --workspace --no-fail-fast
Finished `test` profile [unoptimized + debuginfo] target(s) in 0.17s
Running unittests src/lib.rs (target/debug/deps/radicle-1034629edcccc222)
running 323 tests
test canonical::formatter::test::ascii_control_characters ... ok
test canonical::formatter::test::securesystemslib_asserts ... ok
test canonical::formatter::test::ordered_nested_object ... ok
test cob::cache::tests::test_check_version ... ok
test cob::common::test::test_color ... ok
test cob::cache::migrations::_2::tests::test_patch_json_deserialization ... ok
test cob::cache::tests::test_migrate_to ... ok
test cob::common::test::test_emojis ... ok
test cob::cache::migrations::_2::tests::test_migration_2 ... ok
test cob::common::test::test_title ... ok
test cob::identity::test::prop_json_eq_str ... ok
test cob::identity::test::test_identity_redact_revision ... ok
test cob::identity::test::test_identity_reject_concurrent ... ok
test cob::identity::test::test_identity_remove_delegate_concurrent ... ok
test cob::identity::test::test_identity_update_rejected ... ok
test cob::identity::test::test_identity_updates ... ok
test cob::issue::cache::tests::test_counts ... ok
test cob::issue::cache::tests::test_get ... ok
test cob::issue::cache::tests::test_is_empty ... ok
test cob::issue::cache::tests::test_list ... ok
test cob::issue::cache::tests::test_list_by_status ... ok
test cob::issue::cache::tests::test_remove ... ok
test cob::identity::test::test_identity_updates_concurrent ... ok
test cob::identity::test::test_valid_identity ... ok
test cob::issue::test::test_embeds ... ok
test cob::issue::test::test_embeds_edit ... ok
test cob::identity::test::test_identity_updates_concurrent_outdated ... ok
test cob::issue::test::test_invalid_actions ... ok
test cob::issue::test::test_invalid_tx_reference ... ok
test cob::issue::test::test_invalid_tx ... ok
test cob::issue::test::test_invalid_cob ... ok
test cob::issue::test::test_concurrency ... ok
test cob::issue::test::test_issue_all ... ok
test cob::issue::test::test_issue_comment ... ok
test cob::issue::test::test_issue_create_and_assign ... ok
test cob::issue::test::test_issue_comment_redact ... ok
test cob::issue::test::test_issue_create_and_change_state ... ok
test cob::issue::test::test_issue_create_and_get ... ok
test cob::issue::test::test_issue_create_and_reassign ... ok
test cob::issue::test::test_issue_create_and_unassign ... ok
test cob::issue::test::test_issue_edit ... ok
test cob::issue::test::test_issue_edit_description ... ok
test cob::issue::test::test_issue_multilines ... ok
test cob::issue::test::test_issue_state_serde ... ok
test cob::issue::test::test_ordering ... ok
test cob::patch::actions::test::test_review_edit ... ok
test cob::issue::test::test_issue_react ... ok
test cob::issue::test::test_issue_label ... ok
test cob::issue::test::test_issue_reply ... ok
test cob::patch::cache::tests::test_is_empty ... ok
test cob::patch::cache::tests::test_list ... ok
test cob::patch::cache::tests::test_list_by_status ... ok
test cob::patch::cache::tests::test_get ... ok
test cob::patch::encoding::review::test::test_review_deserialize_summary_migration_null_summary ... ok
test cob::patch::encoding::review::test::test_review_deserialize_summary_migration_with_summary ... ok
test cob::patch::encoding::review::test::test_review_deserialize_summary_migration_without_summary ... ok
test cob::patch::encoding::review::test::test_review_deserialize_summary_v2 ... ok
test cob::patch::encoding::review::test::test_review_summary ... ok
test cob::patch::test::test_json ... ok
test cob::patch::test::test_json_serialization ... ok
test cob::patch::cache::tests::test_remove ... ok
test cob::patch::test::test_patch_create_and_get ... ok
test cob::patch::test::test_patch_discussion ... ok
test cob::patch::cache::tests::test_counts ... ok
test cob::patch::test::test_patch_merge ... ok
test cob::patch::test::test_patch_redact ... ok
test cob::patch::test::test_patch_review_comment ... ok
test cob::patch::test::test_patch_review ... ok
test cob::patch::test::test_patch_review_duplicate ... ok
test cob::patch::test::test_patch_review_remove_summary ... ok
test cob::patch::test::test_patch_review_edit ... ok
test cob::patch::test::test_patch_review_edit_comment ... ok
test cob::patch::test::test_reactions_json_serialization ... ok
test cob::patch::test::test_revision_edit_redact ... ok
test cob::patch::test::test_revision_reaction ... ok
test cob::patch::test::test_revision_review_merge_redacted ... ok
test cob::stream::tests::test_all_from ... ok
test cob::stream::tests::test_all_from_until ... ok
test cob::patch::test::test_patch_review_revision_redact ... ok
test cob::stream::tests::test_all_until ... ok
test cob::stream::tests::test_regression_from_until ... ok
test cob::stream::tests::test_from_until ... ok
test cob::thread::tests::test_comment_edit_missing ... ok
test cob::thread::tests::test_comment_edit_redacted ... ok
test cob::thread::tests::test_comment_redact_missing ... ok
test cob::thread::tests::test_duplicate_comments ... ok
test cob::patch::test::test_patch_update ... ok
test cob::thread::tests::test_edit_comment ... ok
test cob::thread::tests::test_redact_comment ... ok
test git::canonical::quorum::test::merge_base_commutative ... ok
test git::canonical::quorum::test::test_merge_bases ... ok
test cob::thread::tests::test_timeline ... ok
test git::canonical::rules::tests::test_deserialization ... ok
test git::canonical::rules::tests::test_deserialize_extensions ... ok
test git::canonical::rules::tests::test_order ... ok
test git::canonical::rules::tests::test_roundtrip ... ok
test git::canonical::rules::tests::test_canonical ... ok
test git::canonical::rules::tests::test_rule_validate_success ... ok
test git::canonical::rules::tests::test_special_branches ... ok
test git::canonical::tests::test_commit_quorum_fork_of_a_fork ... ok
test git::canonical::tests::test_commit_quorum_forked_merge_commits ... ok
test git::canonical::tests::test_commit_quorum_groups ... ok
test git::canonical::tests::test_commit_quorum_linear ... ok
test git::canonical::tests::test_commit_quorum_merges ... ok
test git::canonical::tests::test_commit_quorum_single ... ok
test git::canonical::tests::test_commit_quorum_three_way_fork ... ok
test git::canonical::tests::test_commit_quorum_two_way_fork ... ok
test git::canonical::tests::test_quorum_different_types ... ok
test git::canonical::rules::tests::test_rule_validate_failures ... ok
test git::canonical::tests::test_tag_quorum ... ok
test git::test::test_version_from_str ... ok
test git::test::test_version_ord ... ok
test identity::did::test::test_did_encode_decode ... ok
test identity::did::test::test_did_vectors ... ok
test cob::patch::cache::tests::test_find_by_revision ... ok
test identity::doc::test::test_canonical_doc ... ok
test identity::doc::test::test_canonical_example ... ok
test identity::doc::test::test_duplicate_dids ... ok
test identity::doc::test::test_future_version_error ... ok
test identity::doc::test::test_is_valid_version ... ok
test identity::doc::test::test_max_delegates ... ok
test identity::doc::test::test_not_found ... ok
test identity::doc::test::test_parse_version ... ok
test identity::doc::test::test_visibility_json ... ok
test identity::doc::update::test::test_can_update_crefs ... ok
test identity::doc::update::test::test_cannot_include_default_branch_rule ... ok
test identity::doc::update::test::test_default_branch_rule_exists_after_verification ... ok
test identity::project::test::test_project_name ... ok
test node::address::store::test::test_alias ... ok
test node::address::store::test::test_disconnected ... ok
test node::address::store::test::test_disconnected_ban ... ok
test node::address::store::test::test_empty ... ok
test node::address::store::test::test_entries ... ok
test node::address::store::test::test_get_none ... ok
test node::address::store::test::test_insert_and_get ... ok
test node::address::store::test::test_insert_and_remove ... ok
test node::address::store::test::test_insert_and_update ... ok
test node::address::store::test::test_insert_duplicate ... ok
test node::address::store::test::test_node_aliases ... ok
test node::address::store::test::test_remove_nothing ... ok
test node::command::test::command_result ... ok
test node::config::test::deserialize_migrating_scope ... ok
test node::config::test::fetch_level_min ... ok
test node::config::test::onion_absent ... ok
test node::config::test::onion_null ... ok
test node::config::test::partial ... ok
test node::config::test::regression_ipv6_address_brackets ... ok
test node::config::test::regression_ipv6_address_no_brackets ... ok
test node::config::test::serialize_migrating_scope ... ok
test node::db::config::test::database_config_valid_combinations ... ok
test node::db::config::test::invalid ... ok
test node::db::test::migration_8::all_ipv6_formatted_dns_addresses_are_retyped ... ok
test node::db::test::migration_8::dns_address_starting_with_bracket_but_missing_closing_bracket_colon_is_unaffected ... ok
test node::db::test::migration_8::dns_address_with_bracket_not_at_start_is_unaffected ... ok
test node::db::test::migration_8::ipv4_address_is_unaffected ... ok
test node::db::test::migration_8::ipv6_formatted_dns_address_is_deleted_when_correct_ipv6_row_already_exists ... ok
test node::db::test::migration_8::ipv6_formatted_dns_address_is_retyped_to_ipv6 ... ok
test node::db::test::migration_8::migration_applies_to_all_nodes ... ok
test node::db::test::migration_8::plain_dns_hostname_without_brackets_is_unaffected ... ok
test node::db::test::migration_8::retype_preserves_address_metadata ... ok
test node::db::test::test_version ... ok
test node::features::test::test_operations ... ok
test node::notifications::store::test::test_branch_notifications ... ok
test node::notifications::store::test::test_clear ... ok
test node::notifications::store::test::test_cob_notifications ... ok
test node::notifications::store::test::test_counts_by_repo ... ok
test node::notifications::store::test::test_duplicate_notifications ... ok
test node::notifications::store::test::test_notification_status ... ok
test node::policy::store::test::test_follow_and_unfollow_node ... ok
test node::policy::store::test::test_node_aliases ... ok
test node::policy::store::test::test_node_policies ... ok
test node::policy::store::test::test_node_policy ... ok
test node::policy::store::test::test_repo_policies ... ok
test node::policy::store::test::test_repo_policy ... ok
test node::policy::store::test::test_seed_and_unseed_repo ... ok
test node::policy::store::test::test_update_alias ... ok
test node::policy::store::test::test_update_scope ... ok
test node::refs::store::test::test_count ... ok
test node::refs::store::test::test_set_and_delete ... ok
test node::refs::store::test::test_set_and_get ... ok
test node::routing::test::test_count ... ok
test git::canonical::tests::test_quorum_properties ... ok
test node::routing::test::test_entries ... ok
test node::routing::test::test_insert_and_get ... ok
test node::routing::test::test_insert_and_get_resources ... ok
test node::routing::test::test_insert_and_remove ... ok
test node::routing::test::test_insert_duplicate ... ok
test node::routing::test::test_insert_existing_updated_time ... ok
test node::routing::test::test_len ... ok
test node::routing::test::test_remove_many ... ok
test node::routing::test::test_remove_redundant ... ok
test node::routing::test::test_update_existing_multi ... ok
test node::sync::announce::test::all_synced_nodes_are_preferred_seeds ... ok
test node::sync::announce::test::announcer_adapts_target_to_reach ... ok
test node::routing::test::test_prune ... ok
test node::sync::announce::test::announcer_preferred_seeds_or_replica_factor ... ok
test node::sync::announce::test::announcer_reached_max_replication_target ... ok
test node::sync::announce::test::announcer_reached_min_replication_target ... ok
test node::sync::announce::test::announcer_reached_preferred_seeds ... ok
test node::sync::announce::test::announcer_synced_with_unknown_node ... ok
test node::sync::announce::test::announcer_with_replication_factor_zero_and_preferred_seeds ... ok
test node::sync::announce::test::announcer_timed_out ... ok
test node::sync::announce::test::construct_node_appears_in_multiple_input_sets ... ok
test node::sync::announce::test::construct_only_preferred_seeds_provided ... ok
test node::sync::announce::test::cannot_construct_announcer ... ok
test node::sync::announce::test::invariant_progress_should_match_state ... ok
test node::sync::announce::test::local_node_in_multiple_sets ... ok
test node::sync::announce::test::local_node_in_preferred_seeds ... ok
test node::sync::announce::test::local_node_in_synced_set ... ok
test node::sync::announce::test::local_node_only_in_all_sets_results_in_no_seeds_error ... ok
test node::sync::announce::test::local_node_in_unsynced_set ... ok
test node::sync::announce::test::preferred_seeds_already_synced ... ok
test node::sync::announce::test::synced_with_same_node_multiple_times ... ok
test node::sync::announce::test::synced_with_local_node_is_ignored ... ok
test node::sync::announce::test::timed_out_after_reaching_success ... ok
test node::sync::fetch::test::all_nodes_are_candidates ... ok
test node::sync::fetch::test::could_not_reach_target ... ok
test node::sync::fetch::test::ignores_duplicates_and_local_node ... ok
test node::sync::fetch::test::all_nodes_are_fetchable ... ok
test node::sync::fetch::test::preferred_seeds_target_returned_over_replicas ... ok
test node::sync::fetch::test::reaches_target_of_max_replicas ... ok
test node::sync::fetch::test::reaches_target_of_preferred_seeds ... ok
test node::sync::fetch::test::reaches_target_of_replicas ... ok
test node::sync::test::ensure_replicas_construction ... ok
test node::sync::test::replicas_constrain_to ... ok
test node::test::test_address ... ok
test node::test::test_alias ... ok
test node::test::test_command_result ... ok
test node::test::test_user_agent ... ok
test node::timestamp::tests::test_timestamp_max ... ok
test profile::test::canonicalize_home ... ok
test profile::test::test_config ... ok
test cob::thread::tests::prop_ordering ... ok
test rad::tests::test_checkout ... ok
test rad::tests::test_fork ... ok
test rad::tests::test_init ... ok
test storage::git::tests::test_references_of ... ok
test storage::git::transport::local::url::test::test_url_parse ... ok
test storage::git::transport::local::url::test::test_url_to_string ... ok
test storage::git::transport::remote::url::test::test_url_parse ... ok
test storage::git::tests::test_sign_refs ... ok
test profile::config::test::schema ... ok
test storage::refs::sigrefs::read::test::commit_reader::identity_root_error ... ok
test storage::refs::sigrefs::read::test::commit_reader::missing_commit ... ok
test storage::refs::sigrefs::read::test::commit_reader::read_ok ... ok
test storage::refs::sigrefs::read::test::commit_reader::too_many_parents ... ok
test storage::refs::sigrefs::read::test::commit_reader::tree_error ... ok
test storage::refs::sigrefs::read::test::identity_root_reader::doc_blob_error ... ok
test storage::refs::sigrefs::read::test::identity_root_reader::missing_identity ... ok
test storage::refs::sigrefs::read::test::identity_root_reader::read_ok_none ... ok
test storage::refs::sigrefs::read::test::identity_root_reader::read_ok_some ... ok
test storage::refs::sigrefs::read::test::resolve_tip::find_reference_error ... ok
test storage::refs::sigrefs::read::test::resolve_tip::missing_sigrefs ... ok
test storage::refs::sigrefs::read::test::resolve_tip::resolve_tip_ok ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::detect_parent::root_without_parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::detect_parent::root_without_root ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::downgrade::parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::downgrade::restore ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::downgrade::root ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::downgrade::root_with_parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::head_commit_error ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::head_verify_mismatched_identity_error ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::head_verify_signature_error ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::invalid_parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::read_ok_no_parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::read_ok_parent ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::read_ok_root ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::replay::alternating ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::replay::chain ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::replay::multiple ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::replay::root_at_head ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::single_commit ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::two_commits ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::walk_commit_error ... ok
test storage::refs::sigrefs::read::test::signed_refs_reader::walk_verify_error ... ok
test storage::refs::sigrefs::read::test::tree_reader::missing_both ... ok
test storage::refs::sigrefs::read::test::tree_reader::missing_refs ... ok
test storage::refs::sigrefs::read::test::tree_reader::missing_signature ... ok
test storage::refs::sigrefs::read::test::tree_reader::parse_refs_error ... ok
test storage::refs::sigrefs::read::test::tree_reader::parse_signature_error ... ok
test storage::refs::sigrefs::read::test::tree_reader::read_ok ... ok
test storage::refs::sigrefs::read::test::tree_reader::read_refs_error ... ok
test storage::refs::sigrefs::read::test::tree_reader::read_signature_error ... ok
test storage::refs::sigrefs::write::test::commit_writer::tree_error ... ok
test storage::refs::sigrefs::write::test::commit_writer::write_commit_error ... ok
test storage::refs::sigrefs::write::test::commit_writer::write_empty_refs ... ok
test storage::refs::sigrefs::write::test::commit_writer::write_root_ok ... ok
test storage::refs::sigrefs::write::test::commit_writer::write_with_parent_ok ... ok
test storage::refs::sigrefs::write::test::head_reader::no_head ... ok
test storage::refs::sigrefs::write::test::head_reader::read_ok ... ok
test storage::refs::sigrefs::write::test::head_reader::reference_error ... ok
test storage::refs::sigrefs::write::test::head_reader::refs_blob_error ... ok
test storage::refs::sigrefs::write::test::head_reader::refs_blob_missing ... ok
test storage::refs::sigrefs::write::test::head_reader::refs_parse_error ... ok
test storage::refs::sigrefs::write::test::head_reader::signature_blob_error ... ok
test storage::refs::sigrefs::write::test::head_reader::signature_blob_missing ... ok
test storage::refs::sigrefs::write::test::head_reader::signature_parse_error ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::commit_error ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::head_error ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::never_write_rad_sigrefs ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::reference_error ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::unchanged ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::unchanged_force_writes_new_commit ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::write_empty_refs ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::write_root_ok ... ok
test storage::refs::sigrefs::write::test::signed_refs_writer::write_with_parent_ok ... ok
test storage::refs::sigrefs::write::test::tree_writer::sign_error ... ok
test storage::refs::sigrefs::write::test::tree_writer::write_ok ... ok
test storage::refs::sigrefs::write::test::tree_writer::write_tree_error ... ok
test storage::refs::tests::prop_canonical_roundtrip ... ok
test storage::refs::tests::test_rid_verification ... ok
test storage::tests::test_storage ... ok
test test::assert::test::assert_with_message ... ok
test test::assert::test::test_assert_no_move ... ok
test test::assert::test::test_assert_panic_0 - should panic ... ok
test test::assert::test::test_assert_panic_1 - should panic ... ok
test test::assert::test::test_assert_panic_2 - should panic ... ok
test test::assert::test::test_assert_succeed ... ok
test test::assert::test::test_panic_message ... ok
test version::test::test_version ... ok
test storage::refs::sigrefs::property::idempotent ... ok
test identity::doc::test::prop_encode_decode ... ok
test storage::refs::sigrefs::property::roundtrip ... ok
test result: ok. 323 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 3.54s
Running unittests src/lib.rs (target/debug/deps/radicle_cli-f88c627dc09e436f)
running 46 tests
test commands::block::args::test::should_not_parse ... ok
test commands::block::args::test::should_parse_rid ... ok
test commands::block::args::test::should_parse_nid ... ok
test commands::clone::args::test::should_parse_rid_non_urn ... ok
test commands::clone::args::test::should_parse_rid_url ... ok
test commands::clone::args::test::should_parse_rid_urn ... ok
test commands::cob::args::test::should_allow_log_pretty_format ... ok
test commands::cob::args::test::should_allow_log_json_format ... ok
test commands::cob::args::test::should_allow_show_json_format ... ok
test commands::cob::args::test::should_allow_update_json_format ... ok
test commands::cob::args::test::should_not_allow_show_pretty_format ... ok
test commands::fork::args::test::should_not_parse_rid_url ... ok
test commands::fork::args::test::should_parse_rid_non_urn ... ok
test commands::cob::args::test::should_not_allow_update_pretty_format ... ok
test commands::fork::args::test::should_parse_rid_urn ... ok
test commands::id::args::test::should_not_clobber_payload_args ... ok
test commands::id::args::test::should_not_parse_single_payloads ... ok
test commands::id::args::test::should_not_parse_single_payload ... ok
test commands::id::args::test::should_not_parse_into_payload - should panic ... ok
test commands::id::args::test::should_parse_into_payload ... ok
test commands::init::args::test::should_not_parse_rid_url ... ok
test commands::id::args::test::should_parse_single_payload ... ok
test commands::id::args::test::should_parse_multiple_payloads ... ok
test commands::init::args::test::should_parse_rid_non_urn ... ok
test commands::init::args::test::should_parse_rid_urn ... ok
test commands::inspect::test::test_tree ... ok
test commands::patch::review::builder::tests::test_review_comments_before ... ok
test commands::patch::review::builder::tests::test_review_comments_basic ... ok
test commands::patch::review::builder::tests::test_review_comments_multiline ... ok
test commands::patch::review::builder::tests::test_review_comments_split_hunk ... ok
test commands::publish::args::test::should_parse_rid_non_urn ... ok
test commands::publish::args::test::should_parse_rid_urn ... ok
test commands::publish::args::test::should_not_parse_rid_url ... ok
test commands::watch::args::test::should_parse_ref_str ... ok
test git::pretty_diff::test::test_pretty ... ignored
test git::ddiff::tests::diff_encode_decode_ddiff_hunk ... ok
test git::unified_diff::test::test_diff_content_encode_decode_content ... ok
test git::unified_diff::test::test_diff_encode_decode_diff ... ok
test terminal::args::test::should_not_parse ... ok
test terminal::args::test::should_parse_nid ... ok
test terminal::args::test::should_parse_rid ... ok
test terminal::format::test::test_strip_comments ... ok
test terminal::patch::test::test_edit_display_message ... ok
test terminal::format::test::test_bytes ... ok
test terminal::patch::test::test_create_display_message ... ok
test terminal::patch::test::test_update_display_message ... ok
test result: ok. 45 passed; 0 failed; 1 ignored; 0 measured; 0 filtered out; finished in 0.01s
Running unittests src/main.rs (target/debug/deps/rad-8443661264181606)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running tests/commands.rs (target/debug/deps/commands-980d3acf3e9a8617)
running 115 tests
test commands::clone::rad_clone_bare ... ok
test commands::clone::rad_clone ... ok
test commands::checkout::rad_checkout ... ok
test commands::clone::rad_clone_all ... ok
test commands::clone::rad_clone_scope ... ok
test commands::clone::rad_clone_unknown ... ok
test commands::clone::rad_clone_directory ... ok
test commands::clone::rad_clone_partial_fail ... ok
test commands::clone::rad_clone_connect ... ok
test commands::cob::rad_cob_multiset ... ok
test commands::clone::test_clone_without_seeds ... ok
test commands::cob::rad_cob_log ... ok
test commands::cob::rad_cob_operations ... ok
test commands::cob::rad_cob_migrate ... ok
test commands::cob::rad_cob_show ... ok
test commands::cob::rad_cob_update_identity ... ok
test commands::cob::rad_cob_update ... ok
test commands::cob::test_cob_deletion ... ok
test commands::cob::test_cob_replication ... ok
test commands::git::git_push_amend ... ok
test commands::git::git_push_and_fetch ... ok
test commands::git::git_push_canonical_annotated_tags ... ok
test commands::git::git_push_canonical_lightweight_tags ... ok
test commands::git::git_push_force_with_lease ... ok
test commands::git::git_push_diverge ... ok
test commands::git::git_push_rollback ... ok
test commands::id::rad_id_collaboration ... ignored, slow
test commands::id::rad_id ... ok
test commands::git::git_push_converge ... ok
test commands::git::git_tag ... ok
test commands::id::rad_id_private ... ok
test commands::id::rad_id_conflict ... ok
test commands::id::rad_id_threshold_soft_fork ... ok
test commands::id::rad_id_threshold ... ok
test commands::id::rad_id_unknown_field ... ok
test commands::id::rad_id_update_delete_field ... ok
test commands::init::rad_init ... ignored, part of many other tests
test commands::id::rad_id_multi_delegate ... ok
test commands::init::rad_init_detached_head ... ok
test commands::id::rad_id_unauthorized_delegate ... ok
test commands::init::rad_init_bare ... ok
test commands::init::rad_init_no_git ... ok
test commands::init::rad_init_existing ... ok
test commands::init::rad_init_existing_bare ... ok
test commands::init::rad_init_no_seed ... ok
test commands::init::rad_init_private ... ok
test commands::init::rad_init_private_no_seed ... ok
test commands::init::rad_init_private_clone ... ok
test commands::inbox::rad_inbox ... ok
test commands::init::rad_init_private_clone_seed ... ok
test commands::init::rad_init_private_seed ... ok
test commands::init::rad_init_sync_not_connected ... ok
test commands::init::rad_init_sync_preferred ... ok
test commands::init::rad_init_with_existing_remote ... ok
test commands::init::rad_publish ... ok
test commands::issue::rad_issue ... ok
test commands::jj::rad_jj_bare ... ok
test commands::jj::rad_jj_colocated_patch ... ok
test commands::issue::rad_issue_list ... ok
test commands::node::rad_node_connect ... ok
test commands::node::rad_node_connect_without_address ... ok
test commands::patch::rad_merge_after_update ... ok
test commands::node::rad_node ... ok
test commands::patch::rad_merge_no_ff ... ok
test commands::patch::rad_merge_via_push ... ok
test commands::patch::rad_patch ... ok
test commands::patch::rad_patch_ahead_behind ... ok
test commands::patch::rad_patch_change_base ... ok
test commands::patch::rad_patch_checkout ... ok
test commands::init::rad_init_sync_and_clone ... ok
test commands::init::rad_init_sync_timeout ... ok
test commands::patch::rad_patch_checkout_revision ... ok
test commands::patch::rad_patch_detached_head ... ok
test commands::patch::rad_patch_checkout_force ... ok
test commands::patch::rad_patch_diff ... ok
test commands::patch::rad_patch_draft ... ok
test commands::patch::rad_patch_edit ... ok
test commands::patch::rad_patch_fetch_2 ... FAILED
test commands::patch::rad_patch_fetch_1 ... ok
test commands::patch::rad_patch_merge_draft ... ok
test commands::patch::rad_patch_revert_merge ... ok
test commands::patch::rad_patch_delete ... ok
test commands::patch::rad_patch_open_explore ... ok
test commands::patch::rad_patch_update ... ok
test commands::patch::rad_patch_via_push ... FAILED
test commands::policy::rad_block ... ok
test commands::policy::rad_seed_and_follow ... ok
test commands::patch::rad_review_by_hunk ... ok
test commands::policy::rad_seed_policy_allow_no_scope ... ok
test commands::policy::rad_seed_scope ... ok
test commands::policy::rad_seed_many ... ok
test commands::policy::rad_unseed ... ok
test commands::policy::rad_unseed_many ... ok
test commands::patch::rad_push_and_pull_patches ... ok
test commands::remote::rad_remote ... ok
test commands::sync::rad_sync_without_node ... ok
test commands::sync::rad_sync ... ok
test commands::utility::framework_home ... ok
test commands::utility::rad_auth ... ok
test commands::utility::rad_auth_errors ... ok
test commands::utility::rad_clean ... ok
test commands::utility::rad_config ... ok
test commands::patch::rad_patch_pull_update ... ok
test commands::utility::rad_diff ... ok
test commands::utility::rad_help ... ok
test commands::utility::rad_inspect ... ok
test commands::utility::rad_key_mismatch ... ok
test commands::utility::rad_self ... ok
test commands::utility::rad_warn_old_nodes ... ok
test commands::sync::rad_fetch ... ok
test commands::watch::rad_watch ... ok
test rad_remote ... ok
test commands::sync::test_replication_via_seed ... ok
test commands::workflow::rad_workflow ... ok
test commands::utility::rad_fork ... ok
failures:
---- commands::patch::rad_patch_fetch_2 stdout ----
1775142832 test: rad-init:6: `rad init --name heartwood --description Radicle Heartwood Protocol & Stack --no-confirm --public -v` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-init:28: `rad init` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-init:35: `rad ls` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-init:46: `rad node inventory` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:6: `git checkout -b alice/1 -q` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:7: `git commit --allow-empty -m Changes #1 -q` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:8: `git push rad -o patch.message=Changes HEAD:refs/patches` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:12: `git checkout master -q` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:13: `git branch -D alice/1 -q` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:14: `git update-ref -d refs/remotes/rad/alice/1` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:15: `git update-ref -d refs/remotes/rad/patches/5e2dedcc5d515fcbc1cca483d3376609fe889bfb` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:16: `git gc --prune=now` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:17: `git branch -r` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:22: `git pull` @ /tmp/.tmpNT8vgX/alice/work
1775142832 test: rad-patch-fetch-2:24: `git branch -r` @ /tmp/.tmpNT8vgX/alice/work
thread 'commands::patch::rad_patch_fetch_2' panicked at crates/radicle-cli-test/src/lib.rs:502:36:
--- Expected
++++ actual: stdout
1 - rad/HEAD -> rad/master
2 1 | rad/master
3 2 | rad/patches/5e2dedcc5d515fcbc1cca483d3376609fe889bfb
Exit status: 0
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
---- commands::patch::rad_patch_via_push stdout ----
1775142833 test: rad-init:6: `rad init --name heartwood --description Radicle Heartwood Protocol & Stack --no-confirm --public -v` @ /tmp/.tmpMKQYgk/alice/work
1775142833 test: rad-init:28: `rad init` @ /tmp/.tmpMKQYgk/alice/work
1775142833 test: rad-init:35: `rad ls` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-init:46: `rad node inventory` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:7: `git checkout -b feature/1` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:9: `git commit -a -m Add things -q --allow-empty` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:10: `git push -o patch.message=Add things #1 -o patch.message=See commits for details. rad HEAD:refs/patches` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:22: `rad patch show 6035d2f582afbe01ff23ea87528ae523d76875b6` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:45: `git branch -vv` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:53: `git status --short --branch` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:55: `git fetch` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:56: `git push` @ /tmp/.tmpMKQYgk/alice/work
1775142834 test: rad-patch-via-push:62: `git show-ref` @ /tmp/.tmpMKQYgk/alice/work
thread 'commands::patch::rad_patch_via_push' panicked at crates/radicle-cli-test/src/lib.rs:502:36:
--- Expected
++++ actual: stdout
1 1 | 42d894a83c9c356552a57af09ccdbd5587a99045 refs/heads/feature/1
2 2 | f2de534b5e81d7c6e2dcaf58c3dd91573c0a0354 refs/heads/master
3 - f2de534b5e81d7c6e2dcaf58c3dd91573c0a0354 refs/remotes/rad/HEAD
4 3 | f2de534b5e81d7c6e2dcaf58c3dd91573c0a0354 refs/remotes/rad/master
5 4 | 42d894a83c9c356552a57af09ccdbd5587a99045 refs/remotes/rad/patches/6035d2f582afbe01ff23ea87528ae523d76875b6
Exit status: 0
failures:
commands::patch::rad_patch_fetch_2
commands::patch::rad_patch_via_push
test result: FAILED. 111 passed; 2 failed; 2 ignored; 0 measured; 0 filtered out; finished in 74.93s
error: test failed, to rerun pass `-p radicle-cli --test commands`
Running unittests src/lib.rs (target/debug/deps/radicle_cli_test-bd3e3c4dcb1548d6)
running 3 tests
test tests::test_parse ... ok
test tests::test_run ... ok
test tests::test_example_spaced_brackets ... ok
test result: ok. 3 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_cob-6fe139a7978b5280)
running 9 tests
test object::tests::test_serde ... ok
test tests::git::roundtrip ... ok
test tests::invalid_parse_refstr ... ok
test tests::git::list_cobs ... ok
test type_name::test::invalid_typenames ... ok
test type_name::test::valid_typenames ... ok
test tests::git::traverse_cobs ... ok
test tests::git::update_cob ... ok
test tests::parse_refstr ... ok
test result: ok. 9 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.01s
Running unittests src/lib.rs (target/debug/deps/radicle_core-e6bfed2c189a2d2c)
running 4 tests
test repo::test::invalid ... ok
test repo::test::valid ... ok
test repo::test::assert_prop_roundtrip_parse ... ok
test repo::serde_impls::test::assert_prop_roundtrip_serde_json ... ok
test result: ok. 4 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_crypto-99d51a8f68c92ad6)
running 11 tests
test ssh::agent::test::test_agent_encoding_sign ... ok
test ssh::agent::test::test_agent_encoding_remove ... ok
test ssh::fmt::test::test_key ... ok
test ssh::fmt::test::test_fingerprint ... ok
test ssh::keystore::tests::test_init_no_passphrase ... ok
test tests::prop_encode_decode ... ok
test tests::test_e25519_dh ... ok
test tests::test_encode_decode ... ok
test tests::prop_key_equality ... ok
test ssh::keystore::tests::test_signer ... ok
test ssh::keystore::tests::test_init_passphrase ... ok
test result: ok. 11 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.83s
Running unittests src/lib.rs (target/debug/deps/radicle_dag-05f7457e7194a495)
running 20 tests
test tests::test_contains ... ok
test tests::test_dependencies ... ok
test tests::test_cycle ... ok
test tests::test_diamond ... ok
test tests::test_fold_diamond ... ok
test tests::test_fold_multiple_roots ... ok
test tests::test_fold_reject ... ok
test tests::test_fold_sorting_1 ... ok
test tests::test_get ... ok
test tests::test_fold_sorting_2 ... ok
test tests::test_is_empty ... ok
test tests::test_merge_1 ... ok
test tests::test_len ... ok
test tests::test_merge_2 ... ok
test tests::test_prune_1 ... ok
test tests::test_complex ... ok
test tests::test_prune_2 ... ok
test tests::test_prune_by_sorting ... ok
test tests::test_remove ... ok
test tests::test_siblings ... ok
test result: ok. 20 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_fetch-d6686b91a80123f8)
running 1 test
test stage::test::valid_refspecs ... ok
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_git_metadata-7266929c438027dc)
running 24 tests
test commit::parse::test::error::invalid_author ... ok
test commit::parse::test::error::invalid_format_continuation_without_preceding_header ... ok
test commit::parse::test::error::invalid_committer ... ok
test commit::parse::test::error::invalid_parent ... ok
test commit::parse::test::error::missing_author ... ok
test commit::parse::test::error::invalid_tree ... ok
test commit::parse::test::error::missing_committer ... ok
test commit::parse::test::error::missing_header_body_separator ... ok
test commit::parse::test::error::missing_tree_empty_header ... ok
test commit::parse::test::success::commit_last_paragraph_kept_in_message_when_not_all_trailers ... ok
test commit::parse::test::success::commit_with_extra_headers ... ok
test commit::parse::test::error::missing_tree_wrong_first_line ... ok
test commit::parse::test::success::commit_gpgsig_is_preserved_and_strip_removes_it ... ok
test commit::parse::test::success::commit_with_multiline_gpgsig ... ok
test commit::parse::test::success::commit_with_single_parent ... ok
test commit::parse::test::success::merge_commit ... ok
test commit::parse::test::success::commit_with_trailers ... ok
test commit::parse::test::success::root_commit ... ok
test commit::parse::test::unit::body_last_paragraph_not_trailers_stays_in_message ... ok
test commit::parse::test::success::roundtrip ... ok
test commit::parse::test::unit::body_no_paragraph_separator_means_no_trailers ... ok
test commit::parse::test::unit::trailers_accepts_empty_input ... ok
test commit::parse::test::unit::trailers_rejects_invalid_token_chars ... ok
test commit::parse::test::unit::trailers_rejects_line_without_separator ... ok
test result: ok. 24 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_git_ref_format-b95a9bd0637298c8)
running 9 tests
test test::component ... ok
test test::component_invalid - should panic ... ok
test test::pattern ... ok
test test::qualified_invalid - should panic ... ok
test test::qualified_pattern ... ok
test test::qualified ... ok
test test::qualified_pattern_invalid - should panic ... ok
test test::refname ... ok
test test::refname_invalid - should panic ... ok
test result: ok. 9 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_localtime-9ecb37b1e50cb4e3)
running 1 test
test serde_impls::test::test_localtime ... ok
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_node-a19843acdcc94364)
running 79 tests
test reactor::timer::tests::test_next ... ok
test reactor::timer::tests::test_wake ... ok
test reactor::timer::tests::test_wake_exact ... ok
test control::tests::test_control_socket ... ok
test control::tests::test_seed_unseed ... ok
test fingerprint::tests::matching ... ok
test tests::e2e::missing_default_branch ... ok
test tests::e2e::fetch_does_not_contain_rad_sigrefs_parent ... ok
test tests::e2e::missing_delegate_default_branch ... ok
test tests::e2e::test_background_foreground_fetch ... ok
test tests::e2e::test_block_prevents_connection ... ok
test tests::e2e::test_block_active_connection ... ok
test tests::e2e::test_block_prevents_fetch ... ok
test tests::e2e::test_channel_reader_limit ... ok
test tests::e2e::test_clone ... ok
test tests::e2e::test_catchup_on_refs_announcements ... ok
test tests::e2e::test_dont_fetch_owned_refs ... ok
test tests::e2e::test_fetch_followed_remotes ... ok
test tests::e2e::test_connection_crossing ... ok
test tests::e2e::test_concurrent_fetches ... ok
test tests::e2e::test_fetch_preserve_owned_refs ... ok
test tests::e2e::test_fetch_unseeded ... ok
test tests::e2e::test_fetch_up_to_date ... ok
test tests::e2e::test_fetch_emits_canonical_ref_update ... ok
test tests::e2e::test_inventory_sync_basic ... ok
test tests::e2e::test_large_fetch ... ok
test tests::e2e::test_migrated_clone ... ok
test tests::e2e::test_missing_remote ... ok
test tests::e2e::test_multiple_offline_inits ... ok
test tests::e2e::test_non_fastforward_identity_doc ... ok
test tests::e2e::test_non_fastforward_sigrefs ... ok
test tests::e2e::test_outdated_delegate_sigrefs ... ok
test tests::e2e::test_outdated_sigrefs ... ok
test tests::e2e::test_replication ... ok
test tests::e2e::test_inventory_sync_bridge ... ok
test tests::e2e::test_replication_invalid ... ok
test tests::e2e::test_inventory_sync_ring ... ok
test tests::e2e::test_replication_ref_in_sigrefs ... ok
test tests::test_announcement_rebroadcast ... ok
test tests::e2e::test_inventory_sync_star ... ok
test tests::test_announcement_rebroadcast_timestamp_filtered ... ok
test tests::test_announcement_rebroadcast_duplicates ... ok
test tests::test_connection_kept_alive ... ok
test tests::test_announcement_relay ... ok
test tests::test_disconnecting_unresponsive_peer ... ok
test tests::test_fetch_missing_inventory_on_gossip ... ok
test tests::test_fetch_missing_inventory_on_schedule ... ok
test tests::test_inbound_connection ... ok
test tests::test_inventory_decode ... ok
test tests::test_init_and_seed ... ok
test tests::test_inventory_relay ... ok
test tests::test_inventory_relay_bad_timestamp ... ok
test tests::test_inventory_sync ... ok
test tests::test_maintain_connections ... ok
test tests::test_maintain_connections_failed_attempt ... ok
test tests::test_maintain_connections_transient ... ok
test tests::test_outbound_connection ... ok
test tests::test_inventory_pruning ... ok
test tests::test_persistent_peer_connect ... ok
test tests::test_persistent_peer_reconnect_success ... ok
test tests::test_persistent_peer_reconnect_attempt ... ok
test tests::test_ping_response ... ok
test tests::test_queued_fetch_from_ann_same_rid ... ok
test tests::test_queued_fetch_max_capacity ... ok
test tests::test_queued_fetch_from_command_same_rid ... ok
test tests::test_redundant_connect ... ok
test tests::test_refs_announcement_fetch_trusted_no_inventory ... ok
test tests::test_refs_announcement_followed ... ok
test tests::prop_inventory_exchange_dense ... ok
test tests::test_refs_announcement_no_subscribe ... ok
test tests::test_refs_announcement_offline ... ok
test tests::test_refs_announcement_relay_private ... ok
test tests::test_announcement_message_amplification ... ok
test tests::test_refs_synced_event ... ok
test wire::test::test_inventory_ann_with_extension ... ok
test wire::test::test_pong_message_with_extension ... ok
test tests::test_refs_announcement_relay_public ... ok
test tests::test_seed_repo_subscribe ... ok
test tests::test_seeding ... ok
test result: ok. 79 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 14.89s
Running unittests src/main.rs (target/debug/deps/radicle_node-91030754a726535d)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_oid-32b41d88203b7116)
running 10 tests
test fmt::test::fixture ... ok
test fmt::test::zero ... ok
test fmt::test::git2 ... ok
test fmt::test::gix ... ok
test git2::test::zero ... ok
test gix::test::zero ... ok
test str::test::fixture ... ok
test str::test::zero ... ok
test str::test::git2_roundtrip ... ok
test str::test::gix_roundrip ... ok
test result: ok. 10 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_protocol-8abd9fe24a797cf4)
running 99 tests
test deserializer::test::test_decode_next ... ok
test deserializer::test::prop_decode_next ... ok
test fetcher::service::tests::test_fetch_coalescing_different_refs ... ok
test deserializer::test::test_unparsed ... ok
test fetcher::test::queue::properties::capacity::bounded ... ok
test fetcher::test::queue::properties::capacity::rejection ... ok
test fetcher::test::queue::properties::dequeue::empty_queue_returns_none ... ok
test fetcher::test::queue::properties::dequeue::enables_reenqueue ... ok
test fetcher::test::queue::properties::capacity::restored_after_dequeue ... ok
test fetcher::test::queue::properties::dequeue::drained_queue_returns_none ... ok
test fetcher::test::queue::properties::capacity::capacity_reached_returns_same_item ... ok
test fetcher::test::queue::properties::fifo::interleaved_operations ... ok
test fetcher::test::queue::properties::fifo::ordering ... ok
test fetcher::test::queue::properties::equality::reflexive ... ok
test fetcher::test::queue::properties::equality::symmetric ... ok
test fetcher::test::queue::properties::merge::different_rid_accepted ... ok
test fetcher::test::queue::properties::merge::combines_refs ... ok
test fetcher::test::queue::properties::merge::longer_timeout_preserved ... ok
test fetcher::test::queue::properties::equality::transitive ... ok
test fetcher::test::queue::properties::merge::does_not_increase_queue_length ... ok
test fetcher::test::queue::unit::capacity_takes_precedence_over_merge_for_new_items ... ok
test fetcher::test::queue::unit::empty_refs_items_can_be_equal ... ok
test fetcher::test::queue::unit::max_timeout_accepted ... ok
test fetcher::test::queue::unit::merge_preserves_position_in_queue ... ok
test fetcher::test::queue::unit::zero_timeout_accepted ... ok
test fetcher::test::queue::properties::merge::empty_refs_fetches_all ... ok
test fetcher::test::state::command::cancel::non_existent_returns_unexpected ... ok
test fetcher::test::state::command::cancel::ongoing_and_queued ... ok
test fetcher::test::state::command::cancel::single_ongoing ... ok
test fetcher::test::state::command::fetch::fetch_after_previous_completed ... ok
test fetcher::test::state::command::fetch::fetch_at_capacity_enqueues ... ok
test fetcher::test::state::command::fetch::fetch_different_repo_same_node_within_capacity ... ok
test fetcher::test::state::command::fetch::fetch_duplicate_returns_already_fetching ... ok
test fetcher::test::state::command::fetch::fetch_queue_merge_empty_refs_fetches_all ... ok
test fetcher::test::state::command::cancel::cancellation_is_isolated ... ok
test fetcher::test::state::command::fetch::fetch_queue_merge_takes_longer_timeout ... ok
test fetcher::test::state::command::fetch::fetch_queue_merges_already_queued ... ok
test fetcher::test::state::command::fetch::fetch_queue_rejected_capacity_reached ... ok
test fetcher::test::state::command::fetch::fetch_same_repo_different_nodes_queues_second ... ok
test fetcher::test::state::command::fetch::fetch_same_repo_different_refs_enqueues ... ok
test fetcher::test::state::command::fetch::fetch_start_first_fetch_for_node ... ok
test fetcher::test::queue::properties::merge::same_rid_merges_anywhere_in_queue ... ok
test fetcher::test::state::command::fetched::complete_single_ongoing ... ok
test fetcher::test::state::command::fetched::complete_then_dequeue_fifo ... ok
test fetcher::test::state::command::fetched::complete_one_of_multiple ... ok
test fetcher::test::state::command::fetched::non_existent_returns_not_found ... ok
test fetcher::test::state::concurrent::fetched_then_cancel ... ok
test fetcher::test::state::concurrent::interleaved_operations ... ok
test fetcher::test::state::config::min_queue_size ... ok
test fetcher::test::state::dequeue::empty_queue_returns_none ... ok
test fetcher::test::state::dequeue::cannot_dequeue_while_node_at_capacity ... ok
test fetcher::test::state::invariant::queue_integrity_after_merge ... ok
test fetcher::test::state::dequeue::maintains_fifo_order ... ok
test fetcher::test::queue::properties::merge::succeed_when_at_capacity ... ok
test service::filter::test::compatible ... ok
test service::filter::test::test_parameters ... ok
test fetcher::test::state::multinode::independent_queues ... ok
test service::filter::test::test_sizes ... ok
test service::limiter::test::test_limiter_different_rates ... ok
test service::limiter::test::test_limiter_multi ... ok
test service::limiter::test::test_limiter_refill ... ok
test service::gossip::store::test::test_announced ... ok
test service::message::tests::test_inventory_limit ... ok
test fetcher::test::state::config::high_concurrency ... ok
test service::message::tests::test_ref_remote_limit ... ok
test wire::frame::test::test_encode_git_large ... ok
test wire::frame::test::test_stream_id ... ok
test fetcher::test::state::multinode::high_count ... ok
test wire::message::tests::prop_roundtrip_address ... ok
test service::message::tests::prop_refs_announcement_signing ... ok
test wire::message::tests::prop_zero_bytes_encode_decode ... ok
test wire::message::tests::test_inv_ann_max_size ... ok
test wire::message::tests::test_node_ann_max_size ... ok
test wire::message::tests::test_ping_encode_size_overflow - should panic ... ok
test wire::message::tests::test_pingpong_encode_max_size ... ok
test wire::message::tests::test_pong_encode_size_overflow - should panic ... ok
test wire::message::tests::prop_roundtrip_message ... ok
test wire::tests::prop_oid ... ok
test wire::tests::prop_roundtrip_filter ... ok
test wire::tests::prop_roundtrip_publickey ... ok
test wire::tests::prop_roundtrip_refs ... ok
test wire::tests::prop_roundtrip_repoid ... ok
test wire::tests::prop_roundtrip_tuple ... ok
test wire::tests::prop_roundtrip_u16 ... ok
test wire::tests::prop_roundtrip_u32 ... ok
test wire::tests::prop_roundtrip_u64 ... ok
test wire::tests::prop_roundtrip_vec ... ok
test wire::tests::prop_signature ... ok
test wire::tests::prop_string ... ok
test wire::tests::test_alias ... ok
test wire::tests::test_bounded_vec_limit ... ok
test wire::tests::test_filter_invalid ... ok
test wire::tests::test_string ... ok
test wire::varint::test::prop_roundtrip_varint ... ok
test wire::varint::test::test_encode_overflow - should panic ... ok
test wire::varint::test::test_encoding ... ok
test wire::message::tests::test_refs_ann_max_size ... ok
test wire::message::tests::prop_message_decoder ... ok
test service::message::tests::test_node_announcement_validate ... ok
test result: ok. 99 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 1.61s
Running unittests src/main.rs (target/debug/deps/git_remote_rad-742be4fde4d7f3e8)
running 12 tests
test protocol::tests::test_capabilities ... ok
test protocol::tests::test_fetch ... ok
test protocol::tests::test_fetch_whitespace ... ok
test protocol::tests::test_invalid ... ok
test protocol::tests::test_empty ... ok
test protocol::tests::test_list_for_push ... ok
test protocol::tests::test_list ... ok
test protocol::tests::test_option ... ok
test protocol::tests::test_option_whitespace_preservation ... ok
test protocol::tests::test_push ... ok
test protocol::tests::test_push_delete ... ok
test protocol::tests::test_push_force ... ok
test result: ok. 12 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/main.rs (target/debug/deps/radicle_schemars-a1f957f3728f7d57)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_signals-edd60ee35e85748c)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_systemd-d0fb15a14ac3d4e9)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_term-51db4123a6bc434a)
running 21 tests
test ansi::tests::colors_disabled ... ok
test ansi::tests::colors_enabled ... ok
test cell::test::test_width ... ok
test ansi::tests::wrapping ... ok
test element::test::test_spaced ... ok
test element::test::test_width ... ok
test table::test::test_table ... ok
test element::test::test_truncate ... ok
test table::test::test_table_border_maximized ... ok
test table::test::test_table_border ... ok
test table::test::test_table_border_truncated ... ok
test table::test::test_table_truncate ... ok
test table::test::test_table_unicode ... ok
test table::test::test_truncate ... ok
test table::test::test_table_unicode_truncate ... ok
test textarea::test::test_wrapping ... ok
test vstack::test::test_vstack ... ok
test textarea::test::test_wrapping_code_block ... ok
test textarea::test::test_wrapping_fenced_block ... ok
test vstack::test::test_vstack_maximize ... ok
test textarea::test::test_wrapping_paragraphs ... ok
test result: ok. 21 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/lib.rs (target/debug/deps/radicle_windows-2c067555aa9e0165)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle
running 1 test
test crates/radicle/src/cob/patch/encoding/review.rs - cob::patch::encoding::review::Review (line 23) ... ignored
test result: ok. 0 passed; 0 failed; 1 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_cli
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_cli_test
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_cob
running 1 test
test crates/radicle-cob/src/backend/stable.rs - backend::stable::with_advanced_timestamp (line 56) ... ignored
test result: ok. 0 passed; 0 failed; 1 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_core
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_crypto
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_dag
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_fetch
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_git_metadata
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_git_ref_format
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_localtime
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_node
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_oid
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_protocol
running 6 tests
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::collect_from (line 30) ... ok
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::truncate (line 50) ... ok
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::max (line 96) ... ok
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::push (line 122) ... ok
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::unbound (line 149) ... ok
test crates/radicle-protocol/src/bounded.rs - bounded::BoundedVec<T,N>::with_capacity (line 66) ... ok
test result: ok. 6 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_signals
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_systemd
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_term
running 1 test
test crates/radicle-term/src/table.rs - table (line 4) ... ok
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests radicle_windows
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
error: 1 target failed:
`-p radicle-cli --test commands`
Exit code: 101
{
"response": "finished",
"result": "failure"
}