正如我之前提到的,编写测试将是 FunC 合约编程中非常重要的一部分。让我们回顾一下上一课创建的测试: ! T9 N. h9 C; Q; L5 o, m
- import { Cell, toNano } from "ton-core";; {( D3 d* R- ^1 w
- import { hex } from "../build/main.compiled.json";6 o- D6 {5 E2 r* P9 {6 e
- import { Blockchain } from "@ton-community/sandbox";
2 d- Z8 [& I5 N( I - import { MainContract } from "../wrappers/MainContract";
: J+ h" v/ c& p7 ]* [: [ - import "@ton-community/test-utils";5 b$ ^5 a$ F, r- w# `2 ?" g
- + @3 B( q$ i4 S/ w2 _: c7 D
- describe("main.fc contract tests", () => {. @7 x r) i7 r2 V1 ~4 b% J
- it("should get the proper most recent sender address", async () => {
' x( ?+ T5 x: H6 O - const blockchain = await Blockchain.create();
% M5 E2 }) H* C9 `, _1 j - const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];2 \) O' [8 c+ D; ~ O9 L# C
- 5 X {2 N( x; Y7 ]
- const myContract = blockchain.openContract(
8 S% P6 ?1 @% Y! @+ o: G" c' k" @2 j - await MainContract.createFromConfig({}, codeCell)- I/ f" R1 c: @6 G# [3 G; r
- );( m! X/ n5 \. w# z4 q3 x
6 k" l: H9 t, _! x3 i# l8 `- \- const senderWallet = await blockchain.treasury("sender");
" r. e$ z0 c$ Z
6 ~1 b7 s1 F' V0 ~- const sentMessageResult = await myContract.sendInternalMessage(% S: c# C" @2 T- n# v0 w3 [
- senderWallet.getSender(),
/ r. Q. ?+ E" {& W - toNano("0.05")$ v0 w& R( M( u# S4 `, j
- );
?- j6 q9 @, L( s/ W5 B - 0 h* B% ]6 t! w# M- E! Y
- expect(sentMessageResult.transactions).toHaveTransaction({* a/ u3 t* a; ^* ?# B
- from: senderWallet.address,# C' Z2 h7 Y8 N
- to: myContract.address,
1 C! Y& C, g) t& |- c - success: true,9 A& w& o6 }( L) t9 Z* c
- });
) c, e( B0 Q8 ?, {/ G$ r# s
5 l! F3 F3 F; U) B. k7 |' E$ X- const data = await myContract.getData();' Y% j! [' w( ~1 \0 r$ I4 Q7 M8 @
- 6 y. g+ a. I: ]0 e) x
- expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());
" y1 ]& u% @; G. h* O9 m4 w - });) S1 d5 P2 {7 |4 s3 m7 V
- });
复制代码值得一提的是,如果我们现在运行 yarn 测试,它将无法工作,因为我们的代码发生了很大变化。为了解决这个问题,我们需要做的如下: 让我们来处理初始数据。正如你所记得的,我们使用 createFromConfig 方法来初始化我们的合约,并通过以下方式将合约的初始状态数据设置为简单的空Cell: - static createFromConfig(config: any, code: Cell, workchain = 0) {9 o) J& \% c. F9 s+ ]
- const data = beginCell().endCell();& o. Q) X# e, {: Q% V
- const init = { code, data };
- z5 q) O0 q9 @6 A! C% Y - const address = contractAddress(workchain, init);: P% ~* I x' ?
- 0 H9 Y. L& h \# b
- return new MainContract(address, init);6 s( X- C; \- Q8 W3 x
- }
复制代码然后在 main.spec.ts 中调用 createFromConfig:
4 Y/ c3 q: r* I4 i: g4 V- const myContract = blockchain.openContract(9 K1 [# C; U% c. l+ e
- await MainContract.createFromConfig({}, codeCell)
/ O7 G6 j8 Y2 O6 C/ Q: `+ y3 p; t - );
复制代码到目前为止,我们传递的是一个空对象作为配置,但现在我们可以传递一些值,这些值实际上是我们初始化合约的适当数据所需要的。 让我们定义一下我们的 config 应该是什么样子。在 wrappers/MainContract.ts 中导入后,我们将定义一个名为 MainContractConfig 的新类型。 - // ... library imports
% W! H2 Q& T4 A( ^ - + N$ U1 e* N. t/ l
- export type MainContractConfig = {# B3 d& k) ?3 _# M* S
- number: number;
7 I, u" e' p; m& d- P. |1 q/ |' ^ - address: Address;
4 ^# @& n! V0 p0 N% x+ ?" [# K - };
复制代码 我们将在同一封装文件中实现的另一个很酷的辅助函数是 mainContractConfigToCell.它将接收 MainContractConfig 类型的对象,将这些参数正确打包到Cell中并返回.我们需要这样做,因为你还记得,c4 存储空间是一个存储Cell:4 E, d5 J7 y$ X: g
- // ... library imports
! R! G7 N6 n+ @" X+ o6 `- n% [ - 7 }" ~3 R8 Z- |5 w; Y
- // ... MainContractConfig type; p' l$ a8 r! m5 q( i/ c/ m1 b
- 9 p+ F: W C, d+ v, N1 |& N
- export function mainContractConfigToCell(config: MainContractConfig): Cell {' }5 f! Z! T3 C9 V4 Y' }
- return beginCell().storeUint(config.number, 32).storeAddress(config.address).endCell();8 L" E: L; v6 E
- }
复制代码 现在让我们更新同一文件中的 createFromConfig,不过是在 MainContract 类中。现在,它必须使用 mainContractConfigToCell 函数来形成初始状态数据 Cell:" ]; P A* E, z
- // ... library imports+ A: K, f2 m( {, _. t1 |
- // ... MainContractConfig type7 [* H! ^6 p9 T* N
- // ... mainContractConfigToCell function
0 }+ ~9 A' k2 _ l) e/ v. p8 B: y# G - $ K) }+ ?$ J4 _) O# a
- export class MainContract implements Contract {
8 u6 T9 X7 x5 H& x+ m9 A9 C$ [ - constructor(
+ @$ U' N2 e: {+ n) Y3 f - readonly address: Address,. T0 |( i% i9 M1 Y, s
- readonly init?: { code: Cell; data: Cell }3 v8 F s, t( X2 \, `
- ) {}; [! d# K) @2 m( j* U7 b8 g( g
$ G4 i( v) k! O: t; ?4 R8 H$ x* Q+ F- static createFromConfig(8 c! j2 o5 B8 N/ c% J* i5 J/ l
- config: MainContractConfig,1 B6 Z6 x1 o" e) u/ ]
- code: Cell,
/ y4 h# T$ [, Q! u - workchain = 01 A2 B0 ?$ ?2 p5 l3 n) _+ @
- ) {
2 K: r R+ W0 g% f - const data = mainContractConfigToCell(config);+ p: e" M; R7 ?; \- w
- const init = { code, data };
0 I) Y! A. F$ T# s1 y# } - const address = contractAddress(workchain, init);
. D/ v6 b* Y9 D+ k! |4 E- S3 A, H! d) d
6 G: h- P2 g" v W5 i- return new MainContract(address, init);, {2 n& C G$ `5 f
- }
1 J2 ^1 O P, v9 B; `; K -
* D l" e# b2 v. u. w& K - // ... rest of the methods1 i5 y; V& a, P; K$ R+ b
- }
复制代码现在,我们的 createFromConfig 方法已经准备就绪,让我们更新一下如何在 tests/main.spec.ts 文件中使用该方法:
& x# u* T8 M+ q) }2 A& s- // ... library imports
7 e" H1 x$ D4 ?% V8 K0 B: J; z
2 \# Y" b+ e5 s0 v- describe("main.fc contract tests", () => {# I2 [4 I7 `% f2 P% X, R
- it("should get the proper most recent sender address", async () => {6 F" J9 J! d* E9 o
- const blockchain = await Blockchain.create();3 q& o. W0 @# t* d0 w
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];( [/ \/ n2 z) z5 U0 M
; l- d* X. L7 \3 n6 Y5 n+ G- const initAddress = await blockchain.treasury("initAddress");
9 l& c: T' H) m X% | |7 L0 n0 }5 l
8 t6 c( t+ U0 `: ]$ M- const myContract = blockchain.openContract(
. N# o% v! m4 F j6 @' D - await MainContract.createFromConfig(( P$ I! r. ^* T5 H3 I* c# ^# b
- {
9 N: z3 f6 M/ K. r - number: 0,
( }! _5 V" A X( p5 ? - address: initAddress.address," t9 Z$ r& Z$ K, s! e& {
- },
- B: M( y( i" I" L - codeCell
1 h& n2 m! Q0 E$ a! b: n- C& ~ - )' k( N* `4 z/ ^6 C7 M
- );. v7 u, h/ [7 x F s! ^
- ( _" p1 E/ Y3 u8 k( o
- // ... the rest of old tests logic
7 \; _) W- a4 Z& N9 R/ C+ S# P -
2 e3 ~ \! R- J0 O - }
复制代码请注意我们是如何从助记词 "initAddress "中创建 initAddress 钱包(金库),并将其地址传递到 createFromConfig 我们的 MainContract. 至此,我们成功地初始化了我们的合约!祝贺您 :) 由于我们现在还要在消息正文中发送特定数据--让我们更新方法 sendInternalMessage 在我们的 wrappers/MainContract.ts. 我们将把它重命名为 sendIncrement,现在它还将接受一个值-increment_by。这将是当前合约存储值递增的一个数字。 我们还要记住,这是我们的合约为获取操作码而解析的同一个正文。因此,我们不要忘记首先存储一个 32 位整数(如合约代码所示,1 表示递增),然后再存储一个 32 位整数的递增值: - async sendIncrement(6 y/ X! k6 }( R6 q- E: g }$ u% i
- provider: ContractProvider,
N) W Z5 o. G+ F9 {& {) y7 B - sender: Sender,& O" D0 _- R4 y, P5 j
- value: bigint,
: N5 m% x, B! Z, R* E+ I/ X - increment_by: number
8 e Y) k! r" W9 s6 J - ) {
6 h! P& O/ O8 R% Z* n2 i1 {9 a - : R0 v$ S4 C' m! |5 L' X
- const msg_body = beginCell()
; O F' `+ [% F: b/ s/ w - .storeUint(1, 32) // OP code! J, ?# c- D! a6 e; m( Y* w% B" c
- .storeUint(increment_by, 32) // increment_by value3 o7 y$ G# \% K% v0 ~1 ]
- .endCell();
0 L6 l( U" C @* j, Z- v) A" j
$ X1 t* U0 `. @- n; W. m- await provider.internal(sender, {
" i, H. F; e8 |# T! l2 o n3 c - value,6 e4 s! E' B! r7 U) a* H: r% q9 K
- sendMode: SendMode.PAY_GAS_SEPARATELY,: m- @! N4 _( N! O$ ^
- body: msg_body,
# }1 R1 B! X1 i/ z1 O2 V& K - });. B \/ J6 W% ^. T6 E% e
- }
复制代码 我们还需要更新封装器中的一个方法-getData。正如你所记得的,我们重新命名了 getter 方法,而且它现在返回两个值。让我们调整一下 wrapper 方法: Q: _% s0 Y: ` z: y2 D2 @9 I) X% ]
- async getData(provider: ContractProvider) {
8 l: u( V7 G1 x4 ~$ L* I: z6 q$ f" p/ ` - const { stack } = await provider.get("get_contract_storage_data", []);
1 u# T1 F h/ y* h+ H& `6 [ - return {0 E; p" r; J9 X& S% x& q k1 S/ S# R0 z
- number: stack.readNumber(),
. o- L# T# G k M5 T. N: |" U - recent_sender: stack.readAddress()," I4 L5 {. x$ l# Q% i; {
- };* K# M5 C$ s( Q
- }
复制代码请注意,我们是如何读取 getter 方法的堆栈结果的。我们使用 readNumber() 读取当前值的整数。 现在,我们可以返回 tests/main.spec.ts,继续完成测试。以下是测试的更新代码: - // ... library imports $ l3 d4 r4 I; u4 S; V
- 9 O; G! ?$ `7 l9 R
- describe("main.fc contract tests", () => {* u: ]& d w; V' K6 C* `
- it("should successfully increase counter in contract and get the proper most recent sender address", async () => {1 g$ a% f" _+ C
- const blockchain = await Blockchain.create();
7 p! ], J$ N1 W! j" U" Y$ A% o - const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
3 @1 D) `: ^7 T" b" J - ! r$ R/ R( v5 S* }
- const initAddress = await blockchain.treasury("initAddress");$ X' b+ g! S' i" a+ j( ^: K% y
- 3 P2 C8 e8 ]$ w# Z" k* ?% Z' B; K
- const myContract = blockchain.openContract(# K/ i w' ]/ Y) f
- await MainContract.createFromConfig(' F u. n5 a/ g1 L4 \
- {
& x0 I. R6 g" S4 Y. k - number: 0,; l0 A/ v3 K; G' B1 v4 v* }) R( m
- address: initAddress.address,2 ]0 @- h5 r! \
- },
! h8 Y( Q: R. g3 ^5 r) s - codeCell
/ O! ^" u% n& [& l) l# f - )
, t i' Z, N m9 y$ O - );
: R8 w+ g6 n% n, t - ( z0 j7 A! e# W+ y! j
- const senderWallet = await blockchain.treasury("sender");+ m9 \- R" M2 |0 h6 o7 k- ^2 u
- n, p1 l2 C- Y' b7 [# u- const sentMessageResult = await myContract.sendIncrement(: A( ~* I, h9 E2 y( X/ r
- senderWallet.getSender(),# S' b7 I) i- R+ R- v" v C
- toNano("0.05"),
2 b' S4 Z: }: Q - 1+ [2 A- f( K3 D) P) U
- );* ]' J* c1 i' E, }! ~% P) |
- , w5 {, P) f3 A/ B( o
- expect(sentMessageResult.transactions).toHaveTransaction({8 r4 d3 u; {$ {; T) y
- from: senderWallet.address,$ p+ L, C" s" B: P
- to: myContract.address,
( I+ J4 h0 b; y) U - success: true,6 w. u! e$ k3 b! D# @
- });
( U7 Q5 U) \% c1 W4 G8 h6 ~
5 {9 j. g2 L& w3 Q- const data = await myContract.getData();! S: j0 b0 V( S6 _7 x8 u1 G+ F
2 a6 i5 {4 Q3 |. _6 @$ R- expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());: B5 V1 C1 Y# k a6 k a
- expect(data.number).toEqual(1);
/ Z- K2 b7 x4 Y4 ` - });
2 J( V* g( X) k$ j/ z- I1 s' o - });
复制代码我们在这里更新了什么? 现在我们可以使用 yarn test 命令。如果一切操作正确,你将在终端中看到以下内容: - PASS tests/main.spec.ts# j4 S; z, Z. }6 [7 K1 B8 P
- main.fc contract tests
, ^( v; p' f N7 w - ✓ should successfully increase counter in contract and get the proper most recent sender address (452 ms)
' `/ H0 f4 I z5 |# o( h- C& @
3 _, {6 t3 l# b9 g9 a- Test Suites: 1 passed, 1 total
7 X, Z' B) U* u5 ]( o - Tests: 1 passed, 1 total6 U& Q7 J3 |/ r
- Snapshots: 0 total
+ m" v0 _+ w& m2 _/ H) i$ N4 a - Time: 4.339 s, estimated 5 s
复制代码
$ X! z" [# M% c7 y. G% l; V0 W/ ^* q4 k: u* }, I( c4 u7 c
|