Skip to main content

Transactions

Transactions let you send Cadence code to the Flow blockchain that permanently alters its state.

We are assuming you have read the Scripts Documentation before this, as transactions are sort of scripts with more required things.

While query is used for sending scripts to the chain, mutate is used for building and sending transactions. Just like scripts, fcl.mutate is a JavaScript Tagged Template Literal that we can pass Cadence code into.

Unlike scripts, they require a little more information, things like a proposer, authorizations and a payer, which may be a little confusing and overwhelming.

Sending Your First Transaction

There is a lot to unpack in the following code snippet. It sends a transaction to the Flow blockchain. For the transaction, the current user is authorizing it as both the proposer and the payer. Something that is unique to Flow is the one paying for the transaction doesn't always need to be the one performing the transaction. Proposers and Payers are special kinds of authorizations that are always required for a transaction. The proposer acts similar to the nonce in Ethereum transactions, and helps prevent repeat attacks. The payer is who will be paying for the transaction. If these are not set, FCL defaults to using the current user for all roles.

fcl.mutate will return a transactionId. We can pass the response directly to fcl.tx and then use the onceExecuted method which resolves a promise when a transaction result is available.


_17
import * as fcl from '@onflow/fcl';
_17
_17
const transactionId = await fcl.mutate({
_17
cadence: `
_17
transaction {
_17
execute {
_17
log("Hello from execute")
_17
}
_17
}
_17
`,
_17
proposer: fcl.currentUser,
_17
payer: fcl.currentUser,
_17
limit: 50,
_17
});
_17
_17
const transaction = await fcl.tx(transactionId).onceExecuted();
_17
console.log(transaction); // The transactions status and events after being executed

Authorizing a Transaction

The below code snippet is the same as the above one, except for one extremely important difference. Our Cadence code this time has a prepare statement, and we are using the fcl.currentUser when constructing our transaction.

The prepare statement's arguments directly map to the order of the authorizations in the authorizations array. Four authorizations means four &Accounts as arguments passed to prepare. In this case though there is only one, and it is the currentUser.

These authorizations are important as you can only access/modify an accounts storage if you have the said accounts authorization.


_21
import * as fcl from '@onflow/fcl';
_21
_21
const transactionId = await fcl.mutate({
_21
cadence: `
_21
transaction {
_21
prepare(acct: &Account) {
_21
log("Hello from prepare")
_21
}
_21
execute {
_21
log("Hello from execute")
_21
}
_21
}
_21
`,
_21
proposer: fcl.currentUser,
_21
payer: fcl.currentUser,
_21
authorizations: [fcl.currentUser],
_21
limit: 50,
_21
});
_21
_21
const transaction = await fcl.tx(transactionId).onceExecuted();
_21
console.log(transaction); // The transactions status and events after being executed

To learn more about mutate, check out the API documentation.

Querying Transaction Results

When querying transaction results (e.g., via HTTP/REST endpoints like GET /v1/transaction_results/{id}), you can provide either:

  • A transaction ID (256-bit hash as hex string)
  • A scheduled transaction ID (UInt64 as decimal string)

The returned result always includes transaction_id as the underlying native transaction ID. For scheduled transactions, this will be the system transaction ID that executed the scheduled callback.

Learn more about Scheduled Transactions.

Transaction Finality

As of FCL v1.15.0, it is now recommended to use use onceExecuted in most cases, leading to a 2.5x reduction in latency when waiting for a transaction result. For example, the following code snippet should be updated from:


_10
import * as fcl from '@onflow/fcl';
_10
const result = await fcl.tx(txId).onceSealed();

to:


_10
import * as fcl from '@onflow/fcl';
_10
const result = await fcl.tx(txId).onceExecuted();

Developers manually subscribing to transaction statuses should update their listeners to treat "executed" as the final status (see the release notes here). For example, the following code snippet should be updated from:


_10
import * as fcl from '@onflow/fcl';
_10
import { TransactionExecutionStatus } from '@onflow/typedefs';
_10
_10
fcl.tx(txId).subscribe((txStatus) => {
_10
if (txStatus.status === TransactionExecutionStatus.SEALED) {
_10
console.log('Transaction executed!');
_10
}
_10
});


_11
import * as fcl from '@onflow/fcl';
_11
import { TransactionExecutionStatus } from '@onflow/typedefs';
_11
_11
fcl.tx(txId).subscribe((txStatus) => {
_11
if (
_11
// SEALED status is no longer necessary
_11
txStatus.status === TransactionExecutionStatus.EXECUTED
_11
) {
_11
console.log('Transaction executed!');
_11
}
_11
});

The "executed" status corresponds to soft finality, indicating that the transaction has been included in a block and a transaction status is available, backed by a cryptographic proof. Only in rare cases should a developer need to wait for "sealed" status in their applications and you can learn more about the different transaction statuses on Flow here.

See the following video for demonstration of how to update your code to wait for "executed" status: