在本课中,我们将编写一些有趣的测试。 首先,我们要更新之前编写的合约的初始化过程,因为我们引入了一个新变量,它应该存储在 c4 存储器中。 9 - Cell下溢。从片断基元读取数据时,试图读取的位或引用数超过总数。 更新我们的合约包装器首先,让我们更新我们的合约包装器,以便它能意识到 c4 存储中的第三个参数。也就是说,我们将更新 MainContractConfig 和 mainContractConfigToCell 函数 - // ...library imports: {/ S7 [+ V/ o9 b7 l% v% X* Y
! M: ~( k+ C3 [6 X5 R- export type MainContractConfig = {! R/ Y6 C( u; ?! a; ~1 Z
- number: number; S+ M3 U' b5 P( K
- address: Address;
8 s3 J$ J2 b R4 ^/ g5 @/ c - owner_address: Address;
7 P7 _* [' L( y' A$ f+ C - };
0 ?: T$ M3 I- [8 L* C - & e$ ?( o# [: h0 R
- export function mainContractConfigToCell(config: MainContractConfig): Cell {
" `; c2 g1 P. y+ {$ t - return beginCell()
; K$ d( \3 a- @3 n; T# e - .storeUint(config.number, 32)
6 I6 M% n+ d; \' z; D1 c' o$ r% T - .storeAddress(config.address)
( Z' j0 C# S( _1 \( P& F- D - .storeAddress(config.owner_address)
2 m! U4 p& s( t- h! _/ V - .endCell();
" g& S1 w5 b& t" ~7 H) n6 z - }% v! c7 I; W6 t0 X0 |0 H0 v
- 9 D, {* i. p! p& a& g" ~" O; h* A3 }
- // ...contact wrapper class
复制代码封装文件中还有一点需要更新-getData 方法,这样它也会返回 owner_address 我们还将创建新方法 getBalance,以测试存款后余额是否增加:
' u6 I4 ` L7 i- async getData(provider: ContractProvider) {
6 E* _- b! i7 \) b* L, k6 O. h2 x - const { stack } = await provider.get("get_contract_storage_data", []);
/ {/ G! x* H/ v1 r9 Y1 ^! J - return {& c% m/ R4 S5 s* l2 I
- number: stack.readNumber(),
# W& P- Q; J# H; Z6 M - recent_sender: stack.readAddress(),
* n' g% g" g# G# N% c - owner_address: stack.readAddress(), P9 b! e, Y. c6 \
- };
2 Y0 b* h; H9 X8 _# c, j. ~' f W" _ - }
; \8 }8 ]0 M- N -
3 a7 Y# f- }' ^) j" ` - async getBalance(provider: ContractProvider) {6 O: ?1 Z. y/ j1 G
- const { stack } = await provider.get("balance", []);
# a: P4 H J I - return {
' [! }- ?" X1 Z: C. ?7 Q) [ - number: stack.readNumber(),
* M& L/ V- A8 R7 g5 m - };7 |3 v; ]- q$ e
- }
复制代码- // ...library imports3 u6 E# t2 B* V
- * f0 z# K( _: `6 ]/ D$ W3 c
- describe("main.fc contract tests", () => {% a/ _. G' k8 I, C* j3 u* X
- it("should get the proper most recent sender address", async () => {
* T q! r! e( f$ ^5 ~5 I' q4 k - const blockchain = await Blockchain.create();0 c; e- h1 y# x* r4 K5 K' [
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];' O) x; ?* X- L
- $ E1 m' T5 b( P+ Z. |
- const initAddress = await blockchain.treasury("initAddress");& g( [! D5 t5 ?: ^6 @; U
- const ownerAddress = await blockchain.treasury("ownerAddress");) L8 ^" k& d* ?, y0 K: m6 N4 m
- + S Q' c! J5 c" N
- const myContract = blockchain.openContract(
2 K/ n( L$ ^/ f4 y) B - await MainContract.createFromConfig(
2 T$ u; T" \, P0 [* p, M - {
$ d$ e! [, r" T5 W+ F2 V - number: 0,
( C. n' l7 X: w7 ]! W - address: initAddress.address,: N* B# h" C6 v1 C: C
- owner_address: ownerAddress.address, // now we create myContract from 3 parameters- a, a: P. X) W4 `6 V. M
- },- y2 j3 E; i& |$ N/ X
- codeCell
# @) Y1 [1 O) U/ u - )9 L0 v2 [8 `9 W; m
- );/ Y; e+ d$ e8 E
d [& b- h7 y* a' y- // ...rest of testing code
复制代码
4 e5 f3 H3 g% _9 Y2 Y# G( V我们引入一个新的金库,它实际上将是合约的所有者,只有这个所有者以后才能提取资金。这些更新在 tests/main.spec.ts 中进行: - // ...library imports: F+ B" t. |, R
- . s6 W; T+ D, A! x' C8 S$ |2 q
- describe("main.fc contract tests", () => { X" s! r5 r) f L, [& l! l, p
- it("should get the proper most recent sender address", async () => {
# n' i7 l5 E& Z - const blockchain = await Blockchain.create();* ~3 `( H5 B; d. Y0 D
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
( ?! ^) H$ ]3 ]0 U$ P: P - 5 @6 `+ q9 D( F2 m" R& l" I
- const initAddress = await blockchain.treasury("initAddress");
$ ^1 |) Z9 Y6 g; u0 a - const ownerAddress = await blockchain.treasury("ownerAddress");
* ]2 a3 f- v& w j! R6 k" ^+ a+ K -
# u/ o! H9 K7 E. N- d - const myContract = blockchain.openContract(
u% I4 C1 T+ Q: m - await MainContract.createFromConfig(
" i# [) b7 a6 i+ N4 _% o) g - {$ y. r+ {0 Z3 b+ y+ n$ T
- number: 0,& J; ^* [ i: m" q
- address: initAddress.address,
- K' {/ \3 S" O) ] y8 A/ E5 T - owner_address: ownerAddress.address, // now we create myContract from 3 parameters* H$ d2 c5 d* E, q% n. f. k. Q T( l
- },
8 v; |4 L7 Q5 P& p - codeCell
# w- y0 h: o. `% C - )4 J& D+ ^$ Z( E J6 t! P
- );
/ [0 I, o ^; H6 `; I8 a9 H; r1 @
/ i- P/ P* S% A5 G; ^- // ...rest of testing code
复制代码现在我们好了。 yarn test 成功通过我们之前编写的测试。 5 s" ?: ]! }# B" w; B/ w3 N3 p# s
存款测试让我们在 main.spec.ts 文件中再添加几个测试。 由于我们必须在每次新测试前重新启动合约,因此我们还将把合约启动引入 Jest 的 beforeEach 函数: - // We need to additionally import SandboxContract and TreasuryContract
- J- s) x# k9 z - import { Blockchain, SandboxContract, TreasuryContract } from "@ton/sandbox";0 T: ^+ F% J3 k% ~# `& t
- // ... other library imports5 r Z- H6 Z4 B- M. S. ?
- describe("main.fc contract tests", () => {, \4 X0 q, S+ x* r& p. }8 E$ P
- let blockchain: Blockchain;
* F9 ^8 Y" k9 L) _. v6 l% y - let myContract: SandboxContract<MainContract>;
! u# i$ h; I9 U7 w4 _$ u! g - let initWallet: SandboxContract<TreasuryContract>;
+ F" W! d7 H( k" G1 f& h' |7 `; f - let ownerWallet: SandboxContract<TreasuryContract>;* ^/ d, H$ U9 u) ]" G
- ' }7 ?! t. s- C4 a
- beforeEach(async () => {: X, x# |. F1 U5 B* e+ S+ p
- blockchain = await Blockchain.create(); @% M8 ~' S9 a
- initWallet = await blockchain.treasury("initWallet");
3 y' ~+ B7 D# ^7 T. m% L* `. F - ownerWallet = await blockchain.treasury("ownerWallet");3 o7 c6 r2 x; X- o; H9 y8 X
- 3 W: ^% z' `1 @4 ^) B7 Q; v
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];& j+ S. l; T s/ e
6 l. u8 O- J! B6 Q6 j3 K8 r. d- myContract = blockchain.openContract(
6 n2 l5 z* Y7 d6 g5 m - await MainContract.createFromConfig(7 `# C" B4 j; U* K$ C
- {3 Y( v8 N/ Z8 i8 H V- \
- number: 0,
! M3 c& p# k2 q- V4 Y* V - address: initWallet.address,
, W7 r/ T! M4 H/ n- Y/ ` V - owner_address: ownerWallet.address,0 k" c3 V/ ]6 s4 M1 {* }+ ~
- },
/ A' H1 a- [8 K) M - codeCell( p4 g0 |) a/ n( u+ z7 V
- )+ v' R- M6 L5 h7 B5 F
- );% ~* |) V. ]+ t* X0 K0 |
- });9 g# m' f4 Q/ ~/ f
- 4 X* n& g" m2 s5 m; u" [; ~
- it("should get the proper most recent sender address", async () => {) Q, U: ^. z$ e& C' O& F1 r( M0 C
- const senderWallet = await blockchain.treasury("sender");
/ u- q7 N6 X" J
9 |3 {% B; \4 i- E5 P- r4 A- const sentMessageResult = await myContract.sendIncrement(' ?' s3 U# |; @8 K# r2 T$ v& n
- senderWallet.getSender(),
_; W% h. e1 I9 r - toNano("0.05"),3 D5 V2 V* i, w& w x" g
- 1" R2 d' s' J/ R- q: H5 R7 |
- );7 [( J( f# x7 B# \
- 6 x# q# M. j! t3 R
- expect(sentMessageResult.transactions).toHaveTransaction({
6 a" N: J) e- ?* c( I - from: senderWallet.address,5 u* f/ ^7 B* z' j
- to: myContract.address,
' B! M3 K: M% S( J# n6 D# T - success: true,
7 Q" S: ^; \- m$ m, l4 g1 N3 j - });
; P* p1 o( @, X) u' G' |% i
6 S" i7 r3 W+ v( H& y- const data = await myContract.getData();
# H4 y8 I. L7 K - + r x1 j, k5 x+ ?1 D
- expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());6 m1 w, ~0 O S6 T4 s
- expect(data.number).toEqual(1);
% D' j% [: T) `+ o7 v; k - });
9 d: p" G3 X" i; K1 b; _ - it("successfully deposits funds", () => {* S( _7 ]1 E1 @' F: k9 N/ u
- // test logic is coming
3 m9 f/ N! D3 Z, o O# `, Q8 H5 p - });3 d/ g: K) n0 K G6 C* L0 i! `
- it("should return deposit funds as no command is sent", () => {
3 k0 r: I8 i% i" b4 h% \% v. w - // test logic is coming
% K: D( ]/ I5 w - });
) s9 x4 i" v( N p( F: O - it("successfully withdraws funds on behalf of owner", () => {4 f& p3 j, \+ A9 p2 m
- // test logic is coming/ v' {) {- d! @( t0 G8 x
- });4 x$ g$ y3 q; v( M8 q9 w9 z( S
- it("fails to withdraw funds on behalf of non-owner", () => {! \- N2 `5 q r/ U, ^" z
- // test logic is coming' Z) H# @0 e6 S! ~4 A0 m4 b
- });; q# m: a. p: g% v! n, F
- it("fails to withdraw funds because lack of balance", () => {) J$ c! Z5 h) J7 b( {8 v4 \
- // test logic is coming
( ]1 W, v+ L. r2 V9 H - });
9 D8 v5 L7 a; n! C - });
复制代码我们只是将第一个测试的部分行移动到了 beforeEach 中,其余的行应留在第一个测试中 it
让我们为存款写一个测试。我们将发送一条 op == 2 的信息,然后检查余额。像往常一样,首先创建一个封装方法,我们将其称为 sendDeposit: - async sendDeposit(provider: ContractProvider, sender: Sender, value: bigint) {
8 c' l7 B7 p9 @ - const msg_body = beginCell(). A0 F' M% L7 W0 X6 D4 W) k) J' ?
- .storeUint(2, 32) // OP code
: t9 A% r! O" x( t8 o - .endCell();0 U# s4 j. C$ a7 M5 t
/ V2 J, P2 U$ \7 x/ T2 S5 @! D! G/ g; s- await provider.internal(sender, {8 Z! z" J0 x8 A$ q
- value,- m6 J. g$ A7 _0 R* f' q
- sendMode: SendMode.PAY_GAS_SEPARATELY,
1 p. m- M2 Y; i7 p - body: msg_body,5 ^8 ]4 j8 Q. j# C
- });
6 R( |" F, |+ [( o: i: v( k+ h7 H - }
复制代码我们将不详细介绍这种方法,因为它非常简单--我们只需在信息正文中传递一个操作码。 下面是我们如何使用该方法运行存款测试。 - it("successfully deposits funds", async () => {0 e4 C$ O, i( L; i: f( z
- const senderWallet = await blockchain.treasury("sender");1 b1 v/ o/ i) l0 t2 K+ G
" Y7 }# M' x. P3 K. y7 i- const depositMessageResult = await myContract.sendDeposit(4 l7 x# Q! S2 M2 J) T5 O- @: ]
- senderWallet.getSender(),
) q$ G; ?6 ~5 r* e! c5 H7 c - toNano("5")
, ]/ T0 u: [7 n2 p) _+ m - );( ~! g6 N8 t2 [$ W2 M
- ) C- a# o T$ `! R# }5 x3 }7 P
- expect(depositMessageResult.transactions).toHaveTransaction({
& d% H" D5 q' o3 X# X2 m9 y) L3 H8 } - from: senderWallet.address,
j# k8 a; }2 i0 l) ^ - to: myContract.address,
, Z' s" Q- e* `; u6 [2 j - success: true,: g/ r v; p, x6 n# V
- });& H+ c4 R# C; i0 l! c! ~- L
% z N0 `. }) ?8 Q' G* q( p9 E% r- const balanceRequest = await myContract.getBalance();! @ y! C2 D# _ \
. U$ g9 f6 W0 G K d* n- expect(balanceRequest.number).toBeGreaterThan(toNano("4.99"));
0 |2 V* B4 c( P, X - });
复制代码请注意我们是如何检查合约余额是否大于 4.99 TON的,因为我们知道手续费会损失一些资金。 我们再写一个存款测试,但这次会出现误报。我们将在没有存款操作码的情况下发送资金,并期待返回交易,因为该操作码未知。 和以往一样,我们还有一个包装器: - async sendNoCodeDeposit( Q# G6 D# L4 a( L$ ^
- provider: ContractProvider,2 ~0 Z2 O1 f" p' o2 U
- sender: Sender,
: g* L" Z: p. W2 `2 ]7 c$ R* m - value: bigint! Q" u- ]% a* f4 V
- ) {
2 e5 k% a. | }: k' {9 s - const msg_body = beginCell().endCell();
+ k# |- m/ a: J. \# O/ j v
5 G ^2 z0 z+ J' G* S- await provider.internal(sender, {
8 L% S* A2 @: ~9 A - value,4 I: h# ^; n/ @& h
- sendMode: SendMode.PAY_GAS_SEPARATELY,4 ^; Q! p! G, }1 K* p; _
- body: msg_body,' S9 z- x" R/ q
- });
, P) x3 C( N5 ^" n% F - }
复制代码下面是测试代码: 7 k: |$ Z) L z* \, |* g7 w
- it("should return funds as no command is sent", async () => {. s/ e2 w5 i1 e; r/ i
- const senderWallet = await blockchain.treasury("sender");7 f1 |% g0 z l6 Z
/ R j6 [9 D) k- const depositMessageResult = await myContract.sendNoCodeDeposit(1 K2 R3 }0 M. n
- senderWallet.getSender(),
# p. R+ F1 Q& p! R# z' O2 l: F& K - toNano("5")
; z' k* e3 S# {" M' y+ h - );6 l/ u. }7 X( r4 u' @" r" ~
- 2 f! ^9 a$ p" B4 W: d
- expect(depositMessageResult.transactions).toHaveTransaction({& j( J. Y" D( x- u2 ~0 \3 L* N
- from: myContract.address,8 A6 d6 C( E' E1 x
- to: senderWallet.address,
) E3 {: h' x+ r1 z - success: true,
' D1 H, R% b+ M7 I - });
/ Y m4 `; E1 P1 H$ ^& }
! v, H0 f/ J8 D- const balanceRequest = await myContract.getBalance();
! V9 u, P/ i) ?( O9 O4 W" Y - 1 n; n0 M: k/ u, x; n- Q+ y; T
- expect(balanceRequest.number).toBe(0);
; p2 l; I( Z9 F - });
复制代码 提款测试让我们先创建一个用于提款的新包装器: - async sendWithdrawalRequest(
: R* O# U2 V. t3 s8 q+ o% F - provider: ContractProvider,% {4 `& R" \2 U' u3 ]
- sender: Sender,! l9 ]! ]" x- d( C j
- value: bigint,$ G- N$ ~0 V3 u6 {; Z6 v4 h: j
- amount: bigint4 }6 g- x" N+ q7 g0 m
- ) {
9 F, i6 A2 K" |4 w3 v" |, k" e3 I) w9 B - const msg_body = beginCell()
2 R/ h4 _5 [7 H0 e/ r) C - .storeUint(3, 32) // OP code
! W' ~+ g9 T6 Q2 |' m z' _ - .storeCoins(amount)
: g4 x! V% O" S7 q - .endCell();' d1 W( ^& }7 n8 V4 e* g& w8 n% a/ z
- 0 e: B# j' b* L5 K9 v& `5 C; U9 @, j
- await provider.internal(sender, {
! W) k* ]5 \: h - value,6 T/ d: }; L+ j" _
- sendMode: SendMode.PAY_GAS_SEPARATELY,
# c& `# ]/ {& @9 U% o' Q* I - body: msg_body,, [ x9 |( f6 b
- });+ @- f% Z# L3 i8 Q% X
- }
复制代码我们输入适当的 msg_body 和所需的提款 amount 。 我们将进行 3 项不同的测试。让我们一起来看看: - it("successfully withdraws funds on behalf of owner", async () => {
6 B- ^/ N9 _4 k, P9 ~# c7 } - const senderWallet = await blockchain.treasury("sender");& T( f) o7 y, G1 A
- , h5 \$ k* N- t
- await myContract.sendDeposit(senderWallet.getSender(), toNano("5"));
0 `6 E, K2 ~ h. ~: f
' ^, Y" y$ [; o/ t+ d- const withdrawalRequestResult = await myContract.sendWithdrawalRequest(% e+ s- V& ?7 @3 r! y9 D* R$ [
- ownerWallet.getSender(),
* I3 p5 p! b, ?) s! j, W - toNano("0.05"), A- X" a: t+ Y3 @7 L" r
- toNano("1")7 l# m7 |$ R4 L1 C
- );
0 J: t/ z8 C! E( V7 J& |
. S/ s& l$ b$ J- expect(withdrawalRequestResult.transactions).toHaveTransaction({
8 ^ m2 Q& E/ G' U - from: myContract.address,
( V3 \1 A9 r/ c - to: ownerWallet.address,( f: Z# Z- g- M5 y* {
- success: true,
* z. C6 X- D9 T5 o - value: toNano(1),1 d& b" J/ L- p+ @" h1 A
- });
* F5 ?8 q' @1 T- E) o+ q - });
复制代码 要成功提款,我们首先要正确存款。然后我们调用 sendWithdrawalRequest 指定我们要提取 1 Ton。请注意,0.05 Ton只是为了支付费用而指定的信息值。
8 F- f) R6 w! x) P+ c' D8 A- it("fails to withdraw funds on behalf of not-owner", async () => {/ ^8 H+ Q) H% k3 a& g% b
- const senderWallet = await blockchain.treasury("sender");
3 m9 T$ `6 Z4 N1 C3 ?
9 N* x) w; |& h0 Z, Y o, c9 J: j- await myContract.sendDeposit(senderWallet.getSender(), toNano("5"));% l+ @# `. z2 \- e1 O" C% h6 L
3 x/ q( Z2 S& B1 \% W- const withdrawalRequestResult = await myContract.sendWithdrawalRequest(4 G8 r! q+ W y
- senderWallet.getSender(),3 R( p5 V( a! ~) ?. P
- toNano("0.5"),
3 R/ j1 L; N; Z7 Y - toNano("1")
7 x$ E8 Z5 N7 m4 R" r8 F2 \ - );
4 U+ K& o1 _+ m7 `# r" d
+ o9 z) k. @+ R: B- expect(withdrawalRequestResult.transactions).toHaveTransaction({1 D. E3 U1 h# V& {+ c
- from: senderWallet.address,
3 A- x% e6 n% F1 A1 X - to: myContract.address, Y6 j/ k7 V4 N% S$ R! l
- success: false,
3 l; F j2 i7 n/ q0 k" u: `" W# Q - exitCode: 103, x1 [/ M+ `, h+ q5 U) J
- });
- a( h* H0 ?# l( A* z/ H - });
复制代码在本次测试中,我们代表发送方钱包而不是所有者钱包发送提款请求(两者的区别在于初始化时我们提供的助记词). 还请注意,我们是如何通过 success: false 和具体的失败原因 exitCode: 103. 为了确保我们做得正确,请看一下我们的合约代码,我们有这样一行代码: - throw_unless(103, equal_slice_bits(sender_address, owner_address));
# y9 d- k* Q2 C
复制代码我们进行最后一项测试--由于没有余额,提款失败: - it("fails to withdraw funds because lack of balance", async () => {/ u, {& V; i0 L- g* W% \
- const withdrawalRequestResult = await myContract.sendWithdrawalRequest(
' V! ]# `0 V" U - ownerWallet.getSender(),
% Z2 n* H- T z6 Q5 H# z) Y6 ` - toNano("0.5"),
& B q; e! V1 i - toNano("1")
. a- o' d- d; ]9 d+ _ - );
- K% K) f/ T& M9 F# N) A) v+ F - 2 I. ~* c V4 m& [$ L
- expect(withdrawalRequestResult.transactions).toHaveTransaction({
+ @2 ~: G. n' c9 k* L - from: ownerWallet.address,7 K! d0 q# O& @
- to: myContract.address,+ \; u/ x4 v6 z
- success: false,
% T6 V) }3 J/ I: e* R, N6 I7 G - exitCode: 104,
9 z/ }( B- T- R. s) H - });- [9 ?: n2 R% |5 n6 R$ ~
- });
复制代码- 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 |