rad:z3gqcJUoA1n9HaHKufZs5FCSGazv5 heartwood406ea09e63659795098c646b2386eb1e28aac457
{
"request": "trigger",
"version": 1,
"event_type": "patch",
"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"
]
},
"action": "Updated",
"patch": {
"id": "80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e",
"author": {
"id": "did:key:z6MkrnXJWPndzPBxpBUaE3L3BnMeWpaQdT1V1FvkoCPFSFS3",
"alias": "yorgos-laptop"
},
"title": "[WIP] Explore relaxing synchronous sqlite flag",
"state": {
"status": "open",
"conflicts": []
},
"before": "4a5a51e6e550456c75d1a2182b0eafb9c3033ec0",
"after": "406ea09e63659795098c646b2386eb1e28aac457",
"commits": [
"406ea09e63659795098c646b2386eb1e28aac457",
"48582815c27aa5bc23f7a53f9b2ac93469c5ea83",
"1233a26842784b3ac7ce36db0904b97e55b07f89",
"a3dda38996d2ed34ca0ffaaaed4a72cbb94a5d5a",
"a65fb8b60be13e2d60fd3ad275d96a80305b79e6",
"7c6d4acad1ebc7e8d512f08834fc7156d97cc99a",
"876ac47281e865378e77c98820b80999076d110d",
"689dccccc90acedccacc3b9ab75174f10b0ce51c",
"4d965bb448c6d89974bd4b8c10450945a9d7d230",
"e5d7381e31ef2d4ac9b599468d7c51b9f05e8b51",
"a1aa3e0d4bc6cc339e13f822b9c3e2fba1218c3a",
"99c4aa6979d2ddfe57ecb7eeebc9513e4c547217",
"b52ea8487c9dae11c41ed09c2c71cab9688f16db",
"4fc1d57299f573b0165ebe38c644bbfe01626192"
],
"target": "b04f487b3ae2efcd23f52f8138abe32e73a718fd",
"labels": [],
"assignees": [],
"revisions": [
{
"id": "80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e",
"author": {
"id": "did:key:z6MkrnXJWPndzPBxpBUaE3L3BnMeWpaQdT1V1FvkoCPFSFS3",
"alias": "yorgos-laptop"
},
"description": "According to the docs [1]: \n\n> WAL mode is safe from corruption with synchronous=NORMAL, and probably DELETE mode is safe too on modern filesystems. WAL mode is always consistent with synchronous=NORMAL, but WAL mode does lose durability. A transaction committed in WAL mode with synchronous=NORMAL might roll back following a power loss or system crash. Transactions are durable across application crashes regardless of the synchronous setting or journal mode.\n\nAlso: \n\n> You lose durability across power lose with synchronous NORMAL in WAL mode, but that is not important for most applications. Transactions are still atomic, consistent, and isolated, which are the most important characteristics in most use cases.\n\nThe tradeoff here is that SQLite will not aggressively fsync() after every transaction, so we should see less disk pressure. \n\n[1] - https://sqlite.org/wal.html",
"base": "b04f487b3ae2efcd23f52f8138abe32e73a718fd",
"oid": "00485a49e4164046381fdade76878cb74ba5b715",
"timestamp": 1771604372
},
{
"id": "5c635af3c0e148e4c9fb9bcb3ca01405e9a273a9",
"author": {
"id": "did:key:z6MkrnXJWPndzPBxpBUaE3L3BnMeWpaQdT1V1FvkoCPFSFS3",
"alias": "yorgos-laptop"
},
"description": "Rebased on 1.6.1",
"base": "4a5a51e6e550456c75d1a2182b0eafb9c3033ec0",
"oid": "406ea09e63659795098c646b2386eb1e28aac457",
"timestamp": 1771604770
}
]
}
}
{
"response": "triggered",
"run_id": {
"id": "eaf55967-fde1-44f9-b51b-a0e811503bc7"
},
"info_url": "https://cci.rad.levitte.org//eaf55967-fde1-44f9-b51b-a0e811503bc7.html"
}
Started at: 2026-02-20 17:26:15.839016+01: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/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/
╭────────────────────────────────────╮
│ heartwood │
│ Radicle Heartwood Protocol & Stack │
│ 141 issues · 25 patches │
╰────────────────────────────────────╯
Run `cd ./.` to go to the repository directory.
Exit code: 0
$ rad patch checkout 80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e
✓ Switched to branch patch/80369b8 at revision 5c635af
✓ Branch patch/80369b8 setup to track rad/patches/80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e
Exit code: 0
$ git config advice.detachedHead false
Exit code: 0
$ git checkout 406ea09e63659795098c646b2386eb1e28aac457
HEAD is now at 406ea09e [WIP] Explore relaxing synchronous flag
Exit code: 0
$ rad patch show 80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e -p
╭──────────────────────────────────────────────────────────────────────────────────╮
│ Title [WIP] Explore relaxing synchronous sqlite flag │
│ Patch 80369b810dd2e4742d5bd4ff1e7e960a7bf1d13e │
│ Author yorgos-laptop z6MkrnX…CPFSFS3 │
│ Head 406ea09e63659795098c646b2386eb1e28aac457 │
│ Base 4a5a51e6e550456c75d1a2182b0eafb9c3033ec0 │
│ Branches patch/80369b8 │
│ Commits ahead 14, behind 156 │
│ Status open │
│ │
│ According to the docs [1]: │
│ │
│ > WAL mode is safe from corruption with synchronous=NORMAL, and probably DELETE │
│ mode is safe too on modern filesystems. WAL mode is always consistent with │
│ synchronous=NORMAL, but WAL mode does lose durability. A transaction committed │
│ in WAL mode with synchronous=NORMAL might roll back following a power loss or │
│ system crash. Transactions are durable across application crashes regardless of │
│ the synchronous setting or journal mode. │
│ │
│ Also: │
│ │
│ > You lose durability across power lose with synchronous NORMAL in WAL mode, but │
│ that is not important for most applications. Transactions are still atomic, │
│ consistent, and isolated, which are the most important characteristics in most │
│ use cases. │
│ │
│ The tradeoff here is that SQLite will not aggressively fsync() after every │
│ transaction, so we should see less disk pressure. │
│ │
│ [1] - https://sqlite.org/wal.html │
├──────────────────────────────────────────────────────────────────────────────────┤
│ 406ea09 [WIP] Explore relaxing synchronous flag │
│ 4858281 CHANGELOG: Release 1.6.1 │
│ 1233a26 node: downgrade `error!` logs │
│ a3dda38 node: downgrade `warn!` logs │
│ a65fb8b node/wire: manage logs for error establishing connection │
│ 7c6d4ac protocol: downgrade `error!` logs │
│ 876ac47 node: update comment │
│ 689dccc node/routing: remove error scenario from Database::prune │
│ 4d965bb protocol: downgrade `warn!` logs │
│ e5d7381 protocol/service: defensive storage.contains check │
│ a1aa3e0 fetch: remove `target: "fetch"` │
│ 99c4aa6 fetch: clean up logging │
│ b52ea84 fetch: surface underlying I/O error │
│ 4fc1d57 Release 1.6.0 CHANGELOG │
├──────────────────────────────────────────────────────────────────────────────────┤
│ ● Revision 80369b8 @ 00485a4 by yorgos-laptop z6MkrnX…CPFSFS3 6 minutes ago │
│ ↑ Revision 5c635af @ 406ea09 by yorgos-laptop z6MkrnX…CPFSFS3 7 seconds ago │
╰──────────────────────────────────────────────────────────────────────────────────╯
commit 406ea09e63659795098c646b2386eb1e28aac457
Author: Yorgos Saslis <yorgos.work@proton.me>
Date: Fri Feb 20 18:08:59 2026 +0200
[WIP] Explore relaxing synchronous flag
According to the docs [1]:
> WAL mode is safe from corruption with synchronous=NORMAL, and probably DELETE mode is safe too on modern filesystems. WAL mode is always consistent with synchronous=NORMAL, but WAL mode does lose durability. A transaction committed in WAL mode with synchronous=NORMAL might roll back following a power loss or system crash. Transactions are durable across application crashes regardless of the synchronous setting or journal mode.
Also:
> You lose durability across power lose with synchronous NORMAL in WAL mode, but that is not important for most applications. Transactions are still atomic, consistent, and isolated, which are the most important characteristics in most use cases.
The tradeoff here is that SQLite will not aggressively fsync() after every transaction, so we should see less disk pressure.
[1] - https://sqlite.org/wal.html
diff --git a/crates/radicle/src/node/db.rs b/crates/radicle/src/node/db.rs
index 423162525..ef45321ab 100644
--- a/crates/radicle/src/node/db.rs
+++ b/crates/radicle/src/node/db.rs
@@ -123,6 +123,7 @@ impl Database {
}
JournalMode::WriteAheadLog => {
self.db.execute("PRAGMA journal_mode = WAL;")?;
+ self.db.execute("PRAGMA synchronous = NORMAL;")?;
}
}
Ok(self)
commit 48582815c27aa5bc23f7a53f9b2ac93469c5ea83
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Tue Jan 20 15:42:14 2026 +0000
CHANGELOG: Release 1.6.1
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 8f3591ef2..1f4c6a074 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -15,6 +15,27 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## Deprecations
+## 1.6.1
+
+## Fixed Bugs
+
+### Improve Logging
+
+The introduction of new logs in `radicle-node` caused too many log lines to be
+output. All logs were evaluated and adjusted to a more suitable log level.
+
+### Improve `Service::fetch_missing_repositories`
+
+If the check for `storage.contains` failed with an error, the whole process of
+fetching missing repositories would fail. Instead, the error is logged and it
+continues to gather repositories to fetch.
+
+### Surface Underlying I/O Error for `radicle-fetch`
+
+When an I/O error would occur within the `gix-transport` crate, the underlying
+error would become opaque. This makes it hard to debug the issue when it occurs.
+Instead, surface the I/O error so that it can be inspected.
+
## 1.6.0
## Release Highlights
commit 1233a26842784b3ac7ce36db0904b97e55b07f89
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 15:24:56 2026 +0000
node: downgrade `error!` logs
Evaluate the `error!` logging in `radicle-node`.
The majority of cases, the `error!` is downgraded to `warn!`. These
cases are generally useful to know if there's an issue, but the
operator cannon necessarily do anything about it.
In a few cases, `debug!` was chosen. These are generally when an error
means that a result is not part of an accumulation, or the system will
eventually correct itself during another event or restart.
diff --git a/crates/radicle-node/src/control.rs b/crates/radicle-node/src/control.rs
index 0be586b25..434ff6c91 100644
--- a/crates/radicle-node/src/control.rs
+++ b/crates/radicle-node/src/control.rs
@@ -50,7 +50,7 @@ where
thread::spawn(&nid, "control", move || {
if let Err((e, mut stream)) = command(stream, handle) {
- log::error!(target: "control", "Command returned error: {e}");
+ log::debug!(target: "control", "Command returned error: {e}");
CommandResult::error(e).to_writer(&mut stream).ok();
@@ -59,7 +59,7 @@ where
}
});
}
- Err(e) => log::error!(target: "control", "Failed to accept incoming connection: {e}"),
+ Err(e) => log::warn!(target: "control", "Failed to accept incoming connection: {e}"),
}
}
log::debug!(target: "control", "Exiting control loop..");
diff --git a/crates/radicle-node/src/reactor.rs b/crates/radicle-node/src/reactor.rs
index 84a2bfd9e..a9d099833 100644
--- a/crates/radicle-node/src/reactor.rs
+++ b/crates/radicle-node/src/reactor.rs
@@ -399,7 +399,7 @@ impl<H: ReactionHandler> Runtime<H> {
}
if let Err(err) = res {
- log::error!(target: "reactor", "Error during polling: {err}");
+ log::warn!(target: "reactor", "Failure during polling: {err}");
self.service.handle_error(Error::Poll(err));
}
@@ -497,7 +497,7 @@ impl<H: ReactionHandler> Runtime<H> {
// Deadlock may happen here if the service will generate events over and over
// in the handle_* calls we may never get out of this loop
if let Err(err) = self.handle_action(action, instant) {
- log::error!(target: "reactor", "Error: {err}");
+ log::warn!(target: "reactor", "Failure: {err}");
self.service.handle_error(err);
}
}
@@ -558,7 +558,7 @@ impl<H: ReactionHandler> Runtime<H> {
}
}
} else {
- log::error!(target: "reactor", token=token.0; "No transport with token {token:?} is known!");
+ log::debug!(target: "reactor", token=token.0; "No transport with token {token:?} is known!");
}
}
Action::SetTimer(duration) => {
diff --git a/crates/radicle-node/src/reactor/transport.rs b/crates/radicle-node/src/reactor/transport.rs
index ef6e141a2..8f0a1ea5f 100644
--- a/crates/radicle-node/src/reactor/transport.rs
+++ b/crates/radicle-node/src/reactor/transport.rs
@@ -265,7 +265,7 @@ impl<S: Session> Transport<S> {
Ok(0)
},
_ => {
- log::error!(target: "transport", "Resource {} failed write operation with message '{err}'", self.display());
+ log::warn!(target: "transport", "Resource {} failed write operation with message '{err}'", self.display());
Err(err)
},
}
diff --git a/crates/radicle-node/src/wire.rs b/crates/radicle-node/src/wire.rs
index b950b13db..e5f46b043 100644
--- a/crates/radicle-node/src/wire.rs
+++ b/crates/radicle-node/src/wire.rs
@@ -361,7 +361,7 @@ where
}
Entry::Occupied(mut e) => match e.get_mut() {
Peer::Disconnecting { nid, link, .. } => {
- log::error!(target: "wire", token=token.0; "Peer is already disconnecting");
+ log::debug!(target: "wire", token=token.0; "Peer is already disconnecting");
nid.map(|n| (n, *link))
}
@@ -527,7 +527,7 @@ where
Ok((connection, peer)) => {
let remote = NetAddr::from(peer);
let InetHost::Ip(ip) = remote.host else {
- log::error!(target: "wire", "Unexpected host type for inbound connection {remote}; dropping..");
+ log::debug!(target: "wire", "Unexpected host type for inbound connection {remote}; dropping..");
drop(connection);
return;
@@ -551,7 +551,7 @@ where
let transport = match Transport::with_session(session, Link::Inbound) {
Ok(transport) => transport,
Err(err) => {
- log::error!(target: "wire", "Failed to create transport for accepted connection: {err}");
+ log::warn!(target: "wire", "Failed to create transport for accepted connection: {err}");
return;
}
};
@@ -592,7 +592,7 @@ where
let nid: NodeId = state.remote_static_key.unwrap();
// Make sure we don't try to connect to ourselves by mistake.
if &nid == self.signer.public_key() {
- log::error!(target: "wire", "Self-connection detected, disconnecting..");
+ log::warn!(target: "wire", "Self-connection detected, disconnecting..");
self.disconnect(token, DisconnectReason::SelfConnection);
return;
@@ -606,7 +606,7 @@ where
assert_eq!(nid, peer.nid);
(peer.addr, Link::Outbound)
} else {
- log::error!(target: "wire", token=token.0; "Session for {nid} not found");
+ log::debug!(target: "wire", token=token.0; "Session for {nid} not found");
return;
};
log::debug!(
@@ -701,8 +701,8 @@ where
metrics.received_bytes += data.len();
if inbox.input(&data).is_err() {
- log::error!(target: "wire", "Maximum inbox size ({MAX_INBOX_SIZE}) reached for peer {nid}");
- log::error!(target: "wire", "Unable to process messages fast enough for peer {nid}; disconnecting..");
+ log::warn!(target: "wire", "Maximum inbox size ({MAX_INBOX_SIZE}) reached for peer {nid}");
+ log::warn!(target: "wire", "Unable to process messages fast enough for peer {nid}; disconnecting..");
self.disconnect(
token,
DisconnectReason::Session(session::Error::Misbehavior),
@@ -739,7 +739,7 @@ where
channels,
};
if let Err(e) = self.worker.try_send(task) {
- log::error!(
+ log::warn!(
target: "wire",
"Worker pool failed to accept incoming fetch request: {e}"
);
@@ -753,7 +753,7 @@ where
log::debug!(target: "wire", "Received `end-of-file` on stream {stream} from {nid}");
if s.channels.send(ChannelEvent::Eof).is_err() {
- log::error!(target: "wire", "Worker is disconnected; cannot send `EOF`");
+ log::debug!(target: "wire", "Worker is disconnected; cannot send `EOF`");
}
} else {
log::debug!(target: "wire", "Ignoring frame on closed or unknown stream {stream}");
@@ -790,7 +790,7 @@ where
metrics.received_git_bytes += data.len();
if s.channels.send(ChannelEvent::Data(data)).is_err() {
- log::error!(target: "wire", "Worker is disconnected; cannot send data");
+ log::warn!(target: "wire", "Worker is disconnected; cannot send data");
}
} else {
log::debug!(target: "wire", "Ignoring frame on closed or unknown stream {stream}");
@@ -801,7 +801,7 @@ where
break;
}
Err(e) => {
- log::error!(target: "wire", "Invalid gossip message from {nid}: {e}");
+ log::warn!(target: "wire", "Invalid gossip message from {nid}: {e}");
if !inbox.is_empty() {
log::debug!(target: "wire", "Dropping read buffer for {nid} with {} bytes", inbox.len());
@@ -874,7 +874,7 @@ where
}
fn handover_listener(&mut self, token: Token, _listener: Self::Listener) {
- log::error!(target: "wire", token=token.0; "Listener handover is not supported");
+ log::warn!(target: "wire", token=token.0; "Listener handover is not supported");
}
fn handover_transport(&mut self, token: Token, transport: Self::Transport) {
@@ -932,7 +932,7 @@ where
continue;
}
None => {
- log::error!(target: "wire", "Dropping {} message(s) to {node_id}: unknown peer", msgs.len());
+ log::debug!(target: "wire", "Dropping {} message(s) to {node_id}: unknown peer", msgs.len());
continue;
}
};
@@ -952,7 +952,7 @@ where
}
Io::Connect(node_id, addr) => {
if self.peers.connected().any(|(_, id)| id == &node_id) {
- log::error!(
+ log::debug!(
target: "wire",
"Attempt to connect to already connected peer {node_id}"
);
@@ -1028,7 +1028,7 @@ where
// is in the service's i/o buffer. Since the service may not purge the
// buffer on disconnect, we should just ignore i/o actions that don't
// have a connected peer.
- log::error!(target: "wire", "Peer {remote} is not connected: dropping fetch");
+ log::debug!(target: "wire", "Peer {remote} is not connected: dropping fetch");
continue;
};
let (stream, channels) =
@@ -1054,7 +1054,7 @@ where
);
}
if let Err(e) = self.worker.try_send(task) {
- log::error!(
+ log::warn!(
target: "wire",
"Worker pool failed to accept outgoing fetch request: {e}"
);
@@ -1231,7 +1231,7 @@ mod logger {
| NotConnected => {
log::info!(target: "wire", "Could not establish connection to {addr}: {err}")
}
- _ => log::error!(target: "wire", "Error establishing connection to {addr}: {err}"),
+ _ => log::warn!(target: "wire", "Failed to establish connection to {addr}: {err}"),
}
}
}
diff --git a/crates/radicle-node/src/worker/fetch.rs b/crates/radicle-node/src/worker/fetch.rs
index 44a803381..f3fb13ed3 100644
--- a/crates/radicle-node/src/worker/fetch.rs
+++ b/crates/radicle-node/src/worker/fetch.rs
@@ -106,7 +106,7 @@ impl Handle {
validations,
} => {
for fail in validations.iter() {
- log::error!(target: "worker", "Validation error: {fail}");
+ log::warn!(target: "worker", "Validation error: {fail}");
}
Err(error::Fetch::Validation {
threshold,
@@ -207,7 +207,7 @@ fn notify(
if let RefUpdate::Skipped { .. } = update {
// Don't notify about skipped refs.
} else if let Err(e) = store.insert(rid, update, now) {
- log::error!(
+ log::debug!(
target: "worker",
"Failed to update notification store for {rid}: {e}"
);
@@ -227,7 +227,7 @@ where
let name = r.name();
let (namespace, qualified) = match radicle::git::parse_ref_namespaced(name) {
Err(e) => {
- log::error!(target: "worker", "Git reference is invalid: {name:?}: {e}");
+ log::debug!(target: "worker", "Git reference is invalid: {name:?}: {e}");
log::debug!(target: "worker", "Skipping refs caching for fetch of {repo}");
break;
}
@@ -247,7 +247,7 @@ where
};
if let Err(e) = result {
- log::error!(target: "worker", "Error updating git refs cache for {name:?}: {e}");
+ log::debug!(target: "worker", "Failed to update git refs cache for {name:?}: {e}");
log::debug!(target: "worker", "Skipping refs caching for fetch of {repo}");
break;
}
@@ -326,7 +326,7 @@ where
}
Err(e) => {
// Object was found, but failed to load. Fall-through.
- log::error!(target: "fetch", "Error loading COB {tid} from storage: {e}");
+ log::debug!(target: "fetch", "Failed to load COB {tid} from storage: {e}");
}
}
// The object has either been removed entirely from the repository,
diff --git a/crates/radicle-node/src/worker/upload_pack.rs b/crates/radicle-node/src/worker/upload_pack.rs
index 50de14ae6..e84206ba4 100644
--- a/crates/radicle-node/src/worker/upload_pack.rs
+++ b/crates/radicle-node/src/worker/upload_pack.rs
@@ -137,7 +137,7 @@ where
}
}
Err(e) => {
- log::error!(target: "worker", "Error on upload-pack channel read for {}: {e}", header.repo);
+ log::debug!(target: "worker", "Failure on upload-pack channel read for {}: {e}", header.repo);
emitter.emit(events::UploadPack::error(header.repo, remote, e).into());
break;
}
commit a3dda38996d2ed34ca0ffaaaed4a72cbb94a5d5a
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 13:46:58 2026 +0000
node: downgrade `warn!` logs
Evaluate the `warn!` logging in `radicle-node`.
If the event is a result of something being suspicious in the running
of the protocol, then it remains a `warn!`.
However, if the system ends up ignoring the event and can continue
without issues, the log level is reduced to `debug!`.
diff --git a/crates/radicle-node/src/main.rs b/crates/radicle-node/src/main.rs
index 9fcd4cc8e..46036647e 100644
--- a/crates/radicle-node/src/main.rs
+++ b/crates/radicle-node/src/main.rs
@@ -327,7 +327,7 @@ fn execute(options: Options) -> Result<(), ExecutionError> {
#[cfg(windows)]
let signals = {
let (_, signals) = chan::bounded(1);
- log::warn!(target: "node", "Signal handlers not installed.");
+ log::info!(target: "node", "No signal handlers were installed, because this is not available on Windows.");
signals
};
diff --git a/crates/radicle-node/src/reactor.rs b/crates/radicle-node/src/reactor.rs
index 467eded31..84a2bfd9e 100644
--- a/crates/radicle-node/src/reactor.rs
+++ b/crates/radicle-node/src/reactor.rs
@@ -483,7 +483,7 @@ impl<H: ReactionHandler> Runtime<H> {
deregistered.push(token);
}
} else if !deregistered.contains(&token) {
- log::warn!(target: "reactor", token=token.0; "Event from unknown token {}: {:?}", token.0, event);
+ log::debug!(target: "reactor", token=token.0; "Event from unknown token {}: {:?}", token.0, event);
}
}
@@ -576,12 +576,12 @@ impl<H: ReactionHandler> Runtime<H> {
fn deregister_listener(&mut self, token: Token) -> Option<H::Listener> {
let Some(mut source) = self.listeners.remove(&token) else {
- log::warn!(target: "reactor", token=token.0; "Deregistering non-registered listener with token {}", token.0);
+ log::debug!(target: "reactor", token=token.0; "Deregistering non-registered listener with token {}", token.0);
return None;
};
if let Err(err) = self.poll.registry().deregister(&mut source) {
- log::warn!(target: "reactor", token=token.0; "Failed to deregister listener with token {} from mio: {err}", token.0);
+ log::debug!(target: "reactor", token=token.0; "Failed to deregister listener with token {} from mio: {err}", token.0);
}
Some(source)
@@ -589,12 +589,12 @@ impl<H: ReactionHandler> Runtime<H> {
fn deregister_transport(&mut self, token: Token) -> Option<H::Transport> {
let Some(mut source) = self.transports.remove(&token) else {
- log::warn!(target: "reactor", token=token.0; "Deregistering non-registered transport with token {}", token.0);
+ log::debug!(target: "reactor", token=token.0; "Deregistering non-registered transport with token {}", token.0);
return None;
};
if let Err(err) = self.poll.registry().deregister(&mut source) {
- log::warn!(target: "reactor", token=token.0; "Failed to deregister transport with token {} from mio: {err}", token.0);
+ log::debug!(target: "reactor", token=token.0; "Failed to deregister transport with token {} from mio: {err}", token.0);
}
Some(source)
diff --git a/crates/radicle-node/src/reactor/transport.rs b/crates/radicle-node/src/reactor/transport.rs
index 6ace822c1..ef6e141a2 100644
--- a/crates/radicle-node/src/reactor/transport.rs
+++ b/crates/radicle-node/src/reactor/transport.rs
@@ -148,7 +148,7 @@ impl<S: Session> Transport<S> {
fn handle_io(&mut self, interest: Interest) -> Option<SessionEvent<S>> {
if self.state == TransportState::Terminated {
- log::warn!(target: "transport", "Transport {self} is terminated, ignoring I/O event");
+ log::debug!(target: "transport", "Transport {self} is terminated, ignoring I/O event");
return None;
}
@@ -217,7 +217,7 @@ impl<S: Session> Transport<S> {
| io::ErrorKind::Interrupted
) =>
{
- log::warn!(target: "transport", "Resource {} was not able to consume any data even though it has announced its write readiness", self.display());
+ log::debug!(target: "transport", "Resource {} was not able to consume any data even though it has announced its write readiness", self.display());
self.write_intent = true;
None
}
@@ -241,7 +241,7 @@ impl<S: Session> Transport<S> {
// when there's data on the socket. We leave it here in case external
// conditions change.
- log::warn!(target: "transport",
+ log::trace!(target: "transport",
"WOULD_BLOCK on resource which had read intent - probably normal thing to happen"
);
None
@@ -261,7 +261,7 @@ impl<S: Session> Transport<S> {
| io::ErrorKind::OutOfMemory
| io::ErrorKind::WriteZero
| io::ErrorKind::Interrupted => {
- log::warn!(target: "transport", "Resource {} kernel buffer is full (system message is '{err}')", self.display());
+ log::trace!(target: "transport", "Resource {} kernel buffer is full (system message is '{err}')", self.display());
Ok(0)
},
_ => {
diff --git a/crates/radicle-node/src/wire.rs b/crates/radicle-node/src/wire.rs
index fc03a85b0..b950b13db 100644
--- a/crates/radicle-node/src/wire.rs
+++ b/crates/radicle-node/src/wire.rs
@@ -247,7 +247,7 @@ impl Peers {
fn insert(&mut self, token: Token, peer: Peer) {
if self.0.insert(token, peer).is_some() {
- log::warn!(target: "wire", token=token.0; "Replacing existing peer");
+ log::debug!(target: "wire", token=token.0; "Replacing existing peer");
}
}
@@ -395,7 +395,7 @@ where
let nid = task.remote;
let Some((fd, peer)) = self.peers.lookup_mut(&nid) else {
- log::warn!(target: "wire", "Peer {nid} not found; ignoring fetch result");
+ log::debug!(target: "wire", "Peer {nid} not found; ignoring fetch result");
return;
};
@@ -420,7 +420,7 @@ where
} else {
// If the peer disconnected, we'll get here, but we still want to let the service know
// about the fetch result, so we don't return here.
- log::warn!(target: "wire", "Peer {nid} is not connected; ignoring fetch result");
+ log::debug!(target: "wire", "Peer {nid} is not connected; ignoring fetch result");
return;
};
@@ -443,11 +443,11 @@ where
fn flush(&mut self, remote: NodeId, stream: StreamId) {
let Some((fd, peer)) = self.peers.lookup_mut(&remote) else {
- log::warn!(target: "wire", "Peer {remote} is not known; ignoring flush");
+ log::debug!(target: "wire", "Peer {remote} is not known; ignoring flush");
return;
};
let Peer::Connected { streams, link, .. } = peer else {
- log::warn!(target: "wire", "Peer {remote} is not connected; ignoring flush");
+ log::debug!(target: "wire", "Peer {remote} is not connected; ignoring flush");
return;
};
let Some(s) = streams.get_mut(&stream) else {
@@ -581,7 +581,7 @@ where
} else if self.inbound.contains(&token) {
log::debug!(target: "wire", token=token.0; "Inbound peer resource registered");
} else {
- log::warn!(target: "wire", token=token.0; "Unknown peer registered");
+ log::debug!(target: "wire", token=token.0; "Unknown peer registered");
}
}
@@ -664,14 +664,14 @@ where
(Outbound, Outbound, _) => token.max(c_token),
};
- log::warn!(
+ log::trace!(
target: "wire", "Established session with token {} conflicts with existing session with token {} for {nid}. Disconnecting session with token {}.", token.0, c_token.0, close.0
);
disconnect.push(close);
}
}
for id in &disconnect {
- log::warn!(
+ log::info!(
target: "wire", token=token.0; "Closing conflicting session with {nid}.."
);
// Disconnect and return the associated NID of the peer, if available.
@@ -726,7 +726,7 @@ where
ChannelsConfig::new(FETCH_TIMEOUT)
.with_reader_limit(reader_limit),
) else {
- log::warn!(target: "wire", "Peer attempted to open already-open stream stream {stream}");
+ log::debug!(target: "wire", "Peer attempted to open already-open stream stream {stream}");
continue;
};
@@ -815,7 +815,7 @@ where
}
}
} else {
- log::warn!(target: "wire", token=token.0; "Dropping message from unconnected peer");
+ log::debug!(target: "wire", token=token.0; "Dropping message from unconnected peer");
}
}
SessionEvent::Terminated(err) => {
@@ -1006,7 +1006,7 @@ where
self.metrics.peer(nid).disconnects += 1;
}
} else {
- log::warn!(target: "wire", "Peer {nid} is not connected: ignoring disconnect");
+ log::debug!(target: "wire", "Peer {nid} is not connected: ignoring disconnect");
}
}
Io::Wakeup(d) => {
diff --git a/crates/radicle-node/src/worker.rs b/crates/radicle-node/src/worker.rs
index 3aae8166a..0449b526f 100644
--- a/crates/radicle-node/src/worker.rs
+++ b/crates/radicle-node/src/worker.rs
@@ -240,7 +240,7 @@ impl Worker {
// N.b. ensure that `git gc` works in debug mode.
debug_assert!(false, "`git gc` failed: {e}");
- log::warn!(target: "worker", "Failed to run `git gc`: {e}");
+ log::debug!(target: "worker", "Failed to run `git gc`: {e}");
}
Ok(result)
}
diff --git a/crates/radicle-node/src/worker/fetch.rs b/crates/radicle-node/src/worker/fetch.rs
index da4256a35..44a803381 100644
--- a/crates/radicle-node/src/worker/fetch.rs
+++ b/crates/radicle-node/src/worker/fetch.rs
@@ -96,7 +96,7 @@ impl Handle {
};
for rejected in result.rejected() {
- log::warn!(target: "worker", "Rejected update for {}", rejected.refname())
+ log::debug!(target: "worker", "Rejected update for {}", rejected.refname())
}
match result {
@@ -119,7 +119,7 @@ impl Handle {
validations,
} => {
for warn in validations {
- log::warn!(target: "worker", "Validation error: {warn}");
+ log::debug!(target: "worker", "Validation error: {warn}");
}
// N.b. We do not go through handle for this since the cloning handle
@@ -133,7 +133,7 @@ impl Handle {
}
}
Err(RepositoryError::Quorum(e)) => {
- log::warn!(target: "worker", "Fetch could not set HEAD: {e}")
+ log::warn!(target: "worker", "Fetch could not set HEAD for {rid}: {e}")
}
Err(e) => return Err(e.into()),
}
@@ -141,7 +141,7 @@ impl Handle {
let canonical = match set_canonical_refs(&repo, &applied) {
Ok(updates) => updates.unwrap_or_default(),
Err(e) => {
- log::warn!(target: "worker", "Failed to set canonical references: {e}");
+ log::warn!(target: "worker", "Failed to set canonical references for {rid}: {e}");
UpdatedCanonicalRefs::default()
}
};
@@ -228,7 +228,7 @@ where
let (namespace, qualified) = match radicle::git::parse_ref_namespaced(name) {
Err(e) => {
log::error!(target: "worker", "Git reference is invalid: {name:?}: {e}");
- log::warn!(target: "worker", "Skipping refs caching for fetch of {repo}");
+ log::debug!(target: "worker", "Skipping refs caching for fetch of {repo}");
break;
}
Ok((n, q)) => (n, q),
@@ -248,7 +248,7 @@ where
if let Err(e) = result {
log::error!(target: "worker", "Error updating git refs cache for {name:?}: {e}");
- log::warn!(target: "worker", "Skipping refs caching for fetch of {repo}");
+ log::debug!(target: "worker", "Skipping refs caching for fetch of {repo}");
break;
}
}
@@ -380,7 +380,7 @@ fn set_canonical_refs(
let canonical = match canonical.find_objects() {
Err(err) => {
- log::warn!(target: "worker", "Failed to find objects for canonical computation: {err}");
+ log::warn!(target: "worker", "Failed to find objects for canonical computation of `{name}`: {err}");
continue;
}
Ok(canonical) => canonical,
@@ -390,7 +390,7 @@ fn set_canonical_refs(
Err(err) => {
log::warn!(
target: "worker",
- "Failed to calculate canonical reference: {err}",
+ "Failed to calculate canonical reference `{name}`: {err}",
);
continue;
}
diff --git a/crates/radicle-node/src/worker/upload_pack.rs b/crates/radicle-node/src/worker/upload_pack.rs
index 9fb867012..50de14ae6 100644
--- a/crates/radicle-node/src/worker/upload_pack.rs
+++ b/crates/radicle-node/src/worker/upload_pack.rs
@@ -97,7 +97,7 @@ where
Ok(n) => {
let mut lock = reporter.lock().expect("FATAL: upload_pack poisoned lock");
if let Err(e) = lock.write_all(&buffer[..n]) {
- log::warn!(target: "worker", "Error reading stdout to upload-pack reporter: {e}");
+ log::debug!(target: "worker", "Failed to write buffer to upload-pack reporter: {e}");
emitter.emit(events::UploadPack::error(header.repo, remote, e).into());
break;
}
@@ -119,7 +119,7 @@ where
Ok(0) => break,
Ok(n) => {
if let Err(e) = stdin.write_all(&buffer[..n]) {
- log::warn!(target: "worker", "Error writing to upload-pack stdin: {e}");
+ log::debug!(target: "worker", "Failed to write to upload-pack stdin: {e}");
break;
}
}
@@ -128,7 +128,7 @@ where
break;
}
Err(e) if e.kind() == io::ErrorKind::TimedOut => {
- log::warn!(target: "worker", "Read channel timed out for upload-pack {}", header.repo);
+ log::debug!(target: "worker", "Read channel timed out for upload-pack {}", header.repo);
// N.b. if the read timed out, ensure that the sender isn't
// still sending messages.
let lock = reporter.lock().expect("FATAL: upload_pack poisoned lock");
commit a65fb8b60be13e2d60fd3ad275d96a80305b79e6
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Wed Dec 17 13:23:27 2025 +0000
node/wire: manage logs for error establishing connection
Instead of logging all IO errors as an error, match on the kind of the
error, logging an info message, otherwise an error.
These errors are informational, and cannot necessarily be resolved by
the node operator – since it is a matter of not being able to connect
to another end.
diff --git a/crates/radicle-node/src/wire.rs b/crates/radicle-node/src/wire.rs
index 6736bc334..fc03a85b0 100644
--- a/crates/radicle-node/src/wire.rs
+++ b/crates/radicle-node/src/wire.rs
@@ -990,7 +990,7 @@ where
.push_back(reactor::Action::RegisterTransport(token, transport));
}
Err(err) => {
- log::error!(target: "wire", "Error establishing connection to {addr}: {err}");
+ logger::establish_connection(&addr, &err);
self.service.disconnected(
node_id,
@@ -1221,6 +1221,21 @@ fn session<G: Ecdh<Pk = NodeId>>(
WireSession::new(proxy, noise)
}
+mod logger {
+ use radicle::node::Address;
+
+ pub fn establish_connection(addr: &Address, err: &std::io::Error) {
+ use std::io::ErrorKind::*;
+ match err.kind() {
+ ConnectionRefused | ConnectionReset | HostUnreachable | ConnectionAborted
+ | NotConnected => {
+ log::info!(target: "wire", "Could not establish connection to {addr}: {err}")
+ }
+ _ => log::error!(target: "wire", "Error establishing connection to {addr}: {err}"),
+ }
+ }
+}
+
#[cfg(test)]
mod test {
use super::*;
commit 7c6d4acad1ebc7e8d512f08834fc7156d97cc99a
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 12:18:27 2026 +0000
protocol: downgrade `error!` logs
Evaluate the `error!` logging in `radicle-protocol`.
The majority of cases, the `error!` is downgraded to `warn!`. These
cases are generally useful to know if there's an issue, but the
operator cannon necessarily do anything about it.
In a few cases, `debug!` was chosen. These are generally when an error
means that a result is not part of an accumulation, or the system will
eventually correct itself during another event or restart.
If the previous message started with "Error", then the wording is
changed to say "Failed to".
diff --git a/crates/radicle-protocol/src/service.rs b/crates/radicle-protocol/src/service.rs
index 053d47968..42fc27d2c 100644
--- a/crates/radicle-protocol/src/service.rs
+++ b/crates/radicle-protocol/src/service.rs
@@ -572,7 +572,7 @@ where
self.filter = Filter::allowed_by(self.policies.seed_policies()?);
// Update and announce new inventory.
if let Err(e) = self.remove_inventory(id) {
- error!(target: "service", "Error updating inventory after unseed: {e}");
+ warn!(target: "service", "Failed to update inventory after unseed: {e}");
}
}
Ok(updated)
@@ -660,7 +660,7 @@ where
Ok(Some(last)) => Some(last.to_local_time()),
Ok(None) => None,
Err(e) => {
- error!(target: "service", "Error getting the latest gossip message from db: {e}");
+ warn!(target: "service", "Failed to get the latest gossip message from db: {e}");
None
}
};
@@ -671,11 +671,13 @@ where
Ok(0) => {
info!(target: "service", "Empty refs database, populating from storage..");
if let Err(e) = self.db.refs_mut().populate(&self.storage) {
- error!(target: "service", "Failed to populate refs database: {e}");
+ warn!(target: "service", "Failed to populate refs database: {e}");
}
}
Ok(n) => debug!(target: "service", "Refs database has {n} cached references"),
- Err(e) => error!(target: "service", "Error checking refs database: {e}"),
+ Err(e) => {
+ warn!(target: "service", "Failed to retrieve count of refs from database: {e}")
+ }
}
let announced = self
@@ -748,7 +750,7 @@ where
let addrs = self.config.connect.clone();
for (id, addr) in addrs.into_iter().map(|ca| ca.into()) {
if let Err(e) = self.connect(id, addr) {
- error!(target: "service", "Service::initialization connection error: {e}");
+ debug!(target: "service", "Service::initialization connection error: {e}");
}
}
// Try to establish some connections.
@@ -804,7 +806,7 @@ where
trace!(target: "service", "Running 'gossip' task...");
if let Err(e) = self.relay_announcements() {
- error!(target: "service", "Error relaying stored announcements: {e}");
+ warn!(target: "service", "Failed to relay stored announcements: {e}");
}
self.outbox.wakeup(GOSSIP_INTERVAL);
self.last_gossip = now;
@@ -813,7 +815,7 @@ where
trace!(target: "service", "Running 'sync' task...");
if let Err(e) = self.fetch_missing_repositories() {
- error!(target: "service", "Error fetching missing inventory: {e}");
+ warn!(target: "service", "Failed to fetch missing inventory: {e}");
}
self.outbox.wakeup(SYNC_INTERVAL);
self.last_sync = now;
@@ -829,14 +831,14 @@ where
trace!(target: "service", "Running 'prune' task...");
if let Err(err) = self.prune_routing_entries(&now) {
- error!(target: "service", "Error pruning routing entries: {err}");
+ warn!(target: "service", "Failed to prune routing entries: {err}");
}
if let Err(err) = self
.db
.gossip_mut()
.prune((now - LocalDuration::from(self.config.limits.gossip_max_age)).into())
{
- error!(target: "service", "Error pruning gossip entries: {err}");
+ warn!(target: "service", "Failed to prune gossip entries: {err}");
}
self.outbox.wakeup(PRUNE_INTERVAL);
@@ -890,7 +892,7 @@ where
resp.send(seeds).ok();
}
Err(e) => {
- error!(target: "service", "Error getting seeds for {rid}: {e}");
+ warn!(target: "service", "Failed to get seeds for {rid}: {e}");
}
},
Command::Fetch(rid, seed, timeout, resp) => {
@@ -933,11 +935,11 @@ where
let doc = match self.storage.get(id) {
Ok(Some(doc)) => doc,
Ok(None) => {
- error!(target: "service", "Error announcing refs: repository {id} not found");
+ warn!(target: "service", "Failed to announce refs: repository {id} not found");
return;
}
Err(e) => {
- error!(target: "service", "Error announcing refs: doc error: {e}");
+ warn!(target: "service", "Failed to announce refs: doc error: {e}");
return;
}
};
@@ -949,7 +951,7 @@ where
}
}
Err(err) => {
- error!(target: "service", "Error announcing refs: {err}");
+ warn!(target: "service", "Failed to announce refs: {err}");
}
}
}
@@ -961,7 +963,7 @@ where
resp.send(updated).ok();
}
Err(e) => {
- error!(target: "service", "Error adding {rid} to inventory: {e}");
+ warn!(target: "service", "Failed to add {rid} to inventory: {e}");
}
},
Command::QueryState(query, sender) => {
@@ -990,7 +992,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Error getting the refs status of {rid}: {e}");
+ warn!(target: "service", "Failed to get the refs status of {rid}: {e}");
}
}
// We didn't try to fetch anything.
@@ -1070,7 +1072,7 @@ where
fn queue_fetch(&mut self, fetch: QueuedFetch) {
let Some(s) = self.sessions.get_mut(&fetch.from) else {
- log::error!(target: "service", "Cannot queue fetch for unknown session {}", fetch.from);
+ log::debug!(target: "service", "Cannot queue fetch for unknown session {}", fetch.from);
return;
};
if let Err(e) = s.queue_fetch(fetch) {
@@ -1139,7 +1141,7 @@ where
result: Result<crate::worker::fetch::FetchResult, crate::worker::FetchError>,
) {
let Some(fetching) = self.fetching.remove(&rid) else {
- error!(target: "service", "Received unexpected fetch result for {rid}, from {remote}");
+ debug!(target: "service", "Received unexpected fetch result for {rid}, from {remote}");
return;
};
debug_assert_eq!(fetching.from, remote);
@@ -1165,7 +1167,7 @@ where
},
};
if sub.send(result).is_err() {
- error!(target: "service", "Error sending fetch result for {rid} from {remote}..");
+ debug!(target: "service", "Failed to send fetch result for {rid} from {remote}..");
} else {
debug!(target: "service", "Sent fetch result for {rid} from {remote}..");
}
@@ -1213,7 +1215,7 @@ where
debug!(target: "service", "Updating and announcing inventory for cloned repository {rid}..");
if let Err(e) = self.add_inventory(rid) {
- error!(target: "service", "Error announcing inventory for {rid}: {e}");
+ warn!(target: "service", "Failed to announce inventory for {rid}: {e}");
}
}
@@ -1224,12 +1226,12 @@ where
// Finally, announce the refs. This is useful for nodes to know what we've synced,
// beyond just knowing that we have added an item to our inventory.
if let Err(e) = self.announce_refs(rid, doc.into(), namespaces, false) {
- error!(target: "service", "Failed to announce new refs: {e}");
+ warn!(target: "service", "Failed to announce new refs: {e}");
}
}
}
Err(err) => {
- error!(target: "service", "Fetch failed for {rid} from {remote}: {err}");
+ warn!(target: "service", "Fetch failed for {rid} from {remote}: {err}");
// For now, we only disconnect the remote in case of timeout. In the future,
// there may be other reasons to disconnect.
@@ -1310,7 +1312,7 @@ where
return false;
}
}
- Err(e) => error!(target: "service", "Error querying ban status for {ip}: {e}"),
+ Err(e) => warn!(target: "service", "Failed to query ban status for {ip}: {e}"),
}
let host: HostName = ip.into();
let tokens = self.config.limits.rate.inbound;
@@ -1379,7 +1381,7 @@ where
.addresses_mut()
.record_ip(&remote, ip, self.clock.into())
{
- log::error!(target: "service", "Error recording IP address for {remote}: {e}");
+ log::debug!(target: "service", "Failed to record IP address for {remote}: {e}");
}
}
}
@@ -1473,7 +1475,7 @@ where
.addresses_mut()
.disconnected(&remote, &addr, severity)
{
- error!(target: "service", "Error updating address store: {e}");
+ debug!(target: "service", "Failed to update address store: {e}");
}
// Only re-attempt outbound connections, since we don't care if an inbound connection
// is dropped.
@@ -1544,7 +1546,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Error looking up node in address book: {e}");
+ debug!(target: "service", "Failed to look up node in address book: {e}");
return Ok(None);
}
}
@@ -1577,7 +1579,7 @@ where
return Ok(None);
}
Err(e) => {
- error!(target: "service", "Error updating gossip entry from {announcer}: {e}");
+ debug!(target: "service", "Failed to update gossip entry from {announcer}: {e}");
return Ok(None);
}
};
@@ -1602,7 +1604,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Error processing inventory from {announcer}: {e}");
+ debug!(target: "service", "Failed to process inventory from {announcer}: {e}");
return Ok(None);
}
}
@@ -1633,7 +1635,7 @@ where
missing.push(*id);
}
}
- Err(e) => error!(
+ Err(e) => debug!(
target: "service",
"Error checking local inventory for {id}: {e}"
),
@@ -1703,7 +1705,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Error updating sync status for {}: {e}", message.rid);
+ debug!(target: "service", "Failed to update sync status for {}: {e}", message.rid);
}
}
}
@@ -1781,7 +1783,7 @@ where
}
Err(err) => {
// An error here is due to a fault in our address store.
- error!(target: "service", "Error processing node announcement from {announcer}: {err}");
+ warn!(target: "service", "Failed to process node announcement from {announcer}: {err}");
}
}
}
@@ -1871,7 +1873,7 @@ where
.gossip_mut()
.set_relay(id, gossip::RelayStatus::Relay)
{
- error!(target: "service", "Error setting relay flag for message: {e}");
+ warn!(target: "service", "Failed to set relay flag for message: {e}");
return Ok(());
}
} else {
@@ -1892,7 +1894,7 @@ where
let ann = match ann {
Ok(a) => a,
Err(e) => {
- error!(target: "service", "Error reading gossip message from store: {e}");
+ debug!(target: "service", "Failed to read gossip message from store: {e}");
continue;
}
};
@@ -1907,7 +1909,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Error querying gossip messages from store: {e}");
+ warn!(target: "service", "Failed to query gossip messages from store: {e}");
}
}
peer.subscribe = Some(subscribe);
@@ -2044,7 +2046,7 @@ where
let now = self.timestamp();
if !self.storage.contains(&rid)? {
- error!(target: "service", "Attempt to add non-existing inventory {rid}: repository not found in storage");
+ debug!(target: "service", "Attempt to add non-existing inventory {rid}: repository not found in storage");
return Ok(false);
}
// Add to our local inventory.
@@ -2188,7 +2190,7 @@ where
r.at,
timestamp.to_local_time(),
) {
- error!(
+ warn!(
target: "service",
"Error updating refs database for `rad/sigrefs` of {} in {rid}: {e}",
r.remote
@@ -2219,7 +2221,7 @@ where
);
// Update our local node's sync status to mark the refs as announced.
if let Err(e) = self.db.seeds_mut().synced(&rid, &ann.node, r.at, timestamp) {
- error!(target: "service", "Error updating sync status for local node: {e}");
+ warn!(target: "service", "Failed to update sync status for local node: {e}");
} else {
debug!(target: "service", "Saved local sync status for {rid}..");
}
@@ -2262,7 +2264,7 @@ where
let timestamp: Timestamp = self.clock.into();
if let Err(e) = self.db.addresses_mut().attempted(&nid, &addr, timestamp) {
- error!(target: "service", "Error updating address book with connection attempt: {e}");
+ warn!(target: "service", "Failed to update address book with connection attempt: {e}");
}
self.sessions.insert(
nid,
@@ -2507,7 +2509,7 @@ where
peers
}
Err(e) => {
- error!(target: "service", "Unable to lookup available peers in address book: {e}");
+ warn!(target: "service", "Unable to lookup available peers in address book: {e}");
Vec::new()
}
}
@@ -2520,7 +2522,7 @@ where
let policy = match policy {
Ok(policy) => policy,
Err(err) => {
- log::error!(target: "protocol::filter", "Failed to read seed policy: {err}");
+ debug!(target: "protocol::filter", "Failed to read seed policy: {err}");
continue;
}
};
@@ -2561,7 +2563,7 @@ where
}
}
Err(e) => {
- error!(target: "service", "Couldn't fetch missing repo {rid}: failed to lookup seeds: {e}");
+ debug!(target: "service", "Couldn't fetch missing repo {rid}: failed to lookup seeds: {e}");
}
}
}
@@ -2580,7 +2582,7 @@ where
.addresses_mut()
.connected(&sess.id, &sess.addr, self.clock.into())
{
- error!(target: "service", "Error updating address book with connection: {e}");
+ warn!(target: "service", "Failed to update address book with connection: {e}");
}
}
}
@@ -2645,7 +2647,7 @@ where
}
for (id, ka) in connect {
if let Err(e) = self.connect(id, ka.addr.clone()) {
- error!(target: "service", "Service::maintain_connections connection error: {e}");
+ warn!(target: "service", "Service::maintain_connections connection error: {e}");
}
}
}
diff --git a/crates/radicle-protocol/src/service/filter.rs b/crates/radicle-protocol/src/service/filter.rs
index 500bb26aa..8a8916321 100644
--- a/crates/radicle-protocol/src/service/filter.rs
+++ b/crates/radicle-protocol/src/service/filter.rs
@@ -70,7 +70,7 @@ impl Filter {
let seed = match seed {
Ok(seed) => seed,
Err(err) => {
- log::error!(target: "protocol::filter", "Failed to read seed policy: {err}");
+ log::debug!(target: "protocol::filter", "Failed to read seed policy: {err}");
continue;
}
};
diff --git a/crates/radicle-protocol/src/service/gossip.rs b/crates/radicle-protocol/src/service/gossip.rs
index 0d04c80f0..45aa0b89a 100644
--- a/crates/radicle-protocol/src/service/gossip.rs
+++ b/crates/radicle-protocol/src/service/gossip.rs
@@ -44,7 +44,7 @@ pub fn inventory(
) -> InventoryAnnouncement {
let inventory = inventory.into_iter().collect::<Vec<_>>();
if inventory.len() > INVENTORY_LIMIT {
- error!(
+ warn!(
target: "service",
"inventory announcement limit ({}) exceeded, other nodes will see only some of your projects",
inventory.len()
diff --git a/crates/radicle-protocol/src/service/io.rs b/crates/radicle-protocol/src/service/io.rs
index cc041f7dc..1862c11d4 100644
--- a/crates/radicle-protocol/src/service/io.rs
+++ b/crates/radicle-protocol/src/service/io.rs
@@ -82,7 +82,7 @@ impl Outbox {
// Store our announcement so that it can be retrieved from us later, just like
// announcements we receive from peers.
if let Err(e) = gossip.announced(&ann.node, &ann) {
- error!(target: "service", "Error updating our gossip store with announced message: {e}");
+ warn!(target: "service", "Failed to update gossip store with announced message: {e}");
}
for peer in peers {
commit 876ac47281e865378e77c98820b80999076d110d
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 12:18:27 2026 +0000
node: update comment
The comment mentions a variable `needs_flush`, which is not `write_intent`.
diff --git a/crates/radicle-node/src/reactor/transport.rs b/crates/radicle-node/src/reactor/transport.rs
index 4d877e694..6ace822c1 100644
--- a/crates/radicle-node/src/reactor/transport.rs
+++ b/crates/radicle-node/src/reactor/transport.rs
@@ -206,7 +206,7 @@ impl<S: Session> Transport<S> {
}
match self.flush() {
Ok(_) => None,
- // In this case, the write could not complete. Leave `needs_flush` set
+ // In this case, the write could not complete. Leave `write_intent` set
// to be notified when the socket is ready to write again.
Err(err)
if matches!(
commit 689dccccc90acedccacc3b9ab75174f10b0ce51c
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 12:07:00 2026 +0000
node/routing: remove error scenario from Database::prune
A call to `Database::prune` with no limit could result in a
domain-defined overflow error.
If the limit is not provided, then it uses `i64::MAX`. It is
reasonable to expect that if the `usize` provided is more than
`i64::MAX` then the `i64::MAX` value is acceptable to use for pruning.
diff --git a/crates/radicle/src/node/routing.rs b/crates/radicle/src/node/routing.rs
index 466010f96..2443e4b26 100644
--- a/crates/radicle/src/node/routing.rs
+++ b/crates/radicle/src/node/routing.rs
@@ -222,10 +222,8 @@ impl Store for Database {
ignore: &NodeId,
) -> Result<usize, Error> {
let limit: i64 = limit
- .unwrap_or(i64::MAX as usize)
- .try_into()
- .map_err(|_| Error::UnitOverflow)?;
-
+ .and_then(|limit| i64::try_from(limit).ok())
+ .unwrap_or(i64::MAX);
let mut stmt = self.db.prepare(
"DELETE FROM routing
WHERE node <> ?1 AND rowid IN
commit 4d965bb448c6d89974bd4b8c10450945a9d7d230
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 12:04:09 2026 +0000
protocol: downgrade `warn!` logs
Evaluate the `warn!` logging in `radicle-protocol`.
If the event is a result of something being suspicious in the running
of the protocol, then it remains a `warn!`.
However, if the system ends up ignoring the event and can continue
without issues, the log level is reduced to `debug!`.
diff --git a/crates/radicle-protocol/src/service.rs b/crates/radicle-protocol/src/service.rs
index a437b4e93..053d47968 100644
--- a/crates/radicle-protocol/src/service.rs
+++ b/crates/radicle-protocol/src/service.rs
@@ -691,7 +691,7 @@ where
// If we're not seeding this repo, just skip it.
if !self.policies.is_seeding(&rid)? {
- warn!(target: "service", "Local repository {rid} is not seeded");
+ debug!(target: "service", "Local repository {rid} is not seeded");
continue;
}
// Add public repositories to inventory.
@@ -1813,7 +1813,7 @@ where
let local = self.node_id();
let relay = self.config.is_relay();
let Some(peer) = self.sessions.get_mut(remote) else {
- warn!(target: "service", "Session not found for {remote}");
+ debug!(target: "service", "Session not found for {remote}");
return Ok(());
};
peer.last_active = self.clock;
@@ -2537,7 +2537,7 @@ where
}
}
Err(err) => {
- log::warn!(target: "protocol::filter", "Failed to check if {rid} exists: {err}");
+ log::debug!(target: "protocol::filter", "Failed to check if {rid} exists: {err}");
continue;
}
}
diff --git a/crates/radicle-protocol/src/service/message.rs b/crates/radicle-protocol/src/service/message.rs
index 68ac318dd..afcea6317 100644
--- a/crates/radicle-protocol/src/service/message.rs
+++ b/crates/radicle-protocol/src/service/message.rs
@@ -217,9 +217,9 @@ impl RefsStatus {
self.want.push(theirs);
}
Err(e) => {
- log::warn!(
+ log::debug!(
target: "service",
- "Error getting cached ref of {repo} for refs status: {e}"
+ "Failed to get cached 'rad/sigrefs' of {} in {repo} for refs status: {e}", theirs.remote,
);
}
}
diff --git a/crates/radicle-protocol/src/service/session.rs b/crates/radicle-protocol/src/service/session.rs
index 024c42ed1..665576563 100644
--- a/crates/radicle-protocol/src/service/session.rs
+++ b/crates/radicle-protocol/src/service/session.rs
@@ -301,7 +301,7 @@ impl Session {
pub fn fetched(&mut self, rid: RepoId) {
if let State::Connected { fetching, .. } = &mut self.state {
if !fetching.remove(&rid) {
- log::warn!(target: "service", "Fetched unknown repository {rid}");
+ log::debug!(target: "service", "Fetched unknown repository {rid}");
}
}
}
@@ -319,7 +319,7 @@ impl Session {
self.last_active = since;
if let State::Connected { .. } = &self.state {
- log::error!(target: "service", "Session {} is already in 'connected' state, resetting..", self.id);
+ log::debug!(target: "service", "Session {} is already in 'connected' state, resetting..", self.id);
};
self.state = State::Connected {
since,
commit e5d7381e31ef2d4ac9b599468d7c51b9f05e8b51
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Wed Dec 17 13:57:25 2025 +0000
protocol/service: defensive storage.contains check
In the case of the `storage.contains` call returning an error, it
would be preferable to log the error and continue checking for missing
repositories.
diff --git a/crates/radicle-protocol/src/service.rs b/crates/radicle-protocol/src/service.rs
index 775d692e7..a437b4e93 100644
--- a/crates/radicle-protocol/src/service.rs
+++ b/crates/radicle-protocol/src/service.rs
@@ -2530,8 +2530,16 @@ where
if !policy.is_allow() {
continue;
}
- if self.storage.contains(&rid)? {
- continue;
+ match self.storage.contains(&rid) {
+ Ok(exists) => {
+ if exists {
+ continue;
+ }
+ }
+ Err(err) => {
+ log::warn!(target: "protocol::filter", "Failed to check if {rid} exists: {err}");
+ continue;
+ }
}
match self.seeds(&rid, [self.node_id()].into()) {
Ok(seeds) => {
commit a1aa3e0d4bc6cc339e13f822b9c3e2fba1218c3a
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 16:31:13 2026 +0000
fetch: remove `target: "fetch"`
Going forward, there will be a preference to not use the `target` in
logging, and allow the default of the crate and module to be used
instead.
diff --git a/crates/radicle-fetch/src/git/packfile.rs b/crates/radicle-fetch/src/git/packfile.rs
index 1b7a6612b..313f84c40 100644
--- a/crates/radicle-fetch/src/git/packfile.rs
+++ b/crates/radicle-fetch/src/git/packfile.rs
@@ -26,7 +26,7 @@ impl Keepfile {
impl Drop for Keepfile {
fn drop(&mut self) {
if let Err(e) = fs::remove_file(&self.path) {
- log::warn!(target: "fetch", "Failed to remove {:?}: {e}", self.path);
+ log::warn!("Failed to remove {:?}: {e}", self.path);
}
}
}
diff --git a/crates/radicle-fetch/src/lib.rs b/crates/radicle-fetch/src/lib.rs
index 17fced5ad..e1e0b0150 100644
--- a/crates/radicle-fetch/src/lib.rs
+++ b/crates/radicle-fetch/src/lib.rs
@@ -88,7 +88,6 @@ where
.map_err(Error::Protocol);
log::debug!(
- target: "fetch",
"Finished pull of {} ({}ms)",
handle.repository().id(),
start.elapsed().as_millis()
@@ -123,16 +122,10 @@ where
match &result {
Ok(_) => {
- log::debug!(
- target: "fetch",
- "Finished clone of {rid} from {remote} ({elapsed}ms)",
- );
+ log::debug!("Finished clone of {rid} from {remote} ({elapsed}ms)",);
}
Err(e) => {
- log::debug!(
- target: "fetch",
- "Clone of {rid} from {remote} failed with '{e}' ({elapsed}ms)",
- );
+ log::debug!("Clone of {rid} from {remote} failed with '{e}' ({elapsed}ms)",);
}
}
result
diff --git a/crates/radicle-fetch/src/policy.rs b/crates/radicle-fetch/src/policy.rs
index e70ee201c..d7ef0684a 100644
--- a/crates/radicle-fetch/src/policy.rs
+++ b/crates/radicle-fetch/src/policy.rs
@@ -20,7 +20,7 @@ impl Allowed {
.map_err(|err| error::Policy::FailedPolicy { rid, err })?;
match entry.policy {
SeedingPolicy::Block => {
- log::info!(target: "fetch", "Attempted to fetch non-seeded repo {rid}");
+ log::info!("Attempted to fetch non-seeded repo {rid}");
Err(error::Policy::BlockedPolicy { rid })
}
SeedingPolicy::Allow { scope: Scope::All } => Ok(Self::All),
@@ -37,7 +37,7 @@ impl Allowed {
let node = match node {
Ok(policy) => policy,
Err(err) => {
- log::debug!(target: "fetch", "Failed to read follow policy for {rid}: {err}");
+ log::debug!("Failed to read follow policy for {rid}: {err}");
continue;
}
};
@@ -81,7 +81,7 @@ impl BlockList {
let entry = match entry {
Ok(entry) => entry,
Err(err) => {
- log::debug!(target: "fetch", "Failed to read follow policy: {err}");
+ log::debug!("Failed to read follow policy: {err}");
continue;
}
};
diff --git a/crates/radicle-fetch/src/state.rs b/crates/radicle-fetch/src/state.rs
index 9b7f6b9a4..42469ab5e 100644
--- a/crates/radicle-fetch/src/state.rs
+++ b/crates/radicle-fetch/src/state.rs
@@ -231,7 +231,7 @@ impl FetchState {
.collect::<Vec<_>>(),
None => vec![],
};
- log::trace!(target: "fetch", "Received refs {refs:?}");
+ log::trace!("Received refs {refs:?}");
step.pre_validate(&refs)?;
let wants_haves = step.wants_haves(handle.repository(), &refs)?;
@@ -242,7 +242,7 @@ impl FetchState {
.fetch(wants_haves, handle.interrupt.clone(), handshake)?;
self.keepfiles.extend(keepfile);
} else {
- log::trace!(target: "fetch", "Nothing to fetch")
+ log::trace!("Nothing to fetch")
};
let mut fetched = BTreeSet::new();
@@ -308,7 +308,7 @@ impl FetchState {
blocked: handle.blocked.clone(),
limit: limit.special,
};
- log::trace!(target: "fetch", "{sigrefs_at:?}");
+ log::trace!("{sigrefs_at:?}");
self.run_stage(handle, handshake, &sigrefs_at)?;
let remotes = refs_at.iter().map(|r| &r.remote);
@@ -317,7 +317,7 @@ impl FetchState {
}
None => {
let followed = handle.allowed();
- log::trace!(target: "fetch", "Followed nodes {followed:?}");
+ log::trace!("Followed nodes {followed:?}");
let special_refs = stage::SpecialRefs {
blocked: handle.blocked.clone(),
remote,
@@ -326,7 +326,7 @@ impl FetchState {
threshold,
limit: limit.special,
};
- log::trace!(target: "fetch", "{special_refs:?}");
+ log::trace!("{special_refs:?}");
let fetched = self.run_stage(handle, handshake, &special_refs)?;
let signed_refs = sigrefs::RemoteRefs::load(
@@ -378,7 +378,7 @@ impl FetchState {
limit: limit.special,
},
)?;
- log::debug!(target: "fetch", "Fetched rad/id ({}ms)", start.elapsed().as_millis());
+ log::debug!("Fetched rad/id ({}ms)", start.elapsed().as_millis());
// N.b. The error case here should not happen. In the case of
// a `clone` we have asked for refs/rad/id and ensured it was
@@ -399,7 +399,7 @@ impl FetchState {
.map(|did| PublicKey::from(*did))
.collect::<BTreeSet<_>>();
- log::trace!(target: "fetch", "Identity delegates {delegates:?}");
+ log::trace!("Identity delegates {delegates:?}");
// The local peer does not need to count towards the threshold
// since they must be valid already.
@@ -418,7 +418,6 @@ impl FetchState {
refs_at,
)?;
log::debug!(
- target: "fetch",
"Fetched data for {} remote(s) ({}ms)",
signed_refs.len(),
start.elapsed().as_millis()
@@ -431,7 +430,6 @@ impl FetchState {
};
self.run_stage(handle, handshake, &data_refs)?;
log::debug!(
- target: "fetch",
"Fetched data refs for {} remotes ({}ms)",
data_refs.remotes.len(),
start.elapsed().as_millis()
@@ -441,9 +439,9 @@ impl FetchState {
// We're finished fetching on this side, and all that's left
// is validation.
match handle.transport.done() {
- Ok(()) => log::debug!(target: "fetch", "Sent done signal to remote {remote}"),
+ Ok(()) => log::debug!("Sent done signal to remote {remote}"),
Err(err) => {
- log::debug!(target: "fetch", "Failed to signal EOF to {remote}: {err}")
+ log::debug!("Failed to signal EOF to {remote}: {err}")
}
}
@@ -472,7 +470,7 @@ impl FetchState {
// private function.
for remote in signed_refs.keys() {
if handle.is_blocked(remote) {
- log::trace!(target: "fetch", "Skipping blocked remote {remote}");
+ log::trace!("Skipping blocked remote {remote}");
continue;
}
@@ -480,12 +478,12 @@ impl FetchState {
.load(&self.as_cached(handle))?;
match remote {
sigrefs::DelegateStatus::NonDelegate { remote, data: None } => {
- log::debug!(target: "fetch", "Pruning non-delegate {remote} tips, missing 'rad/sigrefs'");
+ log::debug!("Pruning non-delegate {remote} tips, missing 'rad/sigrefs'");
failures.push(sigrefs::Validation::MissingRadSigRefs(remote));
self.prune(&remote);
}
sigrefs::DelegateStatus::Delegate { remote, data: None } => {
- log::debug!(target: "fetch", "Pruning delegate {remote} tips, missing 'rad/sigrefs'");
+ log::debug!("Pruning delegate {remote} tips, missing 'rad/sigrefs'");
failures.push(sigrefs::Validation::MissingRadSigRefs(remote));
self.prune(&remote);
// This delegate has removed their `rad/sigrefs`.
@@ -518,7 +516,6 @@ impl FetchState {
let cache = self.as_cached(handle);
if let Some(warns) = sigrefs::validate(&cache, sigrefs)?.as_mut() {
log::debug!(
- target: "fetch",
"Pruning non-delegate {remote} tips, due to validation failures"
);
self.prune(&remote);
@@ -536,7 +533,10 @@ impl FetchState {
{
let ancestry = repository::ancestry(handle.repository(), at, sigrefs.at)?;
if matches!(ancestry, repository::Ancestry::Behind) {
- log::trace!(target: "fetch", "Advertised `rad/sigrefs` {} is behind {at} for {remote}", sigrefs.at);
+ log::trace!(
+ "Advertised `rad/sigrefs` {} is behind {at} for {remote}",
+ sigrefs.at
+ );
self.prune(&remote);
continue;
} else if matches!(ancestry, repository::Ancestry::Diverged) {
@@ -552,7 +552,7 @@ impl FetchState {
let mut fails =
sigrefs::validate(&cache, sigrefs)?.unwrap_or(Validations::default());
if !fails.is_empty() {
- log::debug!(target: "fetch", "Pruning delegate {remote} tips, due to validation failures");
+ log::debug!("Pruning delegate {remote} tips, due to validation failures");
self.prune(&remote);
valid_delegates.remove(&remote);
failed_delegates.insert(remote);
@@ -565,7 +565,6 @@ impl FetchState {
}
}
log::debug!(
- target: "fetch",
"Validated {} remote(s) ({}ms)",
remotes.len(),
start.elapsed().as_millis()
@@ -581,7 +580,7 @@ impl FetchState {
.into_values()
.flat_map(|ups| ups.into_iter()),
)?;
- log::debug!(target: "fetch", "Applied updates ({}ms)", start.elapsed().as_millis());
+ log::debug!("Applied updates ({}ms)", start.elapsed().as_millis());
Ok(FetchResult::Success {
applied,
remotes,
@@ -589,7 +588,6 @@ impl FetchState {
})
} else {
log::debug!(
- target: "fetch",
"Fetch failed: {} failure(s) ({}ms)",
failures.len(),
start.elapsed().as_millis()
diff --git a/crates/radicle-fetch/src/transport.rs b/crates/radicle-fetch/src/transport.rs
index e5e7bf9d6..1e5d13c21 100644
--- a/crates/radicle-fetch/src/transport.rs
+++ b/crates/radicle-fetch/src/transport.rs
@@ -91,7 +91,7 @@ where
/// Perform the handshake with the server side.
#[allow(clippy::result_large_err)]
pub(crate) fn handshake(&mut self) -> Result<handshake::Outcome, handshake::Error> {
- log::trace!(target: "fetch", "Performing handshake for {}", self.repo);
+ log::trace!("Performing handshake for {}", self.repo);
let (read, write) = self.stream.open();
gix_protocol::fetch::handshake(
&mut Connection::new(read, write, self.repo.clone()),
@@ -129,7 +129,6 @@ where
handshake: &handshake::Outcome,
) -> Result<Option<Keepfile>, Error> {
log::trace!(
- target: "fetch",
"Running fetch wants={:?}, haves={:?}",
wants_haves.wants,
wants_haves.haves
@@ -328,8 +327,8 @@ fn agent_name() -> String {
Ok(version) => version,
Err(err) => {
use radicle::git::VERSION_REQUIRED;
- log::debug!(target: "fetch", "The git version could not be determined: {err}");
- log::debug!(target: "fetch", "Pretending that we are on git version {VERSION_REQUIRED}.");
+ log::debug!("The git version could not be determined: {err}");
+ log::debug!("Pretending that we are on git version {VERSION_REQUIRED}.");
VERSION_REQUIRED
}
};
diff --git a/crates/radicle-fetch/src/transport/fetch.rs b/crates/radicle-fetch/src/transport/fetch.rs
index 61538ffa0..81a00351b 100644
--- a/crates/radicle-fetch/src/transport/fetch.rs
+++ b/crates/radicle-fetch/src/transport/fetch.rs
@@ -159,7 +159,7 @@ where
R: io::Read,
W: io::Write,
{
- log::trace!(target: "fetch", "Performing fetch");
+ log::trace!("Performing fetch");
if wants_haves.wants.is_empty() {
return Err(Error::ReadRemainingBytes(io::Error::new(
@@ -214,6 +214,6 @@ where
.and_then(packfile::Keepfile::new);
out.pack = Some(pack_out);
- log::trace!(target: "fetch", "fetched refs: {:?}", out.refs);
+ log::trace!("fetched refs: {:?}", out.refs);
Ok(out)
}
diff --git a/crates/radicle-fetch/src/transport/ls_refs.rs b/crates/radicle-fetch/src/transport/ls_refs.rs
index 6ef18ebba..c088eb875 100644
--- a/crates/radicle-fetch/src/transport/ls_refs.rs
+++ b/crates/radicle-fetch/src/transport/ls_refs.rs
@@ -38,7 +38,7 @@ where
R: io::Read,
W: io::Write,
{
- log::trace!(target: "fetch", "Performing ls-refs: {:?}", config.prefixes);
+ log::trace!("Performing ls-refs: {:?}", config.prefixes);
let handshake::Outcome {
server_protocol_version: protocol,
capabilities,
commit 99c4aa6979d2ddfe57ecb7eeebc9513e4c547217
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Mon Jan 19 11:52:24 2026 +0000
fetch: clean up logging
Evaluate the use of `warn!` and `error!` logging in `radicle-fetch`.
The majority are downgraded to `debug!`, the rest either being
downgraded to `info!` or removed altogether.
diff --git a/crates/radicle-fetch/src/lib.rs b/crates/radicle-fetch/src/lib.rs
index a53a0bfe9..17fced5ad 100644
--- a/crates/radicle-fetch/src/lib.rs
+++ b/crates/radicle-fetch/src/lib.rs
@@ -150,16 +150,11 @@ where
}
fn handle_handshake_err(err: handshake::Error) -> HandshakeError {
- let err = match err {
+ match err {
handshake::Error::Transport(error) => match error {
gix_transport::client::Error::Io(error) => HandshakeError::Io(error),
err => HandshakeError::Gix(handshake::Error::Transport(err)),
},
- err => {
- log::warn!(target: "fetch", "Failed to perform handshake: {err}");
- HandshakeError::Gix(err)
- }
- };
- log::warn!(target: "fetch", "{err}");
- err
+ err => HandshakeError::Gix(err),
+ }
}
diff --git a/crates/radicle-fetch/src/policy.rs b/crates/radicle-fetch/src/policy.rs
index afc07aca1..e70ee201c 100644
--- a/crates/radicle-fetch/src/policy.rs
+++ b/crates/radicle-fetch/src/policy.rs
@@ -20,7 +20,7 @@ impl Allowed {
.map_err(|err| error::Policy::FailedPolicy { rid, err })?;
match entry.policy {
SeedingPolicy::Block => {
- log::error!(target: "fetch", "Attempted to fetch non-seeded repo {rid}");
+ log::info!(target: "fetch", "Attempted to fetch non-seeded repo {rid}");
Err(error::Policy::BlockedPolicy { rid })
}
SeedingPolicy::Allow { scope: Scope::All } => Ok(Self::All),
@@ -37,7 +37,7 @@ impl Allowed {
let node = match node {
Ok(policy) => policy,
Err(err) => {
- log::error!(target: "fetch", "Failed to read follow policy for {rid}: {err}");
+ log::debug!(target: "fetch", "Failed to read follow policy for {rid}: {err}");
continue;
}
};
@@ -81,7 +81,7 @@ impl BlockList {
let entry = match entry {
Ok(entry) => entry,
Err(err) => {
- log::error!(target: "fetch", "Failed to read follow policy: {err}");
+ log::debug!(target: "fetch", "Failed to read follow policy: {err}");
continue;
}
};
diff --git a/crates/radicle-fetch/src/state.rs b/crates/radicle-fetch/src/state.rs
index 2449b6039..9b7f6b9a4 100644
--- a/crates/radicle-fetch/src/state.rs
+++ b/crates/radicle-fetch/src/state.rs
@@ -443,7 +443,7 @@ impl FetchState {
match handle.transport.done() {
Ok(()) => log::debug!(target: "fetch", "Sent done signal to remote {remote}"),
Err(err) => {
- log::warn!(target: "fetch", "Attempted to send done to remote {remote}: {err}")
+ log::debug!(target: "fetch", "Failed to signal EOF to {remote}: {err}")
}
}
@@ -485,7 +485,7 @@ impl FetchState {
self.prune(&remote);
}
sigrefs::DelegateStatus::Delegate { remote, data: None } => {
- log::warn!(target: "fetch", "Pruning delegate {remote} tips, missing 'rad/sigrefs'");
+ log::debug!(target: "fetch", "Pruning delegate {remote} tips, missing 'rad/sigrefs'");
failures.push(sigrefs::Validation::MissingRadSigRefs(remote));
self.prune(&remote);
// This delegate has removed their `rad/sigrefs`.
@@ -552,7 +552,7 @@ impl FetchState {
let mut fails =
sigrefs::validate(&cache, sigrefs)?.unwrap_or(Validations::default());
if !fails.is_empty() {
- log::warn!(target: "fetch", "Pruning delegate {remote} tips, due to validation failures");
+ log::debug!(target: "fetch", "Pruning delegate {remote} tips, due to validation failures");
self.prune(&remote);
valid_delegates.remove(&remote);
failed_delegates.insert(remote);
diff --git a/crates/radicle-fetch/src/transport.rs b/crates/radicle-fetch/src/transport.rs
index f3a46b6b6..e5e7bf9d6 100644
--- a/crates/radicle-fetch/src/transport.rs
+++ b/crates/radicle-fetch/src/transport.rs
@@ -328,8 +328,8 @@ fn agent_name() -> String {
Ok(version) => version,
Err(err) => {
use radicle::git::VERSION_REQUIRED;
- log::warn!(target: "fetch", "The git version could not be determined: {err}");
- log::warn!(target: "fetch", "Pretending that we are on git version {VERSION_REQUIRED}.");
+ log::debug!(target: "fetch", "The git version could not be determined: {err}");
+ log::debug!(target: "fetch", "Pretending that we are on git version {VERSION_REQUIRED}.");
VERSION_REQUIRED
}
};
commit b52ea8487c9dae11c41ed09c2c71cab9688f16db
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Fri Dec 19 10:16:55 2025 +0000
fetch: surface underlying I/O error
The message that is returned by `gix-transport` for I/O errors can be
unhelpful, since it does not provide the reason for what happened.
Instead, surface the error so that it provides more detail for logging.
diff --git a/crates/radicle-cli/examples/rad-init-private-clone.md b/crates/radicle-cli/examples/rad-init-private-clone.md
index a52498812..852538d95 100644
--- a/crates/radicle-cli/examples/rad-init-private-clone.md
+++ b/crates/radicle-cli/examples/rad-init-private-clone.md
@@ -10,7 +10,7 @@ $ rad clone rad:z2ug5mwNKZB8KGpBDRTrWHAMbvHCu --seed z6MknSLrJoTcukLrE435hVNQT4J
Fetching rad:z2ug5mwNKZB8KGpBDRTrWHAMbvHCu from the network, found 1 potential seed(s).
✗ Target not met: could not fetch from [z6MknSLrJoTcukLrE435hVNQT4JUhbvWLX4kUzqkEStBU8Vi], and required 1 more seed(s)
! Warning: Failed to fetch from 1 seed(s).
-! Warning: z6MknSLrJoTcukLrE435hVNQT4JUhbvWLX4kUzqkEStBU8Vi: failed to perform fetch handshake: [..]
+! Warning: z6MknSLrJoTcukLrE435hVNQT4JUhbvWLX4kUzqkEStBU8Vi: an I/O error occurred during the fetch handshake (connection reset)
✗ Error: no seeds found for rad:z2ug5mwNKZB8KGpBDRTrWHAMbvHCu
```
diff --git a/crates/radicle-fetch/src/lib.rs b/crates/radicle-fetch/src/lib.rs
index cd1c20074..a53a0bfe9 100644
--- a/crates/radicle-fetch/src/lib.rs
+++ b/crates/radicle-fetch/src/lib.rs
@@ -9,6 +9,7 @@ mod refs;
mod stage;
mod state;
+use std::io;
use std::time::Instant;
use gix_protocol::handshake;
@@ -28,8 +29,8 @@ use thiserror::Error;
#[derive(Debug, Error)]
pub enum Error {
- #[error("failed to perform fetch handshake: {0}")]
- Handshake(#[from] Box<handshake::Error>),
+ #[error(transparent)]
+ Handshake(Box<HandshakeError>),
#[error("failed to load `rad/id`")]
Identity {
#[source]
@@ -43,6 +44,20 @@ pub enum Error {
ReplicateSelf,
}
+impl From<HandshakeError> for Error {
+ fn from(err: HandshakeError) -> Self {
+ Self::Handshake(Box::new(err))
+ }
+}
+
+#[derive(Debug, Error)]
+pub enum HandshakeError {
+ #[error("failed to perform fetch handshake: {0}")]
+ Gix(handshake::Error),
+ #[error("an I/O error occurred during the fetch handshake ({0})")]
+ Io(io::Error),
+}
+
/// Pull changes from the `remote`.
///
/// It is expected that the local peer has a copy of the repository
@@ -127,11 +142,24 @@ fn perform_handshake<R, S>(handle: &mut Handle<R, S>) -> Result<handshake::Outco
where
S: transport::ConnectionStream,
{
- let result = handle.transport.handshake();
-
- if let Err(err) = &result {
- log::warn!(target: "fetch", "Failed to perform handshake: {err}");
- }
+ handle
+ .transport
+ .handshake()
+ .map_err(handle_handshake_err)
+ .map_err(Error::from)
+}
- Ok(result?)
+fn handle_handshake_err(err: handshake::Error) -> HandshakeError {
+ let err = match err {
+ handshake::Error::Transport(error) => match error {
+ gix_transport::client::Error::Io(error) => HandshakeError::Io(error),
+ err => HandshakeError::Gix(handshake::Error::Transport(err)),
+ },
+ err => {
+ log::warn!(target: "fetch", "Failed to perform handshake: {err}");
+ HandshakeError::Gix(err)
+ }
+ };
+ log::warn!(target: "fetch", "{err}");
+ err
}
diff --git a/crates/radicle-fetch/src/transport.rs b/crates/radicle-fetch/src/transport.rs
index d46c7ec57..f3a46b6b6 100644
--- a/crates/radicle-fetch/src/transport.rs
+++ b/crates/radicle-fetch/src/transport.rs
@@ -89,7 +89,8 @@ where
}
/// Perform the handshake with the server side.
- pub(crate) fn handshake(&mut self) -> Result<handshake::Outcome, Box<handshake::Error>> {
+ #[allow(clippy::result_large_err)]
+ pub(crate) fn handshake(&mut self) -> Result<handshake::Outcome, handshake::Error> {
log::trace!(target: "fetch", "Performing handshake for {}", self.repo);
let (read, write) = self.stream.open();
gix_protocol::fetch::handshake(
@@ -98,7 +99,6 @@ where
vec![],
&mut progress::Discard,
)
- .map_err(Box::new)
}
/// Perform ls-refs with the server side.
commit 4fc1d57299f573b0165ebe38c644bbfe01626192
Author: Fintan Halpenny <fintan.halpenny@gmail.com>
Date: Wed Jan 7 14:11:49 2026 +0000
Release 1.6.0 CHANGELOG
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 9aad309e0..8f3591ef2 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -9,25 +9,127 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## Release Highlights
+## New Features
+
+## Fixed Bugs
+
## Deprecations
+## 1.6.0
+
+## Release Highlights
+
+### Migrating `radicle-node` to `mio`
+
+The crates [`netservices`], [`io-reactor`], and [`popol`] were crucially valuable
+for implementing `radicle-node`. However, they are not ideal dependencies for
+ensuring long-term health of the network I/O layer:
+
+- [`popol`] is only intended to support Unix-like platforms, and support on other
+ platforms, like Windows, is desired.
+- Even though [`io-reactor`] defines the trait [`reactor::poller::Poll`] to
+ potentially support multiple I/O polling mechanisms, there is only one single
+ implementation wrapping [`popol`]. Issues for other polling crates are open
+ since 2023 without tangible progress: [#10 for `mio`], [#9 for `polling`],
+ [#8 for `epoll`]. This suggests that it is not a high priority for the maintainers
+ to integrate with other polling abstractions which might offer better
+ cross-platform compatibility when compared to `popol`.
+- The trait [`reactor::poller::Poll`] can only be implemented for file
+ descriptors which also implement [`std::os::fd::raw::AsRawFd`], which is only
+ implemented on Unix-like platforms and WASI. It is believed that this is
+ leaked from `popol` as the only known implementation of the trait wraps it.
+- To benefit from network effects, it would be nice to see others maintaining crates
+ that depend on `io-reactor`. However, according to crates.io, the
+ [only dependent is `radicle-node`] (via `netservices`). Contrary to that,
+ at the time of writing, `mio` has 494 dependents according to
+ [crates.io][mio reverse dependencies], and, notably, `tokio`, which has
+ 30628 dependents on [crates.io][tokio reverse dependencies], is dependent on
+ [`mio`]. We therefore think that even if `mio` is obsoleted, e.g. by [`a10`]
+ (which is based on [`io_uring`] on Linux and could potentially build on top of
+ [I/O rings on Windows]) the people behind a large network of dependent projects
+ are expected to come up with new ideas and solutions, that Radicle would then
+ benefit from.
+- One downside of using `mio` is that it forces the use of [`mio::Token`] to
+ identify sources (while a type that is `Eq + Clone` might be enough). Another
+ downside is that it forces the use of the types in [`mio::net`] for sockets,
+ which need to be converted to/from [`std::net`] if required. These
+ distinctions are also [noted by cloudhead]. This is acceptable to the team, in
+ order to leverage the benefits of a well-tested and cross-platform network I/O
+ layer.
+
+[`netservices`]: https://crates.io/crates/netservices
+[`io-reactor`]: https://crates.io/crates/io-reactor
+[`popol`]: https://crates.io/crates/popol
+[`reactor::poller::Poll`]: https://docs.rs/io-reactor/0.5.2/reactor/poller/trait.Poll.html
+[#10 for `mio`]: https://github.com/rust-amplify/io-reactor/issues/10
+[#9 for `polling`]: https://github.com/rust-amplify/io-reactor/issues/9
+[#8 for `epoll`]: https://github.com/rust-amplify/io-reactor/issues/8
+[`std::os::fd::raw::AsRawFd`]: https://doc.rust-lang.org/nightly/std/os/fd/raw/trait.AsRawFd.html
+[only dependent is `radicle-node`]: https://crates.io/crates/io-reactor/reverse_dependencies
+[mio reverse dependencies]: https://crates.io/crates/mio/reverse_dependencies
+[tokio reverse dependencies]: https://crates.io/crates/tokio/reverse_dependencies
+[`a10`]: https://crates.io/crates/a10
+[`io_uring`]: https://en.wikipedia.org/wiki/Io_uring
+[I/O rings on Windows]: https://learn.microsoft.com/en-us/windows/win32/api/ioringapi/
+[`mio::Token`]: https://docs.rs/mio/1.0.4/mio/struct.Token.html
+[`mio::net`]: https://docs.rs/mio/1.0.4/mio/net/index.html
+[`std::net`]: https://doc.rust-lang.org/stable/std/net/index.html
+[noted by cloudhead]: https://cloudhead.io/popol/
+
+### Building `radicle-node` on Windows
+
+The efforts to migrate `radicle-node` to use `mio`, alongside changes that fixed
+path canonicalization and supporting Windows pipes, have allowed developers to
+build `radicle-node` on Windows.
+
+We encourage users to try out Radicle on Windows by building from source. At the
+time of writing, there may be undiscovered issues, since this is a nascent time
+for `radicle-node` on Windows. Please report any issues you see via `rad issue`
+or on our [Zulip](https://radicle.zulipchat.com).
+
+### Rust MSRV Update to 1.85
+
+For those who are developing on top of the `heartwood` crates, it is important
+to note that the Minimum Supported Rust Version (MSRV) is now 1.85.
+
## New Features
-- `rad issue` now uses `clap` to parse its command-line arguments.
- This affects error reporting as well as help output.
-- `radicle-node` now supports systemd Credentials (refer to
- <https://systemd.io/CREDENTIALS> for more information) to load:
- 1. The secret key, in addition to the commandline argument
- `--secret` (higher priority than the credential) and the
- configuration file (lower priority than the credential).
- The identifier of the credential is "xyz.radicle.node.secret".
+### Argument Parsing via `clap`
+
+`rad` now uses the `clap` crate for parsing its command-line arguments. This
+brings a brand new look to the help output for the `rad` CLI, and ensures that
+we do not miss documenting options when they are added. Note that this does
+affect error reporting, as they are now reported by `clap` when parsing fails.
+
+#### Shell Completions
+
+With the introduction of `clap`, this helped with the introduction of a command
+`rad completion` to emit shell completions for static information.
+
+### systemd Credentials for `radicle-node`
+
+`radicle-node` now supports systemd Credentials (refer to
+<https://systemd.io/CREDENTIALS> for more information) to load:
+ 1. The secret key, in addition to the commandline argument `--secret`
+ (higher priority than the credential) and the configuration file (lower
+ priority than the credential). The identifier of the credential is
+ "xyz.radicle.node.secret".
2. The optional passphrase for the secret key, in addition to the
environment variable `RAD_PASSPHRASE` (lower priority than the
- credential).
- The identifier of the credential is "xyz.radicle.node.passphrase".
+ credential). The identifier of the credential is
+ "xyz.radicle.node.passphrase".
## Fixed Bugs
+### Fix Bootstrapping
+
+The IP (both IPv4 and IPv6) and the Tor onion addresses were specified for the
+bootstrap nodes. When a new user came to using Radicle, there was a chance that
+their setup did not support IPv6 or Tor, resulting in a failure to connect to
+one of those addresses. The node does not know how to try a follow-up address,
+for the moment, so we have decided to skip Tor addresses when it is not
+configured, and removed the IP addresses in favor of the DNS names.
+
## 1.5.0
## Release Highlights
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 eaf55967-fde1-44f9-b51b-a0e811503bc7 -v /opt/radcis/ci.rad.levitte.org/cci/state/eaf55967-fde1-44f9-b51b-a0e811503bc7/s:/eaf55967-fde1-44f9-b51b-a0e811503bc7/s:ro -v /opt/radcis/ci.rad.levitte.org/cci/state/eaf55967-fde1-44f9-b51b-a0e811503bc7/w:/eaf55967-fde1-44f9-b51b-a0e811503bc7/w -w /eaf55967-fde1-44f9-b51b-a0e811503bc7/w -v /opt/radcis/ci.rad.levitte.org/.radicle:/${id}/.radicle:ro -e RAD_HOME=/${id}/.radicle rust:trixie bash /eaf55967-fde1-44f9-b51b-a0e811503bc7/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 autocfg v1.2.0
Downloaded ecdsa v0.16.9
Downloaded amplify_derive v4.0.0
Downloaded libz-rs-sys v0.5.2
Downloaded icu_provider v1.5.0
Downloaded base16ct v0.2.0
Downloaded ff v0.13.0
Downloaded gix-credentials v0.30.0
Downloaded gix-chunk v0.4.11
Downloaded filetime v0.2.23
Downloaded gix-commitgraph v0.29.0
Downloaded gix-diff v0.53.0
Downloaded flate2 v1.1.1
Downloaded fast-glob v0.3.3
Downloaded fraction v0.15.3
Downloaded memmap2 v0.9.8
Downloaded form_urlencoded v1.2.1
Downloaded fxhash v0.2.1
Downloaded poly1305 v0.8.0
Downloaded ec25519 v0.1.0
Downloaded hash32 v0.3.1
Downloaded gix-config-value v0.15.1
Downloaded jobserver v0.1.31
Downloaded icu_locid v1.5.0
Downloaded maybe-async v0.2.10
Downloaded num-complex v0.4.6
Downloaded data-encoding-macro-internal v0.1.12
Downloaded miniz_oxide v0.8.8
Downloaded memchr v2.7.2
Downloaded once_cell v1.21.3
Downloaded primeorder v0.13.6
Downloaded idna_adapter v1.2.0
Downloaded gix-validate v0.10.0
Downloaded pkcs8 v0.10.2
Downloaded pin-project-lite v0.2.16
Downloaded prodash v30.0.1
Downloaded pretty_assertions v1.4.0
Downloaded ref-cast-impl v1.0.24
Downloaded radicle-surf v0.25.0
Downloaded litemap v0.7.5
Downloaded ppv-lite86 v0.2.17
Downloaded ryu v1.0.17
Downloaded serde_spanned v1.0.0
Downloaded p384 v0.13.0
Downloaded signals_receipts v0.2.0
Downloaded shell-words v1.1.0
Downloaded sqlite3-sys v0.15.2
Downloaded ssh-encoding v0.2.0
Downloaded ref-cast v1.0.24
Downloaded sval v2.14.1
Downloaded walkdir v2.5.0
Downloaded sval_nested v2.14.1
Downloaded sval_serde v2.14.1
Downloaded phf v0.11.3
Downloaded radicle-git-ext v0.10.0
Downloaded tree-sitter-language v0.1.2
Downloaded test-log-macros v0.2.18
Downloaded ssh-cipher v0.2.0
Downloaded tinyvec_macros v0.1.1
Downloaded structured-logger v1.0.4
Downloaded spin v0.9.8
Downloaded sval_ref v2.14.1
Downloaded sha2 v0.10.8
Downloaded parking_lot_core v0.9.12
Downloaded siphasher v0.3.11
Downloaded sec1 v0.7.3
Downloaded schemars v1.0.4
Downloaded utf8parse v0.2.1
Downloaded xattr v1.3.1
Downloaded value-bag-sval2 v1.11.1
Downloaded value-bag-serde1 v1.11.1
Downloaded pkcs1 v0.7.5
Downloaded tree-sitter-css v0.23.1
Downloaded unicode-ident v1.0.12
Downloaded yoke-derive v0.7.5
Downloaded tree-sitter-html v0.23.2
Downloaded zerofrom-derive v0.1.6
Downloaded universal-hash v0.5.1
Downloaded utf16_iter v1.0.5
Downloaded utf8_iter v1.0.4
Downloaded tinystr v0.7.6
Downloaded vsimd v0.8.0
Downloaded unit-prefix v0.5.1
Downloaded tar v0.4.40
Downloaded yoke v0.7.5
Downloaded ssh-key v0.6.6
Downloaded zerovec-derive v0.10.3
Downloaded indicatif v0.18.0
Downloaded zeroize v1.7.0
Downloaded tree-sitter-c v0.23.2
Downloaded tracing-subscriber v0.3.19
Downloaded uuid v1.16.0
Downloaded itertools v0.14.0
Downloaded inquire v0.7.5
Downloaded unicode-segmentation v1.11.0
Downloaded zerofrom v0.1.6
Downloaded zerocopy v0.7.35
Downloaded url v2.5.4
Downloaded tree-sitter-python v0.23.4
Downloaded zlib-rs v0.5.2
Downloaded winnow v0.7.13
Downloaded zerovec v0.10.4
Downloaded vcpkg v0.2.15
Downloaded unicode-normalization v0.1.23
Downloaded unicode-width v0.2.1
Downloaded tree-sitter-md v0.3.2
Downloaded regex-syntax v0.6.29
Downloaded tree-sitter v0.24.4
Downloaded tree-sitter-rust v0.23.2
Downloaded tokio v1.47.1
Downloaded serde_json v1.0.140
Downloaded gimli v0.31.1
Downloaded tree-sitter-bash v0.23.3
Downloaded jsonschema v0.30.0
Downloaded icu_properties_data v1.5.1
Downloaded syn v1.0.109
Downloaded tree-sitter-ruby v0.23.1
Downloaded hashbrown v0.14.3
Downloaded gix-pack v0.60.0
Downloaded mio v1.0.4
Downloaded mio v0.8.11
Downloaded version_check v0.9.4
Downloaded syn v2.0.106
Downloaded tree-sitter-go v0.23.4
Downloaded jiff v0.2.15
Downloaded libc v0.2.174
Downloaded rsa v0.9.6
Downloaded yansi v0.5.1
Downloaded writeable v0.5.5
Downloaded write16 v1.0.0
Downloaded value-bag v1.11.1
Downloaded snapbox v0.4.17
Downloaded smallvec v1.15.1
Downloaded sha3 v0.10.8
Downloaded tree-sitter-typescript v0.23.2
Downloaded sha1-checked v0.10.0
Downloaded toml_writer v1.0.2
Downloaded unicode-width v0.1.11
Downloaded tempfile v3.23.0
Downloaded serde_derive v1.0.219
Downloaded gix-ref v0.53.1
Downloaded gix-protocol v0.51.0
Downloaded systemd-journal-logger v2.2.2
Downloaded tinyvec v1.6.0
Downloaded sharded-slab v0.1.7
Downloaded unicode-display-width v0.3.0
Downloaded timeago v0.4.2
Downloaded similar v2.5.0
Downloaded linux-raw-sys v0.4.13
Downloaded typenum v1.17.0
Downloaded tree-sitter-highlight v0.24.4
Downloaded heapless v0.8.0
Downloaded thiserror-impl v1.0.69
Downloaded sval_fmt v2.14.1
Downloaded rfc6979 v0.4.0
Downloaded tree-sitter-toml-ng v0.6.0
Downloaded num-traits v0.2.19
Downloaded tree-sitter-json v0.24.8
Downloaded tracing-log v0.2.0
Downloaded toml_datetime v0.7.0
Downloaded p256 v0.13.2
Downloaded icu_normalizer_data v1.5.1
Downloaded uuid-simd v0.8.0
Downloaded typeid v1.0.3
Downloaded thread_local v1.1.9
Downloaded synstructure v0.13.1
Downloaded tracing-core v0.1.34
Downloaded tracing v0.1.41
Downloaded toml v0.9.5
Downloaded sval_buffer v2.14.1
Downloaded subtle v2.5.0
Downloaded streaming-iterator v0.1.9
Downloaded socket2 v0.5.7
Downloaded signature v1.6.4
Downloaded signal-hook-registry v1.4.5
Downloaded signal-hook v0.3.18
Downloaded regex-automata v0.4.9
Downloaded rand v0.8.5
Downloaded libgit2-sys v0.17.0+1.8.1
Downloaded indexmap v2.2.6
Downloaded thiserror-impl v2.0.12
Downloaded thiserror v2.0.12
Downloaded thiserror v1.0.69
Downloaded test-log v0.2.18
Downloaded sval_json v2.14.1
Downloaded socks5-client v0.4.1
Downloaded scrypt v0.11.0
Downloaded regex-syntax v0.8.5
Downloaded referencing v0.30.0
Downloaded idna v1.0.3
Downloaded icu_normalizer v1.5.0
Downloaded parking_lot v0.12.5
Downloaded rustix v0.38.34
Downloaded qcheck v1.0.0
Downloaded group v0.13.0
Downloaded icu_locid_transform_data v1.5.1
Downloaded rand_core v0.6.4
Downloaded spki v0.7.3
Downloaded sqlite v0.32.0
Downloaded stable_deref_trait v1.2.0
Downloaded signature v2.2.0
Downloaded sem_safe v0.2.0
Downloaded scopeguard v1.2.0
Downloaded regex-automata v0.1.10
Downloaded proc-macro-error-attr2 v2.0.0
Downloaded sval_dynamic v2.14.1
Downloaded serde_fmt v1.0.3
Downloaded pkg-config v0.3.30
Downloaded gix-hash v0.19.0
Downloaded linux-raw-sys v0.9.4
Downloaded siphasher v1.0.1
Downloaded sha1 v0.10.6
Downloaded radicle-std-ext v0.1.1
Downloaded pem-rfc7468 v0.7.0
Downloaded strsim v0.11.1
Downloaded snapbox-macros v0.3.8
Downloaded signal-hook-mio v0.2.4
Downloaded same-file v1.0.6
Downloaded salsa20 v0.10.2
Downloaded rustc-demangle v0.1.26
Downloaded rand_chacha v0.3.1
Downloaded proc-macro2 v1.0.101
Downloaded portable-atomic v1.11.0
Downloaded num v0.4.3
Downloaded quote v1.0.41
Downloaded paste v1.0.15
Downloaded hmac v0.12.1
Downloaded gix-refspec v0.31.0
Downloaded serde v1.0.219
Downloaded schemars_derive v1.0.4
Downloaded rustix v1.0.7
Downloaded gix-traverse v0.47.0
Downloaded sqlite3-src v0.5.1
Downloaded polyval v0.6.2
Downloaded outref v0.5.2
Downloaded object v0.36.7
Downloaded lexopt v0.3.0
Downloaded itoa v1.0.11
Downloaded gix-fs v0.16.1
Downloaded shlex v1.3.0
Downloaded regex v1.11.1
Downloaded percent-encoding v2.3.1
Downloaded num-rational v0.4.2
Downloaded lock_api v0.4.14
Downloaded localtime v1.3.1
Downloaded inout v0.1.3
Downloaded icu_provider_macros v1.5.0
Downloaded gix-sec v0.12.0
Downloaded emojis v0.6.4
Downloaded serde_derive_internals v0.29.1
Downloaded serde-untagged v0.1.7
Downloaded num-cmp v0.1.0
Downloaded human-panic v2.0.3
Downloaded gix-actor v0.35.4
Downloaded gix-trace v0.1.13
Downloaded gix-shallow v0.5.0
Downloaded icu_locid_transform v1.5.0
Downloaded ghash v0.5.1
Downloaded qcheck-macros v1.0.0
Downloaded os_info v3.12.0
Downloaded nu-ansi-term v0.46.0
Downloaded newline-converter v0.3.0
Downloaded home v0.5.9
Downloaded gix-prompt v0.11.1
Downloaded cyphergraphy v0.3.0
Downloaded num-iter v0.1.45
Downloaded num-bigint v0.4.6
Downloaded libm v0.2.8
Downloaded gix-odb v0.70.0
Downloaded gix-object v0.50.2
Downloaded fastrand v2.3.0
Downloaded crossterm v0.29.0
Downloaded proc-macro-error2 v2.0.1
Downloaded p521 v0.13.3
Downloaded opaque-debug v0.3.1
Downloaded nonempty v0.9.0
Downloaded litrs v0.4.1
Downloaded lazy_static v1.5.0
Downloaded keccak v0.1.5
Downloaded gix-url v0.32.0
Downloaded gix-revision v0.35.0
Downloaded gix-quote v0.6.0
Downloaded gix-hashtable v0.9.0
Downloaded derive_more v2.0.1
Downloaded phf_shared v0.11.3
Downloaded pbkdf2 v0.12.2
Downloaded num-integer v0.1.46
Downloaded log v0.4.27
Downloaded gix-transport v0.48.0
Downloaded diff v0.1.13
Downloaded clap_builder v4.5.44
Downloaded cc v1.2.2
Downloaded iana-time-zone v0.1.60
Downloaded heck v0.5.0
Downloaded gix-packetline v0.19.1
Downloaded gix-date v0.10.5
Downloaded crossbeam-channel v0.5.15
Downloaded chrono v0.4.38
Downloaded overload v0.1.1
Downloaded normalize-line-endings v0.3.0
Downloaded jiff-static v0.2.15
Downloaded gix-negotiate v0.21.0
Downloaded env_logger v0.11.8
Downloaded dunce v1.0.5
Downloaded ct-codecs v1.1.1
Downloaded clap_complete v4.5.60
Downloaded borrow-or-share v0.2.2
Downloaded num-bigint-dig v0.8.4
Downloaded noise-framework v0.4.0
Downloaded libz-sys v1.1.16
Downloaded gix-path v0.10.20
Downloaded erased-serde v0.4.6
Downloaded bytes v1.10.1
Downloaded equivalent v1.0.1
Downloaded either v1.11.0
Downloaded crc32fast v1.5.0
Downloaded bstr v1.12.0
Downloaded matchers v0.1.0
Downloaded faster-hex v0.10.0
Downloaded block-buffer v0.10.4
Downloaded multibase v0.9.1
Downloaded icu_properties v1.5.1
Downloaded fancy-regex v0.14.0
Downloaded clap_lex v0.7.5
Downloaded blowfish v0.9.1
Downloaded bit-vec v0.8.0
Downloaded icu_collections v1.5.0
Downloaded crossterm v0.25.0
Downloaded chacha20poly1305 v0.10.1
Downloaded gix-revwalk v0.21.0
Downloaded displaydoc v0.2.5
Downloaded digest v0.10.7
Downloaded der v0.7.9
Downloaded data-encoding-macro v0.1.14
Downloaded colorchoice v1.0.0
Downloaded bytesize v2.0.1
Downloaded gix-utils v0.3.0
Downloaded gix-lock v18.0.0
Downloaded git-ref-format-macro v0.5.0
Downloaded escargot v0.5.10
Downloaded cypheraddr v0.4.0
Downloaded git2 v0.19.0
Downloaded email_address v0.2.9
Downloaded convert_case v0.7.1
Downloaded clap v4.5.44
Downloaded bitflags v2.9.1
Downloaded base64ct v1.6.0
Downloaded gix-command v0.6.2
Downloaded elliptic-curve v0.13.8
Downloaded derive_more-impl v2.0.1
Downloaded bit-set v0.8.0
Downloaded ctr v0.9.2
Downloaded cpufeatures v0.2.12
Downloaded cipher v0.4.4
Downloaded bytecount v0.6.8
Downloaded bitflags v1.3.2
Downloaded bcrypt-pbkdf v0.10.0
Downloaded hashbrown v0.15.5
Downloaded document-features v0.2.11
Downloaded byteorder v1.5.0
Downloaded block-padding v0.3.3
Downloaded base64 v0.21.7
Downloaded errno v0.3.13
Downloaded anstyle-query v1.0.2
Downloaded gix-features v0.43.1
Downloaded generic-array v0.14.7
Downloaded env_filter v0.1.3
Downloaded crypto-common v0.1.6
Downloaded crypto-bigint v0.5.5
Downloaded colored v2.1.0
Downloaded chacha20 v0.9.1
Downloaded clap_derive v4.5.41
Downloaded ed25519 v1.5.3
Downloaded dyn-clone v1.0.17
Downloaded data-encoding v2.5.0
Downloaded cyphernet v0.5.2
Downloaded const-oid v0.9.6
Downloaded console v0.16.0
Downloaded cfg-if v1.0.0
Downloaded cbc v0.1.2
Downloaded bloomy v1.2.0
Downloaded base32 v0.4.0
Downloaded crossbeam-utils v0.8.19
Downloaded amplify_num v0.5.2
Downloaded git-ref-format v0.5.0
Downloaded getrandom v0.2.15
Downloaded ascii v1.1.0
Downloaded git-ref-format-core v0.5.0
Downloaded fluent-uri v0.3.2
Downloaded base64 v0.22.1
Downloaded anstyle v1.0.11
Downloaded ahash v0.8.11
Downloaded base-x v0.2.11
Downloaded backtrace v0.3.75
Downloaded aho-corasick v1.1.3
Downloaded aes-gcm v0.10.3
Downloaded getrandom v0.3.3
Downloaded arc-swap v1.7.1
Downloaded anstyle-parse v0.2.3
Downloaded amplify_syn v2.0.1
Downloaded anstream v0.6.13
Downloaded aes v0.8.4
Downloaded aead v0.5.2
Downloaded adler2 v2.0.0
Downloaded addr2line v0.24.2
Downloaded anyhow v1.0.82
Downloaded amplify v4.6.0
Downloaded gix-tempfile v18.0.0
Compiling libc v0.2.174
Compiling proc-macro2 v1.0.101
Compiling unicode-ident v1.0.12
Compiling quote v1.0.41
Checking cfg-if v1.0.0
Compiling version_check v0.9.4
Compiling shlex v1.3.0
Checking memchr v2.7.2
Compiling typenum v1.17.0
Compiling syn v2.0.106
Compiling jobserver v0.1.31
Checking getrandom v0.2.15
Compiling generic-array v0.14.7
Compiling cc v1.2.2
Checking rand_core v0.6.4
Compiling serde v1.0.219
Checking aho-corasick v1.1.3
Checking crypto-common v0.1.6
Checking regex-syntax v0.8.5
Checking smallvec v1.15.1
Checking subtle v2.5.0
Checking stable_deref_trait v1.2.0
Checking cpufeatures v0.2.12
Checking once_cell v1.21.3
Compiling parking_lot_core v0.9.12
Checking scopeguard v1.2.0
Checking fastrand v2.3.0
Checking lock_api v0.4.14
Checking block-buffer v0.10.4
Compiling thiserror v2.0.12
Checking digest v0.10.7
Checking parking_lot v0.12.5
Compiling crc32fast v1.5.0
Checking byteorder v1.5.0
Checking regex-automata v0.4.9
Checking tinyvec_macros v0.1.1
Compiling typeid v1.0.3
Checking tinyvec v1.6.0
Checking gix-trace v0.1.13
Checking bitflags v2.9.1
Checking home v0.5.9
Compiling synstructure v0.13.1
Checking unicode-normalization v0.1.23
Checking zlib-rs v0.5.2
Checking gix-utils v0.3.0
Checking same-file v1.0.6
Checking walkdir v2.5.0
Checking prodash v30.0.1
Checking itoa v1.0.11
Checking libz-rs-sys v0.5.2
Compiling heapless v0.8.0
Compiling serde_derive v1.0.219
Compiling thiserror-impl v2.0.12
Checking bstr v1.12.0
Compiling zerofrom-derive v0.1.6
Checking flate2 v1.1.1
Compiling yoke-derive v0.7.5
Checking gix-validate v0.10.0
Checking gix-path v0.10.20
Checking zerofrom v0.1.6
Compiling zerovec-derive v0.10.3
Checking gix-features v0.43.1
Checking yoke v0.7.5
Compiling displaydoc v0.2.5
Checking hash32 v0.3.1
Checking zeroize v1.7.0
Compiling icu_locid_transform_data v1.5.1
Checking litemap v0.7.5
Checking writeable v0.5.5
Checking zerovec v0.10.4
Compiling thiserror v1.0.69
Compiling pkg-config v0.3.30
Checking faster-hex v0.10.0
Compiling thiserror-impl v1.0.69
Compiling icu_provider_macros v1.5.0
Checking tinystr v0.7.6
Checking icu_locid v1.5.0
Compiling rustix v1.0.7
Compiling icu_properties_data v1.5.1
Checking icu_provider v1.5.0
Checking sha1 v0.10.6
Checking block-padding v0.3.3
Compiling icu_normalizer_data v1.5.1
Checking linux-raw-sys v0.9.4
Checking inout v0.1.3
Checking sha1-checked v0.10.0
Checking icu_locid_transform v1.5.0
Checking icu_collections v1.5.0
Compiling syn v1.0.109
Checking erased-serde v0.4.6
Checking serde_fmt v1.0.3
Compiling getrandom v0.3.3
Checking icu_properties v1.5.1
Checking value-bag-serde1 v1.11.1
Checking gix-hash v0.19.0
Checking value-bag v1.11.1
Checking cipher v0.4.4
Checking log v0.4.27
Checking utf16_iter v1.0.5
Checking write16 v1.0.0
Checking utf8_iter v1.0.4
Checking percent-encoding v2.3.1
Checking form_urlencoded v1.2.1
Checking sha2 v0.10.8
Compiling vcpkg v0.2.15
Checking icu_normalizer v1.5.0
Checking idna_adapter v1.2.0
Checking idna v1.0.3
Compiling libz-sys v1.1.16
Checking universal-hash v0.5.1
Checking opaque-debug v0.3.1
Checking url v2.5.4
Checking tempfile v3.23.0
Compiling autocfg v1.2.0
Compiling libgit2-sys v0.17.0+1.8.1
Compiling amplify_syn v2.0.1
Compiling num-traits v0.2.19
Checking hashbrown v0.14.3
Checking signature v1.6.4
Compiling serde_json v1.0.140
Checking equivalent v1.0.1
Checking ed25519 v1.5.3
Checking indexmap v2.2.6
Checking aead v0.5.2
Checking ryu v1.0.17
Checking amplify_num v0.5.2
Compiling ref-cast v1.0.24
Checking ascii v1.1.0
Compiling amplify_derive v4.0.0
Checking ct-codecs v1.1.1
Checking ec25519 v0.1.0
Checking poly1305 v0.8.0
Checking git-ref-format-core v0.5.0
Checking chacha20 v0.9.1
Compiling ref-cast-impl v1.0.24
Checking polyval v0.6.2
Compiling sqlite3-src v0.5.1
Checking amplify v4.6.0
Checking hmac v0.12.1
Checking dyn-clone v1.0.17
Checking keccak v0.1.5
Checking base64ct v1.6.0
Checking cyphergraphy v0.3.0
Checking sha3 v0.10.8
Checking pem-rfc7468 v0.7.0
Checking pbkdf2 v0.12.2
Checking ghash v0.5.1
Checking ctr v0.9.2
Checking aes v0.8.4
Checking rand v0.8.5
Compiling crossbeam-utils v0.8.19
Compiling data-encoding v2.5.0
Checking base32 v0.4.0
Checking cypheraddr v0.4.0
Checking qcheck v1.0.0
Compiling data-encoding-macro-internal v0.1.12
Checking aes-gcm v0.10.3
Checking ssh-encoding v0.2.0
Checking chacha20poly1305 v0.10.1
Checking cbc v0.1.2
Checking blowfish v0.9.1
Compiling serde_derive_internals v0.29.1
Checking data-encoding-macro v0.1.14
Checking bcrypt-pbkdf v0.10.0
Checking ssh-cipher v0.2.0
Checking noise-framework v0.4.0
Compiling schemars_derive v1.0.4
Checking socks5-client v0.4.1
Checking signature v2.2.0
Checking base-x v0.2.11
Checking multibase v0.9.1
Checking ssh-key v0.6.6
Checking cyphernet v0.5.2
Checking crossbeam-channel v0.5.15
Checking schemars v1.0.4
Checking radicle-ssh v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-ssh)
Checking jiff v0.2.15
Checking lazy_static v1.5.0
Checking nonempty v0.9.0
Checking siphasher v1.0.1
Checking anstyle-query v1.0.2
Checking radicle-git-metadata v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-metadata)
Checking radicle-dag v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-dag)
Checking winnow v0.7.13
Checking hashbrown v0.15.5
Checking utf8parse v0.2.1
Checking anstyle-parse v0.2.3
Checking gix-hashtable v0.9.0
Checking radicle-git-ref-format v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-ref-format)
Checking base64 v0.21.7
Checking gix-date v0.10.5
Checking gix-actor v0.35.4
Checking colorchoice v1.0.0
Checking anstyle v1.0.11
Checking iana-time-zone v0.1.60
Checking anstream v0.6.13
Checking chrono v0.4.38
Checking gix-object v0.50.2
Checking colored v2.1.0
Checking serde-untagged v0.1.7
Checking localtime v1.3.1
Checking bytesize v2.0.1
Checking memmap2 v0.9.8
Checking fast-glob v0.3.3
Checking tree-sitter-language v0.1.2
Checking dunce v1.0.5
Checking gix-chunk v0.4.11
Checking gix-fs v0.16.1
Checking gix-commitgraph v0.29.0
Checking gix-tempfile v18.0.0
Checking mio v1.0.4
Checking gix-revwalk v0.21.0
Checking gix-quote v0.6.0
Checking errno v0.3.13
Checking sem_safe v0.2.0
Checking either v1.11.0
Checking shell-words v1.1.0
Checking gix-command v0.6.2
Checking signals_receipts v0.2.0
Compiling object v0.36.7
Compiling signal-hook v0.3.18
Checking gix-lock v18.0.0
Checking gix-url v0.32.0
Checking gix-config-value v0.15.1
Checking gix-sec v0.12.0
Checking signal-hook-registry v1.4.5
Compiling rustix v0.38.34
Checking gimli v0.31.1
Checking adler2 v2.0.0
Checking miniz_oxide v0.8.8
Checking gix-prompt v0.11.1
Checking addr2line v0.24.2
Checking radicle-signals v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-signals)
Checking gix-traverse v0.47.0
Checking gix-revision v0.35.0
Checking gix-diff v0.53.0
Checking mio v0.8.11
Checking gix-packetline v0.19.1
Compiling tree-sitter v0.24.4
Compiling linux-raw-sys v0.4.13
Checking rustc-demangle v0.1.26
Compiling anyhow v1.0.82
Compiling unicode-segmentation v1.11.0
Compiling convert_case v0.7.1
Checking backtrace v0.3.75
Checking gix-transport v0.48.0
Checking signal-hook-mio v0.2.4
Checking gix-pack v0.60.0
Checking gix-refspec v0.31.0
Checking gix-credentials v0.30.0
Checking gix-ref v0.53.1
Checking gix-shallow v0.5.0
Checking gix-negotiate v0.21.0
Checking regex v1.11.1
Compiling maybe-async v0.2.10
Compiling proc-macro-error-attr2 v2.0.0
Checking arc-swap v1.7.1
Compiling portable-atomic v1.11.0
Checking gix-odb v0.70.0
Checking gix-protocol v0.51.0
Compiling proc-macro-error2 v2.0.1
Compiling xattr v1.3.1
Compiling derive_more-impl v2.0.1
Compiling filetime v0.2.23
Checking uuid v1.16.0
Checking bytes v1.10.1
Checking bitflags v1.3.2
Checking unicode-width v0.2.1
Compiling litrs v0.4.1
Checking console v0.16.0
Compiling document-features v0.2.11
Checking crossterm v0.25.0
Checking derive_more v2.0.1
Compiling tar v0.4.40
Compiling git-ref-format-macro v0.5.0
Checking newline-converter v0.3.0
Checking snapbox-macros v0.3.8
Checking salsa20 v0.10.2
Checking fxhash v0.2.1
Checking clap_lex v0.7.5
Compiling heck v0.5.0
Checking streaming-iterator v0.1.9
Checking normalize-line-endings v0.3.0
Checking similar v2.5.0
Checking siphasher v0.3.11
Checking unicode-width v0.1.11
Checking unit-prefix v0.5.1
Checking sqlite3-sys v0.15.2
Checking strsim v0.11.1
Checking indicatif v0.18.0
Checking sqlite v0.32.0
Checking clap_builder v4.5.44
Checking radicle-crypto v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-crypto)
Checking snapbox v0.4.17
Checking inquire v0.7.5
Checking bloomy v1.2.0
Compiling clap_derive v4.5.41
Checking scrypt v0.11.0
Compiling radicle-surf v0.25.0
Checking git-ref-format v0.5.0
Checking crossterm v0.29.0
Checking unicode-display-width v0.3.0
Checking systemd-journal-logger v2.2.2
Checking toml_datetime v0.7.0
Checking serde_spanned v1.0.0
Compiling tree-sitter-go v0.23.4
Compiling tree-sitter-c v0.23.2
Compiling tree-sitter-rust v0.23.2
Compiling tree-sitter-json v0.24.8
Compiling tree-sitter-python v0.23.4
Compiling tree-sitter-ruby v0.23.1
Compiling tree-sitter-css v0.23.1
Compiling tree-sitter-md v0.3.2
Compiling tree-sitter-bash v0.23.3
Compiling tree-sitter-toml-ng v0.6.0
Compiling tree-sitter-typescript v0.23.2
Compiling tree-sitter-html v0.23.2
Checking radicle-std-ext v0.1.1
Checking toml_writer v1.0.2
Checking pin-project-lite v0.2.16
Checking tokio v1.47.1
Checking toml v0.9.5
Checking clap v4.5.44
Checking os_info v3.12.0
Compiling radicle-cli v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli)
Checking diff v0.1.13
Compiling radicle-node v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-node)
Checking yansi v0.5.1
Checking pretty_assertions v1.4.0
Checking human-panic v2.0.3
Checking clap_complete v4.5.60
Checking structured-logger v1.0.4
Checking radicle-systemd v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-systemd)
Checking tree-sitter-highlight v0.24.4
Checking itertools v0.14.0
Checking num-integer v0.1.46
Compiling qcheck-macros v1.0.0
Checking socket2 v0.5.7
Compiling escargot v0.5.10
Checking lexopt v0.3.0
Checking timeago v0.4.2
Checking num-bigint v0.4.6
Compiling ahash v0.8.11
Checking num-iter v0.1.45
Checking num-complex v0.4.6
Checking env_filter v0.1.3
Checking borrow-or-share v0.2.2
Checking zerocopy v0.7.35
Checking num-rational v0.4.2
Checking bit-vec v0.8.0
Checking num v0.4.3
Checking bit-set v0.8.0
Checking fluent-uri v0.3.2
Checking env_logger v0.11.8
Checking phf_shared v0.11.3
Compiling test-log-macros v0.2.18
Compiling paste v1.0.15
Compiling radicle-remote-helper v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-remote-helper)
Checking vsimd v0.8.0
Checking outref v0.5.2
Checking test-log v0.2.18
Checking referencing v0.30.0
Checking uuid-simd v0.8.0
Checking phf v0.11.3
Checking fancy-regex v0.14.0
Checking fraction v0.15.3
Checking email_address v0.2.9
Checking bytecount v0.6.8
Checking base64 v0.22.1
Checking num-cmp v0.1.0
Checking emojis v0.6.4
Checking jsonschema v0.30.0
Checking git2 v0.19.0
Checking radicle-oid v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-oid)
Checking radicle-git-ext v0.10.0
Checking radicle-term v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-term)
Checking radicle-cob v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cob)
Checking radicle v0.20.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle)
Checking radicle-fetch v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-fetch)
Checking radicle-cli-test v0.13.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli-test)
Checking radicle-protocol v0.4.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-protocol)
Checking radicle-schemars v0.6.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-schemars)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 41.97s
+ cargo build --all-targets --workspace
Compiling libc v0.2.174
Compiling cfg-if v1.0.0
Compiling shlex v1.3.0
Compiling memchr v2.7.2
Compiling typenum v1.17.0
Compiling serde v1.0.219
Compiling generic-array v0.14.7
Compiling jobserver v0.1.31
Compiling getrandom v0.2.15
Compiling rand_core v0.6.4
Compiling cc v1.2.2
Compiling aho-corasick v1.1.3
Compiling crypto-common v0.1.6
Compiling regex-syntax v0.8.5
Compiling smallvec v1.15.1
Compiling subtle v2.5.0
Compiling stable_deref_trait v1.2.0
Compiling cpufeatures v0.2.12
Compiling once_cell v1.21.3
Compiling scopeguard v1.2.0
Compiling fastrand v2.3.0
Compiling regex-automata v0.4.9
Compiling lock_api v0.4.14
Compiling parking_lot_core v0.9.12
Compiling block-buffer v0.10.4
Compiling parking_lot v0.12.5
Compiling digest v0.10.7
Compiling thiserror v2.0.12
Compiling bitflags v2.9.1
Compiling byteorder v1.5.0
Compiling tinyvec_macros v0.1.1
Compiling tinyvec v1.6.0
Compiling crc32fast v1.5.0
Compiling gix-trace v0.1.13
Compiling typeid v1.0.3
Compiling home v0.5.9
Compiling erased-serde v0.4.6
Compiling serde_fmt v1.0.3
Compiling zlib-rs v0.5.2
Compiling unicode-normalization v0.1.23
Compiling value-bag-serde1 v1.11.1
Compiling same-file v1.0.6
Compiling walkdir v2.5.0
Compiling gix-utils v0.3.0
Compiling zerofrom v0.1.6
Compiling value-bag v1.11.1
Compiling prodash v30.0.1
Compiling bstr v1.12.0
Compiling log v0.4.27
Compiling gix-validate v0.10.0
Compiling gix-path v0.10.20
Compiling yoke v0.7.5
Compiling zerovec v0.10.4
Compiling itoa v1.0.11
Compiling hash32 v0.3.1
Compiling zeroize v1.7.0
Compiling heapless v0.8.0
Compiling tinystr v0.7.6
Compiling writeable v0.5.5
Compiling litemap v0.7.5
Compiling libz-rs-sys v0.5.2
Compiling icu_locid v1.5.0
Compiling flate2 v1.1.1
Compiling gix-features v0.43.1
Compiling faster-hex v0.10.0
Compiling icu_provider v1.5.0
Compiling icu_locid_transform_data v1.5.1
Compiling sha1 v0.10.6
Compiling block-padding v0.3.3
Compiling linux-raw-sys v0.9.4
Compiling sha1-checked v0.10.0
Compiling thiserror v1.0.69
Compiling inout v0.1.3
Compiling icu_locid_transform v1.5.0
Compiling icu_properties_data v1.5.1
Compiling icu_collections v1.5.0
Compiling gix-hash v0.19.0
Compiling cipher v0.4.4
Compiling icu_properties v1.5.1
Compiling icu_normalizer_data v1.5.1
Compiling rustix v1.0.7
Compiling write16 v1.0.0
Compiling utf16_iter v1.0.5
Compiling utf8_iter v1.0.4
Compiling getrandom v0.3.3
Compiling percent-encoding v2.3.1
Compiling form_urlencoded v1.2.1
Compiling sha2 v0.10.8
Compiling libz-sys v1.1.16
Compiling universal-hash v0.5.1
Compiling opaque-debug v0.3.1
Compiling libgit2-sys v0.17.0+1.8.1
Compiling hashbrown v0.14.3
Compiling equivalent v1.0.1
Compiling signature v1.6.4
Compiling ed25519 v1.5.3
Compiling icu_normalizer v1.5.0
Compiling idna_adapter v1.2.0
Compiling idna v1.0.3
Compiling url v2.5.4
Compiling tempfile v3.23.0
Compiling indexmap v2.2.6
Compiling aead v0.5.2
Compiling ct-codecs v1.1.1
Compiling ryu v1.0.17
Compiling amplify_num v0.5.2
Compiling ascii v1.1.0
Compiling serde_json v1.0.140
Compiling ec25519 v0.1.0
Compiling num-traits v0.2.19
Compiling amplify v4.6.0
Compiling poly1305 v0.8.0
Compiling chacha20 v0.9.1
Compiling git-ref-format-core v0.5.0
Compiling ref-cast v1.0.24
Compiling cyphergraphy v0.3.0
Compiling polyval v0.6.2
Compiling sqlite3-src v0.5.1
Compiling hmac v0.12.1
Compiling base64ct v1.6.0
Compiling dyn-clone v1.0.17
Compiling keccak v0.1.5
Compiling sha3 v0.10.8
Compiling pem-rfc7468 v0.7.0
Compiling pbkdf2 v0.12.2
Compiling ghash v0.5.1
Compiling aes v0.8.4
Compiling ctr v0.9.2
Compiling rand v0.8.5
Compiling base32 v0.4.0
Compiling qcheck v1.0.0
Compiling cypheraddr v0.4.0
Compiling aes-gcm v0.10.3
Compiling ssh-encoding v0.2.0
Compiling chacha20poly1305 v0.10.1
Compiling blowfish v0.9.1
Compiling cbc v0.1.2
Compiling data-encoding v2.5.0
Compiling data-encoding-macro v0.1.14
Compiling ssh-cipher v0.2.0
Compiling bcrypt-pbkdf v0.10.0
Compiling noise-framework v0.4.0
Compiling socks5-client v0.4.1
Compiling crossbeam-utils v0.8.19
Compiling base-x v0.2.11
Compiling signature v2.2.0
Compiling ssh-key v0.6.6
Compiling crossbeam-channel v0.5.15
Compiling multibase v0.9.1
Compiling cyphernet v0.5.2
Compiling schemars v1.0.4
Compiling radicle-ssh v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-ssh)
Compiling jiff v0.2.15
Compiling lazy_static v1.5.0
Compiling nonempty v0.9.0
Compiling anstyle-query v1.0.2
Compiling siphasher v1.0.1
Compiling radicle-git-metadata v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-metadata)
Compiling radicle-dag v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-dag)
Compiling winnow v0.7.13
Compiling utf8parse v0.2.1
Compiling hashbrown v0.15.5
Compiling gix-hashtable v0.9.0
Compiling anstyle-parse v0.2.3
Compiling gix-date v0.10.5
Compiling gix-actor v0.35.4
Compiling radicle-git-ref-format v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-ref-format)
Compiling iana-time-zone v0.1.60
Compiling colorchoice v1.0.0
Compiling base64 v0.21.7
Compiling anstyle v1.0.11
Compiling anstream v0.6.13
Compiling chrono v0.4.38
Compiling gix-object v0.50.2
Compiling colored v2.1.0
Compiling serde-untagged v0.1.7
Compiling bytesize v2.0.1
Compiling localtime v1.3.1
Compiling memmap2 v0.9.8
Compiling tree-sitter-language v0.1.2
Compiling fast-glob v0.3.3
Compiling dunce v1.0.5
Compiling gix-chunk v0.4.11
Compiling adler2 v2.0.0
Compiling gix-commitgraph v0.29.0
Compiling gix-fs v0.16.1
Compiling gix-revwalk v0.21.0
Compiling gix-tempfile v18.0.0
Compiling mio v1.0.4
Compiling gix-quote v0.6.0
Compiling sem_safe v0.2.0
Compiling errno v0.3.13
Compiling unicode-segmentation v1.11.0
Compiling shell-words v1.1.0
Compiling either v1.11.0
Compiling gix-command v0.6.2
Compiling signals_receipts v0.2.0
Compiling gix-lock v18.0.0
Compiling gix-url v0.32.0
Compiling gix-config-value v0.15.1
Compiling gix-sec v0.12.0
Compiling signal-hook-registry v1.4.5
Compiling gimli v0.31.1
Compiling signal-hook v0.3.18
Compiling gix-prompt v0.11.1
Compiling object v0.36.7
Compiling addr2line v0.24.2
Compiling radicle-signals v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-signals)
Compiling gix-traverse v0.47.0
Compiling gix-revision v0.35.0
Compiling miniz_oxide v0.8.8
Compiling gix-diff v0.53.0
Compiling gix-packetline v0.19.1
Compiling mio v0.8.11
Compiling tree-sitter v0.24.4
Compiling rustc-demangle v0.1.26
Compiling backtrace v0.3.75
Compiling rustix v0.38.34
Compiling signal-hook-mio v0.2.4
Compiling gix-transport v0.48.0
Compiling sqlite3-sys v0.15.2
Compiling sqlite v0.32.0
Compiling radicle-crypto v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-crypto)
Compiling gix-pack v0.60.0
Compiling gix-refspec v0.31.0
Compiling gix-credentials v0.30.0
Compiling gix-shallow v0.5.0
Compiling gix-ref v0.53.1
Compiling convert_case v0.7.1
Compiling gix-negotiate v0.21.0
Compiling regex v1.11.1
Compiling arc-swap v1.7.1
Compiling gix-protocol v0.51.0
Compiling gix-odb v0.70.0
Compiling derive_more-impl v2.0.1
Compiling xattr v1.3.1
Compiling uuid v1.16.0
Compiling filetime v0.2.23
Compiling bytes v1.10.1
Compiling unicode-width v0.2.1
Compiling bitflags v1.3.2
Compiling crossterm v0.25.0
Compiling console v0.16.0
Compiling git-ref-format-macro v0.5.0
Compiling tar v0.4.40
Compiling derive_more v2.0.1
Compiling portable-atomic v1.11.0
Compiling anyhow v1.0.82
Compiling newline-converter v0.3.0
Compiling snapbox-macros v0.3.8
Compiling salsa20 v0.10.2
Compiling fxhash v0.2.1
Compiling similar v2.5.0
Compiling siphasher v0.3.11
Compiling unicode-width v0.1.11
Compiling streaming-iterator v0.1.9
Compiling clap_lex v0.7.5
Compiling normalize-line-endings v0.3.0
Compiling unit-prefix v0.5.1
Compiling strsim v0.11.1
Compiling indicatif v0.18.0
Compiling clap_builder v4.5.44
Compiling snapbox v0.4.17
Compiling bloomy v1.2.0
Compiling inquire v0.7.5
Compiling scrypt v0.11.0
Compiling radicle-surf v0.25.0
Compiling crossterm v0.29.0
Compiling git-ref-format v0.5.0
Compiling unicode-display-width v0.3.0
Compiling systemd-journal-logger v2.2.2
Compiling serde_spanned v1.0.0
Compiling toml_datetime v0.7.0
Compiling tree-sitter-bash v0.23.3
Compiling tree-sitter-md v0.3.2
Compiling tree-sitter-ruby v0.23.1
Compiling tree-sitter-go v0.23.4
Compiling tree-sitter-json v0.24.8
Compiling tree-sitter-typescript v0.23.2
Compiling tree-sitter-css v0.23.1
Compiling tree-sitter-html v0.23.2
Compiling tree-sitter-toml-ng v0.6.0
Compiling tree-sitter-rust v0.23.2
Compiling tree-sitter-c v0.23.2
Compiling tree-sitter-python v0.23.4
Compiling pin-project-lite v0.2.16
Compiling radicle-std-ext v0.1.1
Compiling toml_writer v1.0.2
Compiling toml v0.9.5
Compiling tokio v1.47.1
Compiling clap v4.5.44
Compiling os_info v3.12.0
Compiling radicle-cli v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli)
Compiling yansi v0.5.1
Compiling diff v0.1.13
Compiling radicle-node v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-node)
Compiling pretty_assertions v1.4.0
Compiling human-panic v2.0.3
Compiling clap_complete v4.5.60
Compiling structured-logger v1.0.4
Compiling radicle-systemd v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-systemd)
Compiling tree-sitter-highlight v0.24.4
Compiling itertools v0.14.0
Compiling num-integer v0.1.46
Compiling socket2 v0.5.7
Compiling lexopt v0.3.0
Compiling timeago v0.4.2
Compiling escargot v0.5.10
Compiling num-bigint v0.4.6
Compiling num-iter v0.1.45
Compiling num-complex v0.4.6
Compiling env_filter v0.1.3
Compiling num-rational v0.4.2
Compiling borrow-or-share v0.2.2
Compiling bit-vec v0.8.0
Compiling zerocopy v0.7.35
Compiling bit-set v0.8.0
Compiling fluent-uri v0.3.2
Compiling ahash v0.8.11
Compiling num v0.4.3
Compiling env_logger v0.11.8
Compiling phf_shared v0.11.3
Compiling radicle-remote-helper v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-remote-helper)
Compiling outref v0.5.2
Compiling vsimd v0.8.0
Compiling test-log v0.2.18
Compiling referencing v0.30.0
Compiling phf v0.11.3
Compiling uuid-simd v0.8.0
Compiling fraction v0.15.3
Compiling fancy-regex v0.14.0
Compiling email_address v0.2.9
Compiling git2 v0.19.0
Compiling num-cmp v0.1.0
Compiling base64 v0.22.1
Compiling bytecount v0.6.8
Compiling jsonschema v0.30.0
Compiling emojis v0.6.4
Compiling radicle-oid v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-oid)
Compiling radicle-git-ext v0.10.0
Compiling radicle-cob v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cob)
Compiling radicle v0.20.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle)
Compiling radicle-term v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-term)
Compiling radicle-fetch v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-fetch)
Compiling radicle-protocol v0.4.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-protocol)
Compiling radicle-cli-test v0.13.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli-test)
Compiling radicle-schemars v0.6.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-schemars)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 53.50s
+ cargo doc --workspace --no-deps --all-features
Checking regex-automata v0.4.9
Compiling syn v1.0.109
Checking idna v1.0.3
Checking radicle-ssh v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-ssh)
Compiling num-traits v0.2.19
Checking url v2.5.4
Checking radicle-git-metadata v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-metadata)
Checking git2 v0.19.0
Checking radicle-dag v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-dag)
Checking chrono v0.4.38
Checking bstr v1.12.0
Compiling amplify_syn v2.0.1
Checking gix-validate v0.10.0
Checking git-ref-format-core v0.5.0
Checking gix-path v0.10.20
Checking gix-features v0.43.1
Compiling data-encoding-macro-internal v0.1.12
Checking gix-date v0.10.5
Compiling amplify_derive v4.0.0
Checking gix-hash v0.19.0
Checking radicle-git-ref-format v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-ref-format)
Checking gix-actor v0.35.4
Checking data-encoding-macro v0.1.14
Checking multibase v0.9.1
Checking gix-hashtable v0.9.0
Checking radicle-oid v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-oid)
Checking gix-object v0.50.2
Checking gix-commitgraph v0.29.0
Checking gix-fs v0.16.1
Checking gix-quote v0.6.0
Checking gix-tempfile v18.0.0
Checking gix-command v0.6.2
Checking gix-lock v18.0.0
Checking gix-url v0.32.0
Checking gix-config-value v0.15.1
Checking radicle-signals v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-signals)
Checking gix-packetline v0.19.1
Checking gix-prompt v0.11.1
Checking gix-revwalk v0.21.0
Checking gix-diff v0.53.0
Checking regex v1.11.1
Checking gix-traverse v0.47.0
Checking gix-revision v0.35.0
Checking gix-pack v0.60.0
Checking gix-refspec v0.31.0
Checking tree-sitter v0.24.4
Checking amplify v4.6.0
Checking gix-negotiate v0.21.0
Checking gix-transport v0.48.0
Checking gix-credentials v0.30.0
Checking cyphergraphy v0.3.0
Checking gix-ref v0.53.1
Checking git-ref-format v0.5.0
Checking cypheraddr v0.4.0
Checking noise-framework v0.4.0
Checking socks5-client v0.4.1
Checking gix-shallow v0.5.0
Checking cyphernet v0.5.2
Checking radicle-git-ext v0.10.0
Checking radicle-crypto v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-crypto)
Checking gix-odb v0.70.0
Checking uuid v1.16.0
Checking gix-protocol v0.51.0
Checking radicle-cob v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cob)
Compiling radicle-cli v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli)
Checking human-panic v2.0.3
Checking radicle-surf v0.25.0
Checking tree-sitter-toml-ng v0.6.0
Checking radicle-term v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-term)
Checking tree-sitter-highlight v0.24.4
Checking radicle v0.20.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle)
Checking radicle-systemd v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-systemd)
Documenting radicle-systemd v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-systemd)
Documenting radicle v0.20.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle)
Documenting radicle-cob v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cob)
Documenting radicle-crypto v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-crypto)
Documenting radicle-term v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-term)
Documenting radicle-signals v0.11.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-signals)
Documenting radicle-oid v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-oid)
Documenting radicle-git-ref-format v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-ref-format)
Documenting radicle-git-metadata v0.1.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-git-metadata)
Documenting radicle-ssh v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-ssh)
Documenting radicle-dag v0.10.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-dag)
Checking radicle-fetch v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-fetch)
Documenting radicle-cli v0.17.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli)
Documenting radicle-fetch v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-fetch)
Checking radicle-protocol v0.4.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-protocol)
Documenting radicle-protocol v0.4.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-protocol)
Documenting radicle-node v0.16.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-node)
Documenting radicle-schemars v0.6.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-schemars)
Documenting radicle-cli-test v0.13.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli-test)
Documenting radicle-remote-helper v0.14.0 (/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-remote-helper)
Finished `dev` profile [unoptimized + debuginfo] target(s) in 9.15s
Generated /eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/doc/radicle/index.html and 18 other files
+ cargo test --workspace --no-fail-fast
Finished `test` profile [unoptimized + debuginfo] target(s) in 0.15s
Running unittests src/lib.rs (target/debug/deps/radicle-3953555018688ae4)
running 239 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_migrate_to ... ok
test cob::cache::tests::test_check_version ... ok
test cob::cache::migrations::_2::tests::test_patch_json_deserialization ... ok
test cob::common::test::test_color ... ok
test cob::common::test::test_emojis ... ok
test cob::common::test::test_title ... ok
test cob::cache::migrations::_2::tests::test_migration_2 ... 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_remove_delegate_concurrent ... ok
test cob::identity::test::test_identity_update_rejected ... ok
test cob::identity::test::test_identity_reject_concurrent ... 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 ... ok
test cob::issue::test::test_invalid_tx_reference ... 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_comment_redact ... ok
test cob::issue::test::test_issue_create_and_assign ... ok
test cob::issue::test::test_issue_create_and_get ... ok
test cob::issue::test::test_issue_create_and_change_state ... ok
test cob::issue::test::test_issue_create_and_reassign ... ok
test cob::issue::test::test_issue_edit ... ok
test cob::issue::test::test_issue_create_and_unassign ... 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_label ... ok
test cob::issue::test::test_issue_react ... ok
test cob::patch::cache::tests::test_get ... ok
test cob::patch::cache::tests::test_is_empty ... ok
test cob::patch::cache::tests::test_find_by_revision ... ok
test cob::patch::cache::tests::test_list_by_status ... ok
test cob::issue::test::test_issue_reply ... 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_list ... ok
test cob::patch::test::test_patch_create_and_get ... ok
test cob::patch::cache::tests::test_counts ... ok
test cob::patch::test::test_patch_discussion ... ok
test cob::patch::cache::tests::test_remove ... 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_duplicate ... ok
test cob::patch::test::test_patch_review ... ok
test cob::patch::test::test_patch_review_edit ... ok
test cob::patch::test::test_patch_review_remove_summary ... 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::patch::test::test_patch_review_edit_comment ... ok
test cob::stream::tests::test_all_from ... ok
test cob::patch::test::test_patch_review_revision_redact ... ok
test cob::stream::tests::test_all_from_until ... ok
test cob::stream::tests::test_all_until ... ok
test cob::stream::tests::test_regression_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::thread::tests::test_edit_comment ... ok
test cob::stream::tests::test_from_until ... 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 cob::patch::test::test_patch_update ... 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::rules::tests::test_canonical ... ok
test git::canonical::tests::test_quorum_different_types ... 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 identity::doc::id::test::prop_from_str ... ok
test git::canonical::rules::tests::test_rule_validate_failures ... 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::partial ... 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 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_prune ... ok
test node::routing::test::test_remove_many ... ok
test git::canonical::tests::test_quorum_properties ... 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::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_timed_out ... ok
test node::sync::announce::test::announcer_with_replication_factor_zero_and_preferred_seeds ... 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_synced_set ... ok
test node::sync::announce::test::local_node_in_preferred_seeds ... 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::synced_with_local_node_is_ignored ... 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::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::all_nodes_are_fetchable ... ok
test node::sync::fetch::test::ignores_duplicates_and_local_node ... 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::test::ensure_replicas_construction ... ok
test node::sync::test::replicas_constrain_to ... 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 node::sync::fetch::test::reaches_target_of_replicas ... 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 serde_ext::test::test_localtime ... ok
test serde_ext::test::test_localtime_ext ... ok
test rad::tests::test_init ... ok
test storage::git::tests::test_references_of ... ok
test profile::config::test::schema ... 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::refs::tests::prop_canonical_roundtrip ... ok
test storage::git::tests::test_sign_refs ... 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::git::tests::test_remote_refs ... ok
test storage::refs::tests::test_rid_verification ... ok
test identity::doc::test::prop_encode_decode ... ok
test result: ok. 239 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 1.76s
Running unittests src/lib.rs (target/debug/deps/radicle_cli-ecc1df71f0ea5e73)
running 46 tests
test commands::block::args::test::should_not_parse ... 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::block::args::test::should_parse_rid ... ok
test commands::cob::args::test::should_allow_log_json_format ... ok
test commands::cob::args::test::should_allow_log_pretty_format ... ok
test commands::cob::args::test::should_allow_show_json_format ... ok
test commands::cob::args::test::should_not_allow_show_pretty_format ... ok
test commands::cob::args::test::should_allow_update_json_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::fork::args::test::should_parse_rid_urn ... ok
test commands::cob::args::test::should_not_allow_update_pretty_format ... ok
test commands::id::args::test::should_not_parse_into_payload - should panic ... ok
test commands::id::args::test::should_not_clobber_payload_args ... ok
test commands::id::args::test::should_not_parse_single_payload ... ok
test commands::id::args::test::should_parse_into_payload ... ok
test commands::id::args::test::should_not_parse_single_payloads ... ok
test commands::init::args::test::should_not_parse_rid_url ... ok
test commands::id::args::test::should_parse_multiple_payloads ... ok
test commands::init::args::test::should_parse_rid_non_urn ... ok
test commands::id::args::test::should_parse_single_payload ... ok
test commands::inspect::test::test_tree ... ok
test commands::patch::review::builder::tests::test_review_comments_before ... ok
test commands::init::args::test::should_parse_rid_urn ... ok
test commands::patch::review::builder::tests::test_review_comments_basic ... ok
test commands::patch::review::builder::tests::test_review_comments_split_hunk ... ok
test commands::patch::review::builder::tests::test_review_comments_multiline ... ok
test commands::publish::args::test::should_parse_rid_non_urn ... ok
test commands::publish::args::test::should_not_parse_rid_url ... ok
test git::pretty_diff::test::test_pretty ... ignored
test commands::publish::args::test::should_parse_rid_urn ... ok
test git::unified_diff::test::test_diff_content_encode_decode_content ... ok
test commands::watch::args::test::should_parse_ref_str ... ok
test git::ddiff::tests::diff_encode_decode_ddiff_hunk ... 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_bytes ... ok
test terminal::format::test::test_strip_comments ... ok
test git::unified_diff::test::test_diff_encode_decode_diff ... ok
test terminal::patch::test::test_edit_display_message ... 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-7a745e3b233df18f)
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-2324a53f2ec1db87)
running 110 tests
test framework_home ... ok
test git_push_amend ... ok
test git_push_and_fetch ... ok
test git_push_canonical_annotated_tags ... ok
test git_push_canonical_lightweight_tags ... ok
test git_push_force_with_lease ... ok
test git_push_diverge ... ok
test rad_auth ... ok
test rad_auth_errors ... ok
test rad_block ... ok
test git_push_rollback ... ok
test rad_checkout ... ok
test git_tag ... ok
test git_push_converge ... ok
test rad_clone ... ok
test rad_clone_bare ... ok
test rad_clean ... ok
test rad_clone_connect ... ok
test rad_clone_unknown ... ok
test rad_clone_all ... ok
test rad_clone_directory ... ok
test rad_cob_multiset ... ok
test rad_cob_log ... ok
test rad_cob_migrate ... ok
test rad_clone_partial_fail ... ok
test rad_cob_operations ... ok
test rad_config ... ok
test rad_cob_update ... ok
test rad_cob_show ... ok
test rad_cob_update_identity ... ok
test rad_help ... ok
test rad_diff ... ok
test rad_id_collaboration ... ignored, slow
test rad_id ... ok
test rad_id_conflict ... ok
test rad_id_private ... ok
test rad_id_threshold ... ok
test rad_id_multi_delegate ... ok
test rad_id_unknown_field ... ok
test rad_id_threshold_soft_fork ... ok
test rad_id_update_delete_field ... ok
test rad_init ... ignored, part of many other tests
test rad_init_bare ... ok
test rad_init_detached_head ... ok
test rad_init_existing ... ok
test rad_init_existing_bare ... ok
test rad_init_no_git ... ok
test rad_init_no_seed ... ok
test rad_init_private ... ok
test rad_fetch ... ok
test rad_fork ... ok
test rad_inbox ... ok
test rad_init_private_no_seed ... ok
test rad_init_private_clone ... ok
test rad_init_sync_not_connected ... ok
test rad_init_private_seed ... ok
test rad_init_private_clone_seed ... ok
test rad_init_with_existing_remote ... ok
test rad_init_sync_preferred ... ok
test rad_inspect ... ok
test rad_issue ... ok
test rad_jj_bare ... ok
test rad_jj_colocated_patch ... ok
test rad_key_mismatch ... ok
test rad_issue_list ... ok
test rad_merge_no_ff ... ok
test rad_merge_after_update ... ok
test rad_merge_via_push ... ok
test rad_node_connect ... ok
test rad_node_connect_without_address ... ok
test rad_patch ... ok
test rad_node ... ok
test rad_patch_ahead_behind ... ok
test rad_patch_change_base ... ok
test rad_patch_checkout ... ok
test rad_patch_checkout_revision ... ok
test rad_patch_checkout_force ... ok
test rad_init_sync_and_clone ... ok
test rad_patch_detached_head ... ok
test rad_patch_diff ... ok
test rad_init_sync_timeout ... ok
test rad_patch_draft ... ok
test rad_patch_edit ... ok
test rad_patch_fetch_2 ... FAILED
test rad_patch_merge_draft ... ok
test rad_patch_fetch_1 ... ok
test rad_patch_revert_merge ... ok
test rad_patch_delete ... ok
test rad_patch_open_explore ... ok
test rad_patch_update ... ok
test rad_patch_via_push ... FAILED
test rad_publish ... ok
test rad_review_by_hunk ... ok
test rad_seed_and_follow ... ok
test rad_remote ... ok
test rad_self ... ok
test rad_seed_many ... ok
test rad_sync_without_node ... ok
test rad_push_and_pull_patches ... ok
test rad_unseed ... ok
test rad_warn_old_nodes ... ok
test rad_unseed_many ... ok
test rad_watch ... ok
test rad_sync ... ok
test test_clone_without_seeds ... ok
test test_cob_deletion ... ok
test test_cob_replication ... ok
test rad_workflow ... ok
test rad_patch_pull_update ... ok
test test_replication_via_seed ... ok
failures:
---- rad_patch_fetch_2 stdout ----
1771604952 test: Using PATH ["/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli/target/debug", "/usr/local/cargo/bin", "/usr/local/sbin", "/usr/local/bin", "/usr/sbin", "/usr/bin", "/sbin", "/bin", "/tmp/.tmpo6MmtZ/alice/work"]
1771604952 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["init", "--name", "heartwood", "--description", "Radicle Heartwood Protocol & Stack", "--no-confirm", "--public", "-v"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["init"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["ls"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["node", "inventory"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: Using PATH ["/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli/target/debug", "/usr/local/cargo/bin", "/usr/local/sbin", "/usr/local/bin", "/usr/sbin", "/usr/bin", "/sbin", "/bin", "/tmp/.tmpo6MmtZ/alice/work"]
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["checkout", "-b", "alice/1", "-q"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["commit", "--allow-empty", "-m", "Changes #1", "-q"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["push", "rad", "-o", "patch.message=Changes", "HEAD:refs/patches"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["checkout", "master", "-q"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["branch", "-D", "alice/1", "-q"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["update-ref", "-d", "refs/remotes/rad/alice/1"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["update-ref", "-d", "refs/remotes/rad/patches/5e2dedcc5d515fcbc1cca483d3376609fe889bfb"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["gc", "--prune=now"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["branch", "-r"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["pull"] in `/tmp/.tmpo6MmtZ/alice/work`..
1771604952 test: rad-patch-fetch-2.md: Running `git` with ["branch", "-r"] in `/tmp/.tmpo6MmtZ/alice/work`..
thread 'rad_patch_fetch_2' panicked at crates/radicle-cli-test/src/lib.rs:503: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
---- rad_patch_via_push stdout ----
1771604953 test: Using PATH ["/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli/target/debug", "/usr/local/cargo/bin", "/usr/local/sbin", "/usr/local/bin", "/usr/sbin", "/usr/bin", "/sbin", "/bin", "/tmp/.tmpn6Vwy1/alice/work"]
1771604953 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["init", "--name", "heartwood", "--description", "Radicle Heartwood Protocol & Stack", "--no-confirm", "--public", "-v"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["init"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["ls"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-init.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["node", "inventory"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: Using PATH ["/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/crates/radicle-cli/target/debug", "/usr/local/cargo/bin", "/usr/local/sbin", "/usr/local/bin", "/usr/sbin", "/usr/bin", "/sbin", "/bin", "/tmp/.tmpn6Vwy1/alice/work"]
1771604953 test: rad-patch-via-push.md: Running `git` with ["checkout", "-b", "feature/1"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["commit", "-a", "-m", "Add things", "-q", "--allow-empty"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["push", "-o", "patch.message=Add things #1", "-o", "patch.message=See commits for details.", "rad", "HEAD:refs/patches"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `/eaf55967-fde1-44f9-b51b-a0e811503bc7/w/target/debug/rad` with ["patch", "show", "6035d2f582afbe01ff23ea87528ae523d76875b6"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["branch", "-vv"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["status", "--short", "--branch"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["fetch"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["push"] in `/tmp/.tmpn6Vwy1/alice/work`..
1771604953 test: rad-patch-via-push.md: Running `git` with ["show-ref"] in `/tmp/.tmpn6Vwy1/alice/work`..
thread 'rad_patch_via_push' panicked at crates/radicle-cli-test/src/lib.rs:503: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:
rad_patch_fetch_2
rad_patch_via_push
test result: FAILED. 106 passed; 2 failed; 2 ignored; 0 measured; 0 filtered out; finished in 68.62s
error: test failed, to rerun pass `-p radicle-cli --test commands`
Running unittests src/lib.rs (target/debug/deps/radicle_cli_test-a59a60940608d3d7)
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.01s
Running unittests src/lib.rs (target/debug/deps/radicle_cob-f2b3d6c5d7c0ddda)
running 8 tests
test object::tests::test_serde ... ok
test tests::git::roundtrip ... ok
test tests::git::update_cob ... ok
test tests::git::traverse_cobs ... ok
test type_name::test::valid_typenames ... ok
test tests::git::list_cobs ... ok
test tests::invalid_parse_refstr ... ok
test tests::parse_refstr ... ok
test result: ok. 8 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.01s
Running unittests src/lib.rs (target/debug/deps/radicle_crypto-c5b263bc947bec6c)
running 12 tests
test ssh::fmt::test::test_key ... ok
test ssh::fmt::test::test_fingerprint ... ok
test ssh::test::prop_encode_decode_sk ... ok
test ssh::keystore::tests::test_init_no_passphrase ... ok
test ssh::test::test_agent_encoding_sign ... ok
test ssh::test::test_agent_encoding_remove ... 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. 12 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.82s
Running unittests src/lib.rs (target/debug/deps/radicle_dag-05f7457e7194a495)
running 20 tests
test tests::test_dependencies ... ok
test tests::test_contains ... ok
test tests::test_cycle ... ok
test tests::test_diamond ... ok
test tests::test_fold_multiple_roots ... ok
test tests::test_fold_reject ... ok
test tests::test_fold_diamond ... ok
test tests::test_fold_sorting_1 ... ok
test tests::test_fold_sorting_2 ... ok
test tests::test_get ... ok
test tests::test_is_empty ... ok
test tests::test_len ... ok
test tests::test_merge_2 ... ok
test tests::test_merge_1 ... ok
test tests::test_prune_1 ... 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 tests::test_complex ... 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-a7e6b939c4d15a1d)
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_git_metadata-60432466fb6713b9)
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_git_ref_format-35ccca3c145c6922)
running 9 tests
test test::component ... ok
test test::pattern ... ok
test test::qualified ... ok
test test::component_invalid - should panic ... ok
test test::qualified_invalid - should panic ... ok
test test::refname ... ok
test test::qualified_pattern_invalid - should panic ... ok
test test::qualified_pattern ... 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_node-708988a076a0445f)
running 74 tests
test reactor::timer::tests::test_next ... ok
test reactor::timer::tests::test_wake ... ok
test control::tests::test_control_socket ... ok
test reactor::timer::tests::test_wake_exact ... ok
test fingerprint::tests::matching ... ok
test control::tests::test_seed_unseed ... ok
test tests::e2e::missing_default_branch ... ok
test tests::e2e::test_catchup_on_refs_announcements ... ok
test tests::e2e::missing_delegate_default_branch ... ok
test tests::e2e::test_background_foreground_fetch ... ok
test tests::e2e::test_channel_reader_limit ... ok
test tests::e2e::test_clone ... ok
test tests::e2e::test_dont_fetch_owned_refs ... ok
test tests::e2e::test_connection_crossing ... ok
test tests::e2e::test_fetch_followed_remotes ... 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_inventory_sync_basic ... ok
test tests::e2e::test_fetch_emits_canonical_ref_update ... 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_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_replication_invalid ... ok
test tests::e2e::test_replication_ref_in_sigrefs ... ok
test tests::e2e::test_inventory_sync_bridge ... ok
test tests::e2e::test_inventory_sync_ring ... ok
test tests::e2e::test_inventory_sync_star ... ok
test tests::test_announcement_rebroadcast ... ok
test tests::test_announcement_rebroadcast_duplicates ... ok
test tests::test_announcement_relay ... ok
test tests::test_announcement_rebroadcast_timestamp_filtered ... ok
test tests::test_connection_kept_alive ... 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_persistent_peer_connect ... ok
test tests::test_inventory_pruning ... 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_from_command_same_rid ... ok
test tests::test_queued_fetch_max_capacity ... ok
test tests::test_redundant_connect ... ok
test tests::test_refs_announcement_followed ... ok
test tests::test_refs_announcement_fetch_trusted_no_inventory ... 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_refs_announcement_relay_public ... ok
test tests::test_seed_repo_subscribe ... 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_seeding ... ok
test tests::prop_inventory_exchange_dense ... ok
test tests::test_announcement_message_amplification ... ok
test result: ok. 74 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 17.21s
Running unittests src/main.rs (target/debug/deps/radicle_node-7d312b29a9767116)
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-983809b603f6ef8d)
running 10 tests
test fmt::test::fixture ... ok
test fmt::test::zero ... ok
test git2::test::zero ... ok
test gix::test::zero ... ok
test fmt::test::git2 ... ok
test str::test::fixture ... ok
test str::test::zero ... ok
test str::test::git2_roundtrip ... ok
test str::test::gix_roundrip ... ok
test fmt::test::gix ... 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-2bd7009c496516e0)
running 46 tests
test deserializer::test::test_decode_next ... ok
test deserializer::test::test_unparsed ... ok
test service::filter::test::test_parameters ... ok
test deserializer::test::prop_decode_next ... ok
test service::filter::test::compatible ... ok
test service::limiter::test::test_limitter_different_rates ... ok
test service::limiter::test::test_limitter_multi ... ok
test service::gossip::store::test::test_announced ... ok
test service::limiter::test::test_limitter_refill ... ok
test service::filter::test::test_sizes ... 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 service::message::tests::test_inventory_limit ... 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_signed_refs ... 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. 46 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 1.35s
Running unittests src/main.rs (target/debug/deps/git_remote_rad-9b4598e22fb05e53)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests src/main.rs (target/debug/deps/radicle_schemars-ce00fa9f23e76096)
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-bb66f1d798396f61)
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_ssh-4a779b564e98a497)
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-36e3b0b1253d2879)
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-c5564f4668a67e22)
running 21 tests
test cell::test::test_width ... ok
test ansi::tests::colors_disabled ... ok
test ansi::tests::colors_enabled ... 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 ansi::tests::wrapping ... ok
test table::test::test_table_border ... ok
test table::test::test_table_border_truncated ... ok
test table::test::test_table_border_maximized ... ok
test table::test::test_table_truncate ... ok
test table::test::test_table_unicode ... ok
test table::test::test_table_unicode_truncate ... ok
test table::test::test_truncate ... ok
test textarea::test::test_wrapping_fenced_block ... ok
test textarea::test::test_wrapping_code_block ... ok
test textarea::test::test_wrapping ... ok
test textarea::test::test_wrapping_paragraphs ... ok
test vstack::test::test_vstack ... ok
test vstack::test::test_vstack_maximize ... ok
test result: ok. 21 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_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_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>::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>::truncate (line 50) ... ok
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>::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.35s
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_ssh
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.13s
error: 1 target failed:
`-p radicle-cli --test commands`
Exit code: 101
{
"response": "finished",
"result": "failure"
}