Welcome to the Radicle "Heartwood" hacking guide!
We appreciate your interest in contributing to the Radicle project. If you come across a bug or a missing feature, please feel free to submit a patch. This guide is meant as an introduction to the codebase, on how to debug issues, write tests and navigate the repository.
Please make sure to read CONTRIBUTING.md before submitting code, and follow the included guidelines. To download a development version of Radicle, see the README.md.
For an architectural overview of Heartwood, see ARCHITECTURE.md.
The repository is structured in crates, as follows:
radicle
: The Radicle standard library that contains shared libraries used across the project.radicle-cli
: the Radicle command-line interface (rad
).radicle-cli-test
: The Radicle CLI testing framework, for writing documentation tests.radicle-cob
: Radicle Collaborative Objects (COBs). Provides a way of creating and traversing edit histories.radicle-crdt
: Conflict-free replicated datatypes (CRDTs) used for things like discussions and patches.radicle-crypto
: A wrapper around Ed25519 cryptographic signing primitives.radicle-dag
: A simple directed acyclic graph implementation used byradicle-cob
.radicle-node
: The radicle peer-to-peer daemon that enables users to connect to the network and share code.radicle-remote-helper
: A Git remote helper forrad://
remotes.radicle-ssh
: OpenSSH functionality, including a library used to interface withssh-agent
.radicle-term
: A generic terminal library used by the Radicle CLI.radicle-tools
: Tools used to aid in the development of Radicle.
To run the services or the CLI in debug mode, use cargo run -p <package>
.
For example, the equivalent of rad auth
in debug mode would be:
$ cargo run -p radicle-cli --bin rad -- auth
Arguments after the --
are passed directly to the rad
executable.
This is useful if you are running multiple nodes on the same machine. You can also
specify different listen addresses for the peer-to-peer protocol using --listen
.
To view all options, run cargo run -p radicle-node -- --help
.
You may want to set the appropriate environment variables before running these commands to prevent them from interfering with an existing installation of radicle. See the following section on environment variables.
When testing things, if you want to run one or more nodes in isolation from the public Radicle network of nodes:
- In the node
config.json
file:- set
preferredSeeds
to[]
- set
node.connect
to[]
- set
node.network
totest
- set
node.peers.type
tostatic
- set
This allows you to make any changes to the repositories on your nodes without having them replicated to outside your nodes. This would be good to do for any experiments you make that might be burdensome for other nodes, such as creating very large repositories.
When developing radicle, some environment variables may be used to make the development environment more friendly.
RAD_HOME
Set this to a path on your file system where you'd like radicle to store keys
and repositories. Typically you'll want to set this to a temporary folder, eg.
/tmp/radicle
, that can be safely deleted. If set, all radicle data will be
stored within this folder.
RAD_KEYGEN_SEED
Set this to a 32-byte hexadecimal string to generate deterministic Node IDs when creating new profiles. For example, integration tests use the following setting:
RAD_KEYGEN_SEED=ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff
RAD_PASSPHRASE
Set this to the passphrase chosen during profile initialization (rad auth
) to
skip the passphrase prompt. It's recommended to set this while developing to
avoid storing development keys with ssh-agent
.
Logging for radicle-node
is turned on by default. Check the respective
--help
output to set the log level.
When implementing changes to the CLI, or adding a new sub-command, it's a good
idea to add a documentation test. You can find examples of these in
radicle-cli/examples
.
Each documentation test must be accompanied by a regular unit test. These are
located in radicle-cli/tests/commands.rs
. To keep tests deterministic,
environment variables are used. If your document test output is changing on
each test run, make sure to account for any variability in the test environment
(clocks, RNGs, etc.).
When testing the core service logic, eg. the gossip protocol; tests can be
added to radicle-node/src/tests.rs
. These service-level tests simply test
inputs and outputs and do not perform any I/O.
If you find the need to test the replication protocol or networking layer, it's
possible to write an end-to-end test. These tests can be found in
radicle-node/src/tests/e2e.rs
.
Radicle stores git repositories inside $RAD_HOME/storage
, which defaults to
~/.radicle/storage
on UNIX-based operating systems. You can use standard git
tooling to inspect references and other git objects inside storage. Each radicle
repository is stored under its own folder under storage as a bare Git repository.
Once inside a repository folder, the following commands may come in handy.
git show-ref
to show all references:
$ git show-ref
f60b291752bc38be7dfc90c4c4034de13e01a66b refs/heads/master
f60b291752bc38be7dfc90c4c4034de13e01a66b refs/namespaces/z6MkqTY5aQepDGNCrkPqzdmzveX3D4oAmyVXUDDVQaDGdyVH/refs/heads/master
805b7d0df927dcbc4d3911ab07cd497953eecbd1 refs/namespaces/z6MkqTY5aQepDGNCrkPqzdmzveX3D4oAmyVXUDDVQaDGdyVH/refs/rad/id
86136a42a69572015466bac2d974154ee76f0853 refs/namespaces/z6MkqTY5aQepDGNCrkPqzdmzveX3D4oAmyVXUDDVQaDGdyVH/refs/rad/sigrefs
5575035d8b4faf1f18c532b08516f18031dd7b28 refs/namespaces/z6MkuGSynjxM8SLhcsiEPWZgDeGLAVNXf5g7WePmc1Tri1FS/refs/heads/master
805b7d0df927dcbc4d3911ab07cd497953eecbd1 refs/namespaces/z6MkuGSynjxM8SLhcsiEPWZgDeGLAVNXf5g7WePmc1Tri1FS/refs/rad/id
89e4f0baa327595c7b2849189fc8808388a29033 refs/namespaces/z6MkuGSynjxM8SLhcsiEPWZgDeGLAVNXf5g7WePmc1Tri1FS/refs/rad/sigrefs
git cat-file
to examine refs:
$ git cat-file -p f60b291752bc38be7dfc90c4c4034de13e01a66b
tree 1afc38724d2b89264c7b3826d40b0655a95cfab4
author cloudhead <cloudhead@anonymous.xyz> 1678097961 +0100
committer cloudhead <cloudhead@anonymous.xyz> 1678097961 +0100
gpgsig -----BEGIN SSH SIGNATURE-----
U1NIU0lHAAAAAQAAADMAAAALc3NoLWVkMjU1MTkAAAAgvjrQogRxxLjzzWns8+mKJAGzEX
4fm2ALoN7pyvD2ssQAAAADZ2l0AAAAAAAAAAZzaGE1MTIAAABTAAAAC3NzaC1lZDI1NTE5
AAAAQHXhUf7QjXNlgCjDbGSG+zoyIlE4S9/d9qjvG7x9jw8J/fXDVIMkh/Lkp743g7EliM
X+88wqit9BeQoHXuxj2Ao=
-----END SSH SIGNATURE-----
Init
You can also run git ls-remote rad
from inside a working copy to examine the
remote refs in storage.
The radicle node listens on a UNIX domain socket located at
$RAD_HOME/node/control.sock
. Make sure this file is accessible and has the
required permissions for your user to read and write to it.
Radicle uses Ed25519 keys that are located in $RAD_HOME/keys
. These keys are
encoded in the standard OpenSSH format. It's therefore possible to use standard
OpenSSH tools to interact with them, eg. ssh-add
.
Your radicle secret key is protected with a passphrase (See: $RAD_PASSPHRASE
).