57018 sc high handler address format mismatch causes digest divergence and unexecutable messages
Submitted on Oct 22nd 2025 at 17:19:22 UTC by @Rhaydden for Audit Comp | Folks Finance: Wormhole NTT on Algorand
Report ID: #57018
Report Type: Smart Contract
Report severity: High
Target: https://github.com/Folks-Finance/algorand-ntt-contracts/blob/main/ntt_contracts/transceiver/TransceiverManager.py
Impacts:
Permanent freezing of funds
Description
Issue description
There are two incompatible handler formats in play here.
TransceiverManager.attestation_received() parses
message.handler_addressas a uint64 app ID via BytesUtils.safe_convert_bytes32_to_uint64(...) (top 24 bytes zero, last 8 bytes app ID).MessageHandler.execute_message() asserts
Address(message.handler_address.bytes) == Global.current_application_address, expecting a 32 byte Algorand app address.TransceiverManager.calculate_message_digest() includes
message.handler_address.bytesin the keccak input; any change in representation changes the digest.Wormhole consumption flow marks a VAA as consumed before downstream checks. WormholeTransceiver._receive_message() marks the VAA as consumed before forwarding to the manager, so if the manager/handler later fails, the VAA cannot be retried.
Result: a message attested under one handler representation (padded app ID bytes) can fail execution if the handler expects an app address (32 bytes), or vice versa. Because the VAA is consumed first, the message becomes permanently stuck.
Impact
Critical — Permanent freezing of funds.
Messages can be irrecoverably stuck on the destination chain after VAA consumption if attestations or execution fail due to the handler address format mismatch. No re-delivery is possible without manual intervention.
Recommended mitigation steps
Standardize the handler address representation across all components.
A straightforward mitigation: standardize on the padded app ID representation. In MessageHandler.execute_message(), replace the current address equality check with an app ID equality check:
Parse
handler_addresswithBytesUtils.safe_convert_bytes32_to_uint64().Compare the parsed uint64 to
Global.current_application_id.id.
This ensures digest inputs and runtime checks use the same handler format.
Proof of Concept
The included tests demonstrate the digest divergence and execution failure when switching the handler representation.
Test added in: tests/poc/HandlerAddressMismatch.test.ts (see links in report)
Example test excerpt:
test("calculate_message_digest differs between padded app id bytes vs app address bytes", async () => {
const base = getRandomMessageToSend({
destinationChainId: Number(getRandomUInt(MAX_UINT16)),
sourceAddress: getRandomBytes(32),
userAddress: getRandomBytes(32),
});
const paddedHandler = convertNumberToBytes(handlerAppId, 32);
const appAddressBytes = getApplicationAddress(handlerAppId).publicKey;
const sourceChainId = Number(getRandomUInt(MAX_UINT16));
const digestPadded = await transceiverManagerClient.calculateMessageDigest({
args: [{
id: base.id,
userAddress: base.sourceAddress,
sourceChainId,
sourceAddress: base.sourceAddress,
handlerAddress: paddedHandler,
payload: base.payload,
}],
});
const digestAddress = await transceiverManagerClient.calculateMessageDigest({
args: [{
id: base.id,
userAddress: base.sourceAddress,
sourceChainId,
sourceAddress: base.sourceAddress,
handlerAddress: appAddressBytes,
payload: base.payload,
}],
});
expect(Uint8Array.from(digestPadded)).not.toEqual(Uint8Array.from(digestAddress));
});As shown by the PoC,:
executeMessagefails whenhandlerAddressis a 32-byte padded uint64 app ID (handler expects app address bytes).calculate_message_digestdiffers when onlyhandlerAddresschanges (padded app ID vs app address bytes).
Logs
References
TransceiverManager.attestation_received: https://github.com/Folks-Finance/algorand-ntt-contracts//blob/99dfaec5420fa22cfcfaef756f96bfaa5b79701b/ntt_contracts/transceiver/TransceiverManager.py#L189-L218
BytesUtils.safe_convert_bytes32_to_uint64: https://github.com/Folks-Finance/algorand-ntt-contracts//blob/99dfaec5420fa22cfcfaef756f96bfaa5b79701b/folks_contracts/library/BytesUtils.py#L12-L16
MessageHandler.execute_message: https://github.com/Folks-Finance/algorand-ntt-contracts//blob/99dfaec5420fa22cfcfaef756f96bfaa5b79701b/ntt_contracts/transceiver/MessageHandler.py#L35-L65
TransceiverManager.calculate_message_digest: https://github.com/Folks-Finance/algorand-ntt-contracts//blob/99dfaec5420fa22cfcfaef756f96bfaa5b79701b/ntt_contracts/transceiver/TransceiverManager.py#L229-L238
WormholeTransceiver._receive_message: https://github.com/Folks-Finance/algorand-ntt-contracts//blob/99dfaec5420fa22cfcfaef756f96bfaa5b79701b/ntt_contracts/transceiver/WormholeTransceiver.py#L138-L184
Was this helpful?