const bitcore = require("bitcore-lib"); const { verifyMessageSignatureRsv } = require('@stacks/encryption'); const { ECPairFactory } = require('ecpair'); const tinysecp = require('tiny-secp256k1'); const ECPair = ECPairFactory(tinysecp); const verifyData = (commitment, selection, reveal) => { const checks = { 'Timestamp Verification': verifyTimestamps(commitment, selection, reveal), 'Game Nonce Verification': verifyGameNonce(commitment, selection, reveal), 'Game Nonce Signature Verification': verifyGameNonceSignature(commitment, selection), 'Commitment Integrity Verification': verifyCommitmentIntegrity(commitment, reveal), 'Outcome Hash Verification': verifyOutcomeHashValue(selection, reveal), 'Game Outcome Verification': verifyGameOutcome(reveal) }; let allChecksPass = true; for (const [checkName, result] of Object.entries(checks)) { if (result) { console.log(`✅ - ${checkName}`); } else { console.log(`❌ - ${checkName}`); allChecksPass = false; } } if (allChecksPass) { console.log(`Congratulations, you have carried out a direct stateless verification of a trustlessly random coin flip, which you ${reveal.didWin ? '✨won✨' : 'lost'}`); } else { console.log(`This is mathematically impossible, contact a member of the team for help on how to verify.`); } }; const verifyTimestamps = (commitment, selection, reveal) => { return ( commitment.gameTimestamp < selection.selectionTimestamp && selection.selectionTimestamp < reveal.gameTimestamp ) }; const verifyGameNonce = (commitment, selection, reveal) => { return ( commitment.gameNonce === selection.gameNonce && selection.gameNonce === reveal.gameNonce ) } const verifyGameNonceSignature = (commitment, selection) => { if (!commitment.gameNonce === selection.gameNonce) return const gameNonceHash = bitcore.crypto.Hash.sha256(Buffer.from(commitment.gameNonce)).toString('hex') const message = new bitcore.Message(gameNonceHash); var hash = message.magicHash(); // Using Bitcore for message hashing and verification try { var signature = bitcore.crypto.Signature.fromCompact( Buffer.from(selection.signedGameNonce, "base64") ); // recover the public key var ecdsa = new bitcore.crypto.ECDSA(); ecdsa.hashbuf = hash; ecdsa.sig = signature; const pubkeyInSig = ecdsa.toPublicKey(); const pubkeyInSigString = new bitcore.PublicKey( Object.assign({}, pubkeyInSig.toObject(), { compressed: true }) ).toString(); if (pubkeyInSigString != selection.userPublicKey) { return false; } return bitcore.crypto.ECDSA.verify(hash, signature, pubkeyInSig); } catch (e) { const publicKeyBuffer = Buffer.from(selection.userPublicKey, 'hex'); const keyPair = ECPair.fromPublicKey(publicKeyBuffer); const signatureBuffer = Buffer.from(selection.signedGameNonce, "base64"); if (signatureBuffer.length == 65) { const rawSignature = new Uint8Array(signatureBuffer).slice(1); const rawSignatureBuffer = Buffer.from(rawSignature); const isVerified = keyPair.verify(hash, rawSignatureBuffer); return isVerified } else if (signatureBuffer.length == 97) { const verified = verifyMessageSignatureRsv({ message: gameNonceHash, publicKey: selection.userPublicKey, signature: selection.signedGameNonce }); return verified } } } const verifyCommitmentIntegrity = (commitment, reveal) => { const reconstuctedCommitment = bitcore.crypto.Hash.sha256(Buffer.from(reveal.vrn + reveal.gameNonce + reveal.secretNonce)) return (reconstuctedCommitment.toString('hex') === commitment.commitment) } const verifyOutcomeHashValue = (selection, reveal) => { const userData = bitcore.crypto.Hash.sha256(Buffer.from(selection.choice + selection.gameNonce + selection.amount)) const outcomeHash = bitcore.crypto.Hash.sha256(Buffer.from(reveal.vrn + userData)) return outcomeHash.toString('hex') === reveal.outcomeHash } const verifyGameOutcome = (reveal) => { const outcome = parseInt(reveal.outcomeHash[reveal.outcomeHash.length - 1], 16) % 2 === 0 ? "tails" : "heads" return outcome === reveal.outcomeString } (async (commitment, selection, reveal) => { await verifyData(commitment, selection, reveal); })();