2022-03-23 08:50:10 -07:00
|
|
|
#!/bin/bash
|
|
|
|
|
|
|
|
set -euo pipefail
|
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
function usage() {
|
|
|
|
cat<<-EOF >&2
|
|
|
|
Usage:
|
|
|
|
$(basename "$0") [-h] [-n network] <.so file> <account address> -- Verify that the deployed on-chain bytecode matches the local object file
|
2022-03-23 08:50:10 -07:00
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
where:
|
|
|
|
-h show this help text
|
|
|
|
-n set the network (mainnet, testnet, devnet. defaults to \$NETWORK if set)
|
|
|
|
EOF
|
|
|
|
exit 1
|
|
|
|
}
|
2022-03-23 08:50:10 -07:00
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
network=""
|
|
|
|
if [[ -n $NETWORK ]]; then
|
|
|
|
network=$NETWORK
|
|
|
|
fi
|
2022-03-23 08:50:10 -07:00
|
|
|
while getopts ':hn:' option; do
|
|
|
|
case "$option" in
|
2022-07-06 23:21:33 -07:00
|
|
|
h) usage
|
2022-03-23 08:50:10 -07:00
|
|
|
;;
|
|
|
|
n) network=$OPTARG
|
|
|
|
;;
|
|
|
|
:) printf "missing argument for -%s\n" "$OPTARG" >&2
|
2022-07-06 23:21:33 -07:00
|
|
|
usage
|
2022-03-23 08:50:10 -07:00
|
|
|
;;
|
|
|
|
\?) printf "illegal option: -%s\n" "$OPTARG" >&2
|
2022-07-06 23:21:33 -07:00
|
|
|
usage
|
2022-03-23 08:50:10 -07:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
shift $((OPTIND - 1))
|
|
|
|
|
|
|
|
case "$network" in
|
|
|
|
mainnet) moniker="m";;
|
|
|
|
testnet) moniker="d";;
|
|
|
|
devnet) moniker="l";;
|
|
|
|
*) printf "Network not set. Specify with -n\n" >&2
|
2022-07-06 23:21:33 -07:00
|
|
|
usage
|
2022-03-23 08:50:10 -07:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
[ $# -ne 2 ] && usage
|
2022-03-23 08:50:10 -07:00
|
|
|
obj_file=$1
|
|
|
|
sol_addr=$2
|
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
account_json=$(mktemp)
|
|
|
|
account_dump=$(mktemp)
|
|
|
|
|
2022-03-23 08:50:10 -07:00
|
|
|
# Grab account content as JSON
|
2022-07-06 23:21:33 -07:00
|
|
|
solana account "$sol_addr" -u $moniker --output-file "$account_json" --output json-compact >/dev/null
|
2022-03-23 08:50:10 -07:00
|
|
|
# decode the base64 account data to binary
|
2022-07-06 23:21:33 -07:00
|
|
|
jq '.account.data[0]' "$account_json" | sed s/\"//g | base64 -d > "$account_dump"
|
2022-03-23 08:50:10 -07:00
|
|
|
|
|
|
|
# The first 37 bytes are irrelevant, the actual ELF object code starts after,
|
|
|
|
# so we drop these bytes. Presumably those bytes correspond to an encoded rust
|
|
|
|
# enum constructor?
|
|
|
|
# Set the block size to 37 bytes and skip the first block.
|
2022-07-14 15:11:42 -07:00
|
|
|
dd bs=37 skip=1 if="$account_dump" of=/tmp/bytecode.dump 2>/dev/null
|
2022-03-23 08:50:10 -07:00
|
|
|
|
2022-07-06 23:21:33 -07:00
|
|
|
hash1=$(sha256sum /tmp/bytecode.dump | cut -f1 -d' ')
|
|
|
|
hash2=$(sha256sum "$obj_file" | cut -f1 -d' ')
|
2022-03-23 08:50:10 -07:00
|
|
|
|
|
|
|
echo "Deployed bytecode hash (on $network):"
|
2022-07-06 23:21:33 -07:00
|
|
|
echo "$hash1"
|
2022-03-23 08:50:10 -07:00
|
|
|
echo "$obj_file hash:"
|
2022-07-06 23:21:33 -07:00
|
|
|
echo "$hash2"
|
2022-03-23 08:50:10 -07:00
|
|
|
|
|
|
|
if [ "$hash1" == "$hash2" ]; then
|
|
|
|
printf "\033[0;32mSuccessfully verified\033[0m\n";
|
|
|
|
exit 0;
|
|
|
|
else
|
2022-07-06 23:21:33 -07:00
|
|
|
printf "\033[0;31mFailed to verify\033[0m\n" >&2;
|
2022-07-14 15:11:42 -07:00
|
|
|
echo "JSON: $account_json" >&2;
|
|
|
|
echo "Dump: $account_dump" >&2;
|
2022-03-23 08:50:10 -07:00
|
|
|
exit 1;
|
|
|
|
fi
|