123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789 |
- use crate::{
- config::ProgramWorkspace, create_files, override_or_create_files, solidity_template, Files,
- VERSION,
- };
- use anchor_syn::idl::types::Idl;
- use anyhow::Result;
- use clap::{Parser, ValueEnum};
- use heck::{ToLowerCamelCase, ToSnakeCase, ToUpperCamelCase};
- use solana_sdk::{
- pubkey::Pubkey,
- signature::{read_keypair_file, write_keypair_file, Keypair},
- signer::Signer,
- };
- use std::{
- fmt::Write as _,
- fs::{self, File},
- io::Write as _,
- path::Path,
- process::Stdio,
- };
- /// Program initialization template
- #[derive(Clone, Debug, Default, Eq, PartialEq, Parser, ValueEnum)]
- pub enum ProgramTemplate {
- /// Program with a single `lib.rs` file
- #[default]
- Single,
- /// Program with multiple files for instructions, state...
- Multiple,
- }
- /// Create a program from the given name and template.
- pub fn create_program(name: &str, template: ProgramTemplate) -> Result<()> {
- let program_path = Path::new("programs").join(name);
- let common_files = vec![
- ("Cargo.toml".into(), workspace_manifest().into()),
- (program_path.join("Cargo.toml"), cargo_toml(name)),
- (program_path.join("Xargo.toml"), xargo_toml().into()),
- ];
- let template_files = match template {
- ProgramTemplate::Single => create_program_template_single(name, &program_path),
- ProgramTemplate::Multiple => create_program_template_multiple(name, &program_path),
- };
- create_files(&[common_files, template_files].concat())
- }
- /// Create a program with a single `lib.rs` file.
- fn create_program_template_single(name: &str, program_path: &Path) -> Files {
- vec![(
- program_path.join("src").join("lib.rs"),
- format!(
- r#"use anchor_lang::prelude::*;
- declare_id!("{}");
- #[program]
- pub mod {} {{
- use super::*;
- pub fn initialize(ctx: Context<Initialize>) -> Result<()> {{
- Ok(())
- }}
- }}
- #[derive(Accounts)]
- pub struct Initialize {{}}
- "#,
- get_or_create_program_id(name),
- name.to_snake_case(),
- ),
- )]
- }
- /// Create a program with multiple files for instructions, state...
- fn create_program_template_multiple(name: &str, program_path: &Path) -> Files {
- let src_path = program_path.join("src");
- vec![
- (
- src_path.join("lib.rs"),
- format!(
- r#"pub mod constants;
- pub mod error;
- pub mod instructions;
- pub mod state;
- use anchor_lang::prelude::*;
- pub use constants::*;
- pub use instructions::*;
- pub use state::*;
- declare_id!("{}");
- #[program]
- pub mod {} {{
- use super::*;
- pub fn initialize(ctx: Context<Initialize>) -> Result<()> {{
- initialize::handler(ctx)
- }}
- }}
- "#,
- get_or_create_program_id(name),
- name.to_snake_case(),
- ),
- ),
- (
- src_path.join("constants.rs"),
- r#"use anchor_lang::prelude::*;
- #[constant]
- pub const SEED: &str = "anchor";
- "#
- .into(),
- ),
- (
- src_path.join("error.rs"),
- r#"use anchor_lang::prelude::*;
- #[error_code]
- pub enum ErrorCode {
- #[msg("Custom error message")]
- CustomError,
- }
- "#
- .into(),
- ),
- (
- src_path.join("instructions").join("mod.rs"),
- r#"pub mod initialize;
- pub use initialize::*;
- "#
- .into(),
- ),
- (
- src_path.join("instructions").join("initialize.rs"),
- r#"use anchor_lang::prelude::*;
- #[derive(Accounts)]
- pub struct Initialize {}
- pub fn handler(ctx: Context<Initialize>) -> Result<()> {
- Ok(())
- }
- "#
- .into(),
- ),
- (src_path.join("state").join("mod.rs"), r#""#.into()),
- ]
- }
- const fn workspace_manifest() -> &'static str {
- r#"[workspace]
- members = [
- "programs/*"
- ]
- resolver = "2"
- [profile.release]
- overflow-checks = true
- lto = "fat"
- codegen-units = 1
- [profile.release.build-override]
- opt-level = 3
- incremental = false
- codegen-units = 1
- "#
- }
- fn cargo_toml(name: &str) -> String {
- format!(
- r#"[package]
- name = "{0}"
- version = "0.1.0"
- description = "Created with Anchor"
- edition = "2021"
- [lib]
- crate-type = ["cdylib", "lib"]
- name = "{1}"
- [features]
- no-entrypoint = []
- no-idl = []
- no-log-ix-name = []
- cpi = ["no-entrypoint"]
- default = []
- [dependencies]
- anchor-lang = "{2}"
- "#,
- name,
- name.to_snake_case(),
- VERSION,
- )
- }
- fn xargo_toml() -> &'static str {
- r#"[target.bpfel-unknown-unknown.dependencies.std]
- features = []
- "#
- }
- /// Read the program keypair file or create a new one if it doesn't exist.
- pub fn get_or_create_program_id(name: &str) -> Pubkey {
- let keypair_path = Path::new("target")
- .join("deploy")
- .join(format!("{}-keypair.json", name.to_snake_case()));
- read_keypair_file(&keypair_path)
- .unwrap_or_else(|_| {
- let keypair = Keypair::new();
- write_keypair_file(&keypair, keypair_path).expect("Unable to create program keypair");
- keypair
- })
- .pubkey()
- }
- pub fn credentials(token: &str) -> String {
- format!(
- r#"[registry]
- token = "{token}"
- "#
- )
- }
- pub fn idl_ts(idl: &Idl) -> Result<String> {
- let mut idl = idl.clone();
- for acc in idl.accounts.iter_mut() {
- acc.name = acc.name.to_lower_camel_case();
- }
- let idl_json = serde_json::to_string_pretty(&idl)?;
- Ok(format!(
- r#"export type {} = {};
- export const IDL: {} = {};
- "#,
- idl.name.to_upper_camel_case(),
- idl_json,
- idl.name.to_upper_camel_case(),
- idl_json
- ))
- }
- pub fn deploy_js_script_host(cluster_url: &str, script_path: &str) -> String {
- format!(
- r#"
- const anchor = require('@coral-xyz/anchor');
- // Deploy script defined by the user.
- const userScript = require("{script_path}");
- async function main() {{
- const url = "{cluster_url}";
- const preflightCommitment = 'recent';
- const connection = new anchor.web3.Connection(url, preflightCommitment);
- const wallet = anchor.Wallet.local();
- const provider = new anchor.AnchorProvider(connection, wallet, {{
- preflightCommitment,
- commitment: 'recent',
- }});
- // Run the user's deploy script.
- userScript(provider);
- }}
- main();
- "#,
- )
- }
- pub fn deploy_ts_script_host(cluster_url: &str, script_path: &str) -> String {
- format!(
- r#"import * as anchor from '@coral-xyz/anchor';
- // Deploy script defined by the user.
- const userScript = require("{script_path}");
- async function main() {{
- const url = "{cluster_url}";
- const preflightCommitment = 'recent';
- const connection = new anchor.web3.Connection(url, preflightCommitment);
- const wallet = anchor.Wallet.local();
- const provider = new anchor.AnchorProvider(connection, wallet, {{
- preflightCommitment,
- commitment: 'recent',
- }});
- // Run the user's deploy script.
- userScript(provider);
- }}
- main();
- "#,
- )
- }
- pub fn deploy_script() -> &'static str {
- r#"// Migrations are an early feature. Currently, they're nothing more than this
- // single deploy script that's invoked from the CLI, injecting a provider
- // configured from the workspace's Anchor.toml.
- const anchor = require("@coral-xyz/anchor");
- module.exports = async function (provider) {
- // Configure client to use the provider.
- anchor.setProvider(provider);
- // Add your deploy script here.
- };
- "#
- }
- pub fn ts_deploy_script() -> &'static str {
- r#"// Migrations are an early feature. Currently, they're nothing more than this
- // single deploy script that's invoked from the CLI, injecting a provider
- // configured from the workspace's Anchor.toml.
- const anchor = require("@coral-xyz/anchor");
- module.exports = async function (provider) {
- // Configure client to use the provider.
- anchor.setProvider(provider);
- // Add your deploy script here.
- };
- "#
- }
- pub fn mocha(name: &str) -> String {
- format!(
- r#"const anchor = require("@coral-xyz/anchor");
- describe("{}", () => {{
- // Configure the client to use the local cluster.
- anchor.setProvider(anchor.AnchorProvider.env());
- it("Is initialized!", async () => {{
- // Add your test here.
- const program = anchor.workspace.{};
- const tx = await program.methods.initialize().rpc();
- console.log("Your transaction signature", tx);
- }});
- }});
- "#,
- name,
- name.to_upper_camel_case(),
- )
- }
- pub fn jest(name: &str) -> String {
- format!(
- r#"const anchor = require("@coral-xyz/anchor");
- describe("{}", () => {{
- // Configure the client to use the local cluster.
- anchor.setProvider(anchor.AnchorProvider.env());
- it("Is initialized!", async () => {{
- // Add your test here.
- const program = anchor.workspace.{};
- const tx = await program.methods.initialize().rpc();
- console.log("Your transaction signature", tx);
- }});
- }});
- "#,
- name,
- name.to_upper_camel_case(),
- )
- }
- pub fn package_json(jest: bool) -> String {
- if jest {
- format!(
- r#"{{
- "scripts": {{
- "lint:fix": "prettier */*.js \"*/**/*{{.js,.ts}}\" -w",
- "lint": "prettier */*.js \"*/**/*{{.js,.ts}}\" --check"
- }},
- "dependencies": {{
- "@coral-xyz/anchor": "^{VERSION}"
- }},
- "devDependencies": {{
- "jest": "^29.0.3",
- "prettier": "^2.6.2"
- }}
- }}
- "#
- )
- } else {
- format!(
- r#"{{
- "scripts": {{
- "lint:fix": "prettier */*.js \"*/**/*{{.js,.ts}}\" -w",
- "lint": "prettier */*.js \"*/**/*{{.js,.ts}}\" --check"
- }},
- "dependencies": {{
- "@coral-xyz/anchor": "^{VERSION}"
- }},
- "devDependencies": {{
- "chai": "^4.3.4",
- "mocha": "^9.0.3",
- "prettier": "^2.6.2"
- }}
- }}
- "#
- )
- }
- }
- pub fn ts_package_json(jest: bool) -> String {
- if jest {
- format!(
- r#"{{
- "scripts": {{
- "lint:fix": "prettier */*.js \"*/**/*{{.js,.ts}}\" -w",
- "lint": "prettier */*.js \"*/**/*{{.js,.ts}}\" --check"
- }},
- "dependencies": {{
- "@coral-xyz/anchor": "^{VERSION}"
- }},
- "devDependencies": {{
- "@types/bn.js": "^5.1.0",
- "@types/jest": "^29.0.3",
- "jest": "^29.0.3",
- "prettier": "^2.6.2",
- "ts-jest": "^29.0.2",
- "typescript": "^4.3.5"
- }}
- }}
- "#
- )
- } else {
- format!(
- r#"{{
- "scripts": {{
- "lint:fix": "prettier */*.js \"*/**/*{{.js,.ts}}\" -w",
- "lint": "prettier */*.js \"*/**/*{{.js,.ts}}\" --check"
- }},
- "dependencies": {{
- "@coral-xyz/anchor": "^{VERSION}"
- }},
- "devDependencies": {{
- "chai": "^4.3.4",
- "mocha": "^9.0.3",
- "ts-mocha": "^10.0.0",
- "@types/bn.js": "^5.1.0",
- "@types/chai": "^4.3.0",
- "@types/mocha": "^9.0.0",
- "typescript": "^4.3.5",
- "prettier": "^2.6.2"
- }}
- }}
- "#
- )
- }
- }
- pub fn ts_mocha(name: &str) -> String {
- format!(
- r#"import * as anchor from "@coral-xyz/anchor";
- import {{ Program }} from "@coral-xyz/anchor";
- import {{ {} }} from "../target/types/{}";
- describe("{}", () => {{
- // Configure the client to use the local cluster.
- anchor.setProvider(anchor.AnchorProvider.env());
- const program = anchor.workspace.{} as Program<{}>;
- it("Is initialized!", async () => {{
- // Add your test here.
- const tx = await program.methods.initialize().rpc();
- console.log("Your transaction signature", tx);
- }});
- }});
- "#,
- name.to_upper_camel_case(),
- name.to_snake_case(),
- name,
- name.to_upper_camel_case(),
- name.to_upper_camel_case(),
- )
- }
- pub fn ts_jest(name: &str) -> String {
- format!(
- r#"import * as anchor from "@coral-xyz/anchor";
- import {{ Program }} from "@coral-xyz/anchor";
- import {{ {} }} from "../target/types/{}";
- describe("{}", () => {{
- // Configure the client to use the local cluster.
- anchor.setProvider(anchor.AnchorProvider.env());
- const program = anchor.workspace.{} as Program<{}>;
- it("Is initialized!", async () => {{
- // Add your test here.
- const tx = await program.methods.initialize().rpc();
- console.log("Your transaction signature", tx);
- }});
- }});
- "#,
- name.to_upper_camel_case(),
- name.to_snake_case(),
- name,
- name.to_upper_camel_case(),
- name.to_upper_camel_case(),
- )
- }
- pub fn ts_config(jest: bool) -> &'static str {
- if jest {
- r#"{
- "compilerOptions": {
- "types": ["jest"],
- "typeRoots": ["./node_modules/@types"],
- "lib": ["es2015"],
- "module": "commonjs",
- "target": "es6",
- "esModuleInterop": true
- }
- }
- "#
- } else {
- r#"{
- "compilerOptions": {
- "types": ["mocha", "chai"],
- "typeRoots": ["./node_modules/@types"],
- "lib": ["es2015"],
- "module": "commonjs",
- "target": "es6",
- "esModuleInterop": true
- }
- }
- "#
- }
- }
- pub fn git_ignore() -> &'static str {
- r#"
- .anchor
- .DS_Store
- target
- **/*.rs.bk
- node_modules
- test-ledger
- .yarn
- "#
- }
- pub fn prettier_ignore() -> &'static str {
- r#"
- .anchor
- .DS_Store
- target
- node_modules
- dist
- build
- test-ledger
- "#
- }
- pub fn node_shell(
- cluster_url: &str,
- wallet_path: &str,
- programs: Vec<ProgramWorkspace>,
- ) -> Result<String> {
- let mut eval_string = format!(
- r#"
- const anchor = require('@coral-xyz/anchor');
- const web3 = anchor.web3;
- const PublicKey = anchor.web3.PublicKey;
- const Keypair = anchor.web3.Keypair;
- const __wallet = new anchor.Wallet(
- Keypair.fromSecretKey(
- Buffer.from(
- JSON.parse(
- require('fs').readFileSync(
- "{wallet_path}",
- {{
- encoding: "utf-8",
- }},
- ),
- ),
- ),
- ),
- );
- const __connection = new web3.Connection("{cluster_url}", "processed");
- const provider = new anchor.AnchorProvider(__connection, __wallet, {{
- commitment: "processed",
- preflightcommitment: "processed",
- }});
- anchor.setProvider(provider);
- "#,
- );
- for program in programs {
- write!(
- &mut eval_string,
- r#"
- anchor.workspace.{} = new anchor.Program({}, new PublicKey("{}"), provider);
- "#,
- program.name.to_upper_camel_case(),
- serde_json::to_string(&program.idl)?,
- program.program_id
- )?;
- }
- Ok(eval_string)
- }
- /// Test initialization template
- #[derive(Clone, Debug, Default, Eq, PartialEq, Parser, ValueEnum)]
- pub enum TestTemplate {
- /// Generate template for Mocha unit-test
- #[default]
- Mocha,
- /// Generate template for Jest unit-test
- Jest,
- /// Generate template for Rust unit-test
- Rust,
- }
- impl TestTemplate {
- pub fn get_test_script(&self, js: bool) -> &str {
- match &self {
- Self::Mocha => {
- if js {
- "yarn run mocha -t 1000000 tests/"
- } else {
- "yarn run ts-mocha -p ./tsconfig.json -t 1000000 tests/**/*.ts"
- }
- }
- Self::Jest => {
- if js {
- "yarn run jest"
- } else {
- "yarn run jest --preset ts-jest"
- }
- }
- Self::Rust => "cargo test",
- }
- }
- pub fn create_test_files(
- &self,
- project_name: &str,
- js: bool,
- solidity: bool,
- program_id: &str,
- ) -> Result<()> {
- match self {
- Self::Mocha => {
- // Build the test suite.
- fs::create_dir_all("tests")?;
- if js {
- let mut test = File::create(format!("tests/{}.js", &project_name))?;
- if solidity {
- test.write_all(solidity_template::mocha(project_name).as_bytes())?;
- } else {
- test.write_all(mocha(project_name).as_bytes())?;
- }
- } else {
- let mut mocha = File::create(format!("tests/{}.ts", &project_name))?;
- if solidity {
- mocha.write_all(solidity_template::ts_mocha(project_name).as_bytes())?;
- } else {
- mocha.write_all(ts_mocha(project_name).as_bytes())?;
- }
- }
- }
- Self::Jest => {
- // Build the test suite.
- fs::create_dir_all("tests")?;
- let mut test = File::create(format!("tests/{}.test.js", &project_name))?;
- if solidity {
- test.write_all(solidity_template::jest(project_name).as_bytes())?;
- } else {
- test.write_all(jest(project_name).as_bytes())?;
- }
- }
- Self::Rust => {
- // Do not initilize git repo
- let exit = std::process::Command::new("cargo")
- .arg("new")
- .arg("--vcs")
- .arg("none")
- .arg("--lib")
- .arg("tests")
- .stderr(Stdio::inherit())
- .output()
- .map_err(|e| anyhow::format_err!("{}", e.to_string()))?;
- if !exit.status.success() {
- eprintln!("'cargo new --lib tests' failed");
- std::process::exit(exit.status.code().unwrap_or(1));
- }
- let mut files = Vec::new();
- let tests_path = Path::new("tests");
- files.extend(vec![(
- tests_path.join("Cargo.toml"),
- tests_cargo_toml(project_name),
- )]);
- files.extend(create_program_template_rust_test(
- project_name,
- tests_path,
- program_id,
- ));
- override_or_create_files(&files)?;
- }
- }
- Ok(())
- }
- }
- pub fn tests_cargo_toml(name: &str) -> String {
- format!(
- r#"[package]
- name = "tests"
- version = "0.1.0"
- description = "Created with Anchor"
- edition = "2021"
- [dependencies]
- anchor-client = "{0}"
- {1} = {{ version = "0.1.0", path = "../programs/{1}" }}
- "#,
- VERSION, name,
- )
- }
- /// Generate template for Rust unit-test
- fn create_program_template_rust_test(name: &str, tests_path: &Path, program_id: &str) -> Files {
- let src_path = tests_path.join("src");
- vec![
- (
- src_path.join("lib.rs"),
- r#"#[cfg(test)]
- mod test_initialize;
- "#
- .into(),
- ),
- (
- src_path.join("test_initialize.rs"),
- format!(
- r#"use std::str::FromStr;
- use anchor_client::{{
- solana_sdk::{{
- commitment_config::CommitmentConfig, pubkey::Pubkey, signature::read_keypair_file,
- }},
- Client, Cluster,
- }};
- #[test]
- fn test_initialize() {{
- let program_id = "{0}";
- let anchor_wallet = std::env::var("ANCHOR_WALLET").unwrap();
- let payer = read_keypair_file(&anchor_wallet).unwrap();
- let client = Client::new_with_options(Cluster::Localnet, &payer, CommitmentConfig::confirmed());
- let program_id = Pubkey::from_str(program_id).unwrap();
- let program = client.program(program_id).unwrap();
- let tx = program
- .request()
- .accounts({1}::accounts::Initialize {{}})
- .args({1}::instruction::Initialize {{}})
- .send()
- .expect("");
- println!("Your transaction signature {{}}", tx);
- }}
- "#,
- program_id,
- name.to_snake_case(),
- ),
- ),
- ]
- }
|