Skip to content

Commit

Permalink
docs(soroban-test): document all public interfaces
Browse files Browse the repository at this point in the history
  • Loading branch information
chadoh committed Feb 27, 2023
1 parent d541ce8 commit 5cb695a
Show file tree
Hide file tree
Showing 2 changed files with 158 additions and 2 deletions.
125 changes: 123 additions & 2 deletions cmd/crates/soroban-test/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,26 @@ pub enum Error {
FsError(#[from] fs_extra::error::Error),
}

/// A TestEnv is a contained process for a specific test, with its own ENV and
/// its own TempDir where it will save test-specific configuration.
/// The primary interface to your tests. Creates an isolated process with its own temporary
/// directory, storing all config and output there.
///
/// # Example:
///
/// use soroban_test::{TestEnv, Wasm};
/// const WASM: &Wasm = &Wasm::Release("my_contract");
///
/// #[test]
/// fn invoke_and_read() {
/// TestEnv::with_default(|e| {
/// e.new_cmd("contract")
/// .arg("invoke")
/// .arg("--wasm")
/// .arg(&WASM.path())
/// .args(["--fn", "some_fn"])
/// .assert()
/// .stderr("");
/// });
/// }
pub struct TestEnv {
pub temp_dir: TempDir,
}
Expand All @@ -29,26 +47,55 @@ impl Default for TestEnv {
}

impl TestEnv {
/// Initialize a TestEnv with default settings. Takes a closure to execute within this TestEnv.
///
/// For now, this is the primary interface to create and use a TestEnv. In the future, TestEnv
/// may provide an alternate, more customizable method of initialization.
///
/// # Example
///
/// In a test function:
///
/// TestEnv::with_default(|e| {
/// println!("{:#?}", e.new_cmd("version").ok());
/// });
pub fn with_default<F: FnOnce(&TestEnv)>(f: F) {
let test_env = TestEnv::default();
f(&test_env)
}

/// Initialize a TestEnv with default settings and return a Result with this TestEnv or an
/// error. You probably want `with_default` instead, which makes use of `new` internally.
pub fn new() -> Result<TestEnv, Error> {
TempDir::new()
.map_err(Error::TempDir)
.map(|temp_dir| TestEnv { temp_dir })
}

/// Start building a new `soroban` command, skipping the repetitive `soroban` starting word and
/// setting the current directory to the one for this TestEnv.
///
/// # Example
///
/// TestEnv::with_default(|e| {
/// println!("{:#?}", e.new_cmd("version").ok());
/// });
///
/// Note that you don't need `e.new_cmd("soroban").arg("version")`.
pub fn new_cmd(&self, name: &str) -> Command {
let mut this = Command::cargo_bin("soroban").unwrap_or_else(|_| Command::new("soroban"));
this.arg(name);
this.current_dir(&self.temp_dir);
this
}

/// Get the location of the temporary directory created by this TestEnv.
pub fn dir(&self) -> &TempDir {
&self.temp_dir
}

/// Generate new identity for testing. Names the identity `test`. Uses a hard-coded all-zero
/// seed.
pub fn gen_test_identity(&self) {
self.new_cmd("config")
.arg("identity")
Expand All @@ -60,6 +107,13 @@ impl TestEnv {
.success();
}

/// Return a public key of the identity named `test`, which needs to first be created with
/// `gen_test_identity`. The `test` identity is stored as a seed, which can be used to generate
/// multiple public keys. The specific key generated depends on the `hd_path` supplied.
/// Specifying the same `hd_path` will always generate the same key.
///
/// The phrase "HD path" comes from the larger world of crypto wallets:
/// https://www.ledger.com/academy/crypto/what-are-hierarchical-deterministic-hd-wallets
pub fn test_address(&self, hd_path: usize) -> String {
self.new_cmd("config")
.args("identity address test --hd-path".split(' '))
Expand All @@ -68,6 +122,8 @@ impl TestEnv {
.stdout_as_str()
}

/// Fork TestEnv, return a Result with the new TestEnv or an error. Might be useful to create
/// multiple tests that use the same setup.
pub fn fork(&self) -> Result<TestEnv, Error> {
let this = TestEnv::new()?;
self.save(&this.temp_dir)?;
Expand All @@ -89,8 +145,43 @@ pub fn temp_ledger_file() -> OsString {
.into()
}

/// Import this trait into your file to extend `assert_cmd` with extra utility functions.
///
/// `assert_cmd` is the CLI command builder powering soroban_test.
///
/// You don't need to do anything else with `AssertExt` other than import it; it magically extends
/// the command builder with `stdout_as_str` and other helpers.
///
/// # Example:
///
/// use soroban_test::{AssertExt, TestEnv, Wasm};
///
/// const WASM: &Wasm = &Wasm::Release("my_contract");
///
/// #[test]
/// fn invoke() {
/// TestEnv::with_default(|e| {
/// let stdout = e
/// .new_cmd("contract")
/// .arg("install")
/// .arg("--wasm")
/// .arg(&WASM.path())
/// .assert()
/// .stdout_as_str();
/// println!("{stdout}");
/// }
/// }
///
/// Note that you need the `.assert()`, which is where `assert_cmd` executes the command and
/// returns a struct to make assertions on.
pub trait AssertExt {
/// If the command emits to STDOUT, this will return its output as a `&str`, with leading and
/// trailing whitespace trimmed.
fn stdout_as_str(&self) -> String;

/// If the command emits to STDERR, this will return its output as a `&str`, with leading and
/// trailing whitespace trimmed.
fn stderr_as_str(&self) -> String;
}

impl AssertExt for Assert {
Expand All @@ -100,8 +191,38 @@ impl AssertExt for Assert {
.trim()
.to_owned()
}
fn stderr_as_str(&self) -> String {
String::from_utf8(self.get_output().stderr.clone())
.expect("failed to make str")
.trim()
.to_owned()
}
}

/// Import this trait into your file to extend `assert_cmd` with extra utility functions.
///
/// `assert_cmd` is the CLI command builder powering soroban_test.
///
/// You don't need to do anything else with `CommandExt` other than import it; it magically extends
/// the command builder with `json_arg` and other helpers.
///
/// # Example:
///
/// use serde_json::json;
/// use soroban_test::{AssertExt, TestEnv};
///
/// #[test]
/// fn invoke() {
/// TestEnv::with_default(|e| {
/// e.new_cmd("contract")
/// .arg("invoke")
/// .args(["--id", "0"])
/// .json_arg(json!({"ease": true, "pain": false}));
/// }
/// }
pub trait CommandExt {
/// Pass json (constructed with the serde_json::json macro or similar) as a segment to your
/// command.
fn json_arg<A>(&mut self, j: A) -> &mut Self
where
A: Display;
Expand Down
35 changes: 35 additions & 0 deletions cmd/crates/soroban-test/src/wasm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,38 @@ pub enum Error {
Xdr(#[from] xdr::Error),
}

/// Easily include a built Wasm artifact from your project's `target` folder
///
/// # Example
///
/// If your workspace includes a member with name "my-contract" (that is, with `name =
/// "my-contract"` in its Cargo.toml), or if this is the name of your root project, then in your
/// test you can use:
///
/// use soroban_test::{TestEnv, Wasm};
/// const WASM: &Wasm = &Wasm::Release("my_contract");
///
/// #[test]
/// fn invoke_and_read() {
/// TestEnv::with_default(|e| {
/// let output = e.new_cmd("contract")
/// .arg("invoke")
/// .arg("--wasm")
/// .arg(&WASM.path())
/// .ok();
/// })
/// }
pub enum Wasm<'a> {
/// Takes a filename. Look inside `target/wasm32-unknown-unknown/release` for a Wasm file with
/// the given name.
Release(&'a str),

/// Takes a `profile` and a filename. Look inside `target/wasm32-unknown-unknown/{profile}` for
/// a given Wasm file with the given name.
///
/// # Example
///
/// const WASM: &Wasm = &Wasm::Custom("debug", "my_contract");
Custom(&'a str, &'a str),
}

Expand All @@ -26,6 +56,7 @@ fn find_target_dir() -> Option<PathBuf> {
}

impl Wasm<'_> {
/// The file path to the specified Wasm file
pub fn path(&self) -> PathBuf {
let path = find_target_dir().unwrap().join("wasm32-unknown-unknown");
let mut path = match self {
Expand All @@ -37,10 +68,14 @@ impl Wasm<'_> {
std::env::current_dir().unwrap().join(path)
}

/// The bytes of the specified Wasm file
pub fn bytes(&self) -> Vec<u8> {
fs::read(self.path()).unwrap()
}

/// The derived hash of the specified Wasm file which will be used to refer to a contract
/// "installed" in the Soroban blockchain (that is, to contract bytes that have been uploaded,
/// and which zero or more contract instances may refer to for their behavior).
pub fn hash(&self) -> Result<xdr::Hash, Error> {
let args_xdr = InstallContractCodeArgs {
code: self.bytes().try_into()?,
Expand Down

0 comments on commit 5cb695a

Please sign in to comment.