
Security News
The Changelog Podcast: Practical Steps to Stay Safe on npm
Learn the essential steps every developer should take to stay secure on npm and reduce exposure to supply chain attacks.
@ton-community/tx-emulator
Advanced tools
This package allows you to emulate arbitrary smart contracts, send messages to them and run get methods on them as if they were deployed on a real network.
The key difference of this package from ton-contract-executor is the fact that the latter only emulates the compute phase of the contract - it does not know about any other phases and thus does not know anything about fees and balances (in a sense that it does not know whether a contract's balance will be enough to process all the out messages that it produces). On the other hand, this package emulates all the phases of a contract, and as a result, the emulation is much closer to what would happen in a real network.
Requires node 16.
yarn add @ton-community/tx-emulator ton ton-core ton-crypto
or
npm i @ton-community/tx-emulator ton ton-core ton-crypto
To use this package, you need to create an instance of the Blockchain class using the static method Blockchain.create as follows:
import { Blockchain } from "@ton-community/tx-emulator";
const blockchain = await Blockchain.create()
After that, you can use the low level methods on Blockchain (such as sendMessage) to emulate any messages that you want, but the recommended way to use it is to write wrappers for your contract using the Contract interface from ton-core. Then you can use blockchain.openContract on instances of such contracts, and they will be wrapped in a Proxy that will supply a ContractProvider as a first argument to all its methods starting with either get or send. Also all send methods will get Promisified and will return results of running the blockchain message queue along with the original method's result in the result field.
A good example of this is the treasury contract that is basically a built-in highload wallet meant to help you write tests for your systems of smart contracts. When blockchain.treasury is called, an instance of TreasuryContract is created and blockchain.openContract is called to "open" it. After that, when you call treasury.send, Blockchain automatically supplies the first provider argument.
For your own contracts, you can draw inspiration from the contracts in the NFT collection example - all of them use the provider.internal method to send internal messages using the treasuries passed in from the unit test file.
Here is an excerpt of that from NftItem.ts:
import { Address, beginCell, Cell, Contract, ContractProvider, Sender, toNano, Builder } from "ton-core";
class NftItem implements Contract {
async sendTransfer(provider: ContractProvider, via: Sender, params: {
value?: bigint
to: Address
responseTo?: Address
forwardAmount?: bigint
forwardBody?: Cell | Builder
}) {
await provider.internal(via, {
value: params.value ?? toNano('0.05'),
body: beginCell()
.storeUint(0x5fcc3d14, 32) // op
.storeUint(0, 64) // query id
.storeAddress(params.to)
.storeAddress(params.responseTo)
.storeBit(false) // custom payload
.storeCoins(params.forwardAmount ?? 0n)
.storeMaybeRef(params.forwardBody)
.endCell()
})
}
}
When you call nftItem.sendTransfer(treasury.getSender(), { to: recipient }) (with nftItem being an "opened" instance of NftItem), an external message to the wallet represented by treasury will be pushed onto the message queue, then processed, generating an internal message to the NFT contract.
Here is another excerpt that shows the way to interact with get methods from wrappers:
import { Contract, ContractProvider } from "ton-core";
export type NftItemData = {
inited: boolean
index: number
collection: Address | null
owner: Address | null
content: Cell | null
}
class NftItem implements Contract {
async getData(provider: ContractProvider): Promise<NftItemData> {
const { stack } = await provider.get('get_nft_data', [])
return {
inited: stack.readBoolean(),
index: stack.readNumber(),
collection: stack.readAddressOpt(),
owner: stack.readAddressOpt(),
content: stack.readCellOpt(),
}
}
}
When you call nftItem.getData() (note that just like in the sendTransfer method, you don't need to supply the provider argument - it's done for you on "opened" instances), the provider will query the smart contract contained in blockchain and parse the data according to the code. Note that unlike the send methods, get methods on "opened" instances will return the original result as-is to the caller.
Note that all of the methods of contracts that you want to "open" that start with get or send NEED to accept provider: ContractProvider as a first argument (even if not used) due to how the wrapper works.
This package also registers a custom jest and chai matcher for transactions returned from a blockchain emulation. Here is an excerpt of how it's used in the NFT collection example mentioned above:
const buyResult = await buyer.send({
to: sale.address,
value: price * 2n,
sendMode: SendMode.PAY_GAS_SEPARATLY,
})
expect(buyResult.transactions).toHaveTransaction({
from: sale.address,
to: marketplace.address,
value: fee,
})
expect(buyResult.transactions).toHaveTransaction({
from: sale.address,
to: collection.address,
value: fee,
})
(.toHaveTransaction is the jest matcher; for chai you need to use expect(...).transaction or expect(...).to.have.transaction)
The matcher supports the following fields:
export type FlatTransaction = {
from?: Address
to: Address
value?: bigint
body: Cell
initData?: Cell
initCode?: Cell
lt: bigint
now: number
outMessagesCount: number
oldStatus: AccountStatus
endStatus: AccountStatus
totalFees?: bigint
aborted?: boolean
destroyed?: boolean
exitCode?: number
success?: boolean
}
But you can omit those you're not interested in, and you can also pass in functions accepting those types returning booleans (true meaning good) to check for example number ranges, message opcodes, etc. Note however that if a field is optional (like from?: Address), then the function needs to accept the optional type, too.
By default, this package will use its stored network configuration to emulate messages. However, you can set any configuration you want when creating the Blockchain instance by passing the configuration cell in the optional params argument in the config field.
Special thanks to @dungeon-master-666 for their C++ code of the emulator.
This package is released under the MIT License.
FAQs
TON transaction emulator
We found that @ton-community/tx-emulator demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 4 open source maintainers collaborating on the project.
Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Security News
Learn the essential steps every developer should take to stay secure on npm and reduce exposure to supply chain attacks.

Security News
Experts push back on new claims about AI-driven ransomware, warning that hype and sponsored research are distorting how the threat is understood.

Security News
Ruby's creator Matz assumes control of RubyGems and Bundler repositories while former maintainers agree to step back and transfer all rights to end the dispute.