English 简体中文 繁體中文 한국 사람 日本語 Deutsch русский بالعربية TÜRKÇE português คนไทย french

简体中文 繁體中文 English 日本語 Deutsch 한국 사람 بالعربية TÜRKÇE คนไทย Français русский

回答

收藏

4.4 存款/提款逻辑测试

开源社区 开源社区 8483 人阅读 | 0 人回复 | 2025-03-08

在本课中,我们将编写一些有趣的测试。
首先,我们要更新之前编写的合约的初始化过程,因为我们引入了一个新变量,它应该存储在 c4 存储器中。
如果您现在尝试运行我们的测试 - yarn test 您可以看到测试将失败。您还将看到发生在我们合约上的交易跟踪,其中一个交易的错误代码为 9. Have a look at the exit codes documentation.
9 - Cell下溢。从片断基元读取数据时,试图读取的位或引用数超过总数。
更新我们的合约包装器
首先,让我们更新我们的合约包装器,以便它能意识到 c4 存储中的第三个参数。也就是说,我们将更新 MainContractConfig 和 mainContractConfigToCell 函数
  1. // ...library imports: {/ S7 [+ V/ o9 b7 l% v% X* Y

  2. ! M: ~( k+ C3 [6 X5 R
  3. export type MainContractConfig = {! R/ Y6 C( u; ?! a; ~1 Z
  4.   number: number;  S+ M3 U' b5 P( K
  5.   address: Address;
    8 s3 J$ J2 b  R4 ^/ g5 @/ c
  6.   owner_address: Address;
    7 P7 _* [' L( y' A$ f+ C
  7. };
    0 ?: T$ M3 I- [8 L* C
  8. & e$ ?( o# [: h0 R
  9. export function mainContractConfigToCell(config: MainContractConfig): Cell {
    " `; c2 g1 P. y+ {$ t
  10.   return beginCell()
    ; K$ d( \3 a- @3 n; T# e
  11.     .storeUint(config.number, 32)
    6 I6 M% n+ d; \' z; D1 c' o$ r% T
  12.     .storeAddress(config.address)
    ( Z' j0 C# S( _1 \( P& F- D
  13.     .storeAddress(config.owner_address)
    2 m! U4 p& s( t- h! _/ V
  14.     .endCell();
    " g& S1 w5 b& t" ~7 H) n6 z
  15. }% v! c7 I; W6 t0 X0 |0 H0 v
  16. 9 D, {* i. p! p& a& g" ~" O; h* A3 }
  17. // ...contact wrapper class
复制代码
封装文件中还有一点需要更新-getData 方法,这样它也会返回 owner_address 我们还将创建新方法 getBalance,以测试存款后余额是否增加:

' u6 I4 `  L7 i
  1. async getData(provider: ContractProvider) {
    6 E* _- b! i7 \) b* L, k6 O. h2 x
  2.     const { stack } = await provider.get("get_contract_storage_data", []);
    / {/ G! x* H/ v1 r9 Y1 ^! J
  3.     return {& c% m/ R4 S5 s* l2 I
  4.       number: stack.readNumber(),
    # W& P- Q; J# H; Z6 M
  5.       recent_sender: stack.readAddress(),
    * n' g% g" g# G# N% c
  6.       owner_address: stack.readAddress(),  P9 b! e, Y. c6 \
  7.     };
    2 Y0 b* h; H9 X8 _# c, j. ~' f  W" _
  8.   }
    ; \8 }8 ]0 M- N
  9.   
    3 a7 Y# f- }' ^) j" `
  10. async getBalance(provider: ContractProvider) {6 O: ?1 Z. y/ j1 G
  11.     const { stack } = await provider.get("balance", []);
    # a: P4 H  J  I
  12.     return {
    ' [! }- ?" X1 Z: C. ?7 Q) [
  13.       number: stack.readNumber(),
    * M& L/ V- A8 R7 g5 m
  14.     };7 |3 v; ]- q$ e
  15. }
复制代码
  1. // ...library imports3 u6 E# t2 B* V
  2. * f0 z# K( _: `6 ]/ D$ W3 c
  3. describe("main.fc contract tests", () => {% a/ _. G' k8 I, C* j3 u* X
  4.   it("should get the proper most recent sender address", async () => {
    * T  q! r! e( f$ ^5 ~5 I' q4 k
  5.     const blockchain = await Blockchain.create();0 c; e- h1 y# x* r4 K5 K' [
  6.     const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];' O) x; ?* X- L
  7. $ E1 m' T5 b( P+ Z. |
  8.     const initAddress = await blockchain.treasury("initAddress");& g( [! D5 t5 ?: ^6 @; U
  9.     const ownerAddress = await blockchain.treasury("ownerAddress");) L8 ^" k& d* ?, y0 K: m6 N4 m
  10.     + S  Q' c! J5 c" N
  11.     const myContract = blockchain.openContract(
    2 K/ n( L$ ^/ f4 y) B
  12.       await MainContract.createFromConfig(
    2 T$ u; T" \, P0 [* p, M
  13.         {
    $ d$ e! [, r" T5 W+ F2 V
  14.           number: 0,
    ( C. n' l7 X: w7 ]! W
  15.           address: initAddress.address,: N* B# h" C6 v1 C: C
  16.           owner_address: ownerAddress.address, // now we create myContract from 3 parameters- a, a: P. X) W4 `6 V. M
  17.         },- y2 j3 E; i& |$ N/ X
  18.         codeCell
    # @) Y1 [1 O) U/ u
  19.       )9 L0 v2 [8 `9 W; m
  20.     );/ Y; e+ d$ e8 E

  21.   d  [& b- h7 y* a' y
  22. // ...rest of testing code
复制代码

4 e5 f3 H3 g% _9 Y2 Y# G( V
我们引入一个新的金库,它实际上将是合约的所有者,只有这个所有者以后才能提取资金。这些更新在 tests/main.spec.ts 中进行:
  1. // ...library imports: F+ B" t. |, R
  2. . s6 W; T+ D, A! x' C8 S$ |2 q
  3. describe("main.fc contract tests", () => {  X" s! r5 r) f  L, [& l! l, p
  4.   it("should get the proper most recent sender address", async () => {
    # n' i7 l5 E& Z
  5.     const blockchain = await Blockchain.create();* ~3 `( H5 B; d. Y0 D
  6.     const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
    ( ?! ^) H$ ]3 ]0 U$ P: P
  7. 5 @6 `+ q9 D( F2 m" R& l" I
  8.     const initAddress = await blockchain.treasury("initAddress");
    $ ^1 |) Z9 Y6 g; u0 a
  9.     const ownerAddress = await blockchain.treasury("ownerAddress");
    * ]2 a3 f- v& w  j! R6 k" ^+ a+ K
  10.    
    # u/ o! H9 K7 E. N- d
  11.     const myContract = blockchain.openContract(
      u% I4 C1 T+ Q: m
  12.       await MainContract.createFromConfig(
    " i# [) b7 a6 i+ N4 _% o) g
  13.         {$ y. r+ {0 Z3 b+ y+ n$ T
  14.           number: 0,& J; ^* [  i: m" q
  15.           address: initAddress.address,
    - K' {/ \3 S" O) ]  y8 A/ E5 T
  16.           owner_address: ownerAddress.address, // now we create myContract from 3 parameters* H$ d2 c5 d* E, q% n. f. k. Q  T( l
  17.         },
    8 v; |4 L7 Q5 P& p
  18.         codeCell
    # w- y0 h: o. `% C
  19.       )4 J& D+ ^$ Z( E  J6 t! P
  20.     );
    / [0 I, o  ^; H6 `; I8 a9 H; r1 @

  21. / i- P/ P* S% A5 G; ^
  22. // ...rest of testing code
复制代码
现在我们好了。 yarn test 成功通过我们之前编写的测试。
5 s" ?: ]! }# B" w; B/ w3 N3 p# s
存款测试
让我们在 main.spec.ts 文件中再添加几个测试。
由于我们必须在每次新测试前重新启动合约,因此我们还将把合约启动引入 Jest 的 beforeEach 函数:
  1. // We need to additionally import SandboxContract and TreasuryContract
    - J- s) x# k9 z
  2. import { Blockchain, SandboxContract, TreasuryContract } from "@ton/sandbox";0 T: ^+ F% J3 k% ~# `& t
  3. // ... other library imports5 r  Z- H6 Z4 B- M. S. ?
  4. describe("main.fc contract tests", () => {, \4 X0 q, S+ x* r& p. }8 E$ P
  5.   let blockchain: Blockchain;
    * F9 ^8 Y" k9 L) _. v6 l% y
  6.   let myContract: SandboxContract<MainContract>;
    ! u# i$ h; I9 U7 w4 _$ u! g
  7.   let initWallet: SandboxContract<TreasuryContract>;
    + F" W! d7 H( k" G1 f& h' |7 `; f
  8.   let ownerWallet: SandboxContract<TreasuryContract>;* ^/ d, H$ U9 u) ]" G
  9. ' }7 ?! t. s- C4 a
  10.   beforeEach(async () => {: X, x# |. F1 U5 B* e+ S+ p
  11.     blockchain = await Blockchain.create();  @% M8 ~' S9 a
  12.     initWallet = await blockchain.treasury("initWallet");
    3 y' ~+ B7 D# ^7 T. m% L* `. F
  13.     ownerWallet = await blockchain.treasury("ownerWallet");3 o7 c6 r2 x; X- o; H9 y8 X
  14. 3 W: ^% z' `1 @4 ^) B7 Q; v
  15.     const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];& j+ S. l; T  s/ e

  16. 6 l. u8 O- J! B6 Q6 j3 K8 r. d
  17.     myContract = blockchain.openContract(
    6 n2 l5 z* Y7 d6 g5 m
  18.       await MainContract.createFromConfig(7 `# C" B4 j; U* K$ C
  19.         {3 Y( v8 N/ Z8 i8 H  V- \
  20.           number: 0,
    ! M3 c& p# k2 q- V4 Y* V
  21.           address: initWallet.address,
    , W7 r/ T! M4 H/ n- Y/ `  V
  22.           owner_address: ownerWallet.address,0 k" c3 V/ ]6 s4 M1 {* }+ ~
  23.         },
    / A' H1 a- [8 K) M
  24.         codeCell( p4 g0 |) a/ n( u+ z7 V
  25.       )+ v' R- M6 L5 h7 B5 F
  26.     );% ~* |) V. ]+ t* X0 K0 |
  27.   });9 g# m' f4 Q/ ~/ f
  28. 4 X* n& g" m2 s5 m; u" [; ~
  29.   it("should get the proper most recent sender address", async () => {) Q, U: ^. z$ e& C' O& F1 r( M0 C
  30.     const senderWallet = await blockchain.treasury("sender");
    / u- q7 N6 X" J

  31. 9 |3 {% B; \4 i- E5 P- r4 A
  32.     const sentMessageResult = await myContract.sendIncrement(' ?' s3 U# |; @8 K# r2 T$ v& n
  33.       senderWallet.getSender(),
      _; W% h. e1 I9 r
  34.       toNano("0.05"),3 D5 V2 V* i, w& w  x" g
  35.       1" R2 d' s' J/ R- q: H5 R7 |
  36.     );7 [( J( f# x7 B# \
  37. 6 x# q# M. j! t3 R
  38.     expect(sentMessageResult.transactions).toHaveTransaction({
    6 a" N: J) e- ?* c( I
  39.       from: senderWallet.address,5 u* f/ ^7 B* z' j
  40.       to: myContract.address,
    ' B! M3 K: M% S( J# n6 D# T
  41.       success: true,
    7 Q" S: ^; \- m$ m, l4 g1 N3 j
  42.     });
    ; P* p1 o( @, X) u' G' |% i

  43. 6 S" i7 r3 W+ v( H& y
  44.     const data = await myContract.getData();
    # H4 y8 I. L7 K
  45. + r  x1 j, k5 x+ ?1 D
  46.     expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());6 m1 w, ~0 O  S6 T4 s
  47.     expect(data.number).toEqual(1);
    % D' j% [: T) `+ o7 v; k
  48.   });
    9 d: p" G3 X" i; K1 b; _
  49.   it("successfully deposits funds", () => {* S( _7 ]1 E1 @' F: k9 N/ u
  50.     // test logic is coming
    3 m9 f/ N! D3 Z, o  O# `, Q8 H5 p
  51.   });3 d/ g: K) n0 K  G6 C* L0 i! `
  52.   it("should return deposit funds as no command is sent", () => {
    3 k0 r: I8 i% i" b4 h% \% v. w
  53.     // test logic is coming
    % K: D( ]/ I5 w
  54.   });
    ) s9 x4 i" v( N  p( F: O
  55.   it("successfully withdraws funds on behalf of owner", () => {4 f& p3 j, \+ A9 p2 m
  56.     // test logic is coming/ v' {) {- d! @( t0 G8 x
  57.   });4 x$ g$ y3 q; v( M8 q9 w9 z( S
  58.   it("fails to withdraw funds on behalf of non-owner", () => {! \- N2 `5 q  r/ U, ^" z
  59.     // test logic is coming' Z) H# @0 e6 S! ~4 A0 m4 b
  60.   });; q# m: a. p: g% v! n, F
  61.   it("fails to withdraw funds because lack of balance", () => {) J$ c! Z5 h) J7 b( {8 v4 \
  62.     // test logic is coming
    ( ]1 W, v+ L. r2 V9 H
  63.   });
    9 D8 v5 L7 a; n! C
  64. });
复制代码
我们只是将第一个测试的部分行移动到了 beforeEach 中,其余的行应留在第一个测试中 it
让我们为存款写一个测试。我们将发送一条 op == 2 的信息,然后检查余额。像往常一样,首先创建一个封装方法,我们将其称为 sendDeposit:
  1. async sendDeposit(provider: ContractProvider, sender: Sender, value: bigint) {
    8 c' l7 B7 p9 @
  2.     const msg_body = beginCell(). A0 F' M% L7 W0 X6 D4 W) k) J' ?
  3.       .storeUint(2, 32) // OP code
    : t9 A% r! O" x( t8 o
  4.       .endCell();0 U# s4 j. C$ a7 M5 t

  5. / V2 J, P2 U$ \7 x/ T2 S5 @! D! G/ g; s
  6.     await provider.internal(sender, {8 Z! z" J0 x8 A$ q
  7.       value,- m6 J. g$ A7 _0 R* f' q
  8.       sendMode: SendMode.PAY_GAS_SEPARATELY,
    1 p. m- M2 Y; i7 p
  9.       body: msg_body,5 ^8 ]4 j8 Q. j# C
  10.     });
    6 R( |" F, |+ [( o: i: v( k+ h7 H
  11.   }
复制代码
我们将不详细介绍这种方法,因为它非常简单--我们只需在信息正文中传递一个操作码。
下面是我们如何使用该方法运行存款测试。
  1. it("successfully deposits funds", async () => {0 e4 C$ O, i( L; i: f( z
  2.     const senderWallet = await blockchain.treasury("sender");1 b1 v/ o/ i) l0 t2 K+ G

  3. " Y7 }# M' x. P3 K. y7 i
  4.     const depositMessageResult = await myContract.sendDeposit(4 l7 x# Q! S2 M2 J) T5 O- @: ]
  5.       senderWallet.getSender(),
    ) q$ G; ?6 ~5 r* e! c5 H7 c
  6.       toNano("5")
    , ]/ T0 u: [7 n2 p) _+ m
  7.     );( ~! g6 N8 t2 [$ W2 M
  8. ) C- a# o  T$ `! R# }5 x3 }7 P
  9.     expect(depositMessageResult.transactions).toHaveTransaction({
    & d% H" D5 q' o3 X# X2 m9 y) L3 H8 }
  10.       from: senderWallet.address,
      j# k8 a; }2 i0 l) ^
  11.       to: myContract.address,
    , Z' s" Q- e* `; u6 [2 j
  12.       success: true,: g/ r  v; p, x6 n# V
  13.     });& H+ c4 R# C; i0 l! c! ~- L

  14. % z  N0 `. }) ?8 Q' G* q( p9 E% r
  15.     const balanceRequest = await myContract.getBalance();! @  y! C2 D# _  \

  16. . U$ g9 f6 W0 G  K  d* n
  17.     expect(balanceRequest.number).toBeGreaterThan(toNano("4.99"));
    0 |2 V* B4 c( P, X
  18.   });
复制代码
请注意我们是如何检查合约余额是否大于 4.99 TON的,因为我们知道手续费会损失一些资金。
我们再写一个存款测试,但这次会出现误报。我们将在没有存款操作码的情况下发送资金,并期待返回交易,因为该操作码未知。
和以往一样,我们还有一个包装器:
  1. async sendNoCodeDeposit(  Q# G6 D# L4 a( L$ ^
  2.     provider: ContractProvider,2 ~0 Z2 O1 f" p' o2 U
  3.     sender: Sender,
    : g* L" Z: p. W2 `2 ]7 c$ R* m
  4.     value: bigint! Q" u- ]% a* f4 V
  5.   ) {
    2 e5 k% a. |  }: k' {9 s
  6.     const msg_body = beginCell().endCell();
    + k# |- m/ a: J. \# O/ j  v

  7. 5 G  ^2 z0 z+ J' G* S
  8.     await provider.internal(sender, {
    8 L% S* A2 @: ~9 A
  9.       value,4 I: h# ^; n/ @& h
  10.       sendMode: SendMode.PAY_GAS_SEPARATELY,4 ^; Q! p! G, }1 K* p; _
  11.       body: msg_body,' S9 z- x" R/ q
  12.     });
    , P) x3 C( N5 ^" n% F
  13.   }
复制代码
下面是测试代码:
7 k: |$ Z) L  z* \, |* g7 w
  1. it("should return funds as no command is sent", async () => {. s/ e2 w5 i1 e; r/ i
  2.     const senderWallet = await blockchain.treasury("sender");7 f1 |% g0 z  l6 Z

  3. / R  j6 [9 D) k
  4.     const depositMessageResult = await myContract.sendNoCodeDeposit(1 K2 R3 }0 M. n
  5.       senderWallet.getSender(),
    # p. R+ F1 Q& p! R# z' O2 l: F& K
  6.       toNano("5")
    ; z' k* e3 S# {" M' y+ h
  7.     );6 l/ u. }7 X( r4 u' @" r" ~
  8. 2 f! ^9 a$ p" B4 W: d
  9.     expect(depositMessageResult.transactions).toHaveTransaction({& j( J. Y" D( x- u2 ~0 \3 L* N
  10.       from: myContract.address,8 A6 d6 C( E' E1 x
  11.       to: senderWallet.address,
    ) E3 {: h' x+ r1 z
  12.       success: true,
    ' D1 H, R% b+ M7 I
  13.     });
    / Y  m4 `; E1 P1 H$ ^& }

  14. ! v, H0 f/ J8 D
  15.     const balanceRequest = await myContract.getBalance();
    ! V9 u, P/ i) ?( O9 O4 W" Y
  16. 1 n; n0 M: k/ u, x; n- Q+ y; T
  17.     expect(balanceRequest.number).toBe(0);
    ; p2 l; I( Z9 F
  18. });
复制代码
提款测试
让我们先创建一个用于提款的新包装器:
  1. async sendWithdrawalRequest(
    : R* O# U2 V. t3 s8 q+ o% F
  2.     provider: ContractProvider,% {4 `& R" \2 U' u3 ]
  3.     sender: Sender,! l9 ]! ]" x- d( C  j
  4.     value: bigint,$ G- N$ ~0 V3 u6 {; Z6 v4 h: j
  5.     amount: bigint4 }6 g- x" N+ q7 g0 m
  6.   ) {
    9 F, i6 A2 K" |4 w3 v" |, k" e3 I) w9 B
  7.     const msg_body = beginCell()
    2 R/ h4 _5 [7 H0 e/ r) C
  8.       .storeUint(3, 32) // OP code
    ! W' ~+ g9 T6 Q2 |' m  z' _
  9.       .storeCoins(amount)
    : g4 x! V% O" S7 q
  10.       .endCell();' d1 W( ^& }7 n8 V4 e* g& w8 n% a/ z
  11. 0 e: B# j' b* L5 K9 v& `5 C; U9 @, j
  12.     await provider.internal(sender, {
    ! W) k* ]5 \: h
  13.       value,6 T/ d: }; L+ j" _
  14.       sendMode: SendMode.PAY_GAS_SEPARATELY,
    # c& `# ]/ {& @9 U% o' Q* I
  15.       body: msg_body,, [  x9 |( f6 b
  16.     });+ @- f% Z# L3 i8 Q% X
  17.   }
复制代码
我们输入适当的 msg_body 和所需的提款 amount 。
我们将进行 3 项不同的测试。让我们一起来看看:
  1. it("successfully withdraws funds on behalf of owner", async () => {
    6 B- ^/ N9 _4 k, P9 ~# c7 }
  2.     const senderWallet = await blockchain.treasury("sender");& T( f) o7 y, G1 A
  3. , h5 \$ k* N- t
  4.     await myContract.sendDeposit(senderWallet.getSender(), toNano("5"));
    0 `6 E, K2 ~  h. ~: f

  5. ' ^, Y" y$ [; o/ t+ d
  6.     const withdrawalRequestResult = await myContract.sendWithdrawalRequest(% e+ s- V& ?7 @3 r! y9 D* R$ [
  7.       ownerWallet.getSender(),
    * I3 p5 p! b, ?) s! j, W
  8.       toNano("0.05"),  A- X" a: t+ Y3 @7 L" r
  9.       toNano("1")7 l# m7 |$ R4 L1 C
  10.     );
    0 J: t/ z8 C! E( V7 J& |

  11. . S/ s& l$ b$ J
  12.     expect(withdrawalRequestResult.transactions).toHaveTransaction({
    8 ^  m2 Q& E/ G' U
  13.       from: myContract.address,
    ( V3 \1 A9 r/ c
  14.       to: ownerWallet.address,( f: Z# Z- g- M5 y* {
  15.       success: true,
    * z. C6 X- D9 T5 o
  16.       value: toNano(1),1 d& b" J/ L- p+ @" h1 A
  17.     });
    * F5 ?8 q' @1 T- E) o+ q
  18.   });
复制代码
要成功提款,我们首先要正确存款。然后我们调用 sendWithdrawalRequest 指定我们要提取 1 Ton。请注意,0.05 Ton只是为了支付费用而指定的信息值。
8 F- f) R6 w! x) P+ c' D8 A
  1. it("fails to withdraw funds on behalf of not-owner", async () => {/ ^8 H+ Q) H% k3 a& g% b
  2.     const senderWallet = await blockchain.treasury("sender");
    3 m9 T$ `6 Z4 N1 C3 ?

  3. 9 N* x) w; |& h0 Z, Y  o, c9 J: j
  4.     await myContract.sendDeposit(senderWallet.getSender(), toNano("5"));% l+ @# `. z2 \- e1 O" C% h6 L

  5. 3 x/ q( Z2 S& B1 \% W
  6.     const withdrawalRequestResult = await myContract.sendWithdrawalRequest(4 G8 r! q+ W  y
  7.       senderWallet.getSender(),3 R( p5 V( a! ~) ?. P
  8.       toNano("0.5"),
    3 R/ j1 L; N; Z7 Y
  9.       toNano("1")
    7 x$ E8 Z5 N7 m4 R" r8 F2 \
  10.     );
    4 U+ K& o1 _+ m7 `# r" d

  11. + o9 z) k. @+ R: B
  12.     expect(withdrawalRequestResult.transactions).toHaveTransaction({1 D. E3 U1 h# V& {+ c
  13.       from: senderWallet.address,
    3 A- x% e6 n% F1 A1 X
  14.       to: myContract.address,  Y6 j/ k7 V4 N% S$ R! l
  15.       success: false,
    3 l; F  j2 i7 n/ q0 k" u: `" W# Q
  16.       exitCode: 103,  x1 [/ M+ `, h+ q5 U) J
  17.     });
    - a( h* H0 ?# l( A* z/ H
  18.   });
复制代码
在本次测试中,我们代表发送方钱包而不是所有者钱包发送提款请求(两者的区别在于初始化时我们提供的助记词).
还请注意,我们是如何通过 success: false 和具体的失败原因 exitCode: 103. 为了确保我们做得正确,请看一下我们的合约代码,我们有这样一行代码:
  1. throw_unless(103, equal_slice_bits(sender_address, owner_address));
    # y9 d- k* Q2 C
复制代码
我们进行最后一项测试--由于没有余额,提款失败:
  1. it("fails to withdraw funds because lack of balance", async () => {/ u, {& V; i0 L- g* W% \
  2.     const withdrawalRequestResult = await myContract.sendWithdrawalRequest(
    ' V! ]# `0 V" U
  3.       ownerWallet.getSender(),
    % Z2 n* H- T  z6 Q5 H# z) Y6 `
  4.       toNano("0.5"),
    & B  q; e! V1 i
  5.       toNano("1")
    . a- o' d- d; ]9 d+ _
  6.     );
    - K% K) f/ T& M9 F# N) A) v+ F
  7. 2 I. ~* c  V4 m& [$ L
  8.     expect(withdrawalRequestResult.transactions).toHaveTransaction({
    + @2 ~: G. n' c9 k* L
  9.       from: ownerWallet.address,7 K! d0 q# O& @
  10.       to: myContract.address,+ \; u/ x4 v6 z
  11.       success: false,
    % T6 V) }3 J/ I: e* R, N6 I7 G
  12.       exitCode: 104,
    9 z/ }( B- T- R. s) H
  13.     });- [9 ?: n2 R% |5 n6 R$ ~
  14.   });
复制代码
  1. throw_unless(104, balance >= withdraw_amount);& w: B* C3 Y& r1 U8 w
复制代码
就是这样
几章之后,我们将学习如何构建网络客户端,以便与合约进行交互。
' G% R, u7 ]% l1 b' d! v

- K) E/ b# |+ N; o) `2 N
分享到:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则