create_contract.rs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. // SPDX-License-Identifier: Apache-2.0
  2. use crate::{
  3. account_new, build_solidity, create_program_address, Account, AccountState, BorshToken,
  4. };
  5. use base58::{FromBase58, ToBase58};
  6. #[test]
  7. fn simple_create_contract_no_seed() {
  8. let mut vm = build_solidity(
  9. r#"
  10. contract bar0 {
  11. function test_other(address foo, address payer) public returns (bar1) {
  12. bar1 x = new bar1{address: foo}("yo from bar0", payer);
  13. return x;
  14. }
  15. function call_bar1_at_address(bar1 a, string x) public {
  16. a.say_hello(x);
  17. }
  18. }
  19. @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
  20. contract bar1 {
  21. @payer(payer)
  22. constructor(string v, address payer) {
  23. print("bar1 says: " + v);
  24. }
  25. function say_hello(string v) public {
  26. print("Hello {}".format(v));
  27. }
  28. }"#,
  29. );
  30. vm.set_program(0);
  31. vm.constructor(&[]);
  32. let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
  33. .from_base58()
  34. .unwrap()
  35. .try_into()
  36. .unwrap();
  37. let acc = account_new();
  38. let payer = account_new();
  39. println!("new account: {}", acc.to_base58());
  40. vm.account_data.insert(
  41. acc,
  42. AccountState {
  43. data: Vec::new(),
  44. owner: Some(program_id),
  45. lamports: 0,
  46. },
  47. );
  48. let bar1 = vm
  49. .function(
  50. "test_other",
  51. &[BorshToken::Address(acc), BorshToken::Address(payer)],
  52. )
  53. .unwrap();
  54. assert_eq!(vm.logs, "bar1 says: yo from bar0");
  55. assert_eq!(vm.account_data[&acc].data.len(), 16);
  56. vm.logs.truncate(0);
  57. vm.function(
  58. "call_bar1_at_address",
  59. &[bar1, BorshToken::String(String::from("xywoleh"))],
  60. );
  61. assert_eq!(vm.logs, "Hello xywoleh");
  62. }
  63. #[test]
  64. fn simple_create_contract() {
  65. let mut vm = build_solidity(
  66. r#"
  67. contract bar0 {
  68. function test_other(address foo, address payer) public returns (bar1) {
  69. bar1 x = new bar1{address: foo}("yo from bar0", payer);
  70. return x;
  71. }
  72. function call_bar1_at_address(bar1 a, string x) public {
  73. a.say_hello(x);
  74. }
  75. }
  76. @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
  77. contract bar1 {
  78. @payer(pay)
  79. constructor(string v, address pay) {
  80. print("bar1 says: " + v);
  81. }
  82. function say_hello(string v) public {
  83. print("Hello {}".format(v));
  84. }
  85. }"#,
  86. );
  87. vm.set_program(0);
  88. vm.constructor(&[]);
  89. let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
  90. .from_base58()
  91. .unwrap()
  92. .try_into()
  93. .unwrap();
  94. let seed = vm.create_pda(&program_id);
  95. let payer = account_new();
  96. let bar1 = vm
  97. .function(
  98. "test_other",
  99. &[BorshToken::Address(seed.0), BorshToken::Address(payer)],
  100. )
  101. .unwrap();
  102. assert_eq!(vm.logs, "bar1 says: yo from bar0");
  103. vm.logs.truncate(0);
  104. println!("next test, {bar1:?}");
  105. vm.function(
  106. "call_bar1_at_address",
  107. &[bar1, BorshToken::String(String::from("xywoleh"))],
  108. );
  109. assert_eq!(vm.logs, "Hello xywoleh");
  110. }
  111. #[test]
  112. fn create_contract_wrong_program_id() {
  113. let mut vm = build_solidity(
  114. r#"
  115. @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
  116. contract bar0 {}
  117. "#,
  118. );
  119. vm.constructor(&[]);
  120. let program = &vm.programs[0].program;
  121. let code = vm.account_data[program].data.clone();
  122. let mut vm = build_solidity(
  123. r#"
  124. @program_id("25UGQeMKp1YH8dR1WBtaj26iqfc49xjwfvLnUKavcz8E")
  125. contract bar0 {}
  126. "#,
  127. );
  128. let program = &vm.programs[0].program;
  129. vm.account_data.get_mut(program).unwrap().data = code;
  130. vm.constructor_expected(7 << 32, &[]);
  131. assert_eq!(
  132. vm.logs,
  133. "program_id should be CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
  134. );
  135. }
  136. #[test]
  137. fn create_contract_with_payer() {
  138. let mut vm = build_solidity(
  139. r#"
  140. contract x {
  141. uint64 v;
  142. @payer(p)
  143. constructor(address p) {
  144. v = 102;
  145. }
  146. function f() public returns (uint64) {
  147. return v;
  148. }
  149. }"#,
  150. );
  151. // 'remove' the data account
  152. let data = vm.programs[0].data;
  153. vm.account_data.get_mut(&data).unwrap().data.truncate(0);
  154. let payer = account_new();
  155. vm.constructor(&[BorshToken::Address(payer)]);
  156. let ret = vm.function("f", &[]).unwrap();
  157. assert_eq!(
  158. ret,
  159. BorshToken::Uint {
  160. width: 64,
  161. value: 102.into()
  162. }
  163. );
  164. }
  165. #[test]
  166. // 64424509440 = 15 << 32 (ERROR_NEW_ACCOUNT_NEEDED)
  167. fn missing_contract() {
  168. let mut vm = build_solidity(
  169. r#"
  170. contract bar0 {
  171. function test_other(address foo) public returns (bar1) {
  172. bar1 x = new bar1{address: foo}("yo from bar0");
  173. return x;
  174. }
  175. function call_bar1_at_address(bar1 a, string x) public {
  176. a.say_hello(x);
  177. }
  178. }
  179. @program_id("7vJKRaKLGCNUPuHWdeHCTknkYf3dHXXEZ6ri7dc6ngeV")
  180. contract bar1 {
  181. constructor(string v) {
  182. print("bar1 says: " + v);
  183. }
  184. function say_hello(string v) public {
  185. print("Hello {}".format(v));
  186. }
  187. }"#,
  188. );
  189. vm.set_program(0);
  190. vm.constructor(&[]);
  191. let missing = account_new();
  192. let res = vm.function_must_fail("test_other", &[BorshToken::Address(missing)]);
  193. assert_eq!(res.unwrap(), 64424509440);
  194. }
  195. #[test]
  196. fn two_contracts() {
  197. let mut vm = build_solidity(
  198. r#"
  199. contract bar0 {
  200. function test_other(address a, address b) public returns (bar1) {
  201. bar1 x = new bar1{address: a}("yo from bar0");
  202. bar1 y = new bar1{address: b}("hi from bar0");
  203. return x;
  204. }
  205. }
  206. @program_id("CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT")
  207. contract bar1 {
  208. @payer(address"3wvhRNAJSDCk5Mub8NEcShszRrHHVDHsuSUdAcL2aaMV")
  209. constructor(string v) {
  210. print("bar1 says: " + v);
  211. }
  212. }"#,
  213. );
  214. vm.set_program(0);
  215. vm.constructor(&[]);
  216. let program_id: Account = "CPDgqnhHDCsjFkJKMturRQ1QeM9EXZg3EYCeDoRP8pdT"
  217. .from_base58()
  218. .unwrap()
  219. .try_into()
  220. .unwrap();
  221. let seed1 = vm.create_pda(&program_id);
  222. let seed2 = vm.create_pda(&program_id);
  223. let _bar1 = vm.function(
  224. "test_other",
  225. &[BorshToken::Address(seed1.0), BorshToken::Address(seed2.0)],
  226. );
  227. assert_eq!(vm.logs, "bar1 says: yo from bar0bar1 says: hi from bar0");
  228. vm.logs.truncate(0);
  229. }
  230. #[test]
  231. fn account_too_small() {
  232. let mut vm = build_solidity(
  233. r#"
  234. contract bar {
  235. int[200] foo1;
  236. }"#,
  237. );
  238. let data = vm.stack[0].data;
  239. vm.account_data.get_mut(&data).unwrap().data.truncate(100);
  240. vm.constructor_expected(5 << 32, &[]);
  241. }
  242. #[test]
  243. fn account_with_space() {
  244. let mut vm = build_solidity(
  245. r#"
  246. contract bar {
  247. @space(102 * x)
  248. @payer(payer)
  249. constructor(uint64 x, address payer) {}
  250. function hello() public returns (bool) {
  251. return true;
  252. }
  253. }
  254. "#,
  255. );
  256. let data = vm.stack[0].data;
  257. vm.account_data.get_mut(&data).unwrap().data.truncate(0);
  258. let payer = account_new();
  259. vm.constructor(&[
  260. BorshToken::Uint {
  261. width: 64,
  262. value: 3.into(),
  263. },
  264. BorshToken::Address(payer),
  265. ]);
  266. assert_eq!(vm.account_data.get_mut(&data).unwrap().data.len(), 3 * 102);
  267. let ret = vm.function("hello", &[]).unwrap();
  268. assert_eq!(ret, BorshToken::Bool(true));
  269. }
  270. #[test]
  271. fn account_with_seed() {
  272. let mut vm = build_solidity(
  273. r#"
  274. contract bar {
  275. @space(511 + 102)
  276. @seed(seed)
  277. @payer(payer)
  278. constructor(bytes seed, address payer) {}
  279. function hello() public returns (bool) {
  280. return true;
  281. }
  282. }
  283. "#,
  284. );
  285. let program_id = vm.stack[0].program;
  286. let seed = vm.create_pda(&program_id);
  287. vm.stack[0].data = seed.0;
  288. let payer = account_new();
  289. vm.constructor(&[BorshToken::Bytes(seed.1), BorshToken::Address(payer)]);
  290. assert_eq!(
  291. vm.account_data.get_mut(&seed.0).unwrap().data.len(),
  292. 511 + 102
  293. );
  294. let ret = vm.function("hello", &[]).unwrap();
  295. assert_eq!(ret, BorshToken::Bool(true));
  296. }
  297. #[test]
  298. fn account_with_seed_bump() {
  299. let mut vm = build_solidity(
  300. r#"
  301. contract bar {
  302. @space(511 + 102)
  303. @seed(seed)
  304. @bump(b)
  305. @payer(payer)
  306. constructor(bytes seed, address payer, byte b) {}
  307. function hello() public returns (bool) {
  308. return true;
  309. }
  310. }
  311. "#,
  312. );
  313. let program_id = vm.stack[0].program;
  314. let mut seed = vm.create_pda(&program_id);
  315. let bump = seed.1.pop().unwrap();
  316. vm.stack[0].data = seed.0;
  317. let payer = account_new();
  318. vm.constructor(&[
  319. BorshToken::Bytes(seed.1),
  320. BorshToken::Address(payer),
  321. BorshToken::Uint {
  322. width: 8,
  323. value: bump.into(),
  324. },
  325. ]);
  326. assert_eq!(
  327. vm.account_data.get_mut(&seed.0).unwrap().data.len(),
  328. 511 + 102
  329. );
  330. let ret = vm.function("hello", &[]).unwrap();
  331. assert_eq!(ret, BorshToken::Bool(true));
  332. }
  333. #[test]
  334. fn account_with_seed_bump_literals() {
  335. let mut vm = build_solidity(
  336. r#"
  337. @program_id("vS5Tf8mnHGbUCMLQWrnvsFvwHLfA5p3yQM3ozxPckn8")
  338. contract bar {
  339. @space(2 << 8 + 4)
  340. @seed("meh")
  341. @bump(33) // 33 = ascii !
  342. @payer(address"vS5Tf8mnHGbUCMLQWrnvsFvwHLfA5p3yQM3ozxPckn8")
  343. constructor() {}
  344. function hello() public returns (bool) {
  345. return true;
  346. }
  347. }
  348. "#,
  349. );
  350. let program_id = vm.stack[0].program;
  351. let account = create_program_address(&program_id, &[b"meh!"]);
  352. vm.create_empty_account(&account.0, &program_id);
  353. vm.stack[0].data = account.0;
  354. vm.constructor(&[]);
  355. assert_eq!(
  356. vm.account_data.get_mut(&account.0).unwrap().data.len(),
  357. 8192
  358. );
  359. let ret = vm.function("hello", &[]).unwrap();
  360. assert_eq!(ret, BorshToken::Bool(true));
  361. }
  362. #[test]
  363. fn create_child() {
  364. let mut vm = build_solidity(
  365. r#"
  366. contract creator {
  367. Child public c;
  368. function create_child(address child, address payer) public {
  369. print("Going to create child");
  370. c = new Child{address: child}(payer);
  371. c.say_hello();
  372. }
  373. }
  374. @program_id("Chi1d5XD6nTAp2EyaNGqMxZzUjh6NvhXRxbGHP3D1RaT")
  375. contract Child {
  376. @payer(payer)
  377. @space(511 + 7)
  378. constructor(address payer) {
  379. print("In child constructor");
  380. }
  381. function say_hello() pure public {
  382. print("Hello there");
  383. }
  384. }"#,
  385. );
  386. vm.set_program(0);
  387. vm.constructor(&[]);
  388. let payer = account_new();
  389. let program_id = vm.stack[0].program;
  390. let seed = vm.create_pda(&program_id);
  391. vm.function(
  392. "create_child",
  393. &[BorshToken::Address(seed.0), BorshToken::Address(payer)],
  394. );
  395. assert_eq!(
  396. vm.logs,
  397. "Going to create childIn child constructorHello there"
  398. );
  399. }