diff --git a/src/Data/Collector.ts b/src/Data/Collector.ts index 9026cba..74db31d 100644 --- a/src/Data/Collector.ts +++ b/src/Data/Collector.ts @@ -31,7 +31,6 @@ import { accountSpecificHash, verifyAccountHash } from '../shardeum/calculateAcc import { verifyAppReceiptData } from '../shardeum/verifyAppReceiptData' import { Cycle as DbCycle } from '../dbstore/types' import { Utils as StringUtils } from '@shardus/types' -import { offloadReceipt } from '../primary-process' import { verifyPayload } from '../types/ajv/Helpers' import { AJVSchemaEnum } from '../types/enum/AJVSchemaEnum' @@ -63,6 +62,64 @@ export interface ReceiptVerificationResult { nestedCounterMessages?: string[] } +const verifyReceiptMajority = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, + executionGroupNodes: ConsensusNodeInfo[], + minConfirmations: number = config.RECEIPT_CONFIRMATIONS +): Promise<{ success: boolean; newReceipt?: Receipt.ArchiverReceipt }> => { + /** + * Note: + * Currently, only the non-global receipt flow is implemented in `verifyReceiptMajority`, + * `verifyReceiptOffline`, and `verifyNonGlobalTxReceiptWithValidators`. In the future, + * global receipt methods can be added to maintain consistency. As of now, only offline + * verification for global receipts is available, and `verifyGlobalTxReceiptWithValidators` + * is not yet implemented. + */ + + // If robustQuery is disabled, do offline verification + if (!config.useRobustQueryForReceipt) { + return verifyReceiptOffline(receipt, executionGroupNodes, minConfirmations) + } + return verifyReceiptWithValidators(receipt, executionGroupNodes, minConfirmations) +} + +// Offline receipt verification +/** + * Note: + * The `verifyReceiptOffline` function is responsible for verifying receipts + * without querying external validators, which is useful when the robust query + * feature is disabled. It currently supports only non-global receipt verification. + * Future enhancements should include validation logic for global receipts to ensure + * comprehensive receipt verification. This will help maintain consistency and reliability + * across all types of receipts processed by the system. + * + * The function delegates the verification process to `verifyNonGlobalTxReceiptOffline` + * based on the whether the receipt is global or not. + */ +const verifyReceiptOffline = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, + executionGroupNodes: ConsensusNodeInfo[], + minConfirmations: number +): Promise<{ success: boolean; newReceipt?: Receipt.ArchiverReceipt }> => { + return verifyNonGlobalTxReceiptOffline(receipt, executionGroupNodes, minConfirmations) +} + +/** + * Note: + * The `verifyReceiptWithValidators` function currently supports only + * non-global receipt verification. Future enhancements should include + * validation logic for global receipts to ensure comprehensive receipt + * verification. This will help in maintaining consistency and reliability + * across all types of receipts processed by the system. + */ +const verifyReceiptWithValidators = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, + executionGroupNodes: ConsensusNodeInfo[], + minConfirmations: number = config.RECEIPT_CONFIRMATIONS +): Promise<{ success: boolean; newReceipt?: Receipt.ArchiverReceipt }> => { + return verifyNonGlobalTxReceiptWithValidators(receipt, executionGroupNodes, minConfirmations) +} + /** * Calls the /get-tx-receipt endpoint of the nodes in the execution group of the receipt to verify the receipt. If "RECEIPT_CONFIRMATIONS" number of nodes return the same receipt, the receipt is deemed valid. * @param receipt @@ -70,8 +127,8 @@ export interface ReceiptVerificationResult { * @param minConfirmations * @returns boolean */ -const isReceiptRobust = async ( - receipt: Receipt.ArchiverReceipt, +const verifyNonGlobalTxReceiptWithValidators = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, executionGroupNodes: ConsensusNodeInfo[], minConfirmations: number = config.RECEIPT_CONFIRMATIONS ): Promise<{ success: boolean; newReceipt?: Receipt.ArchiverReceipt }> => { @@ -272,6 +329,134 @@ const isReceiptRobust = async ( return { success: true } } +// Offline global receipt verification +const verifyGlobalTxreceiptOffline = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt +): Promise<{ success: boolean; requiredSignatures?: number }> => { + const appliedReceipt = receipt.signedReceipt as P2PTypes.GlobalAccountsTypes.GlobalTxReceipt + const result = { success: false } + const { txId, timestamp } = receipt.tx + const { executionShardKey, cycle } = receipt + const cycleShardData = shardValuesByCycle.get(cycle) + const { homePartition } = ShardFunction.addressToPartition(cycleShardData.shardGlobals, executionShardKey) + const { signs } = appliedReceipt + // Refer to https://github.com/shardeum/shardus-core/blob/7d8877b7e1a5b18140f898a64b932182d8a35298/src/p2p/GlobalAccounts.ts#L397 + let votingGroupCount = cycleShardData.shardGlobals.nodesPerConsenusGroup + if (votingGroupCount > cycleShardData.nodes.length) { + if (nestedCountersInstance) + nestedCountersInstance.countEvent('receipt', 'votingGroupCount_greater_than_nodes_length') + Logger.mainLogger.error( + 'votingGroupCount_greater_than_nodes_length', + votingGroupCount, + cycleShardData.nodes.length + ) + votingGroupCount = cycleShardData.nodes.length + } + let isReceiptMajority = (signs.length / votingGroupCount) * 100 >= config.requiredMajorityVotesPercentage + if (!isReceiptMajority) { + Logger.mainLogger.error( + `Invalid receipt globalModification signs count is less than ${config.requiredMajorityVotesPercentage}% of the votingGroupCount, ${signs.length}, ${votingGroupCount}` + ) + if (nestedCountersInstance) + nestedCountersInstance.countEvent( + 'receipt', + `Invalid_receipt_globalModification_signs_count_less_than_${config.requiredMajorityVotesPercentage}%` + ) + return result + } + + const nodeMap = new Map() + // Fill the map with nodes keyed by their public keys + cycleShardData.nodes.forEach((node) => { + if (node.publicKey) { + nodeMap.set(node.publicKey, node) + } + }) + // Using a set to store the unique signers to avoid duplicates + const uniqueSigners = new Set() + for (const sign of signs) { + const { owner: nodePubKey } = sign + // Get the node id from the public key + const node = nodeMap.get(nodePubKey) + if (node == null) { + Logger.mainLogger.error( + `The node with public key ${nodePubKey} of the receipt ${txId} with ${timestamp} is not in the active nodesList of cycle ${cycle}` + ) + if (nestedCountersInstance) + nestedCountersInstance.countEvent('receipt', 'globalModification_sign_owner_not_in_active_nodesList') + continue + } + // Check if the node is in the execution group + if (!cycleShardData.parititionShardDataMap.get(homePartition).coveredBy[node.id]) { + Logger.mainLogger.error( + `The node with public key ${nodePubKey} of the receipt ${txId} with ${timestamp} is not in the execution group of the tx` + ) + if (nestedCountersInstance) + nestedCountersInstance.countEvent( + 'receipt', + 'globalModification_sign_node_not_in_execution_group_of_tx' + ) + continue + } + + uniqueSigners.add(nodePubKey) + } + isReceiptMajority = (uniqueSigners.size / votingGroupCount) * 100 >= config.requiredMajorityVotesPercentage + if (!isReceiptMajority) { + Logger.mainLogger.error( + `Invalid receipt globalModification valid signs count is less than votingGroupCount ${uniqueSigners.size}, ${votingGroupCount}` + ) + if (nestedCountersInstance) + nestedCountersInstance.countEvent( + 'receipt', + 'Invalid_receipt_globalModification_valid_signs_count_less_than_votingGroupCount' + ) + return result + } + const requiredSignatures = Math.floor(votingGroupCount * (config.requiredMajorityVotesPercentage / 100)) + return { success: true, requiredSignatures } +} + +const verifyNonGlobalTxReceiptOffline = async ( + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, + executionGroupNodes: ConsensusNodeInfo[], + minConfirmations: number +): Promise<{ success: boolean; newReceipt?: Receipt.ArchiverReceipt }> => { + // Code for normal receipts verification + const normalReceipt = receipt.signedReceipt as Receipt.SignedReceipt + const validSigners = new Set() + + if (!normalReceipt.signaturePack || !Array.isArray(normalReceipt.signaturePack)) { + return { success: false } + } + + for (const signature of normalReceipt.signaturePack) { + if (!signature || !signature.owner) continue + + const node = executionGroupNodes.find((n) => n.publicKey.toLowerCase() === signature.owner.toLowerCase()) + if (!node) continue + + try { + const voteHash = calculateVoteHash(normalReceipt.proposal) + const appliedVoteHash = { + txid: receipt.tx.txId, + voteHash + } + + if (Crypto.verify({ ...appliedVoteHash, sign: signature })) { + validSigners.add(signature.owner) + } + } catch (error) { + console.error('Error verifying signature:', error) + continue + } + } + + return { + success: validSigners.size >= minConfirmations + } +} + /** * Validate type and field existence of the receipt data before processing it further * @param receipt @@ -306,7 +491,7 @@ export const validateReceiptType = (receipt: Receipt.Receipt | Receipt.ArchiverR export const verifyReceiptData = async ( - receipt: Receipt.ArchiverReceipt, + receipt: Receipt.Receipt | Receipt.ArchiverReceipt, checkReceiptRobust = true ): Promise<{ success: boolean; requiredSignatures?: number; newReceipt?: Receipt.ArchiverReceipt }> => { const result = { success: false } @@ -336,90 +521,27 @@ export const verifyReceiptData = async ( } // Determine the home partition index of the primary account (executionShardKey) const { homePartition } = ShardFunction.addressToPartition(cycleShardData.shardGlobals, executionShardKey) - if (globalModification) { - const appliedReceipt = receipt.signedReceipt as P2PTypes.GlobalAccountsTypes.GlobalTxReceipt - const { signs } = appliedReceipt - // Refer to https://github.com/shardeum/shardus-core/blob/7d8877b7e1a5b18140f898a64b932182d8a35298/src/p2p/GlobalAccounts.ts#L397 - let votingGroupCount = cycleShardData.shardGlobals.nodesPerConsenusGroup - if (votingGroupCount > cycleShardData.nodes.length) { - if (nestedCountersInstance) - nestedCountersInstance.countEvent('receipt', 'votingGroupCount_greater_than_nodes_length') - Logger.mainLogger.error( - 'votingGroupCount_greater_than_nodes_length', - votingGroupCount, - cycleShardData.nodes.length - ) - votingGroupCount = cycleShardData.nodes.length - } - let isReceiptMajority = - (signs.length / votingGroupCount) * 100 >= config.requiredMajorityVotesPercentage - if (!isReceiptMajority) { - Logger.mainLogger.error( - `Invalid receipt globalModification signs count is less than ${config.requiredMajorityVotesPercentage}% of the votingGroupCount, ${signs.length}, ${votingGroupCount}` - ) - if (nestedCountersInstance) - nestedCountersInstance.countEvent( - 'receipt', - `Invalid_receipt_globalModification_signs_count_less_than_${config.requiredMajorityVotesPercentage}%` - ) - return result - } - - const nodeMap = new Map() - // Fill the map with nodes keyed by their public keys - cycleShardData.nodes.forEach((node) => { - if (node.publicKey) { - nodeMap.set(node.publicKey, node) - } - }) - // Using a set to store the unique signers to avoid duplicates - const uniqueSigners = new Set() - for (const sign of signs) { - const { owner: nodePubKey } = sign - // Get the node id from the public key - const node = nodeMap.get(nodePubKey) - if (node == null) { - Logger.mainLogger.error( - `The node with public key ${nodePubKey} of the receipt ${txId} with ${timestamp} is not in the active nodesList of cycle ${cycle}` - ) - if (nestedCountersInstance) - nestedCountersInstance.countEvent( - 'receipt', - 'globalModification_sign_owner_not_in_active_nodesList' - ) - continue - } - // Check if the node is in the execution group - if (!cycleShardData.parititionShardDataMap.get(homePartition).coveredBy[node.id]) { - Logger.mainLogger.error( - `The node with public key ${nodePubKey} of the receipt ${txId} with ${timestamp} is not in the execution group of the tx` - ) - if (nestedCountersInstance) - nestedCountersInstance.countEvent( - 'receipt', - 'globalModification_sign_node_not_in_execution_group_of_tx' - ) - continue - } - uniqueSigners.add(nodePubKey) - } - isReceiptMajority = - (uniqueSigners.size / votingGroupCount) * 100 >= config.requiredMajorityVotesPercentage - if (!isReceiptMajority) { - Logger.mainLogger.error( - `Invalid receipt globalModification valid signs count is less than votingGroupCount ${uniqueSigners.size}, ${votingGroupCount}` + let globalReceiptValidationErrors + try { + // Validate if receipt is a global modification receipt using AJV + globalReceiptValidationErrors = verifyPayload(AJVSchemaEnum.GlobalTxReceipt, receipt?.signedReceipt) + } catch (error) { + globalReceiptValidationErrors = true + if (nestedCountersInstance) + nestedCountersInstance.countEvent( + 'receipt', + `Failed to validate receipt schema txId: ${txId}, cycle: ${cycle}, timestamp: ${timestamp}, error: ${error}` ) - if (nestedCountersInstance) - nestedCountersInstance.countEvent( - 'receipt', - 'Invalid_receipt_globalModification_valid_signs_count_less_than_votingGroupCount' - ) - return result - } - const requiredSignatures = Math.floor(votingGroupCount * (config.requiredMajorityVotesPercentage / 100)) - return { success: true, requiredSignatures } - + Logger.mainLogger.error( + `Failed to validate receipt schema txId: ${txId}, cycle: ${cycle}, timestamp: ${timestamp}, error: ${error}` + ) + return result + } + + // If the receipt is a global modification receipt, validate the receipt + if (!globalReceiptValidationErrors) { + return verifyGlobalTxreceiptOffline(receipt) } const { signaturePack } = receipt.signedReceipt as Receipt.SignedReceipt if (config.newPOQReceipt === false) { @@ -585,7 +707,7 @@ export const verifyReceiptData = async ( nodesPerConsensusGroup > config.RECEIPT_CONFIRMATIONS ? config.RECEIPT_CONFIRMATIONS : Math.ceil(config.RECEIPT_CONFIRMATIONS / 2) // 3 out of 5 nodes - const { success, newReceipt } = await isReceiptRobust(receipt, executionGroupNodes, minConfirmations) + const { success, newReceipt } = await verifyReceiptMajority(receipt, executionGroupNodes, minConfirmations) if (!success) { Logger.mainLogger.error('Invalid receipt: Robust check failed') if (nestedCountersInstance) @@ -605,7 +727,22 @@ const verifyAppliedReceiptSignatures = ( const result = { success: false, failedReasons, nestedCounterMessages } const { globalModification, cycle, executionShardKey } = receipt const { txId: txid, timestamp } = receipt.tx - if (globalModification) { + let globalReceiptValidationErrors // This is used to store the validation errors of the globalTxReceipt + + try { + globalReceiptValidationErrors = verifyPayload(AJVSchemaEnum.GlobalTxReceipt, receipt?.signedReceipt) + } catch (error) { + globalReceiptValidationErrors = true + failedReasons.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + nestedCounterMessages.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + return result + } + // If the globalReceiptValidationErrors is null, then the receipt is a globalModification receipt + if (!globalReceiptValidationErrors) { const appliedReceipt = receipt.signedReceipt as P2PTypes.GlobalAccountsTypes.GlobalTxReceipt // Refer to https://github.com/shardeum/shardus-core/blob/7d8877b7e1a5b18140f898a64b932182d8a35298/src/p2p/GlobalAccounts.ts#L294 @@ -892,11 +1029,21 @@ export const storeReceiptData = async ( } if (newReceipt) receipt = newReceipt - if (profilerInstance) profilerInstance.profileSectionStart('Offload_receipt') - if (nestedCountersInstance) nestedCountersInstance.countEvent('receipt', 'Offload_receipt') + if (profilerInstance) profilerInstance.profileSectionStart('Verify_archiver_receipt') + if (nestedCountersInstance) nestedCountersInstance.countEvent('receipt', 'Verify_archiver_receipt') const start_time = process.hrtime() // console.log('offloading receipt', txId, timestamp) - const result = await offloadReceipt(txId, timestamp, requiredSignatures, receipt) + // const result = await offloadReceipt(txId, timestamp, requiredSignatures, receipt) + let result + try { + result = await verifyArchiverReceipt(receipt, requiredSignatures) + } catch (error) { + receiptsInValidationMap.delete(txId) + if (nestedCountersInstance) + nestedCountersInstance.countEvent('receipt', 'Invalid_receipt_verification_failed') + if (profilerInstance) profilerInstance.profileSectionEnd('Verify_archiver_receipt') + continue + } // console.log('offload receipt result', txId, timestamp, result) const end_time = process.hrtime(start_time) const time_taken = end_time[0] * 1000 + end_time[1] / 1000000 diff --git a/src/primary-process/index.ts b/src/primary-process/index.ts index be97969..d446971 100644 --- a/src/primary-process/index.ts +++ b/src/primary-process/index.ts @@ -6,6 +6,8 @@ import { config } from '../Config' import { EventEmitter } from 'events' import { StateManager, Utils as StringUtils } from '@shardus/types' import * as Utils from '../Utils' +import { verifyPayload } from '../types/ajv/Helpers' +import { AJVSchemaEnum } from '../types/enum/AJVSchemaEnum' const MAX_WORKERS = cpus().length - 1 // Leaving 1 core for the master process @@ -39,6 +41,11 @@ let currentWorker = 0 const emitter = new EventEmitter() +/** + * @deprecated This method is currently used in server.ts but will be removed as part of cleanup. + * Do not use this method in new code. + */ + export const setupWorkerProcesses = (cluster: Cluster): void => { console.log(`Master ${process.pid} is running`) // Set interval to check receipt count every 15 seconds @@ -223,103 +230,3 @@ const forwardReceiptVerificationResult = ( }) }) } - -export const offloadReceipt = async ( - txId: string, - timestamp: number, - requiredSignatures: number, - receipt: ArchiverReceipt -): Promise => { - receivedReceiptCount++ // Increment the counter for each receipt received - receiptLoadTraker++ // Increment the receipt load tracker - let verificationResult: ReceiptVerificationResult - - // Check if offloading is disabled globally or for global modifications - if ( - config.disableOffloadReceipt || - (config.disableOffloadReceiptForGlobalModification && receipt.globalModification) - ) { - mainProcessReceiptTracker++ - if (config.workerProcessesDebugLog) console.log('Verifying on the main program', txId, timestamp) - verificationResult = await verifyArchiverReceipt(receipt, requiredSignatures) - mainProcessReceiptTracker-- - verifiedReceiptCount++ - if (verificationResult.success) { - successReceiptCount++ - } else { - failureReceiptCount++ - } - return verificationResult - } - - // Existing logic for offloading - if (workers.length === 0 && mainProcessReceiptTracker > config.receiptLoadTrakerLimit) { - // If there are extra workers available, put them to the workers list - if (extraWorkers.size > 0) { - console.log( - `offloadReceipt - Extra workers available: ${extraWorkers.size}, moving them to workers list` - ) - // Move the extra workers to the workers list - for (const [pid, worker] of extraWorkers) { - workers.push(worker) - extraWorkers.delete(pid) - } - } - // // If there are still no workers available, add randon wait time (0-1 second) and proceed - // if (workers.length === 0 && mainProcessReceiptTracker > config.receiptLoadTrakerLimit) { - // await Utils.sleep(Math.floor(Math.random() * 1000)) - // } - } - if (workers.length === 0) { - mainProcessReceiptTracker++ - if (config.workerProcessesDebugLog) console.log('Verifying on the main program 1', txId, timestamp) - verificationResult = await verifyArchiverReceipt(receipt, requiredSignatures) - mainProcessReceiptTracker-- - } else { - mainProcessReceiptTracker = 0 - // Forward the request to a worker in a round-robin fashion - let worker = workers[currentWorker] - currentWorker = (currentWorker + 1) % workers.length - if (!worker) { - console.error('No worker available to process the receipt 1') - worker = workers[currentWorker] - currentWorker = (currentWorker + 1) % workers.length - if (worker) { - console.log('Verifying on the worker process 2', txId, timestamp, worker.process.pid) - const cloneReceipt = Utils.deepCopy(receipt) - delete cloneReceipt.tx.originalTxData - delete cloneReceipt.executionShardKey - const stringifiedReceipt = StringUtils.safeStringify(cloneReceipt) - worker.send({ - type: 'receipt-verification', - data: { stringifiedReceipt, requiredSignatures }, - }) - verificationResult = await forwardReceiptVerificationResult(txId, timestamp, worker) - } else { - console.error('No worker available to process the receipt 2') - // Verifying the receipt in the main thread - console.log('Verifying on the main program 2', txId, timestamp) - verificationResult = await verifyArchiverReceipt(receipt, requiredSignatures) - } - } else { - if (config.workerProcessesDebugLog) - console.log('Verifying on the worker process 1', txId, timestamp, worker.process.pid) - const cloneReceipt = Utils.deepCopy(receipt) - delete cloneReceipt.tx.originalTxData - delete cloneReceipt.executionShardKey - const stringifiedReceipt = StringUtils.safeStringify(cloneReceipt) - worker.send({ - type: 'receipt-verification', - data: { stringifiedReceipt, requiredSignatures }, - }) - verificationResult = await forwardReceiptVerificationResult(txId, timestamp, worker) - } - } - verifiedReceiptCount++ - if (verificationResult.success) { - successReceiptCount++ - } else { - failureReceiptCount++ - } - return verificationResult -} diff --git a/src/server.ts b/src/server.ts index 731bffb..b136a9b 100644 --- a/src/server.ts +++ b/src/server.ts @@ -42,8 +42,6 @@ import { setShutdownCycleRecord, cycleRecordWithShutDownMode } from './Data/Cycl import { registerRoutes } from './API' import { Utils as StringUtils } from '@shardus/types' import { healthCheckRouter } from './routes/healthCheck' -import { setupWorkerProcesses } from './primary-process' -import { initWorkerProcess } from './worker-process' import { initializeTickets } from './routes/tickets'; import { initAjvSchemas } from './types/ajv/Helpers' import { initializeSerialization } from './utils/serialization/SchemaHelpers' @@ -93,7 +91,7 @@ async function start(): Promise { if (!cluster.isPrimary) { // Initialize state from config await State.initFromConfig(config, false, false) - await initWorkerProcess() + // await initWorkerProcess() return } @@ -497,7 +495,7 @@ async function startServer(): Promise { Logger.mainLogger.info(`Worker ${process.pid}: Archive-server is listening on http://0.0.0.0:${config.ARCHIVER_PORT}`) State.setActive() Collector.scheduleMissingTxsDataQuery() - setupWorkerProcesses(cluster) + // setupWorkerProcesses(cluster) } ) } diff --git a/src/shardeum/calculateAccountHash.ts b/src/shardeum/calculateAccountHash.ts index d4f6b20..0688048 100644 --- a/src/shardeum/calculateAccountHash.ts +++ b/src/shardeum/calculateAccountHash.ts @@ -1,5 +1,7 @@ import * as crypto from '../Crypto' import { ArchiverReceipt, SignedReceipt } from '../dbstore/receipts' +import { verifyPayload } from '../types/ajv/Helpers' +import { AJVSchemaEnum } from '../types/enum/AJVSchemaEnum' import { verifyGlobalTxAccountChange } from './verifyGlobalTxReceipt' // account types in Shardeum @@ -64,7 +66,20 @@ export const verifyAccountHash = ( nestedCounterMessages = [] ): boolean => { try { - if (receipt.globalModification) { + let globalReceiptValidationErrors // This is used to store the validation errors of the globalTxReceipt + try { + globalReceiptValidationErrors = verifyPayload(AJVSchemaEnum.GlobalTxReceipt, receipt?.signedReceipt) + } catch (error) { + globalReceiptValidationErrors = true + failedReasons.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + nestedCounterMessages.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + return false + } + if (!globalReceiptValidationErrors) { const result = verifyGlobalTxAccountChange(receipt, failedReasons, nestedCounterMessages) if (!result) return false return true diff --git a/src/shardeum/verifyAppReceiptData.ts b/src/shardeum/verifyAppReceiptData.ts index 1cbdb0d..c531cd8 100644 --- a/src/shardeum/verifyAppReceiptData.ts +++ b/src/shardeum/verifyAppReceiptData.ts @@ -1,6 +1,8 @@ import * as crypto from '../Crypto' import { ArchiverReceipt, Receipt, SignedReceipt } from '../dbstore/receipts' import { Utils as StringUtils } from '@shardus/types' +import { verifyPayload } from '../types/ajv/Helpers' +import { AJVSchemaEnum } from '../types/enum/AJVSchemaEnum' export type ShardeumReceipt = object & { amountSpent: string @@ -14,8 +16,23 @@ export const verifyAppReceiptData = async ( nestedCounterMessages = [] ): Promise<{ valid: boolean; needToSave: boolean }> => { let result = { valid: false, needToSave: false } - const { appReceiptData, globalModification } = receipt - if (globalModification) return { valid: true, needToSave: true } + const { appReceiptData } = receipt + let globalReceiptValidationErrors // This is used to store the validation errors of the globalTxReceipt + try { + globalReceiptValidationErrors = verifyPayload(AJVSchemaEnum.GlobalTxReceipt, receipt?.signedReceipt) + } catch (error) { + globalReceiptValidationErrors = true + failedReasons.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + nestedCounterMessages.push( + `Invalid Global Tx Receipt error: ${error}. txId ${receipt.tx.txId} , cycle ${receipt.cycle} , timestamp ${receipt.tx.timestamp}` + ) + return result + } + if (!globalReceiptValidationErrors) { + return { valid: true, needToSave: true } + } const signedReceipt = receipt.signedReceipt as SignedReceipt const newShardeumReceipt = appReceiptData.data as ShardeumReceipt if (!newShardeumReceipt.amountSpent || !newShardeumReceipt.readableReceipt) { diff --git a/src/types/ajv/OriginalTxData.ts b/src/types/ajv/OriginalTxData.ts index 707a014..d5054ae 100644 --- a/src/types/ajv/OriginalTxData.ts +++ b/src/types/ajv/OriginalTxData.ts @@ -8,7 +8,7 @@ const schemaOriginalTxData = { properties: { txId: { type: 'string' }, // txId must be a string timestamp: { type: 'integer', minimum: 0 }, // timestamp must be an integer - cycle: { type: 'integer', minimum: 0 }, // cycle must be an integer + cycle: { type: 'integer', minimum: -1 }, // cycle must be an integer originalTxData: { type: 'object' }, // originalTxData must be an object // Uncomment if sign is required: // sign: { type: 'string' } // Sign (if used) must be a string diff --git a/src/types/ajv/Receipts.ts b/src/types/ajv/Receipts.ts index 921742d..dbb6412 100644 --- a/src/types/ajv/Receipts.ts +++ b/src/types/ajv/Receipts.ts @@ -195,4 +195,5 @@ function addSchemas(): void { // addSchema('ReceiptTx', schemaTx); addSchema(AJVSchemaEnum.ArchiverReceipt, schemaArchiverReceipt); addSchema(AJVSchemaEnum.Receipt, schemaReceipt); + addSchema(AJVSchemaEnum.GlobalTxReceipt, schemaGlobalTxReceipt); } diff --git a/src/types/enum/AJVSchemaEnum.ts b/src/types/enum/AJVSchemaEnum.ts index 4a3f1da..10efa34 100644 --- a/src/types/enum/AJVSchemaEnum.ts +++ b/src/types/enum/AJVSchemaEnum.ts @@ -2,5 +2,6 @@ export enum AJVSchemaEnum { Receipt = 'Receipt', AccountsCopy = 'AccountsCopy', ArchiverReceipt = 'ArchiverReceipt', - OriginalTxData = 'OriginalTxData' + OriginalTxData = 'OriginalTxData', + GlobalTxReceipt = 'GlobalTxReceipt', }