使用测试网 您已经听说过测试网,它是一个独立的网络,用于测试开发人员可能遇到的所有问题。 该机器人会询问您的地址,因此您需要有一个可以测试 TON 币的钱包。你可能熟悉 Tonhub 和 Tonkeeper 等钱包。Tonkeeper 和 Tonhub 都嵌入了测试网功能。你可以同时使用 Tonhub 或 Tonkeeper,但为了方便举例说明,我将使用 TonWhales 团队制作的 Tonhub 钱包。它同时适用于 iOS 和 Android. 请点击其中一个链接,将其下载到您的手机上。 安装好 Tonhub 后,点击底部菜单最后一个标签上的应用程序版本号约 7 次,该标签上有一个名为 "更多 "的齿轮图标。然后,点击网络选择器,将网络切换到测试网(Testnet)。现在,返回 "主页 "选项并复制您的 TON 币钱包地址--使用此地址在以下设备上接收测试网代币 testnet giver Telegram bot. 5 f, l2 l! N9 [7 m0 D. m9 r9 m
部署脚本现在,让我们在我们的 scripts 资料夹 - deploy.ts. 大家都记得,要部署合约,我们需要: 计算合约的未来地址 创建合约的初始状态(代码和初始数据) 编写一条携带初始状态的信息 向区块链上的合约未来地址发送所编写的信息 9 y8 a# W2 X# x- l! T4 q) F
下面是我们的部署脚本在实际发送部署信息之前的样子: - import {hex} from "../build/main.compiled.json";/ @( G, w$ M! @
- import {Cell, contractAddress, StateInit} from "ton";" x1 q1 F- e1 b, e' q. Q1 i4 s! ^
- % ]: v* O* }: F
- async function deployScript() {
$ x7 W) l1 b+ h. H& Z' L! ]
) z; }* z( j- O& B2 y# O& W! W- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
2 R" ]6 s" S4 j. ^( D0 g - const dataCell = new Cell();
5 C/ a {! J! X# W5 V - 6 P& j2 ~$ L0 d* ]- b! q' @3 A
- const stateInit: StateInit = {
% a8 V |1 Z$ j4 J/ Z/ _ - code: codeCell,
, k* d- K2 o7 r* J5 ~; W - data: dataCell,, J9 T2 n0 W! Y
- };
( P5 {4 r/ I" i; L6 @: b' I& X: u - [+ E0 W, E7 Z9 u _' ^0 u
- const stateInitBuilder = beginCell();! b4 X) P2 [' v$ _) J" ?
- storeStateInit(stateInit)(stateInitBuilder);* e7 n9 L" [" J" j% Y
- const stateInitCell = stateInitBuilder.endCell();8 U$ p) ?/ r5 S4 }* X7 ?
- : r9 L4 N9 B" q* d) M
- const address = contractAddress(0, {
6 l$ n% c5 ]0 u0 }/ d4 h. g - code: codeCell,
( y4 R. X1 B; O) W' U - data: dataCell,6 G5 @0 J! j. w) U
- });
1 M4 S$ S4 q2 p1 g - ( `* \0 P: t$ C( h% N
- }
5 a1 {: l# v# ?1 m; T0 c - 8 G8 d% s/ v( ]7 _) f8 f
- deployScript();
复制代码我们已经熟悉了所有类型和实体,只有一个除外 - StateInit 类型和 storeStateInit 功能. storeStateInit 是由 @ton/core 库,它可以帮助我们创建一个包含 StateInit 的Cell。 让我来详细介绍一下这些线路的情况: - const stateInit: StateInit = {
$ C: u( k7 X% T$ ~% X | - code: codeCell,. H8 x4 S# K. Q8 H
- data: dataCell,
3 R9 s' U K& w8 G w. r - };
* g9 b3 O3 l. F8 a
+ ~8 f/ O& \; I5 L0 c5 Z3 Z- const stateInitBuilder = beginCell();
: u2 T. F: k# ^* y& ^9 ?% D5 z - storeStateInit(stateInit)(stateInitBuilder);
2 k8 m7 D. M0 a9 D; i( O/ \; C - const stateInitCell = stateInitBuilder.endCell();
复制代码这里使用的一个重要概念是构建器(Builder)。要成功创建 FunC 合约并使用 TypeScript 对其进行操作,您需要了解这一点。正如你在代码中看到的,我们使用 beginCell 函数返回一个 Builder(你可以悬停在 stateInitBuilder 变量上查看它的类型)。 我们将 stateInit 传递到 storeStateInit - 这样就有了一个函数,可以将提供的 StateInit 写入构建器。我们立即执行该函数,并传递上一步创建的构建器。 此时,我们已经有了一个已写入所需 InitState 数据的构建器。我们只需运行其 .endCell() 函数,就能将构建器最终转化为一个包含 StateInit 的单元格。 我们可以通过手动构建这样一个Cell来达到同样的效果。代码如下: - const stateInitCell = beginCell()
3 k0 j& ^- v6 @) t j0 Z. h - .storeBit(false) // split_depth - Parameter for the highload contracts, defines behaviour of splitting into multiple instances in different shards. Currently StateInit used without it.( o0 |2 d7 n M$ U% n
- .storeBit(false) // special - Used for invoking smart contracts in every new block of the blockchain. Available only in the masterchain. Regular user's contracts used without it.% v; r- j0 }( P! O: a0 F6 d
- .storeMaybeRef(codeCell) // code - Contract's serialized code.
! x0 W7 i+ k# v - .storeMaybeRef(dataCell) // data - Contract initial data.2 B$ D: \" N$ A, B2 r' f
- .storeUint(0, 1) // library - Currently StateInit is used without libs
复制代码 不过,这需要我们深入了解 InitState 的实体。您可以阅读 here in documentation., g, s$ x+ `5 O9 y, e1 Z3 l
m) U" E. B2 o- A `3 N' P" a
使用 StateInit 创建信息为了发送信息,我们需要本课开始时安装的 Tonhub 钱包。Tonhub 有一个很酷的应用程序接口(API),可以实现与钱包的深度链接,这意味着我们可以编写一个链接,其中包含发送消息所需的所有参数,然后在 Tonhub 中打开这个链接,签署发送消息的实际交易。这需要安装一个 qs 库,它将帮助我们编写带有复杂参数的 url。 运行以下命令进行安装: - yarn add qs @types/qs --dev
+ p/ R9 ]0 Q5 t
复制代码因为我们的 Tonhub 钱包在手机上,而链接在桌面上,所以我们只需生成该链接的二维码,然后在终端中显示,用 Tonhub 扫描并发送部署信息。为此,我们将使用 qrcode-terminal 库。 运行以下命令进行安装: - yarn add qrcode-terminal @types/qrcode-terminal --dev
3 g( h3 y* [3 j7 I9 A# r; J
复制代码有了这两个库的帮助,我们的部署脚本现在看起来就像这样: % |- R; _0 _; l4 @2 z
- import { hex } from "../build/main.compiled.json";
7 ?4 _1 N. ?7 \" X/ W; W - import { beginCell, Cell, contractAddress, StateInit, storeStateInit, toNano } from "ton";
3 Y4 ~6 c' W5 v3 m9 [ - import qs from "qs";. c% `; M& Q+ z8 F
- import qrcode from "qrcode-terminal";
5 K- C' J) h2 f. Z( S; a$ i; d - 1 D0 z4 H+ D$ o8 S9 g
& N. ]' O8 v }- async function deployScript() {9 f0 ]7 ]# m8 |5 ~4 F
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];5 `" W6 W; s6 h5 e/ f. M4 ]9 z
- const dataCell = new Cell();/ k! _: K1 k& _
- ' U) n! |# w$ L0 o
- const stateInit: StateInit = {& I( z9 W/ l; j( e4 A F
- code: codeCell,' ^3 `, {7 L0 z% ^: p; |
- data: dataCell,0 ^) \) `: p6 b7 M
- };; Z. {* ^' T- u4 F3 Z
. ~5 a" l) x+ p1 q6 ^- s1 P, e- const stateInitBuilder = beginCell();5 z& b3 q5 S" N* v
- storeStateInit(stateInit)(stateInitBuilder);2 j4 H. k- s! O+ s
- const stateInitCell = stateInitBuilder.endCell();
- ]: X. ]! R% ] - % R2 a1 d& s! a, P! y m2 _
- const address = contractAddress(0, {
' P) H$ z% |1 i* T$ C - code: codeCell,
7 e1 l( S+ g* V' d; P - data: dataCell,. U1 u; M2 _" y+ W7 X' w
- });! `/ q9 g2 y2 T7 h8 n- i: Y* C
7 z' h' p( l& X5 P% ~- let link =
; R9 N$ s# H. q) I& W' [/ W' V - `https://tonhub.com/transfer/` +/ B( ~% C! n! O( Z' }* G) r0 H* P* k
- address.toString({
3 k; G3 S4 \$ J# _, w) `+ a- C - testOnly: true,
& u0 l3 o' J3 B$ @4 t - }) + "?" +% f" G/ g8 }7 w4 e; v; ~) ]( S6 s
- qs.stringify({
d3 v$ w8 i0 d* R - text: "Deploy contract",7 W# F% \! L& O3 @0 t: b
- amount: toNano(1).toString(10),
. a5 t" M+ X( A- G4 F y - init: stateInitCell.toBoc({ idx: false }).toString("base64"),( v5 e2 F% X7 S6 T: Z g" P
- });2 R1 p0 r2 y) b' H9 k y# X+ v
- s7 J2 \. |5 f3 ^/ X
- qrcode.generate(link, { small: true }, (code) => {
) K: m2 q& [# Z& \$ ?" c" E& b - console.log(code);" S' B8 n7 \; r; i6 G
- });2 Q; C: D) E7 U5 d
- }4 a7 i0 R! F( n; b' i6 B
- 9 k% e) D P! t7 J# U# D6 e
- deployScript();
复制代码正如我们在之前的课程中所做的那样,让我们添加信息丰富的控制台日志,因为你可能想在所有项目中重复使用这些脚本。 现在,我们的代码将如下所示: - import { hex } from "../build/main.compiled.json";1 x$ T% w3 m4 ~3 C: ?- |5 k# U
- import {8 W% D; n- Q$ d# k& q \
- beginCell,8 o# z8 b0 R/ m0 Y! F# W
- Cell,
( y' {. {- M) I6 U& b- _; m# L - contractAddress,
: u Z# R0 E F- K2 Z7 R! j& @ - StateInit,
" }0 h- Z4 d" S5 D2 N - storeStateInit,
; |. x8 T% \$ c - toNano,
* u5 j6 l$ b% l' }) A. n - } from "ton";
6 F0 d) P c' M$ m# K4 G - import qs from "qs";
, u' D; V4 \+ n6 z% j$ ?: u' T2 c - import qrcode from "qrcode-terminal";
$ ~' t$ q: W& E; A - ; C3 E& Y8 j; D& c' d9 n# K" u* G
- # p9 A8 | n. F9 U
- async function deployScript() {
1 k( i( \7 @* Y0 o4 m3 J5 l; ~ - console.log(
6 z! G0 R7 b2 J& X/ Q8 B& |" @1 l - "================================================================="9 l5 S7 T9 S) W3 i& I( b
- );% W1 T- d+ Q: W
- console.log("Deploy script is running, let's deploy our main.fc contract...");/ s3 C8 c% z/ a. ^
& F4 j& H6 ~4 Z/ r% T- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
@ M8 ^& [0 ~, c- B+ G - const dataCell = new Cell();
2 M; T0 ?9 x6 z - / t Y0 p- x+ Y
- const stateInit: StateInit = {
. Y% T, s3 A* S0 h( A) a - code: codeCell,+ m6 @5 h( d2 |# Y
- data: dataCell,! W4 L3 D: V& _& ^. ]2 B, r+ ]) X
- };; n e- ^! q& W, U7 h
" L4 T) X7 o3 S0 C" |+ f- const stateInitBuilder = beginCell();( s7 n# F5 E: o% \7 L
- storeStateInit(stateInit)(stateInitBuilder);
8 s8 u7 c1 n4 L( a0 t5 Y. n - const stateInitCell = stateInitBuilder.endCell();
- k$ q* g6 `" x% j7 ~ - ( z( x+ x. Z: p' K7 j
- const address = contractAddress(0, {
8 B7 S" g- u3 R0 M& {- T - code: codeCell,; f* w+ ^% d* M* `8 o8 ~5 o
- data: dataCell,
/ J# ^7 `! [$ ^ - });+ W6 P( q5 m# h4 a/ f- _
* v/ N' ` h: c9 u9 k8 X- console.log(
- v* r3 r# j" g; K5 _7 s7 ?. B- R - `The address of the contract is following: ${address.toString()}`9 }6 L: \! F" }& x9 r/ r1 I/ M
- );
7 E' x Q% c- ^ - console.log(`Please scan the QR code below to deploy the contract:`);* |; N+ W9 u- y: [, c# g8 B
3 u8 h! z2 h4 ^. [ j0 W- let link =- o4 A4 Y$ q! B
- `https://tonhub.com/transfer/` +
% ^. T, B7 h& O% { - address.toString({6 e1 T1 _! I; ^8 x4 i1 v$ v
- testOnly: true,# n' k9 \9 T3 w& t% X
- }) +
+ F5 l' f7 j) p8 p$ J* S0 V - "?" +
- l9 l+ V: o0 h l; f - qs.stringify({1 J4 F! x3 T% x4 y) Z* y, S& Z
- text: "Deploy contract",
; \3 m$ l2 @' I" b* e" ~' K4 P! f - amount: toNano(1).toString(10),$ p+ e) G0 p7 [ c( N. s/ v. R
- init: stateInitCell.toBoc({ idx: false }).toString("base64"),$ r) j& q" U9 w& M3 X5 X" ~& P
- });. D* v4 @% g6 U. k, u
* p! i4 D8 G. ~0 b% X- qrcode.generate(link, { small: true }, (code) => {
) y) D7 I& ~4 f4 C* L - console.log(code);8 x6 o3 h/ V8 p: {9 Q2 }7 g2 t
- });) Z# L$ F r, M. p
- }0 E8 O, E+ u8 i/ q
- 6 Y% [+ D: \$ G e
- deployScript();
复制代码 同样,我们将立即在 package.json 文件中创建另一个脚本运行快捷方式。让我们把这个脚本叫做 deploy:testnet,这样以后我们就可以在部署主网时使用 deploy 这个名字了:
6 D# e+ d+ `0 V' R- {7 }7 K: `) ^! s! Z& N+ R! r, \, m
- ... our previous package.json keys5 }9 Q& f: U& s, V
- "scripts": {% h9 ?) B* w- a0 W' u8 I
- ... previous scripts keys
( M: M% V* Y0 ?, h3 w - "deploy:testnet": "yarn compile && ts-node ./scripts/deploy.ts"
* H& E( B/ \* S! E - }8 {& A [# Q/ W; ~
- }
复制代码让我们运行部署脚本 (yarn deploy:testnet) 并使用通汇手机应用程序扫描二维码。确保您在 Tonhub 的这个钱包里有您的测试网 TON 代币。 在浏览器中打开测试网版本,并在搜索栏中输入智能合约的地址。这将打开智能合约页面。如果您已经按照我说的一步步完成了所有操作,您将会看到一笔 1 Ton的单笔交易,该交易已经将我们的合约部署到了区块链上。 恭喜您 您的第一份 TON 智能合约已部署完毕! 部署后的链上测试是的,你已经成功了,但还有一件重要的事情需要你认真对待,那就是部署后的链上测试。一旦我们的合约部署完成,我们要确保功能的行为符合我们的预期。为此,我们将编写一个脚本来实际测试我们的功能。 不过,在运行该测试之前,我们需要确保我们的合约已经部署。为此,我们需要访问 TON 区块链的当前状态。有一个很棒的工具可以让我们连接到几乎所有可能类型的 TON 区块链 API - TON Access 由 Orbs team 提供. 首先,创建一个文件 onchaintest.ts 在我们的 scripts 文件夹。 在 package.json 文件中添加脚本运行快捷方式: - {
1 C0 J0 q$ e; U4 t - ... our previous package.json keys
' Q7 l+ B! w' x7 v- }/ ~' x - "scripts": {: i: D5 d7 M5 G( ^" p$ G
- ... previous scripts keys
# H6 w6 J# C0 L0 a" k3 d. G9 o4 _ - "onchaintest": "ts-node ./scripts/onchaintest.ts"
# M; q- c$ `7 ?% e$ k - }
: Z7 H) b- L z! N t - }
复制代码要检查合约是否已部署,我们需要它的地址。我们希望我们的设置使用方便,只需最少的手工操作,因此我们不会对部署前看到的地址进行手动编码。我们将独立计算地址,因为我们知道地址可能会根据合约代码及其初始状态数据发生变化。 当我们要获取合约的地址时,我们的初始代码就是这样的: - import { Cell, contractAddress } from "ton";, S- [; ]* v: o; Y8 H% b/ V
- import { hex } from "../build/main.compiled.json";
2 X* {9 \3 X9 a - % R2 c2 W6 M! k- e1 L
- async function onchainTestScript() {
, b' v+ X( u% w0 S - const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
# k; H. x4 z/ Y; @ - const dataCell = new Cell();
* F }7 C: W/ p# D
4 R( I% U9 Q* F- const address = contractAddress(0, {4 Q5 y- e# g, N4 x) z
- code: codeCell,
: ~8 h$ q ?3 D - data: dataCell,
' i/ z7 e8 Q; M8 I$ X* h2 N6 W - });
! }* @ ]; ?2 q9 k, W! c - }
* V+ S- q& g, n8 m% v
; a4 Q5 A! ? a [4 M% v0 w' Y- onchainTestScript();
复制代码 首先安装 @orbs-network/ton-access 库:
, @. h5 ~5 W t" }6 Z6 ~$ p- yarn add @orbs-network/ton-access --save0 w+ e0 E, U7 ?3 ^% Z
复制代码现在,我们更新代码,加入通过地址检查网络上合约状态的逻辑: 5 }6 z: t1 ]9 S6 E. C8 D) f" q
- import { Cell, contractAddress } from "ton";
+ z: N. F0 o" \ - import { hex } from "../build/main.compiled.json";, z+ i5 x: F) m8 y1 |
- import { getHttpV4Endpoint } from "@orbs-network/ton-access";
* I$ h& s0 T* ^) B3 @8 d
3 h- z; j ]- O7 y/ Q, d- async function onchainTestScript() {4 {+ E4 ]; o) [0 Q# U# }- L
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
) @: V( m: E, r - const dataCell = new Cell();9 y- H# _- E0 _7 g
& C t& V! C, E W1 S2 j i: j7 S1 s- const address = contractAddress({
! M* n$ f) S4 ?+ U8 G- T7 b3 B - workchain: 0,
1 I1 p1 l5 Z! O# W. |; L" P6 ^ - initialCode: codeCell,
/ H& I$ l) ?1 W/ F* q( X+ { - initialData: dataCell,! _' v m; T/ F
- });( {4 }3 }& y, I8 C# X& Y. J
- / Z/ G3 O3 ]! @2 n
- const endpoint = await getHttpV4Endpoint({
/ i- E6 U0 a& {8 }; a - network: "testnet",) h( q. E4 W, m
- });* O" D9 C$ e' K" V; W
- const client4 = new TonClient4({ endpoint });
( w' r( b0 p% X2 {! g# g - + {. o9 r" ~& p0 f
- const latestBlock = await client4.getLastBlock();
/ J3 r9 m( F8 a/ D" E5 i - let status = await client4.getAccount(latestBlock.last.seqno, address);9 W' E. D3 D+ f3 \: I: D) c; @
- - s$ d5 V V. N2 y* w* c
- if (status.account.state.type !== "active") {
; r; C' k( B+ m% U3 [ - console.log("Contract is not active"); Z1 {: S3 I ~$ G; b( p( s
- return;
3 b; X& E- j/ D, v3 q/ [: \& W5 z) f - }
+ a% c, R' j+ D1 A
5 D3 x3 c5 q7 U+ _. x# p( H& |- }* H T+ N, y) |# H! W3 j
- # z5 ]& H% m1 `5 u4 Q
- onchainTestScript()
复制代码我们再深入研究一下这段代码。这里有什么? 我们使用 getHttpV4Endpoint 函数连接到 Orbs 团队托管的 API 端点,并获取 v4 协议的客户端对象。您可以了解有关 v4 协议的更多信息 here # X9 h( h5 N* `# ^
到目前为止,我们必须将网络配置参数设置为 testnet。
因此,在合约尚未部署的情况下,我们可以停止脚本。 现在让我们开始实际测试。我们将创建一个简单交易的链接,向合约地址发送 1 Ton。与部署过程中一样,我们将显示该链接的二维码,您可以用 Tonhub 钱包手机应用扫描。 - let link =
( S. }) N P4 ^1 V- S6 C - `https://tonhub.com/transfer/` +, D1 M1 G3 f( n7 b5 f/ r; k/ V0 r5 d
- address.toString({
* h w. P9 Z. j - testOnly: true,
% E( I" F6 R. @# r1 W! r - }) +
: Z5 k# p. }! d( f* u - "?" +7 `" q, D, f3 } s+ h# |4 [7 ^
- qs.stringify({
( l/ ~, h! q0 c$ \ - text: "Simple test transaction",
. U/ q8 q$ E0 w( W$ Y - amount: toNano(1).toString(10),7 j# C8 u/ P! L) k* V- C9 ~
- });2 g* w+ H- W$ g
- * F% w8 ^5 S; g4 X# Y) }
- qrcode.generate(link, { small: true }, (code) => {
; D0 ]% J4 |$ B/ W - console.log(code);
9 O2 E: H# }- W t, |2 \# ^ - });
复制代码我们还必须导入 toNano 从 @ton/core 库. 它可以将Ton币转换为成nanos。一个TON币有 1 000 000 000 nanos 中。在TON币中,TON币的数量总是nanos为单位。为了便于使用,大多数情况下我们会将nanos单位作为一个字符串来使用。 另外,请注意,我们使用 toString 方法,提供指定地址来自测试网的 config 参数。 在显示二维码后,我们的脚本将立即开始调用合约上的 getter 方法,以接收最新的发件人地址。我们将只在最新发件人地址发生变化时在控制台记录日志,以避免在控制台记录相同的结果。 让我们先来看看获取最新发件人地址的逻辑代码,我们将分析其中发生的情况,然后将其集成到我们的脚本中: - let recent_sender_archive: Address;7 g% M' j7 z: H0 l( s
- - E9 Q7 c8 `: k6 p2 ?; W/ R
- setInterval(async () => {
) w+ c1 C9 B6 C6 P - const latestBlock = await client4.getLastBlock();
4 k( B( h- x8 {# h6 [$ H6 C - const { exitCode, result } = await client4.runMethod(5 }; x2 v6 t8 p3 h# y) l9 y+ @. n
- latestBlock.last.seqno,
6 p" K! k0 L$ c: _: T+ O/ M7 I: y# q - address,
/ A7 r; [: Y n- s1 O2 ]: J0 s8 o - "get_the_latest_sender"
$ l& r7 ], ~ q( Q - );- H$ [$ W) ]0 C7 p2 ^0 H
- % \& n$ ~7 h* |
- if (exitCode !== 0) {
, X c' N2 g' M6 ` - console.log("Running getter method failed");4 }) h8 Z h$ J! `4 R3 B7 }' M% i
- return;
7 g7 M7 F* A* k+ j - } d, Y" m3 S9 _* P/ U7 R1 j' b
- if (result[0].type !== "slice") {
8 x! j6 A, K! Z% u - console.log("Unknown result type");3 D& G4 z+ k/ E
- return;
" y$ k1 s+ \: R* j+ E( ?- v - }
4 m; m/ `. ]7 J, _* y3 v+ F% X
O5 i9 E7 Y% ^* R% f- let most_recent_sender = result[0].cell.beginParse().loadAddress();# Y* o" j9 N$ ~8 c
' G# ?# [9 D" _8 ]6 S* C- if ($ |0 K$ w1 p: E+ n5 T6 w
- most_recent_sender &&/ i X% F1 e; D5 _
- most_recent_sender.toString() !== recent_sender_archive?.toString()) w; {2 v) m$ Q; ^
- ) {! J( k E5 l; ~* r
- console.log(/ d$ ~+ }1 J: X A
- "New recent sender found: " +
3 o; [: ~: W0 x0 U2 v9 X( z - most_recent_sender.toString({ testOnly: true })/ ]7 J3 V" o) p3 N
- );+ G2 i6 q" ]. S/ }
- recent_sender_archive = most_recent_sender;
3 e- G: h2 b- U) O - }2 E2 h3 @8 @4 }& L9 Y
- }, 2000);
复制代码我们在这段代码中究竟要做什么: 我们将设置该函数每 2 秒重复一次 每次执行检查时,我们需要用 getLastBlock 获取最新区块的序列号 我们使用 runMethod 调用 getter 方法,我们感兴趣的参数是 exitCode 和 result。 如果调用不成功或接收到的数据不属于片段类型(如您所见,TON 中的地址总是一个片段),我们将确保脚本停止运行 我们获取 getter 方法的结果,即它的第一个项目 ( result 是一个数组,因为在其他情况下,getter 方法返回的结果可能不止一个) 我们采取 cell 参数的 result, 打开它进行解析,然后从中读取地址 我们会将地址的字符串版本与之前可能收到的发件人的字符串版本进行比较。如果两者不相等,我们会在控制台记录新的 most recent sender's address.
" Z. t& q7 H8 h& V3 b& C6 Q1 ~" e. w ) o! l! G1 m# I7 q; e
我们看看最终的代码是怎样的:
4 `# H3 x; s3 L1 V0 P5 i9 h
: i) G' H {3 U; o5 a) a! }- import { Address, Cell, contractAddress, toNano } from "ton-core";
3 p3 k# s- z8 f( \1 O - import { hex } from "../build/main.compiled.json";) {+ _ v& k0 _" o" j8 I
- import { getHttpV4Endpoint } from "@orbs-network/ton-access";' ^' e. T9 V, N5 w+ u" I
- import { TonClient4 } from "ton";" R, s4 Q$ G" v: E9 z
- import qs from "qs";
+ i6 `7 M0 G+ _ c' e - import qrcode from "qrcode-terminal";
2 n5 ^2 {* u% O5 b) k! I8 V$ z. f- U
! d' j2 |; H2 D2 s/ t1 c- async function onchainTestScript() {: T3 G0 h# }9 s& v9 K9 y, N* \
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
. D! r: F8 W0 A - const dataCell = new Cell();8 P% k# S) w3 U$ T. I! n& p/ l3 @
+ `4 Z$ o! J: I4 k) j1 r- const address = contractAddress(0, {, m* o& K) A% L8 W0 U: E! Y
- code: codeCell,% g; i" G1 }" q, J( v7 m# W. Z
- data: dataCell,. Z. w9 c' |/ S$ y7 U% s) Y
- });" u) Y1 V6 y- i$ E
6 p& B+ V7 m5 ]2 s1 N- const endpoint = await getHttpV4Endpoint({5 d) S* l$ `3 @9 w3 A0 Y4 N
- network: "testnet",
! s7 I$ g* n' o) H# u% V. ? - });
6 h, W9 r. x8 [8 F0 f - const client4 = new TonClient4({ endpoint });2 e7 {: u: E% V3 k2 @$ [, E
- ( W& |* L5 a" W* r2 N- i: I
- const latestBlock = await client4.getLastBlock();" ^& K9 A+ I6 E2 U6 ^' }1 w- D
- let status = await client4.getAccount(latestBlock.last.seqno, address);" W2 G+ b, k" Q$ j+ L; |
- 3 L/ @6 p$ @# V
- if (status.account.state.type !== "active") {
) F, Y& r: J7 ]" ?8 L - console.log("Contract is not active");; Z7 y: e+ V# P0 b3 Q
- return;
( ^% Q' c2 T; I# o/ T3 X - }
' r h9 R0 X8 h' h - % k, S! i& f/ O
- let link =
: K$ P5 A/ ^9 B. _* t* m0 x - `https://test.tonhub.com/transfer/` +$ @; c4 H) K! Q) C% e) f8 T; k" k
- address.toString({
- E. M1 u8 k. }3 Y! g: e p1 ` - testOnly: true,
! X$ m! p9 L: a - }) +
, m) C) [2 I* B6 s - "?" +
4 n/ W# I, u& L - qs.stringify({* E! w9 a- A9 Y9 i& ^8 [& X
- text: "Simple test transaction",
* G( @- i$ k4 L: ?8 A4 {( ?& P g7 K - amount: toNano(1).toString(10),
3 b/ b4 L5 `: G1 M% I& ~' g - });, T; B6 b- z( c) d4 c
- : G- ]* ]4 Y( |
- qrcode.generate(link, { small: true }, (code) => {
8 q" B7 v9 y0 ]/ |: G- j; \ - console.log(code);
, E R4 M o5 P" k' k U' ~ - });. b9 Y3 E% P* U* m
P" [. c" N, i6 `- let recent_sender_archive: Address;! T t+ {# H1 O) ~$ @+ A' y
- - x' v) {" F! ~
- setInterval(async () => {
* s4 C# n) a( L- A% d% Y - const latestBlock = await client4.getLastBlock();$ D/ S) z' G7 S$ y) R2 Q
- const { exitCode, result } = await client4.runMethod() k3 [; H' `$ I% g* |' @ n# i
- latestBlock.last.seqno,9 B, N' p/ g- R. Z5 l& R: B
- address,0 u+ c* ]+ Y2 Q- s
- "get_the_latest_sender"; H1 v- K6 ^/ {# g
- );
9 ]( K$ E: R6 [7 ?% O6 n
- m2 Y' M& h& A+ Y- if (exitCode !== 0) {3 O8 j2 e7 [: o4 _0 W a
- console.log("Running getter method failed");
! B C" y" J0 @& W9 Y$ N - return;, a0 f3 P, H7 z5 H o( I" Z7 L
- }5 b* X4 i9 r$ |- u, {* B
- if (result[0].type !== "slice") {
4 C e- I }; O* M - console.log("Unknown result type");0 b/ P9 w( N! E, X
- return;. b, g1 ~$ g/ ^4 ^$ ?* E( }( j
- }
7 x# ?9 i$ A/ H; x2 l% V - 0 V& E: u1 G+ y0 _, r
- let most_recent_sender = result[0].cell.beginParse().loadAddress();6 m" M7 ~5 X3 I" {4 u
& [2 T# C% K2 U* t+ j- j- if (9 j. d& L* t% l7 e
- most_recent_sender &&
% V7 W% [ g! t, w7 v - most_recent_sender.toString() !== recent_sender_archive?.toString()
0 g3 {" Q$ {) i; e3 [6 j - ) {
* j. |2 t& ~: ]" a" N5 f5 m3 ^; ` - console.log(
- {6 k8 l5 K( T6 y* } - "New recent sender found: " +' R7 h2 q+ R* s" r4 N
- most_recent_sender.toString({ testOnly: true })
4 o) A/ h+ t8 _* m - );6 B$ P9 Q% k4 Z# ]- |( F0 f7 l
- recent_sender_archive = most_recent_sender;
+ u* q6 o7 @- t$ Q! U) u - }
$ d: r( r9 |5 _2 | - }, 2000);
+ |/ W0 L& q1 d# i2 y" l - }( F f( q! a5 X; z, M+ }
0 m6 L' P u. u$ N" j3 g6 n- onchainTestScript();
复制代码 我们继续运行它!
% Q2 n& ^7 v: E P% \$ e% `- yarn onchaintest7 }+ @1 e6 o, s8 U: b1 V& ^
复制代码如果您的操作正确无误,您将在终端看到一个二维码和最新的发件人地址。它必须等于您的 Tonhub 钱包地址,也就是您用来部署合约的地址。 1 K5 x' q4 F% b N; ^( ]
! P" ?4 S4 z0 f8 g在部署合约时,我们将代码和初始状态连同 1 TON 一起发送,因此在部署合约后,代码立即被执行,并将发送者保存到 c4 存储中。
; n+ i# ]7 n" @; J3 u8 I1 f) h为确保我们的脚本按预期运行,您可以在 Tonhub 上创建另一个钱包,在其中装入 测试网 TON 币,然后再次将其发送到我们的合约中。几秒钟后,你就会在控制台中看到更新,因为最近的发送者地址发生了变化。
! G: r# `" r* Y/ W! c( \) ?) y# i' G
部署到主网让我们调整一下脚本,以便将合约部署到主网。我们必须更新所有包含测试网/主网差异的地方的代码。 现在我们更新 deploy.ts 文件的代码: - import { hex } from "../build/main.compiled.json";
* w" E& M9 n) [* Q& [ - import {' X n6 t J, g# D2 _+ I. R
- beginCell,5 @/ y" z9 ~! z
- Cell,
' q7 }8 U# [9 M& D/ p% S" j D - contractAddress,$ Q$ U, l' \9 a
- StateInit,
& I+ r" i3 |+ |2 `5 Q* N - storeStateInit,6 o4 H4 L! r |/ e4 R- s
- toNano,$ W" Z$ m4 S+ I5 t# p
- } from "ton";
6 k% H6 D% `5 H ~ F3 A - import qs from "qs";! j7 f7 d( X) r6 J0 C9 e7 j
- import qrcode from "qrcode-terminal";8 ~0 \: O8 ^* d2 f* W1 x% Y
- , }/ M H4 a6 A0 `/ b. p/ O
- 7 x& z, R; c* @2 K9 a: v6 B: |
- async function deployScript() {
& t' J/ m$ ]* C1 S9 L& } - console.log(# l6 b6 V% H, u0 k- s& d" b2 i
- "================================================================="" b" R! P% g r
- );/ x! I# r @/ m' _ H
- console.log("Deploy script is running, let's deploy our main.fc contract...");
; l% e6 J9 n; t0 K1 U
* G" z: b4 O% f: ^. @2 _6 p- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
, T; `& ?. y9 w: p# H - const dataCell = new Cell();0 A x+ `; L' @# X( g
; s6 }( U' ^' t' N" q' Q( \4 o1 }- const stateInit: StateInit = {" B" m6 E" Y f$ a% T
- code: codeCell,
" H9 k5 j0 S8 Z& Q* c; u - data: dataCell,
+ o2 I6 H7 Q( y) }' I5 k( C - };8 T2 f5 u8 e& V) b/ k0 C: Q
, @9 x6 P7 i4 ?& ]; W- const stateInitBuilder = beginCell();
\3 n, ?1 E5 l& b: [ - storeStateInit(stateInit)(stateInitBuilder);
8 E. H" @2 s8 e& z. J - const stateInitCell = stateInitBuilder.endCell();
% a4 Y$ S) N2 U
8 O/ @+ q) K! H/ i* u" C- const address = contractAddress(0, {# Y7 @" P( y( r7 J
- code: codeCell,7 g3 U; t3 f: s$ y$ f" C
- data: dataCell,
( R4 o, v$ ]3 I- V) [ - });
# C4 [9 F6 U% C% K v: C - 0 e3 y( |: i7 t/ I: `
- console.log(1 \2 L2 b) W) O F* ?1 n7 X
- `The address of the contract is following: ${address.toString()}`
5 w1 E6 i9 B7 ] V" F- | - );
; V+ ]. r& o% p* Q - console.log(`Please scan the QR code below to deploy the contract:`);
' {( S* U9 I1 M& w& O
3 u( b& z# [4 v- i1 J- let link =
8 X, ]( z2 X# w# _! u - `https://tonhub.com/transfer/` +
+ l4 ^( k0 c# t - address.toString({6 z' i7 B8 B; _3 |2 b) X
- testOnly: false,
_- t+ Y3 |; A S% Z8 k5 N - }) +
4 D- i" W* b5 E - "?" +0 P% f; b$ b2 B! m
- qs.stringify({2 O* N) V$ k+ ]' R
- text: "Deploy contract",
0 E7 ~. x0 O) E4 }$ z) O, L) c - amount: toNano(1).toString(10),, O+ V& B& V+ H0 f( K
- init: stateInitCell.toBoc({ idx: false }).toString("base64"),* G9 @: x$ b! ]: }
- });+ t! T2 p" D+ b \/ v
/ d. V" c: n* T- v+ K( j1 R- qrcode.generate(link, { small: true }, (code) => {
' V2 C1 W; N; w- _ h# C' l9 J - console.log(code);
8 i! W! w% z; v - });! ]! [6 D6 N( O) M
- }
2 c3 {9 V" T7 t - 0 ]9 [1 H/ r3 Y
- deployScript();
复制代码我们只需修改一个地方的代码: 现在,让我们用同样的方法更新 onchaintest.ts 脚本: - import { Cell, contractAddress, Address, TonClient4, toNano } from "@ton/ton";
6 k* ]+ ~2 ~* t; @7 b1 {! B - import { hex } from "../build/main.compiled.json";
9 S# }7 F$ k# m: x* { - import { getHttpV4Endpoint } from "@orbs-network/ton-access";
5 b$ E, F4 e" `# w( T3 @ - import qs from "qs";
/ y$ n( w* k q# D - import qrcode from "qrcode-terminal";
: f8 o5 \; K2 f2 I/ `" x
% t3 G% e b& A$ s: |7 w* D
; M! Q9 L3 M5 m( \# t- async function onchainTestScript() {$ v, n1 j G. C
- const codeCell = Cell.fromBoc(Buffer.from(hex, "hex"))[0];
1 @9 Z1 k& D- b8 t: H; w5 F - const dataCell = new Cell();1 M) z2 p& ?% f3 N/ Y% I3 x
8 S. n* D5 o$ U8 Q
8 c6 ]& m: a9 ~) D4 L$ @- const address = contractAddress(0, {
0 L. v2 y8 @" P8 v - code: codeCell,
, r# Z: w& S7 R( p* R - data: dataCell,* h4 L: o4 B0 }* @( @3 S u
- });
! @4 q' J+ S7 l0 z# L - ; M" E3 u4 ^0 `$ m! e$ T5 v
- , g" X, N5 s5 \: s: E8 M0 v& _
* D. i1 W, X9 k, O6 ^- h, n) Q4 O
% X1 x0 b. m8 a- const endpoint = await getHttpV4Endpoint({9 b2 V g! v) o7 s
- network: "mainnet", e- E/ e i1 Q0 x+ `. e
- });0 g) C: D( _2 j( L; z' S( o' G+ ^/ a
- const client4 = new TonClient4({ endpoint });, W$ p! y5 a, `+ T- j" z
- . Z- |5 E! X4 \# ?4 ~! j; X' ]
1 J2 M: ?5 j% r7 V5 D- const latestBlock = await client4.getLastBlock();, C3 ]4 t# r* ~7 G) k. U
- let status = await client4.getAccount(latestBlock.last.seqno, address); ]' [/ t2 A7 L* Q3 O% T: ?
- . V0 o& {5 W: c! J# ]. T
- ; b3 G! [: K' _& l7 M0 s: A2 `$ k
- if (status.account.state.type !== "active") {5 U+ F7 w2 ^; o# s' r! H
- console.log("Contract is not active");
5 K! l+ V* ]0 ^, j* f - return;
' n) J0 }6 ]$ _" }( H - }1 f) ?0 Y& X( ^
- ! X5 a. z0 _2 ?
- 0 Y- c( ~ |4 t: Y+ H
- let link =
a0 i7 s( q" e" x0 H3 I2 f ^ - `https://tonhub.com/transfer/` +4 [, @' U u) I* g
- address.toString({ testOnly: false }) +
" k7 V( X: f y" s - "?" +6 r/ l& P; A8 G& J0 X
- qs.stringify({+ M$ ~' Z2 f; @* ] Z- i; v
- text: "Simple test transaction",
% p, I: y. c9 h3 Z& O - amount: toNano(1).toString(10),
( s7 u4 F4 }0 O8 k) E - });/ Q0 O5 ^% o+ u+ W" n% o, o# T
: [" e' @5 d+ y- 6 P6 b% d8 m$ }6 } T* D
- qrcode.generate(link, { small: true }, (code) => {
$ I9 ~1 [9 C& J( b8 q - console.log(code);
. T% ]$ |" Q( V! W - });+ U D/ s% W& S" H" j
- 3 _/ b7 G$ H+ M/ j; S6 y2 Y/ h
- 1 ]2 a% b1 D/ D) M* p
- let recent_sender_archive: Address;
1 E5 Z6 Z+ W8 ?7 e - ) Q" b$ x. Q# W: o- X' @
- 4 W; O3 @0 b* @" z$ l$ n
- setInterval(async () => {1 G! I# @4 z& k0 i2 X/ p; [
- const latestBlock = await client4.getLastBlock();- K# F; K& f8 L. L& @
- const { exitCode, result } = await client4.runMethod(
9 x \9 D) v( W/ L/ Q1 g. g - latestBlock.last.seqno,
3 @) @% k9 F6 v3 f/ b. }" m, ^ - address,
! [$ Y) l/ R0 y& D+ X - "get_the_latest_sender"
' T1 b, k- r) ^ - );% `/ ~1 t1 w# k4 D
3 h& [- E- Q2 o1 u$ J3 z9 X" d7 o- $ w b, _$ [) S5 A' {
- if (exitCode !== 0) {7 W1 B! j( ]) P" I# C4 L
- console.log("Running getter method failed");( ^! I" a6 `% i0 n
- return;$ W3 e' C( n! O1 C( a& X
- }
5 d3 ~$ _* b' U, n - if (result[0].type !== "slice") {
" z( H1 D. L+ @ - console.log("Unknown result type");
& k: ~" H) F" m% E: [8 W. C9 l# J: M - return;2 I7 D* ~4 G4 J* T$ `9 U8 }
- }
; n& S; m `* b. ^: m
7 v# m- S, H& l. J, |- , y. u5 t0 `0 E5 r- T# n
- let most_recent_sender = result[0].cell.beginParse().loadAddress();, K1 c1 s. e8 g6 h' @
- % j Q* n8 V% Y, p
8 q8 `3 B2 q7 a9 Y# L2 ^& ?- if (% w, j+ M+ s* p$ {: }( H
- most_recent_sender &&7 k( P% G c+ W5 ~! ^* K& |
- most_recent_sender.toString() !== recent_sender_archive?.toString()
8 F+ u/ c& D8 R- X0 M4 E - ) {+ f: m/ s* E* `; V) n( [
- console.log(
" f* h9 J; N; B$ P - "New recent sender found: " +" g# t+ X& @ @2 m8 a" t
- most_recent_sender.toString({
% N( J: P% q( \6 Y' b' ~9 I - testOnly: false,
( T( Q, T6 M7 V, [/ ~% I; Q - })1 e# k) ~9 e6 o2 u' E) O' i
- );
0 H: ? s+ \" q+ @4 U - recent_sender_archive = most_recent_sender;! g$ P5 g( e7 o2 z) M
- }
" t) J3 O, T9 B$ Y7 M0 w. L% V0 q; K - }, 2000);
" z+ P* U* I! F; J3 m; j# d; C0 R - }
: L4 Q- c- L8 X' z3 {. }* A% }
2 t4 E. ~+ T9 n7 z+ z9 _( F
1 k4 d9 X% `1 g: \5 p! z; E- onchainTestScript()
复制代码同样,我们还需要修改 onchaintest.ts 脚本中的两处内容: 现在,我们可以在测试网和主网上部署和运行链上测试! 务必牢记,进行主网部署,您需要将Tonhub钱包切换回主网. 这适用于 iOS 以及 Android.
你成功了 说真的,我为你感到骄傲!我们已经做了大量的工作,但好在你现在已经了解了开发和部署合约的整个过程。 在下一章中,我们将深入探讨更复杂的 FunC 逻辑,并为其编写本地测试。 / `8 L) z& P* `# z* F/ r
1 j% I+ M2 o4 B: V' ~* \% C
|