Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/GBI-1840 - P2SH address support #18

Merged
merged 3 commits into from
May 2, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
109 changes: 80 additions & 29 deletions contracts/BtcUtils.sol
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;

import "./OpCodes.sol";

/**
* @title BtcUtils
* @notice This library contains functionality to make easier to work with Bitcoin transactions in Solidity.
Expand All @@ -10,11 +12,20 @@ pragma solidity ^0.8.18;
library BtcUtils {
uint8 private constant MAX_COMPACT_SIZE_LENGTH = 252;
uint8 private constant MAX_BYTES_USED_FOR_COMPACT_SIZE = 8;

uint8 private constant OUTPOINT_SIZE = 36;
uint8 private constant OUTPUT_VALUE_SIZE = 8;

uint8 private constant PUBKEY_HASH_SIZE = 20;
uint8 private constant PUBKEY_HASH_START = 3;
uint8 private constant CHECK_BYTES_FROM_HASH = 4;
bytes1 private constant PUBKEY_HASH_MAINNET_BYTE = 0x00;
bytes1 private constant PUBKEY_HASH_TESTNET_BYTE = 0x6f;

uint8 private constant SCRIPT_HASH_SIZE = 20;
uint8 private constant SCRIPT_HASH_START = 2;
bytes1 private constant SCRIPT_HASH_MAINNET_BYTE = 0x05;
bytes1 private constant SCRIPT_HASH_TESTNET_BYTE = 0xc4;



/**
Expand Down Expand Up @@ -78,50 +89,88 @@ library BtcUtils {
return result;
}

/// @notice Parse a raw pay-to-public-key-hash output script to get the corresponding address
/// @notice Parse a raw output script whose type is not known by the consumer. The function will
/// return the corresponding address if the type of the script is supported or an error if not.
/// For the addresses that are encoded with base58check the checksum bytes are not included in
/// the resulting byte array
/// @param outputScript the fragment of the raw transaction containing the raw output script
/// @param mainnet if the address to generate is from mainnet or testnet
/// @return The address bytes regarless of the output script type
function outputScriptToAddress(bytes calldata outputScript, bool mainnet) public pure returns (bytes memory) {
if (isP2PKHOutput(outputScript)) {
return parsePayToPubKeyHash(outputScript, mainnet);
}
if (isP2SHOutput(outputScript)) {
return parsePayToScriptHash(outputScript, mainnet);
}
// TODO add here P2WPKH, P2WSH and P2TR
revert("Unsupported script type");
}

/// @notice Check if a raw output script is a pay-to-public-key-hash output
/// @param pkScript the fragment of the raw transaction containing the raw output script
/// @return Whether the script has a pay-to-public-key-hash output structure or not
function isP2PKHOutput(bytes memory pkScript) public pure returns (bool) {
return pkScript.length == 25 &&
pkScript[0] == OpCodes.OP_DUP &&
pkScript[1] == OpCodes.OP_HASH160 &&
uint8(pkScript[2]) == PUBKEY_HASH_SIZE &&
pkScript[23] == OpCodes.OP_EQUALVERIFY &&
pkScript[24] == OpCodes.OP_CHECKSIG;
}

/// @notice Check if a raw output script is a pay-to-script-hash output
/// @param pkScript the fragment of the raw transaction containing the raw output script
/// @return Whether the script has a pay-to-script-hash output structure or not
function isP2SHOutput(bytes memory pkScript) public pure returns (bool) {
return pkScript.length == 23 &&
pkScript[0] == OpCodes.OP_HASH160 &&
uint8(pkScript[1]) == SCRIPT_HASH_SIZE &&
pkScript[22] == OpCodes.OP_EQUAL;
}

/// @notice Parse a raw pay-to-public-key-hash output script to get the corresponding address,
/// the resulting byte array doesn't include the checksum bytes of the base58check encoding at
/// the end
/// @param outputScript the fragment of the raw transaction containing the raw output script
/// @param mainnet if the address to generate is from mainnet or testnet
/// @return The address generated using the receiver's public key hash
function parsePayToPubKeyHash(bytes calldata outputScript, bool mainnet) public pure returns (bytes memory) {
require(outputScript.length == 25, "Script has not the required length");
require(
outputScript[0] == 0x76 && // OP_DUP
outputScript[1] == 0xa9 && // OP_HASH160
outputScript[2] == 0x14 && // pubKeyHashSize, should be always 14 (20B)
outputScript[23] == 0x88 && // OP_EQUALVERIFY
outputScript[24] == 0xac, // OP_CHECKSIG
"Script has not the required structure"
);
require(isP2PKHOutput(outputScript), "Script hasn't the required structure");

bytes memory destinationAddress = new bytes(PUBKEY_HASH_SIZE);
bytes memory destinationAddress = new bytes(PUBKEY_HASH_SIZE + 1);
for(uint8 i = PUBKEY_HASH_START; i < PUBKEY_HASH_SIZE + PUBKEY_HASH_START; i++) {
destinationAddress[i - PUBKEY_HASH_START] = outputScript[i];
destinationAddress[i - PUBKEY_HASH_START + 1] = outputScript[i];
}

uint8 versionByte = mainnet? 0x00 : 0x6f;
bytes memory result = addVersionByte(bytes1(versionByte), destinationAddress);

return result;
destinationAddress[0] = mainnet? PUBKEY_HASH_MAINNET_BYTE : PUBKEY_HASH_TESTNET_BYTE;
return destinationAddress;
}

function addVersionByte(bytes1 versionByte, bytes memory source) private pure returns (bytes memory) {
bytes memory dataWithVersion = new bytes(source.length + 1);
dataWithVersion[0] = versionByte;
/// @notice Parse a raw pay-to-script-hash output script to get the corresponding address,
/// the resulting byte array doesn't include the checksum bytes of the base58check encoding at
/// the end
/// @param outputScript the fragment of the raw transaction containing the raw output script
/// @param mainnet if the address to generate is from mainnet or testnet
/// @return The address generated using the script hash
function parsePayToScriptHash(bytes calldata outputScript, bool mainnet) public pure returns (bytes memory) {
require(isP2SHOutput(outputScript), "Script hasn't the required structure");

uint8 i;
for (i = 0; i < source.length; i++) {
dataWithVersion[i + 1] = source[i];
bytes memory destinationAddress = new bytes(SCRIPT_HASH_SIZE + 1);
for(uint8 i = SCRIPT_HASH_START; i < SCRIPT_HASH_SIZE + SCRIPT_HASH_START; i++) {
destinationAddress[i - SCRIPT_HASH_START + 1] = outputScript[i];
}

return dataWithVersion;
destinationAddress[0] = mainnet? SCRIPT_HASH_MAINNET_BYTE : SCRIPT_HASH_TESTNET_BYTE;
return destinationAddress;
}

/// @notice Parse a raw null-data output script to get its content
/// @param outputScript the fragment of the raw transaction containing the raw output script
/// @return The content embedded inside the script
function parseNullDataScript(bytes calldata outputScript) public pure returns (bytes memory) {
require(outputScript.length > 1,"Invalid size");
require(outputScript[0] == 0x6a, "Not OP_RETURN");
require(outputScript[0] == OpCodes.OP_RETURN, "Not OP_RETURN");
return outputScript[1:];
}

Expand Down Expand Up @@ -165,7 +214,8 @@ library BtcUtils {
(uint32(uint8(bs[offset + 3])) << 24);
}

/// @notice Check if a pay-to-script-hash address belogs to a specific script
/// @notice Check if a pay-to-script-hash address belogs to a specific script, expects the address
/// bytes to include the 4 checksum bytes at the end
/// @param p2sh the pay-to-script-hash address
/// @param script the script to check
/// @param mainnet flag to specify if its a mainnet address
Expand All @@ -174,14 +224,15 @@ library BtcUtils {
return p2sh.length == 25 && keccak256(p2sh) == keccak256(getP2SHAddressFromScript(script, mainnet));
}

/// @notice Generate a pay-to-script-hash address from a script
/// @notice Generate a pay-to-script-hash address from a script, the resulting byte array already contains
/// the 4 checksum bytes at the end of it
/// @param script the script to generate the address from
/// @param mainnet flag to specify if the output should be a mainnet address
/// @return The address generate from the script
function getP2SHAddressFromScript(bytes calldata script, bool mainnet) public pure returns (bytes memory) {
bytes20 scriptHash = ripemd160(abi.encodePacked(sha256(script)));
uint8 versionByte = mainnet ? 0x5 : 0xc4;
bytes memory versionAndHash = bytes.concat(bytes1(versionByte), scriptHash);
bytes1 versionByte = mainnet ? SCRIPT_HASH_MAINNET_BYTE : SCRIPT_HASH_TESTNET_BYTE;
bytes memory versionAndHash = bytes.concat(versionByte, scriptHash);
bytes4 checksum = bytes4(sha256(abi.encodePacked(sha256(versionAndHash))));
return bytes.concat(versionAndHash, checksum);
}
Expand Down
12 changes: 12 additions & 0 deletions contracts/OpCodes.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;


library OpCodes {
bytes1 public constant OP_DUP = 0x76;
bytes1 public constant OP_HASH160 = 0xa9;
bytes1 public constant OP_EQUALVERIFY = 0x88;
bytes1 public constant OP_CHECKSIG = 0xac;
bytes1 public constant OP_RETURN = 0x6a;
bytes1 public constant OP_EQUAL = 0x87;
}
1 change: 1 addition & 0 deletions hardhat.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { HardhatUserConfig } from "hardhat/config";
import "@nomicfoundation/hardhat-toolbox";
import "@nomiclabs/hardhat-solhint";
import "hardhat-gas-reporter";
import 'solidity-coverage';

const config: HardhatUserConfig = {
solidity: "0.8.18",
Expand Down
Loading
Loading