plugins/midnight-dapp/skills/transaction-flows/SKILL.md
Use when submitting transactions to Midnight contracts, tracking transaction status, implementing retry logic, building responsive UIs during proof generation, or understanding the build-prove-submit-confirm flow.
npx skillsauth add aaronbassett/midnight-knowledgebase midnight-dapp:transaction-flowsInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
4 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Submit, track, and handle transactions through the complete Midnight lifecycle: build, prove, submit, and confirm.
Midnight transactions go through four distinct phases:
contract.callTx.* and witness datawalletAPI.submitTransaction()// Complete transaction flow
const tx = await contract.callTx.transfer(recipient, amount, witnesses);
const provenTx = await walletAPI.balanceAndProveTransaction(tx, newCoins);
const txHash = await walletAPI.submitTransaction(provenTx);
// Wait for confirmation...
Unlike Ethereum's instant signing, Midnight proof generation takes several seconds. Your UI must:
| Document | Description | |----------|-------------| | transaction-lifecycle.md | Complete transaction flow with state transitions | | signing-flow.md | Wallet signing and proof generation details | | confirmation-patterns.md | Polling, finality, and UI update strategies | | web3-comparison.md | Ethereum transaction patterns vs Midnight |
| Example | Description | |---------|-------------| | submit-transaction/ | Complete transaction submission flow | | tx-status-tracker/ | Transaction status tracking UI | | retry-patterns/ | Exponential backoff and retry UI |
import type { WitnessContext } from "@midnight-ntwrk/midnight-js-types";
// Define witnesses for private data
const witnesses = {
get_secret: ({ privateState }: WitnessContext<PrivateState>): bigint => {
return privateState.secret;
},
};
// Build transaction
const tx = await contract.callTx.transfer(recipient, amount, witnesses);
// Show loading - this takes several seconds
setStatus("Generating proof...");
// Prove transaction (invokes local proof server)
const provenTx = await walletAPI.balanceAndProveTransaction(tx, newCoins);
// Submit to network
setStatus("Submitting...");
const txHash = await walletAPI.submitTransaction(provenTx);
setStatus("Submitted!");
console.log("Transaction hash:", txHash);
function useTxStatus(txHash: string | null) {
const [status, setStatus] = useState<TxStatus>("pending");
useEffect(() => {
if (!txHash) return;
const checkStatus = async () => {
// Poll for transaction confirmation
const result = await checkTransaction(txHash);
setStatus(result.status);
};
const interval = setInterval(checkStatus, 3000);
return () => clearInterval(interval);
}, [txHash]);
return status;
}
async function submitWithRetry(
buildTx: () => Promise<Transaction>,
maxRetries = 3
): Promise<string> {
let lastError: Error | null = null;
for (let attempt = 1; attempt <= maxRetries; attempt++) {
try {
const tx = await buildTx();
const provenTx = await walletAPI.balanceAndProveTransaction(tx, newCoins);
return await walletAPI.submitTransaction(provenTx);
} catch (error) {
lastError = error as Error;
// Don't retry user rejection
if (error.message.includes("rejected")) {
throw error;
}
// Exponential backoff
if (attempt < maxRetries) {
await new Promise((r) => setTimeout(r, 1000 * Math.pow(2, attempt)));
}
}
}
throw lastError ?? new Error("Transaction failed");
}
type TxState =
| { status: "idle" }
| { status: "building" }
| { status: "proving" }
| { status: "submitting" }
| { status: "pending"; txHash: string }
| { status: "confirmed"; txHash: string }
| { status: "failed"; error: Error };
function txReducer(state: TxState, action: TxAction): TxState {
switch (action.type) {
case "START":
return { status: "building" };
case "PROVING":
return { status: "proving" };
case "SUBMITTING":
return { status: "submitting" };
case "SUBMITTED":
return { status: "pending", txHash: action.txHash };
case "CONFIRMED":
return { status: "confirmed", txHash: action.txHash };
case "FAILED":
return { status: "failed", error: action.error };
case "RESET":
return { status: "idle" };
default:
return state;
}
}
async function transferWithOptimisticUpdate(
recipient: Uint8Array,
amount: bigint
) {
// Optimistically update UI
const previousBalance = balance;
setBalance((prev) => prev - amount);
try {
const tx = await contract.callTx.transfer(recipient, amount, witnesses);
const provenTx = await walletAPI.balanceAndProveTransaction(tx, newCoins);
await walletAPI.submitTransaction(provenTx);
// Success - optimistic update was correct
} catch (error) {
// Revert optimistic update
setBalance(previousBalance);
throw error;
}
}
| Concern | Mitigation | |---------|------------| | Proof generation (5-30s) | Show progress UI, allow cancellation | | Network latency | Implement retry with exponential backoff | | Timeout during proof | Set appropriate timeout (60s+), retry on timeout | | User waits for confirmation | Show pending state, allow background monitoring |
wallet-integration - Wallet connection before transactionsproof-handling - Witness construction and proof generationstate-management - Reading state before/after transactionserror-handling - Transaction error messages and recovery/dapp-check - Validates transaction provider configuration/dapp-debug transactions - Diagnose transaction submission issuestools
Use when setting up Midnight development environment, installing Compact compiler and developer tools, configuring proof server, verifying prerequisites, or getting started with Midnight development.
tools
--- name: midnight-tooling:midnight-debugging description: Use when encountering Midnight errors like "compact: command not found", "ERR_UNSUPPORTED_DIR_IMPORT", version mismatches, proof server failures, "@midnight-ntwrk" package errors, or compilation failures. --- # Midnight Environment Debugging Expert knowledge for identifying and resolving common Midnight development toolchain issues. ## Diagnostic Approach When encountering Midnight-related errors, follow this systematic approach: 1.
tools
Use when checking Midnight version compatibility, understanding pragma language_version, verifying compiler and runtime version relationships, or troubleshooting version mismatch errors between Midnight components.
tools
Use when setting up CI/CD for Midnight projects, configuring GitHub Actions for Compact contract compilation, running TypeScript tests in CI, validating version consistency, or automating contract builds.