This project adheres to the Contributor Covenant code of conduct. By participating, you are expected to uphold this code. Please report unacceptable behavior to [email protected].
- Ensure the bug was not already reported by searching on GitHub under Issues.
- If you're unable to find an open issue addressing the problem, open a new one. Be sure to include:
- a title,
- a clear description,
- as much relevant information as possible,
- a code sample or an executable test case demonstrating the expected behavior that is not occurring.
Development flow:
- Create feature branches using develop as a starting point to start new work;
- Submit a new pull request to the
develop
branch- please ensure the PR description clearly describes the problem and solution and include the relevant issue number if applicable.
Release workflow:
- Create a release branch;
- Tag the commit on Github releases;
- Update
main
branch to point to the "newest" release (by version number); - Update
docs
branch to include documentation based on the "newest" release (by version number).
-
Install brew package manager
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
-
Install
llvm
,cmake
andbinaryen
brew install llvm cmake binaryen
-
Install Rust toolchain
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh source $HOME/.cargo/env rustup target add wasm32-unknown-unknown rustup +nightly target add wasm32-unknown-unknown
-
Install Scrypto CLIs
cargo install --git https://github.com/radixdlt/radixdlt-scrypto --branch develop radix-clis
-
(Recommended) Install VSCode and the following plugins
-
(Optional) Install
cargo nextest
to speedup test execution time Installationcargo install cargo-nextest
more details: cargo-nextest
-
(Optional) Install
sccache
to speedup compilation times.- Installation
cargo install sccache
- Configuration
Two options available:- via environmental variable:
bash:zsh:echo 'export RUSTC_WRAPPER=sccache' >> ~/.profile
echo 'export RUSTC_WRAPPER=sccache' >> ~/.zshrc
- via cargo configuration file
define
build.rustc-wrapper
in the cargo configuration file. For example, you can set it globally in$HOME/.cargo/config.toml
by adding:[build] rustc-wrapper = "/path/to/sccache"
- via environmental variable:
more details: sccache - Shared Compilation Cache
-
(Optional) Enable LFS to pull assets under
assets-lfs
- Install
git-lfs
:- MacOS
brew install git-lfs
- Ubuntu
curl -s https://packagecloud.io/install/repositories/github/git-lfs/script.deb.sh | sudo bash sudo apt-get install git-lfs
- MacOS
- Pull
git lfs install git lfs pull
- Install
Bash scripts that might be of help:
format.sh
- Formats the entire repobuild.sh
- Builds main packagestest.sh
- Runs the essential teststest_extra.sh
- Runs the additional testsassets/update-assets.sh
- UpdatesAccount
/Faucet
scrypto packages (needed when your change would affect the output WASM)
radix-blueprint-schema-init
: Blueprint schema initialization structures, used by Radix Package Definition (RPD).radix-clis
: Various CLI tools, likeresim
,scrypto
,rtmc
andrtmd
.radix-common-derive
: Macros for definingDecimal
andPreciseDecimal
.radix-common
: Common libraries used by Radix Engine and Scrypto.radix-engine
: The Radix Engine implementation.radix-native-sdk
: Library to assist native blueprint development.radix-sbor-derives
: Macros for encoding and decoding Scrypto SBOR and Manifest SBOR data.radix-substate-store-impls
: Various substate store implementations.radix-substate-store-interface
: The interface of any substate store.radix-substate-store-queries
: Interprets data in substate data by injecting high-level knowledge.radix-transaction-scenarios
: Defines various transaction scenarios, for testing.radix-transactions
: Radix transaction manifest compiler, transaction models, signing and validating logic.sbor-derive
: Macros for encoding and decoding SBOR data.sbor
: A generic binary data format, upon which Scrypto SBOR and Manifest SBOR are built.scrypto-derive
: Macros for defining blueprints.scrypto-test
: Library for testing Scrypto blueprints.scrypto
: Scrypto language abstraction.
- Feature -
feature/cool-bananas
- Development -
develop
- Release -
release/0.1.0
- Hotfix -
release/0.1.1
Branch main
always points to the latest release.
Feature branches are where the main work happens. The goal is to keep them as independent from each other as possible. They can be based on a previous release or from develop.
develop branch is not a place to dump WIP features
It’s important to remark that feature branches should only be merged to develop once they are complete and ideally tested in a test network.
This branch acts as staging for new releases, and are where most of QA should happen.
When QA gives the green light, a new release branch is created
These branches will stay alive forever, or at least while we support the release, thereby allowing us to release security hotfixes for older versions.
If QA discovers a bug with any of the features before a release happens, it is fixed in the feature branch taken from the release branch and then merged into the release again.
These changes should immediately be propagated to the current release candidate branch.
Hotfix branches are for providing emergency security fixes to older versions and should be treated like release branches.
The hotifx should be created for the oldest affected release, and then merged downstream into the next release or release candidate, repeated until up to date.
We use the default code style specified by rustfmt.
A convenience script is also provided to format the whole code base:
./format.sh
You're also highly recommended to install the git hooks
git config core.hooksPath .githooks
Please follow the convention below for commit messages:
- Separate subject from body with a blank line
- Limit the subject line to 50 characters
- Capitalise the subject line
- Do not end the subject line with a period
- Use the imperative mood in the subject line
- Wrap the body at 72 characters
- Use the body to explain what and why vs. how, separating paragraphs with an empty line.
Since the Radix Engine is used in a consensus-driven environment, all results of a particular transaction (e.g. final state changes and emitted events) must always be exactly the same, no matter where and when the transaction is executed.
However, some wide-spread programming concepts and data structures are inherently non-deterministic, so - by convention - we choose to not use them at all (rather than to analyze their potential impact on non-deterministic results on a per-usage basis).
Apart from some obvious "things to avoid" (like, using a random value, or a wall-clock), please observe the detailed rules below:
We explicitly ban the plain HashMap
and HashSet
usage from production code, including macro
definitions (i.e. these very structs may only be used in tests and test utilities).
However, hash-based structures are useful and have wonderful runtime characteristics, and actually only introduce non-deterministic behaviors when iterated over. Hence, we provide the following ways to use them:
- Inside macro definitions, use the tree-based replacements (
BTreeMap
andBTreeSet
).- The reasoning is: macros are evaluated during compilation, so we do not need O(1) runtime performance, and we prefer to avoid pulling in dependencies to other alternatives.
- If you do not need to iterate over the collection (e.g. use a
HashMap
only in a "put + get" manner), then use our custom wrapperNonIterMap
, which exposes the deterministic part of theHashMap
's API (i.e. excludes iteration). - If elements of the iterated collection have some well-defined, intuitive natural ordering, then
use the tree-based replacements (
BTreeMap
andBTreeSet
). - If you need to iterate over the collection in some custom order (e.g. order of insertion), or
if you do not care about the order at all, then use the indexed alternative,
IndexMap
(we export it from an external library).- It is essentially a
Vec
, with aHashMap
on the side (for O(1) access), so it can accommodate arbitrary (re-)ordering of elements (having methods like.move_index(from, to)
andsort_by()
).
- It is essentially a