UNPKG

@nori-zk/o1js-zk-utils

Version:

o1js-zk-utils supporting Nori Bridge

169 lines 7.4 kB
import { UInt64, UInt8 } from 'o1js'; import { wordToBytes } from '@nori-zk/proof-conversion/min'; import { Bytes32, } from './types.js'; export function uint8ArrayToBigIntBE(bytes) { return bytes.reduce((acc, byte) => (acc << 8n) + BigInt(byte), 0n); } export function uint8ArrayToBigIntLE(bytes) { return bytes.reduceRight((acc, byte) => (acc << 8n) + BigInt(byte), 0n); } export function fieldToHexBE(field) { const bytesLE = wordToBytes(field, 32); // This is LE const bytesBE = bytesLE.reverse(); return `0x${bytesBE .map((byte) => byte.toBigInt().toString(16).padStart(2, '0')) .join('')}`; } export function fieldToBigIntBE(field) { const bytesLE = wordToBytes(field, 32); // This is LE const bytesBE = bytesLE.reverse(); return bytesBE.reduce((acc, byte) => (acc << 8n) + byte.toBigInt(), 0n); } export function fieldToHexLE(field) { const bytesLE = wordToBytes(field, 32); // This is LE return `0x${bytesLE .map((byte) => byte.toBigInt().toString(16).padStart(2, '0')) .join('')}`; } export function fieldToBigIntLE(field) { const bytesLE = wordToBytes(field, 32); // This is LE return bytesLE.reduce((acc, byte) => (acc << 8n) + byte.toBigInt(), 0n); } // DEPRECATED export function padUInt64To32Bytes(num) { let unpadded = []; unpadded = wordToBytes(num.toFields()[0]); return [...unpadded, ...Array(24).fill(UInt8.from(0))].reverse(); } // This is explicitly here for validation puposes not supposed to be provable. function toBigIntFromBytes(bytes) { let result = 0n; for (const byte of bytes) { result = (result << 8n) | BigInt(byte); } return result; } // This is explicitly here for validation puposes not supposed to be provable. const MAX_U64 = (1n << 64n) - 1n; function assertUint64(value) { if (value < 0n || value > MAX_U64) { throw new RangeError(`Value out of range for u64: '${value}'.`); } } // Proof decoder const proofOffsets = { inputSlot: 0, inputStoreHash: 8, outputSlot: 40, outputStoreHash: 48, executionStateRoot: 80, verifiedContractStorageSlotsRoot: 112, nextSyncCommitteeHash: 144, }; const proofTotalLength = 176; export function decodeConsensusMptProof(ethSP1Proof) { const proofData = new Uint8Array(ethSP1Proof.public_values.buffer.data // Buffer.from() this is nodejs specific and seemingly redundant ); if (proofData.length !== proofTotalLength) { throw new Error(`Byte slice must be exactly ${proofTotalLength} bytes, got '${proofData.length}'.`); } const inputSlotSlice = proofData.slice(proofOffsets.inputSlot, proofOffsets.inputStoreHash); const inputSlot = toBigIntFromBytes(inputSlotSlice); assertUint64(inputSlot); const inputStoreHashSlice = proofData.slice(proofOffsets.inputStoreHash, proofOffsets.outputSlot); const outputSlotSlice = proofData.slice(proofOffsets.outputSlot, proofOffsets.outputStoreHash); const outputSlot = toBigIntFromBytes(outputSlotSlice); assertUint64(outputSlot); const outputStoreHashSlice = proofData.slice(proofOffsets.outputStoreHash, proofOffsets.executionStateRoot); const executionStateRootSlice = proofData.slice(proofOffsets.executionStateRoot, proofOffsets.verifiedContractStorageSlotsRoot); const verifiedContractStorageSlotsRootSlice = proofData.slice(proofOffsets.verifiedContractStorageSlotsRoot, proofOffsets.nextSyncCommitteeHash); const nextSyncCommitteeHashSlice = proofData.slice(proofOffsets.nextSyncCommitteeHash, proofTotalLength); const provables = { inputSlot: UInt64.from(inputSlot), inputStoreHash: Bytes32.from(inputStoreHashSlice), outputSlot: UInt64.from(outputSlot), outputStoreHash: Bytes32.from(outputStoreHashSlice), executionStateRoot: Bytes32.from(executionStateRootSlice), verifiedContractDepositsRoot: Bytes32.from(verifiedContractStorageSlotsRootSlice), nextSyncCommitteeHash: Bytes32.from(nextSyncCommitteeHashSlice), }; return provables; } // Compile and verify contracts utility export async function compileAndVerifyContracts(logger, // Logger fix this later contracts) { try { const results = {}; const mismatches = []; for (const { name, program, integrityHash } of contracts) { logger.log(`Compiling ${name} contract.`); console.time(`${name} compile`); const compiled = await program.compile(); console.timeEnd(`${name} compile`); const verificationKey = compiled.verificationKey; const calculatedHash = verificationKey.hash.toString(); logger.log(`${name} contract vk hash compiled: '${calculatedHash}'`); results[`${name}VerificationKey`] = verificationKey; if (calculatedHash !== integrityHash) { mismatches.push(`${name}: Computed hash '${calculatedHash}' ` + `doesn't match expected hash '${integrityHash}'`); } } if (mismatches.length > 0) { const errorMessage = [ 'Verification key hash mismatch detected:', ...mismatches, '', `Refusing to start. Try clearing your o1js cache directory, typically found at '~/.cache/o1js'. Or do you need to run 'npm run bake-vk-hashes' in the eth-processor or o1js-zk-utils nori-bridge-sdk folder and commit the change?`, ].join('\n'); throw new Error(errorMessage); } logger.log('All contracts compiled and verified successfully.'); return results; } catch (err) { logger.error(`Error compiling contracts:\n${String(err)}`); console.error(err.stack); throw err; } } export function vkToVkSafe(vk) { const { data, hash } = vk; return { hashStr: hash.toBigInt().toString(), data, }; } export async function compileAndOptionallyVerifyContracts(logger, contracts) { const entries = []; const mismatches = []; for (const c of contracts) { const { name, program, integrityHash } = c; logger.log(`Compiling ${name} contract/program.`); console.time(`${name} compiled`); const compiled = await program.compile(); console.timeEnd(`${name} compiled`); const vk = compiled.verificationKey; const hashStr = vk.hash.toBigInt().toString(); logger.log(`${name} contract/program vk hash compiled: '${hashStr}'`); // Validate only if integrityHash is provided if (integrityHash && hashStr !== integrityHash) { mismatches.push(`${name}: Computed hash '${hashStr}' doesn't match expected hash '${integrityHash}'`); } const mappedKey = `${name}VerificationKey`; entries.push([mappedKey, vk]); } if (mismatches.length > 0) { const errorMessage = [ 'Verification key hash mismatch detected:', ...mismatches, '', `Refusing to start. Try clearing your o1js cache directory, typically found at '~/.cache/o1js'. Or do you need to run 'npm run bake-vk-hashes' and commit the changes?`, ].join('\n'); throw new Error(errorMessage); } logger.log('All contracts compiled successfully.'); return Object.fromEntries(entries); } //# sourceMappingURL=utils.js.map