正如我之前提到的,编写测试将是 FunC 合约编程中非常重要的一部分。让我们回顾一下上一课创建的测试: + f# e9 C# X' V1 r
- import { Cell, toNano } from "ton-core";
- @' g' q$ S3 A# t - import { hex } from "../build/main.compiled.json";; L7 T/ [1 n* I
- import { Blockchain } from "@ton-community/sandbox";
8 R+ A" o" ]7 ?1 S+ O- A - import { MainContract } from "../wrappers/MainContract";7 ^* t* x( x: E6 ]2 j' ~
- import "@ton-community/test-utils";
+ Y+ I8 Y) |5 v' d. s - : w8 ^2 H: r) D3 t1 m! n) m4 _
- describe("main.fc contract tests", () => {! Y( V+ N8 D& \" [* v% R( h
- it("should get the proper most recent sender address", async () => {, n9 I, O- N! V ~4 ~0 R
- const blockchain = await Blockchain.create();
5 |8 P5 Y$ N9 Z4 Q m3 y$ T; n: K - const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];: w" \, t! {. t9 {2 W" s
- * X7 r; P& _" L& R; P- \
- const myContract = blockchain.openContract(
6 G2 Y# o% N2 k; ]- L/ r* h7 p - await MainContract.createFromConfig({}, codeCell)
/ ~! q: A+ h; J2 ] Q% h+ V - );
) t X: k3 J9 X$ l# r - ; i* m, J s- Y( `% H/ x) k8 |
- const senderWallet = await blockchain.treasury("sender");4 g9 j& }; D) z5 M
- & {& H, I, D. k1 E: }
- const sentMessageResult = await myContract.sendInternalMessage(
) t/ r: D, S% g( |4 H. ] - senderWallet.getSender(),+ V: O( b4 _- x" F& q$ ~$ h; G
- toNano("0.05")! {3 [$ W7 Z- g: C* j1 @$ q
- );1 I+ @3 k5 G" p. ?" e( L8 l7 H
, R# s( A: o2 C3 {8 \8 M- expect(sentMessageResult.transactions).toHaveTransaction({/ C+ V4 x+ j5 |) y# {& i# z& @
- from: senderWallet.address,. O1 @' p- Z, U- D
- to: myContract.address,1 S) ]- H! J3 Q! l( B: r
- success: true,
5 z9 C+ e2 y: y3 K2 k$ y) k - });6 L# ~! f& L0 S' I
; c4 Z$ q- U) U- const data = await myContract.getData();8 P- c3 A1 F: [; B
- # \; ]+ t, r1 }! x$ T8 ^- p
- expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());
/ r# D Z- {- j - });7 }/ h- ? U s: T
- });
复制代码值得一提的是,如果我们现在运行 yarn 测试,它将无法工作,因为我们的代码发生了很大变化。为了解决这个问题,我们需要做的如下: 让我们来处理初始数据。正如你所记得的,我们使用 createFromConfig 方法来初始化我们的合约,并通过以下方式将合约的初始状态数据设置为简单的空Cell: - static createFromConfig(config: any, code: Cell, workchain = 0) {
0 E: j q; a# q% g, H' o - const data = beginCell().endCell();# K0 s, S+ N5 g2 s; ^% n
- const init = { code, data };! g9 d# ~. C7 e. e+ x! v
- const address = contractAddress(workchain, init);! [) \/ _6 D8 I5 U1 N9 a4 m
6 I4 ]% }- d/ q1 w: E- return new MainContract(address, init);2 B3 W+ s& C |. W5 p5 s
- }
复制代码然后在 main.spec.ts 中调用 createFromConfig:
1 Z J6 i& J/ b1 w/ t& U# q- const myContract = blockchain.openContract(2 y" n& g$ g6 \. \) L- k" Y
- await MainContract.createFromConfig({}, codeCell)/ f5 L: ^% h( j; C% s8 H
- );
复制代码到目前为止,我们传递的是一个空对象作为配置,但现在我们可以传递一些值,这些值实际上是我们初始化合约的适当数据所需要的。 让我们定义一下我们的 config 应该是什么样子。在 wrappers/MainContract.ts 中导入后,我们将定义一个名为 MainContractConfig 的新类型。 - // ... library imports; S. w i* c+ C2 J r
1 E, N4 w" O$ Y1 D. ^# f* o+ b- export type MainContractConfig = {
% t# t; k# o' G d - number: number;* j+ a0 ~7 f( H( a! @2 _
- address: Address;& ?' c' w' V# Y/ |8 ?! c
- };
复制代码 我们将在同一封装文件中实现的另一个很酷的辅助函数是 mainContractConfigToCell.它将接收 MainContractConfig 类型的对象,将这些参数正确打包到Cell中并返回.我们需要这样做,因为你还记得,c4 存储空间是一个存储Cell:" T" E+ [. k' Z- F/ D
- // ... library imports$ j& X8 k! p! i: G+ L& j
- & Q h+ h$ ~4 K8 A3 L t( Y
- // ... MainContractConfig type
9 O3 G8 v/ V( y
8 m a0 i$ f+ b5 N! d6 `- export function mainContractConfigToCell(config: MainContractConfig): Cell {
) G# P3 r5 e; |! ?4 Q - return beginCell().storeUint(config.number, 32).storeAddress(config.address).endCell();
* |5 T2 |' Z( ? - }
复制代码 现在让我们更新同一文件中的 createFromConfig,不过是在 MainContract 类中。现在,它必须使用 mainContractConfigToCell 函数来形成初始状态数据 Cell:
: Q: }: X! g$ T% K8 I; l2 G- // ... library imports
( F4 j& ]3 h) U- `. w, P - // ... MainContractConfig type
1 o/ v; ~. j+ @/ e# m4 D - // ... mainContractConfigToCell function9 ]* \- w# m5 r" J
- 9 d. e0 d2 m% m4 W0 t6 B9 `
- export class MainContract implements Contract {* f1 j( [; O/ }% Y& M. V0 {! B
- constructor(
0 b9 a: p5 T8 Y( |' N - readonly address: Address,
9 g; E5 ~+ [/ z0 ^ - readonly init?: { code: Cell; data: Cell }2 }# p8 T- w& S' E& H3 G
- ) {}! r, ~9 U1 P% u) u
- 2 @: c9 V g( Q5 r
- static createFromConfig(. m- M0 M+ p' v, S9 a" n; J
- config: MainContractConfig, Q5 l1 L5 @; [% f: k' a$ c1 o$ G
- code: Cell,
" m+ C8 w0 D0 b4 k* a+ K, o - workchain = 0% q! g3 n" V( l/ Z& h) u# y
- ) {! K, p* S/ k# j) o0 F. v$ o
- const data = mainContractConfigToCell(config);, H% M0 f7 O: |: \
- const init = { code, data };$ \. W( Y- P: x! u
- const address = contractAddress(workchain, init);: n; q, C! `4 v/ r7 {% p" f- Y
. H+ g: E8 [: T! R, [, k1 j, B% f2 u2 D- return new MainContract(address, init);
. M& O" U# Z/ ^9 e - }, f$ l5 d2 h$ S, K" z( H
- * X( y7 S F; Z& R2 |0 Y5 ?
- // ... rest of the methods
! l7 T$ z2 t% h. i - }
复制代码现在,我们的 createFromConfig 方法已经准备就绪,让我们更新一下如何在 tests/main.spec.ts 文件中使用该方法:
$ n, d8 H+ D" C- // ... library imports( L/ ?5 N8 Q# |" u; K. ^
4 B6 l B' O' d8 C! p- describe("main.fc contract tests", () => {8 Z* N3 a2 M3 L5 _
- it("should get the proper most recent sender address", async () => {" {8 k1 k" u' N
- const blockchain = await Blockchain.create();4 k c% r7 g7 z4 D
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
# S9 J2 f" D! I: \4 @) p/ P - & v# J1 L% C& Q& D# t5 x" K
- const initAddress = await blockchain.treasury("initAddress");
) [$ d7 _$ S# R0 s - 0 F; i' V a* m1 @' U( Y
- const myContract = blockchain.openContract(
0 b: b7 c! K4 N3 x% ^& Y/ I% A - await MainContract.createFromConfig(
" K; I2 {# d7 u2 g - {
1 w$ x2 G7 x$ B6 T0 q7 D6 j - number: 0,
0 g" S+ A+ M3 V0 T! \/ ? - address: initAddress.address,$ H o) E5 [; d3 `, D* d4 L
- },& D+ d: A9 Q8 }' d
- codeCell
+ X* K% m- r, \' j) I% a" R0 I - )
% z0 {* |. L+ Q3 u) p( ` - );2 s$ M* X, T# v$ I" G1 F
-
8 h# v; N) q4 x - // ... the rest of old tests logic
5 F/ X4 p$ L; ]( @# }$ v N -
6 l# T% k4 H* N& B - }
复制代码请注意我们是如何从助记词 "initAddress "中创建 initAddress 钱包(金库),并将其地址传递到 createFromConfig 我们的 MainContract. 至此,我们成功地初始化了我们的合约!祝贺您 :) 由于我们现在还要在消息正文中发送特定数据--让我们更新方法 sendInternalMessage 在我们的 wrappers/MainContract.ts. 我们将把它重命名为 sendIncrement,现在它还将接受一个值-increment_by。这将是当前合约存储值递增的一个数字。 我们还要记住,这是我们的合约为获取操作码而解析的同一个正文。因此,我们不要忘记首先存储一个 32 位整数(如合约代码所示,1 表示递增),然后再存储一个 32 位整数的递增值: - async sendIncrement(: ^7 D" f4 W; s
- provider: ContractProvider,
# s0 d' {+ f% k' z: x9 i - sender: Sender,/ O% K; z5 t" T& Q% D) C
- value: bigint,
& T7 a: B* u- U8 o - increment_by: number4 F" b% I. g8 o+ z
- ) {1 E- M* c3 E5 {; L: ?/ M$ r$ @& S
- 2 p; K0 T! e' T6 z0 n
- const msg_body = beginCell()
0 k2 Y7 C1 g6 m* i" w - .storeUint(1, 32) // OP code7 ? t3 ~( l( N' h3 F4 |
- .storeUint(increment_by, 32) // increment_by value$ V. K- O, r6 }" y; {
- .endCell();; U; M. k) j3 Y) D6 @
! i5 E, {( T6 n& T2 Y- await provider.internal(sender, {
$ D- l' x/ h8 u) n - value,4 C# v: l4 z( z# D3 O
- sendMode: SendMode.PAY_GAS_SEPARATELY,
: M! d4 H7 G& f: c, R - body: msg_body,' A8 S' h( H5 i2 f$ Q8 m
- });" z: {) w8 {2 l- T* n# Q
- }
复制代码 我们还需要更新封装器中的一个方法-getData。正如你所记得的,我们重新命名了 getter 方法,而且它现在返回两个值。让我们调整一下 wrapper 方法:3 E" n5 D' e8 g8 c, r/ v" @% F
- async getData(provider: ContractProvider) {3 I7 O7 q; H- |0 W \
- const { stack } = await provider.get("get_contract_storage_data", []);
# m0 N S1 p# @$ Q! c0 d - return {& }) \6 v! D% ~3 P; k
- number: stack.readNumber(),3 C3 j& H& G" ~8 m
- recent_sender: stack.readAddress(),
3 D' w2 {& }/ f - };
; d1 \/ L$ b( H9 i; {) Z3 i* G - }
复制代码请注意,我们是如何读取 getter 方法的堆栈结果的。我们使用 readNumber() 读取当前值的整数。 现在,我们可以返回 tests/main.spec.ts,继续完成测试。以下是测试的更新代码: - // ... library imports & {0 p4 Y2 y& M: {! h8 G
- n- y1 Y( F) D5 T! Z' L3 x- describe("main.fc contract tests", () => {
- S% K8 b3 d9 D+ t" |* d) a - it("should successfully increase counter in contract and get the proper most recent sender address", async () => {
& t2 w5 T8 v8 s& H - const blockchain = await Blockchain.create();
8 X1 u4 {/ t% G$ Q9 G7 E+ k" I - const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];$ G7 \1 m8 K8 j. T
- & q% q$ D) i& g4 W+ X
- const initAddress = await blockchain.treasury("initAddress");
( _: U$ h# d O% }9 Z( H1 T - ) x3 c/ K* P; H5 _3 C$ @4 ]
- const myContract = blockchain.openContract(0 N' k4 P/ J' R
- await MainContract.createFromConfig(7 u4 H! U$ D: A: Q, u
- {
: C* D7 P. ~+ k: e5 w5 @ - number: 0,/ U8 c* e" y+ s/ ~! Z/ Z* L
- address: initAddress.address,9 ]/ C/ c B3 B# L) A/ C9 s1 O
- },- Q' v h j4 b9 w: `
- codeCell
5 P' A* ]* y8 r' B - )
; G, `1 Q ]7 T; b, ~ - );9 ~0 _; W. D( j1 ~/ j0 @. D
9 X( @/ j( D: J/ b- const senderWallet = await blockchain.treasury("sender");; i( I$ I6 A2 q M1 u3 y F8 {: V
& B# C/ K1 g* I7 f3 j- const sentMessageResult = await myContract.sendIncrement(/ ]& J; w, n E! W4 R- ?
- senderWallet.getSender(),5 k) r a# D, Y- Z6 X2 Q+ g& D
- toNano("0.05"),
0 s9 ~" ~. o% ~* ~! b. n6 E0 \ - 1
: ]1 k2 A, `( ~% i0 E8 Z - );
' ^9 I% Y; }7 S5 Z+ ? - # b& T1 p( f2 q; v" u
- expect(sentMessageResult.transactions).toHaveTransaction({
9 I, B5 F! C9 Z8 f - from: senderWallet.address,4 {3 F' ^8 D" `( Z
- to: myContract.address,
) ]* W) S5 X0 o - success: true,$ n- N! u! W- o; N) C+ Y7 F& B
- });1 p: v. }- C" \9 p$ B0 b
- 8 R! R. G6 l/ Y( O& Q0 h9 v* w
- const data = await myContract.getData();
# T/ _9 i: b U! [& |
4 W0 L# y* [& G- expect(data.recent_sender.toString()).toBe(senderWallet.address.toString());% K0 \. x/ d# M! @& Q6 t+ W' y
- expect(data.number).toEqual(1);
3 r. ~6 d$ N8 S! G8 c* f0 C7 V$ O - });2 I/ m# K3 j" C. i3 f N- R3 E7 _6 }
- });
复制代码我们在这里更新了什么? 现在我们可以使用 yarn test 命令。如果一切操作正确,你将在终端中看到以下内容: - PASS tests/main.spec.ts
) w. `5 O: P1 \1 V! z' ? - main.fc contract tests
; E; I! `' ]: B' K- @4 L- @# T - ✓ should successfully increase counter in contract and get the proper most recent sender address (452 ms)
6 L" I" E1 a6 i) x* |5 H - 5 i2 f% ?5 t( y1 ^: a2 R
- Test Suites: 1 passed, 1 total
: v* \/ Z: T" {- l4 Q/ _: b - Tests: 1 passed, 1 total
n; u1 m3 ]2 q - Snapshots: 0 total
- |2 t, X9 ]0 p% s - Time: 4.339 s, estimated 5 s
复制代码 9 t$ N' a8 _" e# X/ A6 H) a
7 ?; u. y% e6 d, @! o
|