use std::{ collections::BTreeMap, env::consts::{DLL_PREFIX, DLL_SUFFIX}, }; use clap::{Args, Subcommand}; use xshell::cmd; use crate::{build_docs, sh, workspace, DenyWarnings, Result, NIGHTLY}; const WASM_TIMEOUT_ENV_KEY: &str = "WASM_BINDGEN_TEST_TIMEOUT"; const WASM_TIMEOUT_VALUE: &str = "120"; #[derive(Args)] pub struct CiArgs { #[clap(subcommand)] cmd: Option, } #[derive(Subcommand)] enum CiCommand { /// Check style Style, /// Check for typos Typos, /// Check clippy lints Clippy, /// Check documentation Docs, /// Run tests with a specific feature set TestFeatures { #[clap(subcommand)] cmd: Option, }, /// Run clippy checks for the wasm target Wasm { #[clap(subcommand)] cmd: Option, }, /// Run tests with `wasm-pack test` WasmPack { #[clap(subcommand)] cmd: Option, }, /// Run tests for the different crypto crate features TestCrypto, /// Check that bindings can be generated Bindings, /// Check that the examples compile Examples, } #[derive(Subcommand, PartialEq, Eq, PartialOrd, Ord)] enum FeatureSet { NoEncryption, NoSqlite, NoEncryptionAndSqlite, SqliteCryptostore, RustlsTls, Markdown, Socks, SsoLogin, } #[derive(Subcommand, PartialEq, Eq, PartialOrd, Ord)] #[allow(clippy::enum_variant_names)] enum WasmFeatureSet { /// Check `matrix-sdk-qrcode` crate MatrixSdkQrcode, /// Check `matrix-sdk-base` crate MatrixSdkBase, /// Check `matrix-sdk-common` crate MatrixSdkCommon, /// Check `matrix-sdk` crate with no default features MatrixSdkNoDefault, /// Check `matrix-sdk-ui` crate MatrixSdkUi, /// Check `matrix-sdk` crate with `indexeddb` feature (but not /// `e2e-encryption`) MatrixSdkIndexeddbStoresNoCrypto, /// Check `matrix-sdk` crate with `indexeddb` and `e2e-encryption` features MatrixSdkIndexeddbStores, /// Check `matrix-sdk-indexeddb` crate with all features IndexeddbAllFeatures, /// Check `matrix-sdk-indexeddb` crate with `e2e-encryption` feature IndexeddbCrypto, /// Check `matrix-sdk-indexeddb` crate with `state-store` feature IndexeddbState, /// Equivalent to `indexeddb-all-features`, `indexeddb-crypto` and /// `indexeddb-state` Indexeddb, } impl CiArgs { pub fn run(self) -> Result<()> { let sh = sh(); let _p = sh.push_dir(workspace::root_path()?); match self.cmd { Some(cmd) => match cmd { CiCommand::Style => check_style(), CiCommand::Typos => check_typos(), CiCommand::Clippy => check_clippy(), CiCommand::Docs => check_docs(), CiCommand::TestFeatures { cmd } => run_feature_tests(cmd), CiCommand::Wasm { cmd } => run_wasm_checks(cmd), CiCommand::WasmPack { cmd } => run_wasm_pack_tests(cmd), CiCommand::TestCrypto => run_crypto_tests(), CiCommand::Bindings => check_bindings(), CiCommand::Examples => check_examples(), }, None => { check_style()?; check_clippy()?; check_typos()?; check_docs()?; run_feature_tests(None)?; run_wasm_checks(None)?; run_crypto_tests()?; check_examples()?; Ok(()) } } } } fn check_bindings() -> Result<()> { let sh = sh(); cmd!(sh, "rustup run stable cargo build -p matrix-sdk-crypto-ffi -p matrix-sdk-ffi").run()?; cmd!( sh, " rustup run stable cargo run -p uniffi-bindgen -- generate --library --language kotlin --language swift --out-dir target/generated-bindings target/debug/{DLL_PREFIX}matrix_sdk_ffi{DLL_SUFFIX} " ) .run()?; cmd!( sh, " rustup run stable cargo run -p uniffi-bindgen -- generate --library --language kotlin --language swift --out-dir target/generated-bindings target/debug/{DLL_PREFIX}matrix_sdk_crypto_ffi{DLL_SUFFIX} " ) .run()?; Ok(()) } fn check_examples() -> Result<()> { let sh = sh(); cmd!(sh, "rustup run stable cargo check -p example-*").run()?; Ok(()) } fn check_style() -> Result<()> { let sh = sh(); cmd!(sh, "rustup run {NIGHTLY} cargo fmt -- --check").run()?; Ok(()) } fn check_typos() -> Result<()> { let sh = sh(); // FIXME: Print install instructions if command-not-found (needs an xshell // change: https://github.com/matklad/xshell/issues/46) cmd!(sh, "typos").run()?; Ok(()) } fn check_clippy() -> Result<()> { let sh = sh(); cmd!(sh, "rustup run {NIGHTLY} cargo clippy --all-targets --features testing -- -D warnings") .run()?; cmd!( sh, "rustup run {NIGHTLY} cargo clippy --workspace --all-targets --exclude matrix-sdk-crypto --exclude xtask --no-default-features --features native-tls,sso-login,testing -- -D warnings" ) .run()?; cmd!( sh, "rustup run {NIGHTLY} cargo clippy --all-targets -p matrix-sdk-crypto --no-default-features -- -D warnings" ) .run()?; Ok(()) } fn check_docs() -> Result<()> { build_docs([], DenyWarnings::Yes) } fn run_feature_tests(cmd: Option) -> Result<()> { let args = BTreeMap::from([ (FeatureSet::NoEncryption, "--no-default-features --features sqlite,native-tls,testing"), ( FeatureSet::NoSqlite, "--no-default-features --features e2e-encryption,native-tls,testing", ), (FeatureSet::NoEncryptionAndSqlite, "--no-default-features --features native-tls,testing"), ( FeatureSet::SqliteCryptostore, "--no-default-features --features e2e-encryption,sqlite,native-tls,testing", ), (FeatureSet::RustlsTls, "--no-default-features --features rustls-tls,testing"), (FeatureSet::Markdown, "--features markdown,testing"), (FeatureSet::Socks, "--features socks,testing"), (FeatureSet::SsoLogin, "--features sso-login,testing"), ]); let sh = sh(); let run = |arg_set: &str| { cmd!(sh, "rustup run stable cargo nextest run -p matrix-sdk") .args(arg_set.split_whitespace()) .run()?; cmd!(sh, "rustup run stable cargo test --doc -p matrix-sdk") .args(arg_set.split_whitespace()) .run() }; match cmd { Some(cmd) => { run(args[&cmd])?; } None => { for &arg_set in args.values() { run(arg_set)?; } } } Ok(()) } fn run_crypto_tests() -> Result<()> { let sh = sh(); cmd!(sh, "rustup run stable cargo clippy -p matrix-sdk-crypto -- -D warnings").run()?; cmd!(sh, "rustup run stable cargo nextest run -p matrix-sdk-crypto --no-default-features --features testing").run()?; cmd!(sh, "rustup run stable cargo nextest run -p matrix-sdk-crypto --features=testing") .run()?; cmd!(sh, "rustup run stable cargo test --doc -p matrix-sdk-crypto --features=testing").run()?; cmd!( sh, "rustup run stable cargo clippy -p matrix-sdk-crypto --features=experimental-algorithms -- -D warnings" ) .run()?; cmd!( sh, "rustup run stable cargo nextest run -p matrix-sdk-crypto --features=experimental-algorithms,testing" ).run()?; cmd!( sh, "rustup run stable cargo test --doc -p matrix-sdk-crypto --features=experimental-algorithms,testing" ) .run()?; cmd!(sh, "rustup run stable cargo nextest run -p matrix-sdk-crypto-ffi").run()?; cmd!( sh, "rustup run stable cargo nextest run -p matrix-sdk-sqlite --features crypto-store,testing" ) .run()?; Ok(()) } fn run_wasm_checks(cmd: Option) -> Result<()> { if let Some(WasmFeatureSet::Indexeddb) = cmd { run_wasm_checks(Some(WasmFeatureSet::IndexeddbAllFeatures))?; run_wasm_checks(Some(WasmFeatureSet::IndexeddbCrypto))?; run_wasm_checks(Some(WasmFeatureSet::IndexeddbState))?; return Ok(()); } let args = BTreeMap::from([ (WasmFeatureSet::MatrixSdkQrcode, "-p matrix-sdk-qrcode --features js"), ( WasmFeatureSet::MatrixSdkNoDefault, "-p matrix-sdk --no-default-features --features js,rustls-tls", ), (WasmFeatureSet::MatrixSdkBase, "-p matrix-sdk-base --features js,test-send-sync"), (WasmFeatureSet::MatrixSdkCommon, "-p matrix-sdk-common --features js"), (WasmFeatureSet::MatrixSdkUi, "-p matrix-sdk-ui --features js"), ( WasmFeatureSet::MatrixSdkIndexeddbStoresNoCrypto, "-p matrix-sdk --no-default-features --features js,indexeddb,rustls-tls", ), ( WasmFeatureSet::MatrixSdkIndexeddbStores, "-p matrix-sdk --no-default-features --features js,indexeddb,e2e-encryption,rustls-tls", ), (WasmFeatureSet::IndexeddbAllFeatures, "-p matrix-sdk-indexeddb"), ( WasmFeatureSet::IndexeddbCrypto, "-p matrix-sdk-indexeddb --no-default-features --features e2e-encryption", ), ( WasmFeatureSet::IndexeddbState, "-p matrix-sdk-indexeddb --no-default-features --features state-store", ), ]); let sh = sh(); let run = |arg_set: &str| { cmd!(sh, "rustup run stable cargo clippy --target wasm32-unknown-unknown") .args(arg_set.split_whitespace()) .args(["--", "-D", "warnings"]) .env(WASM_TIMEOUT_ENV_KEY, WASM_TIMEOUT_VALUE) .run() }; match cmd { Some(cmd) => { run(args[&cmd])?; } None => { for &arg_set in args.values() { run(arg_set)?; } } } Ok(()) } fn run_wasm_pack_tests(cmd: Option) -> Result<()> { if let Some(WasmFeatureSet::Indexeddb) = cmd { run_wasm_pack_tests(Some(WasmFeatureSet::IndexeddbAllFeatures))?; run_wasm_pack_tests(Some(WasmFeatureSet::IndexeddbCrypto))?; run_wasm_pack_tests(Some(WasmFeatureSet::IndexeddbState))?; return Ok(()); } let args = BTreeMap::from([ (WasmFeatureSet::MatrixSdkQrcode, ("crates/matrix-sdk-qrcode", "--features js")), ( WasmFeatureSet::MatrixSdkNoDefault, ("crates/matrix-sdk", "--no-default-features --features js,rustls-tls --lib"), ), (WasmFeatureSet::MatrixSdkBase, ("crates/matrix-sdk-base", "--features js")), (WasmFeatureSet::MatrixSdkCommon, ("crates/matrix-sdk-common", "--features js")), ( WasmFeatureSet::MatrixSdkIndexeddbStoresNoCrypto, ("crates/matrix-sdk", "--no-default-features --features js,indexeddb,rustls-tls --lib"), ), ( WasmFeatureSet::MatrixSdkIndexeddbStores, ( "crates/matrix-sdk", "--no-default-features --features js,indexeddb,e2e-encryption,rustls-tls,testing --lib", ), ), ( WasmFeatureSet::IndexeddbAllFeatures, ("crates/matrix-sdk-indexeddb", ""), ), ( WasmFeatureSet::IndexeddbCrypto, ("crates/matrix-sdk-indexeddb", "--no-default-features --features e2e-encryption"), ), ( WasmFeatureSet::IndexeddbState, ("crates/matrix-sdk-indexeddb", "--no-default-features --features state-store"), ), ]); let sh = sh(); let run = |(folder, arg_set): (&str, &str)| { let _pwd = sh.push_dir(folder); cmd!(sh, "pwd").env(WASM_TIMEOUT_ENV_KEY, WASM_TIMEOUT_VALUE).run()?; // print dir so we know what might have failed cmd!(sh, "wasm-pack test --node -- ") .args(arg_set.split_whitespace()) .env(WASM_TIMEOUT_ENV_KEY, WASM_TIMEOUT_VALUE) .run()?; cmd!(sh, "wasm-pack test --firefox --headless --") .args(arg_set.split_whitespace()) .env(WASM_TIMEOUT_ENV_KEY, WASM_TIMEOUT_VALUE) .run() }; match cmd { Some(cmd) => { run(args[&cmd])?; } None => { for &arg_set in args.values() { run(arg_set)?; } } } Ok(()) }