| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801 |
- // SPDX-License-Identifier: Apache-2.0
- use crate::{
- account_new, build_solidity, create_program_address, Account, AccountMeta, AccountState,
- BorshToken, Pubkey,
- };
- use base58::{FromBase58, ToBase58};
- #[test]
- fn simple_create_contract_no_seed() {
- let mut vm = build_solidity(
- r#"
- contract bar0 {
- function test_other(address foo) external returns (bar1) {
- bar1 x = new bar1{address: foo}("yo from bar0");
- return x;
- }
- function call_bar1_at_address(bar1 a, string x) public {
- a.say_hello(x);
- }
- }
- @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
- contract bar1 {
- @payer(payer)
- constructor(string v) {
- print("bar1 says: " + v);
- }
- function say_hello(string v) public {
- print("Hello {}".format(v));
- }
- }"#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
- .from_base58()
- .unwrap()
- .try_into()
- .unwrap();
- let acc = account_new();
- let payer = account_new();
- println!("new account: {}", acc.to_base58());
- vm.account_data.insert(payer, AccountState::default());
- vm.account_data.insert(
- acc,
- AccountState {
- data: Vec::new(),
- owner: Some(program_id),
- lamports: 0,
- },
- );
- let bar1 = vm
- .function("test_other")
- .arguments(&[BorshToken::Address(acc)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("payer", payer),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[AccountMeta {
- pubkey: Pubkey(acc),
- is_writable: true,
- is_signer: true,
- }])
- .call()
- .unwrap();
- assert_eq!(vm.logs, "bar1 says: yo from bar0");
- assert_eq!(vm.account_data[&acc].data.len(), 16);
- vm.logs.truncate(0);
- vm.function("call_bar1_at_address")
- .arguments(&[bar1, BorshToken::String(String::from("xywoleh"))])
- .accounts(vec![
- ("dataAccount", data_account),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[
- AccountMeta {
- pubkey: Pubkey(acc),
- is_signer: false,
- is_writable: false,
- },
- AccountMeta {
- pubkey: Pubkey(program_id),
- is_writable: false,
- is_signer: false,
- },
- ])
- .call();
- assert_eq!(vm.logs, "Hello xywoleh");
- }
- #[test]
- fn simple_create_contract() {
- let mut vm = build_solidity(
- r#"
- contract bar0 {
- function test_other(address foo) external returns (bar1) {
- bar1 x = new bar1{address: foo}("yo from bar0");
- return x;
- }
- function call_bar1_at_address(bar1 a, string x) public {
- a.say_hello(x);
- }
- }
- @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
- contract bar1 {
- @payer(pay)
- constructor(string v) {
- print("bar1 says: " + v);
- }
- function say_hello(string v) public {
- print("Hello {}".format(v));
- }
- }"#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
- .from_base58()
- .unwrap()
- .try_into()
- .unwrap();
- let seed = vm.create_pda(&program_id, 7);
- let payer = account_new();
- vm.account_data.insert(payer, AccountState::default());
- let bar1 = vm
- .function("test_other")
- .arguments(&[BorshToken::Address(seed.0)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("pay", payer),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[AccountMeta {
- pubkey: Pubkey(seed.0),
- is_signer: false,
- is_writable: true,
- }])
- .call()
- .unwrap();
- assert_eq!(vm.logs, "bar1 says: yo from bar0");
- vm.logs.truncate(0);
- println!("next test, {bar1:?}");
- vm.function("call_bar1_at_address")
- .arguments(&[bar1, BorshToken::String(String::from("xywoleh"))])
- .accounts(vec![
- ("dataAccount", data_account),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[
- AccountMeta {
- pubkey: Pubkey(seed.0),
- is_signer: false,
- is_writable: false,
- },
- AccountMeta {
- pubkey: Pubkey(program_id),
- is_writable: false,
- is_signer: false,
- },
- ])
- .call();
- assert_eq!(vm.logs, "Hello xywoleh");
- }
- #[test]
- fn create_contract_wrong_program_id() {
- let mut vm = build_solidity(
- r#"
- @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
- contract bar0 {}
- "#,
- );
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let program = &vm.programs[0].id;
- let code = vm.account_data[program].data.clone();
- let mut vm = build_solidity(
- r#"
- @program_id("25UGQeMKp1YH8dR1WBtaj26iqfc49xjwfvLnUKavcz8E")
- contract bar0 {}
- "#,
- );
- let program = &vm.programs[0].id;
- vm.account_data.get_mut(program).unwrap().data = code;
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .expected(7 << 32)
- .call();
- assert_eq!(
- vm.logs,
- "program_id should be CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
- );
- }
- #[test]
- fn call_constructor_twice() {
- let mut vm = build_solidity(
- r#"
- @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
- contract bar0 {}
- "#,
- );
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .expected(2)
- .call();
- }
- #[test]
- fn create_contract_with_payer() {
- let mut vm = build_solidity(
- r#"
- contract x {
- uint64 v;
- @payer(p)
- constructor() {
- v = 102;
- }
- function f() public returns (uint64) {
- return v;
- }
- }"#,
- );
- let payer = account_new();
- vm.account_data.insert(payer, AccountState::default());
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![
- ("dataAccount", data_account),
- ("p", payer),
- ("systemProgram", [0; 32]),
- ])
- .call();
- let ret = vm
- .function("f")
- .accounts(vec![("dataAccount", data_account)])
- .call()
- .unwrap();
- assert_eq!(
- ret,
- BorshToken::Uint {
- width: 64,
- value: 102.into()
- }
- );
- }
- #[test]
- #[should_panic(expected = "external call failed")]
- // 64424509440 = 15 << 32 (ERROR_NEW_ACCOUNT_NEEDED)
- fn missing_contract() {
- let mut vm = build_solidity(
- r#"
- contract bar0 {
- function test_other(address foo) external returns (bar1) {
- bar1 x = new bar1{address: foo}("yo from bar0");
- return x;
- }
- function call_bar1_at_address(bar1 a, string x) public {
- a.say_hello(x);
- }
- }
- @program_id("7vJKRaKLGCNUPuHWdeHCTknkYf3dHXXEZ6ri7dc6ngeV")
- contract bar1 {
- constructor(string v) {
- print("bar1 says: " + v);
- }
- function say_hello(string v) public {
- print("Hello {}".format(v));
- }
- }"#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let missing = account_new();
- vm.logs.clear();
- vm.account_data.insert(missing, AccountState::default());
- let program_id: Account = "7vJKRaKLGCNUPuHWdeHCTknkYf3dHXXEZ6ri7dc6ngeV"
- .from_base58()
- .unwrap()
- .try_into()
- .unwrap();
- // There is no payer account, so the external call fails.
- let _ = vm
- .function("test_other")
- .arguments(&[BorshToken::Address(missing)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[
- AccountMeta {
- pubkey: Pubkey(missing),
- is_signer: true,
- is_writable: false,
- },
- AccountMeta {
- pubkey: Pubkey(program_id),
- is_writable: false,
- is_signer: false,
- },
- ])
- .must_fail();
- }
- #[test]
- fn two_contracts() {
- let mut vm = build_solidity(
- r#"
- contract bar0 {
- function test_other(address a, address b) external returns (bar1) {
- bar1 x = new bar1{address: a}("yo from bar0");
- bar1 y = new bar1{address: b}("hi from bar0");
- return x;
- }
- }
- @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
- contract bar1 {
- @payer(payer_account)
- constructor(string v) {
- print("bar1 says: " + v);
- }
- }"#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
- .from_base58()
- .unwrap()
- .try_into()
- .unwrap();
- let seed1 = vm.create_pda(&program_id, 5);
- let seed2 = vm.create_pda(&program_id, 5);
- let payer = account_new();
- vm.account_data.insert(seed1.0, AccountState::default());
- vm.account_data.insert(seed2.0, AccountState::default());
- vm.account_data.insert(payer, AccountState::default());
- let _bar1 = vm
- .function("test_other")
- .arguments(&[BorshToken::Address(seed1.0), BorshToken::Address(seed2.0)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("systemProgram", [0; 32]),
- ("payer_account", payer),
- ])
- .remaining_accounts(&[
- AccountMeta {
- pubkey: Pubkey(seed1.0),
- is_signer: true,
- is_writable: true,
- },
- AccountMeta {
- pubkey: Pubkey(seed2.0),
- is_signer: true,
- is_writable: true,
- },
- AccountMeta {
- pubkey: Pubkey(program_id),
- is_writable: false,
- is_signer: false,
- },
- ])
- .call();
- assert_eq!(vm.logs, "bar1 says: yo from bar0bar1 says: hi from bar0");
- vm.logs.truncate(0);
- }
- #[test]
- fn account_too_small() {
- let mut vm = build_solidity(
- r#"
- contract bar {
- int[200] foo1;
- }"#,
- );
- let data_account = vm.initialize_data_account();
- vm.account_data
- .get_mut(&data_account)
- .unwrap()
- .data
- .truncate(100);
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .expected(5 << 32)
- .call();
- }
- #[test]
- fn account_with_space() {
- let mut vm = build_solidity(
- r#"
- contract bar {
- @payer(payer)
- constructor(@space uint64 x) {}
- function hello() public returns (bool) {
- return true;
- }
- }
- "#,
- );
- let data_account = vm.initialize_data_account();
- vm.account_data
- .get_mut(&data_account)
- .unwrap()
- .data
- .truncate(0);
- let payer = account_new();
- vm.account_data.insert(payer, AccountState::default());
- vm.function("new")
- .accounts(vec![
- ("dataAccount", data_account),
- ("payer", payer),
- ("systemProgram", [0; 32]),
- ])
- .arguments(&[BorshToken::Uint {
- width: 64,
- value: 306.into(),
- }])
- .call();
- assert_eq!(
- vm.account_data.get_mut(&data_account).unwrap().data.len(),
- 306
- );
- let ret = vm
- .function("hello")
- .accounts(vec![("dataAccount", data_account)])
- .call()
- .unwrap();
- assert_eq!(ret, BorshToken::Bool(true));
- }
- #[test]
- fn account_with_seed() {
- let mut vm = build_solidity(
- r#"
- contract bar {
- @space(511 + 102)
- @payer(payer)
- constructor(@seed bytes seed) {}
- function hello() public returns (bool) {
- return true;
- }
- }
- "#,
- );
- let program_id = vm.stack[0].id;
- let seed = vm.create_pda(&program_id, 7);
- let payer = account_new();
- vm.account_data.insert(payer, AccountState::default());
- vm.function("new")
- .accounts(vec![
- ("dataAccount", seed.0),
- ("payer", payer),
- ("systemProgram", [0; 32]),
- ])
- .arguments(&[BorshToken::Bytes(seed.1)])
- .call();
- assert_eq!(
- vm.account_data.get_mut(&seed.0).unwrap().data.len(),
- 511 + 102
- );
- let ret = vm
- .function("hello")
- .accounts(vec![("dataAccount", seed.0)])
- .call()
- .unwrap();
- assert_eq!(ret, BorshToken::Bool(true));
- }
- #[test]
- fn account_with_seed_bump() {
- let mut vm = build_solidity(
- r#"
- contract bar {
- @space(511 + 102)
- @payer(payer)
- constructor(@seed address seed, @seed bytes2 seed2, @bump byte b) {}
- function hello() public returns (bool) {
- return true;
- }
- }
- "#,
- );
- let program_id = vm.stack[0].id;
- let (address, full_seed) = vm.create_pda(&program_id, 35);
- let bump = full_seed[34];
- let seed_addr = &full_seed[0..32];
- let seed2 = &full_seed[32..34];
- let payer = account_new();
- vm.account_data.insert(payer, AccountState::default());
- vm.function("new")
- .arguments(&[
- BorshToken::Address(seed_addr.try_into().unwrap()),
- BorshToken::FixedBytes(seed2.to_vec()),
- BorshToken::Uint {
- width: 8,
- value: bump.into(),
- },
- ])
- .accounts(vec![
- ("dataAccount", address),
- ("payer", payer),
- ("systemProgram", [0; 32]),
- ])
- .call();
- assert_eq!(
- vm.account_data.get_mut(&address).unwrap().data.len(),
- 511 + 102
- );
- let ret = vm
- .function("hello")
- .accounts(vec![("dataAccount", address)])
- .call()
- .unwrap();
- assert_eq!(ret, BorshToken::Bool(true));
- }
- #[test]
- fn account_with_seed_bump_literals() {
- let mut vm = build_solidity(
- r#"
- @program_id("vS5Tf8mnHGbUCMLQWrnvsFvwHLfA5p3yQM3ozxPckn8")
- contract bar {
- @space(2 << 8 + 4)
- @seed("meh")
- @bump(33) // 33 = ascii !
- @payer(my_account)
- constructor() {}
- function hello() public returns (bool) {
- return true;
- }
- }
- "#,
- );
- let program_id = vm.stack[0].id;
- let account = create_program_address(&program_id, &[b"meh!"]);
- let payer = account_new();
- vm.create_empty_account(&account.0, &program_id);
- vm.account_data.insert(payer, AccountState::default());
- vm.function("new")
- .accounts(vec![
- ("dataAccount", account.0),
- ("my_account", payer),
- ("systemProgram", [0; 32]),
- ])
- .call();
- assert_eq!(
- vm.account_data.get_mut(&account.0).unwrap().data.len(),
- 8192
- );
- let ret = vm
- .function("hello")
- .accounts(vec![("dataAccount", account.0)])
- .call()
- .unwrap();
- assert_eq!(ret, BorshToken::Bool(true));
- }
- #[test]
- fn create_child() {
- let mut vm = build_solidity(
- r#"
- contract creator {
- Child public c;
- function create_child(address child) external {
- print("Going to create child");
- c = new Child{address: child}();
- c.say_hello();
- }
- }
- @program_id("Chi1d5XD6nTAp2EyaNGqMxZzUjh6NvhXRxbGHP3D1RaT")
- contract Child {
- @payer(payer)
- @space(511 + 7)
- constructor() {
- print("In child constructor");
- }
- function say_hello() pure public {
- print("Hello there");
- }
- }"#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let payer = account_new();
- let program_id = vm.stack[0].id;
- let seed = vm.create_pda(&program_id, 7);
- vm.account_data.insert(payer, AccountState::default());
- vm.account_data.insert(seed.0, AccountState::default());
- vm.function("create_child")
- .arguments(&[BorshToken::Address(seed.0)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("payer", payer),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[AccountMeta {
- pubkey: Pubkey(seed.0),
- is_signer: true,
- is_writable: true,
- }])
- .call();
- assert_eq!(
- vm.logs,
- "Going to create childIn child constructorHello there"
- );
- }
- #[test]
- fn create_child_with_meta() {
- let mut vm = build_solidity(
- r#"
- import 'solana';
- contract creator {
- Child public c;
- function create_child_with_meta(address child, address payer) public {
- print("Going to create child");
- AccountMeta[2] metas = [
- AccountMeta({pubkey: child, is_signer: true, is_writable: true}),
- AccountMeta({pubkey: payer, is_signer: true, is_writable: true})
- // Passing the system account here crashes the VM, even if I add it to vm.account_data
- // AccountMeta({pubkey: address"11111111111111111111111111111111", is_writable: false, is_signer: false})
- ];
- c = new Child{accounts: metas}();
- c.say_hello();
- }
- }
- @program_id("Chi1d5XD6nTAp2EyaNGqMxZzUjh6NvhXRxbGHP3D1RaT")
- contract Child {
- @payer(payer)
- @space(511 + 7)
- constructor() {
- print("In child constructor");
- }
- function say_hello() pure public {
- print("Hello there");
- }
- }
- "#,
- );
- vm.set_program(0);
- let data_account = vm.initialize_data_account();
- vm.function("new")
- .accounts(vec![("dataAccount", data_account)])
- .call();
- let payer = account_new();
- let program_id = vm.stack[0].id;
- let seed = vm.create_pda(&program_id, 7);
- vm.account_data.insert(seed.0, AccountState::default());
- vm.account_data.insert(payer, AccountState::default());
- vm.function("create_child_with_meta")
- .arguments(&[BorshToken::Address(seed.0), BorshToken::Address(payer)])
- .accounts(vec![
- ("dataAccount", data_account),
- ("systemProgram", [0; 32]),
- ])
- .remaining_accounts(&[
- AccountMeta {
- pubkey: Pubkey(seed.0),
- is_signer: false,
- is_writable: false,
- },
- AccountMeta {
- pubkey: Pubkey(payer),
- is_signer: true,
- is_writable: false,
- },
- ])
- .call();
- assert_eq!(
- vm.logs,
- "Going to create childIn child constructorHello there"
- );
- }
|