| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772 | mod rust_template;use crate::rust_template::{create_component, create_system};use anchor_cli::config;use anchor_cli::config::{    BootstrapMode, Config, ConfigOverride, GenesisEntry, ProgramArch, ProgramDeployment,    TestValidator, Validator, WithPath,};use anchor_client::Cluster;use anchor_syn::idl::types::Idl;use anyhow::{anyhow, Result};use clap::{Parser, Subcommand};use heck::{ToKebabCase, ToSnakeCase};use std::collections::BTreeMap;use std::fs::{self, create_dir_all, File, OpenOptions};use std::io::Write;use std::io::{self, BufRead};use std::path::{Path, PathBuf};use std::process::Stdio;pub const VERSION: &str = env!("CARGO_PKG_VERSION");pub const ANCHOR_VERSION: &str = anchor_cli::VERSION;pub const WORLD_PROGRAM: &str = "WorLD15A7CrDwLcLy4fRqtaTb9fbd8o8iqiEMUDse2n";#[derive(Debug, Subcommand)]pub enum BoltCommand {    #[clap(about = "Create a new component")]    Component(ComponentCommand),    #[clap(about = "Create a new system")]    System(SystemCommand),    // Include all existing commands from anchor_cli::Command    #[clap(flatten)]    Anchor(anchor_cli::Command),}#[derive(Debug, Parser)]pub struct InitCommand {    #[clap(short, long, help = "Workspace name")]    pub workspace_name: String,}#[derive(Debug, Parser)]pub struct ComponentCommand {    pub name: String,}#[derive(Debug, Parser)]pub struct SystemCommand {    pub name: String,}#[derive(Debug, Parser)]#[clap(version = VERSION)]pub struct Opts {    /// Rebuild the auto-generated types    #[clap(global = true, long, action)]    pub rebuild_types: bool,    #[clap(flatten)]    pub cfg_override: ConfigOverride,    #[clap(subcommand)]    pub command: BoltCommand,}pub fn entry(opts: Opts) -> Result<()> {    match opts.command {        BoltCommand::Anchor(command) => match command {            anchor_cli::Command::Init {                name,                javascript,                solidity,                no_git,                jest,                template,                force,            } => init(                &opts.cfg_override,                name,                javascript,                solidity,                no_git,                jest,                template,                force,            ),            anchor_cli::Command::Build {                idl,                idl_ts,                verifiable,                program_name,                solana_version,                docker_image,                bootstrap,                cargo_args,                env,                skip_lint,                no_docs,                arch,            } => build(                &opts.cfg_override,                idl,                idl_ts,                verifiable,                skip_lint,                program_name,                solana_version,                docker_image,                bootstrap,                None,                None,                env,                cargo_args,                no_docs,                arch,                opts.rebuild_types,            ),            _ => {                let opts = anchor_cli::Opts {                    cfg_override: opts.cfg_override,                    command,                };                anchor_cli::entry(opts)            }        },        BoltCommand::Component(command) => new_component(&opts.cfg_override, command.name),        BoltCommand::System(command) => new_system(&opts.cfg_override, command.name),    }}// Bolt Init#[allow(clippy::too_many_arguments)]fn init(    cfg_override: &ConfigOverride,    name: String,    javascript: bool,    solidity: bool,    no_git: bool,    jest: bool,    template: anchor_cli::rust_template::ProgramTemplate,    force: bool,) -> Result<()> {    if !force && Config::discover(cfg_override)?.is_some() {        return Err(anyhow!("Workspace already initialized"));    }    // We need to format different cases for the dir and the name    let rust_name = name.to_snake_case();    let project_name = if name == rust_name {        rust_name.clone()    } else {        name.to_kebab_case()    };    // Additional keywords that have not been added to the `syn` crate as reserved words    // https://github.com/dtolnay/syn/pull/1098    let extra_keywords = ["async", "await", "try"];    let component_name = "position";    let system_name = "movement";    // Anchor converts to snake case before writing the program name    if syn::parse_str::<syn::Ident>(&rust_name).is_err()        || extra_keywords.contains(&rust_name.as_str())    {        return Err(anyhow!(            "Anchor workspace name must be a valid Rust identifier. It may not be a Rust reserved word, start with a digit, or include certain disallowed characters. See https://doc.rust-lang.org/reference/identifiers.html for more detail.",        ));    }    if force {        fs::create_dir_all(&project_name)?;    } else {        fs::create_dir(&project_name)?;    }    std::env::set_current_dir(&project_name)?;    fs::create_dir_all("app")?;    let mut cfg = Config::default();    if jest {        cfg.scripts.insert(            "test".to_owned(),            if javascript {                "yarn run jest"            } else {                "yarn run jest --preset ts-jest"            }            .to_owned(),        );    } else {        cfg.scripts.insert(            "test".to_owned(),            if javascript {                "yarn run mocha -t 1000000 tests/"            } else {                "yarn run ts-mocha -p ./tsconfig.json -t 1000000 tests/**/*.ts"            }            .to_owned(),        );    }    let mut localnet = BTreeMap::new();    let program_id = anchor_cli::rust_template::get_or_create_program_id(&rust_name);    localnet.insert(        rust_name,        ProgramDeployment {            address: program_id,            path: None,            idl: None,        },    );    if !solidity {        let component_id = anchor_cli::rust_template::get_or_create_program_id(component_name);        let system_id = anchor_cli::rust_template::get_or_create_program_id(system_name);        localnet.insert(            component_name.to_owned(),            ProgramDeployment {                address: component_id,                path: None,                idl: None,            },        );        localnet.insert(            system_name.to_owned(),            ProgramDeployment {                address: system_id,                path: None,                idl: None,            },        );        cfg.workspace.members.push("programs/*".to_owned());        cfg.workspace            .members            .push("programs-ecs/components/*".to_owned());        cfg.workspace            .members            .push("programs-ecs/systems/*".to_owned());    }    // Setup the test validator to clone Bolt programs from devnet    let validator = Validator {        url: Some("https://rpc.magicblock.app/devnet/".to_owned()),        rpc_port: 8899,        bind_address: "0.0.0.0".to_owned(),        ledger: ".bolt/test-ledger".to_owned(),        account: Some(vec![            // Registry account            anchor_cli::config::AccountEntry {                address: "EHLkWwAT9oebVv9ht3mtqrvHhRVMKrt54tF3MfHTey2K".to_owned(),                filename: "tests/fixtures/registry.json".to_owned(),            },        ]),        ..Default::default()    };    let test_validator = TestValidator {        startup_wait: 5000,        shutdown_wait: 2000,        validator: Some(validator),        genesis: Some(vec![GenesisEntry {            address: WORLD_PROGRAM.to_owned(),            program: "tests/fixtures/world.so".to_owned(),            upgradeable: Some(false),        }]),        ..Default::default()    };    cfg.test_validator = Some(test_validator);    cfg.programs.insert(Cluster::Localnet, localnet);    let toml = cfg.to_string();    fs::write("Anchor.toml", toml)?;    // Initialize .gitignore file    fs::write(".gitignore", rust_template::git_ignore())?;    // Initialize .prettierignore file    fs::write(".prettierignore", rust_template::prettier_ignore())?;    // Remove the default programs if `--force` is passed    if force {        let programs_path = std::env::current_dir()?            .join(if solidity { "solidity" } else { "programs" })            .join(&project_name);        fs::create_dir_all(&programs_path)?;        fs::remove_dir_all(&programs_path)?;        let programs_ecs_path = std::env::current_dir()?            .join("programs-ecs")            .join(&project_name);        fs::create_dir_all(&programs_ecs_path)?;        fs::remove_dir_all(&programs_ecs_path)?;    }    // Build the program.    if solidity {        anchor_cli::solidity_template::create_program(&project_name)?;    } else {        create_system(system_name)?;        create_component(component_name)?;        anchor_cli::rust_template::create_program(&project_name, template)?;        // Add the component as a dependency to the system        std::process::Command::new("cargo")            .arg("add")            .arg("--package")            .arg(system_name)            .arg("--path")            .arg(format!("programs-ecs/components/{}", component_name))            .arg("--features")            .arg("cpi")            .stdout(std::process::Stdio::null())            .stderr(std::process::Stdio::null())            .spawn()            .map_err(|e| {                anyhow::format_err!(                    "error adding component as dependency to the system: {}",                    e.to_string()                )            })?;    }    // Build the test suite.    fs::create_dir_all("tests/fixtures")?;    // Build the migrations directory.    fs::create_dir_all("migrations")?;    // Create the registry account    fs::write(        "tests/fixtures/registry.json",        rust_template::registry_account(),    )?;    // Dump the World program into tests/fixtures/world.so    std::process::Command::new("solana")        .arg("program")        .arg("dump")        .arg("-u")        .arg("d")        .arg(WORLD_PROGRAM)        .arg("tests/fixtures/world.so")        .stdout(Stdio::inherit())        .stderr(Stdio::inherit())        .spawn()        .map_err(|e| anyhow::format_err!("solana program dump failed: {}", e.to_string()))?;    if javascript {        // Build javascript config        let mut package_json = File::create("package.json")?;        package_json.write_all(rust_template::package_json(jest).as_bytes())?;        if jest {            let mut test = File::create(format!("tests/{}.test.js", &project_name))?;            if solidity {                test.write_all(anchor_cli::solidity_template::jest(&project_name).as_bytes())?;            } else {                test.write_all(rust_template::jest(&project_name).as_bytes())?;            }        } else {            let mut test = File::create(format!("tests/{}.js", &project_name))?;            if solidity {                test.write_all(anchor_cli::solidity_template::mocha(&project_name).as_bytes())?;            } else {                test.write_all(rust_template::mocha(&project_name).as_bytes())?;            }        }        let mut deploy = File::create("migrations/deploy.js")?;        deploy.write_all(anchor_cli::rust_template::deploy_script().as_bytes())?;    } else {        // Build typescript config        let mut ts_config = File::create("tsconfig.json")?;        ts_config.write_all(anchor_cli::rust_template::ts_config(jest).as_bytes())?;        let mut ts_package_json = File::create("package.json")?;        ts_package_json.write_all(rust_template::ts_package_json(jest).as_bytes())?;        let mut deploy = File::create("migrations/deploy.ts")?;        deploy.write_all(anchor_cli::rust_template::ts_deploy_script().as_bytes())?;        let mut mocha = File::create(format!("tests/{}.ts", &project_name))?;        if solidity {            mocha.write_all(anchor_cli::solidity_template::ts_mocha(&project_name).as_bytes())?;        } else {            mocha.write_all(rust_template::ts_mocha(&project_name).as_bytes())?;        }    }    let yarn_result = install_node_modules("yarn")?;    if !yarn_result.status.success() {        println!("Failed yarn install will attempt to npm install");        install_node_modules("npm")?;    }    if !no_git {        let git_result = std::process::Command::new("git")            .arg("init")            .stdout(Stdio::inherit())            .stderr(Stdio::inherit())            .output()            .map_err(|e| anyhow::format_err!("git init failed: {}", e.to_string()))?;        if !git_result.status.success() {            eprintln!("Failed to automatically initialize a new git repository");        }    }    println!("{project_name} initialized");    Ok(())}#[allow(clippy::too_many_arguments)]pub fn build(    cfg_override: &ConfigOverride,    idl: Option<String>,    idl_ts: Option<String>,    verifiable: bool,    skip_lint: bool,    program_name: Option<String>,    solana_version: Option<String>,    docker_image: Option<String>,    bootstrap: BootstrapMode,    stdout: Option<File>,    stderr: Option<File>,    env_vars: Vec<String>,    cargo_args: Vec<String>,    no_docs: bool,    arch: ProgramArch,    rebuild_types: bool,) -> Result<()> {    let cfg = Config::discover(cfg_override)?.expect("Not in workspace.");    let types_path = "crates/types/src";    // If rebuild_types is true and the types directory exists, remove it    if rebuild_types && Path::new(types_path).exists() {        fs::remove_dir_all(            PathBuf::from(types_path)                .parent()                .ok_or_else(|| anyhow::format_err!("Failed to remove types directory"))?,        )?;    }    create_dir_all(types_path)?;    build_dynamic_types(cfg, cfg_override, types_path)?;    // Build the programs    anchor_cli::build(        cfg_override,        idl,        idl_ts,        verifiable,        skip_lint,        program_name,        solana_version,        docker_image,        bootstrap,        stdout,        stderr,        env_vars,        cargo_args,        no_docs,        arch,    )}// Install node modulesfn install_node_modules(cmd: &str) -> Result<std::process::Output> {    let mut command = std::process::Command::new(if cfg!(target_os = "windows") {        "cmd"    } else {        cmd    });    if cfg!(target_os = "windows") {        command.arg(format!("/C {} install", cmd));    } else {        command.arg("install");    }    command        .stdout(Stdio::inherit())        .stderr(Stdio::inherit())        .output()        .map_err(|e| anyhow::format_err!("{} install failed: {}", cmd, e.to_string()))}// Create a new component from the templatefn new_component(cfg_override: &ConfigOverride, name: String) -> Result<()> {    with_workspace(cfg_override, |cfg| {        match cfg.path().parent() {            None => {                println!("Unable to make new component");            }            Some(parent) => {                std::env::set_current_dir(parent)?;                let cluster = cfg.provider.cluster.clone();                let programs = cfg.programs.entry(cluster).or_default();                if programs.contains_key(&name) {                    return Err(anyhow!("Program already exists"));                }                programs.insert(                    name.clone(),                    ProgramDeployment {                        address: {                            create_component(&name)?;                            anchor_cli::rust_template::get_or_create_program_id(&name)                        },                        path: None,                        idl: None,                    },                );                let toml = cfg.to_string();                fs::write("Anchor.toml", toml)?;                println!("Created new component: {}", name);            }        };        Ok(())    })}// Create a new system from the templatefn new_system(cfg_override: &ConfigOverride, name: String) -> Result<()> {    with_workspace(cfg_override, |cfg| {        match cfg.path().parent() {            None => {                println!("Unable to make new system");            }            Some(parent) => {                std::env::set_current_dir(parent)?;                let cluster = cfg.provider.cluster.clone();                let programs = cfg.programs.entry(cluster).or_default();                if programs.contains_key(&name) {                    return Err(anyhow!("Program already exists"));                }                programs.insert(                    name.clone(),                    anchor_cli::config::ProgramDeployment {                        address: {                            rust_template::create_system(&name)?;                            anchor_cli::rust_template::get_or_create_program_id(&name)                        },                        path: None,                        idl: None,                    },                );                let toml = cfg.to_string();                fs::write("Anchor.toml", toml)?;                println!("Created new system: {}", name);            }        };        Ok(())    })}// with_workspace ensures the current working directory is always the top level// workspace directory, i.e., where the `Anchor.toml` file is located, before// and after the closure invocation.//// The closure passed into this function must never change the working directory// to be outside the workspace. Doing so will have undefined behavior.fn with_workspace<R>(    cfg_override: &ConfigOverride,    f: impl FnOnce(&mut WithPath<Config>) -> R,) -> R {    set_workspace_dir_or_exit();    let mut cfg = Config::discover(cfg_override)        .expect("Previously set the workspace dir")        .expect("Anchor.toml must always exist");    let r = f(&mut cfg);    set_workspace_dir_or_exit();    r}fn set_workspace_dir_or_exit() {    let d = match Config::discover(&ConfigOverride::default()) {        Err(err) => {            println!("Workspace configuration error: {err}");            std::process::exit(1);        }        Ok(d) => d,    };    match d {        None => {            println!("Not in anchor workspace.");            std::process::exit(1);        }        Some(cfg) => {            match cfg.path().parent() {                None => {                    println!("Unable to make new program");                }                Some(parent) => {                    if std::env::set_current_dir(parent).is_err() {                        println!("Not in anchor workspace.");                        std::process::exit(1);                    }                }            };        }    }}fn discover_cluster_url(cfg_override: &ConfigOverride) -> Result<String> {    let url = match Config::discover(cfg_override)? {        Some(cfg) => cluster_url(&cfg, &cfg.test_validator),        None => {            if let Some(cluster) = cfg_override.cluster.clone() {                cluster.url().to_string()            } else {                config::get_solana_cfg_url()?            }        }    };    Ok(url)}fn cluster_url(cfg: &Config, test_validator: &Option<TestValidator>) -> String {    let is_localnet = cfg.provider.cluster == Cluster::Localnet;    match is_localnet {        // Cluster is Localnet, assume the intent is to use the configuration        // for solana-test-validator        true => test_validator_rpc_url(test_validator),        false => cfg.provider.cluster.url().to_string(),    }}// Return the URL that solana-test-validator should be running on given the// configurationfn test_validator_rpc_url(test_validator: &Option<TestValidator>) -> String {    match test_validator {        Some(TestValidator {            validator: Some(validator),            ..        }) => format!("http://{}:{}", validator.bind_address, validator.rpc_port),        _ => "http://127.0.0.1:8899".to_string(),    }}fn build_dynamic_types(    cfg: WithPath<Config>,    cfg_override: &ConfigOverride,    types_path: &str,) -> Result<()> {    let cur_dir = std::env::current_dir()?;    for p in cfg.get_rust_program_list()? {        process_program_path(&p, cfg_override, types_path)?;    }    let types_path = PathBuf::from(types_path);    let cargo_path = types_path        .parent()        .unwrap_or(&types_path)        .join("Cargo.toml");    if !cargo_path.exists() {        let mut file = File::create(cargo_path)?;        file.write_all(rust_template::types_cargo_toml().as_bytes())?;    }    std::env::set_current_dir(cur_dir)?;    Ok(())}fn process_program_path(    program_path: &Path,    cfg_override: &ConfigOverride,    types_path: &str,) -> Result<()> {    let lib_rs_path = Path::new(types_path).join("lib.rs");    let file = File::open(program_path.join("src").join("lib.rs"))?;    let lines = io::BufReader::new(file).lines();    let mut contains_dynamic_components = false;    for line in lines.map_while(Result::ok) {        if let Some(component_id) = extract_component_id(&line) {            let file_path = PathBuf::from(format!("{}/component_{}.rs", types_path, component_id));            if !file_path.exists() {                println!("Generating type for Component: {}", component_id);                generate_component_type_file(&file_path, cfg_override, component_id)?;                append_component_to_lib_rs(&lib_rs_path, component_id)?;            }            contains_dynamic_components = true;        }    }    if contains_dynamic_components {        let program_name = program_path.file_name().unwrap().to_str().unwrap();        add_types_crate_dependency(program_name, &types_path.replace("/src", ""))?;    }    Ok(())}fn add_types_crate_dependency(program_name: &str, types_path: &str) -> Result<()> {    std::process::Command::new("cargo")        .arg("add")        .arg("--package")        .arg(program_name)        .arg("--path")        .arg(types_path)        .stdout(Stdio::null())        .stderr(Stdio::null())        .spawn()        .map_err(|e| {            anyhow::format_err!(                "error adding types as dependency to the program: {}",                e.to_string()            )        })?;    Ok(())}fn extract_component_id(line: &str) -> Option<&str> {    let component_id_marker = "#[component_id(";    line.find(component_id_marker).map(|start| {        let start = start + component_id_marker.len();        let end = line[start..].find(')').unwrap() + start;        line[start..end].trim_matches('"')    })}fn fetch_idl_for_component(component_id: &str, url: &str) -> Result<String> {    let output = std::process::Command::new("bolt")        .arg("idl")        .arg("fetch")        .arg(component_id)        .arg("--provider.cluster")        .arg(url)        .stdout(Stdio::piped())        .stderr(Stdio::piped())        .output()?;    if output.status.success() {        let idl_string = String::from_utf8(output.stdout)            .map_err(|e| anyhow!("Failed to decode IDL output as UTF-8: {}", e))?            .to_string();        Ok(idl_string)    } else {        let error_message = String::from_utf8(output.stderr)            .unwrap_or(format!(                "Error trying to dynamically generate the type \            for component {}, unable to fetch the idl. \nEnsure that the idl is available. Specify \            the appropriate cluster using the --provider.cluster option",                component_id            ))            .to_string();        Err(anyhow!("Command failed with error: {}", error_message))    }}fn generate_component_type_file(    file_path: &Path,    cfg_override: &ConfigOverride,    component_id: &str,) -> Result<()> {    let url = discover_cluster_url(cfg_override)?;    let idl_string = fetch_idl_for_component(component_id, &url)?;    let idl: Idl = serde_json::from_str(&idl_string)?;    let mut file = File::create(file_path)?;    file.write_all(rust_template::component_type(&idl, component_id)?.as_bytes())?;    Ok(())}fn append_component_to_lib_rs(lib_rs_path: &Path, component_id: &str) -> Result<()> {    let mut file = OpenOptions::new()        .create(true)        .append(true)        .open(lib_rs_path)?;    file.write_all(rust_template::component_type_import(component_id).as_bytes())?;    Ok(())}
 |