// Copyright 2018 The Grin Developers // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. //! Lightweight readonly view into output MMR for convenience. use crate::core::core::pmmr::ReadonlyPMMR; use crate::core::core::{Block, Input, Output, Transaction}; use crate::core::ser::PMMRIndexHashable; use crate::error::{Error, ErrorKind}; use crate::store::Batch; use grin_store::pmmr::PMMRBackend; /// Readonly view of the UTXO set (based on output MMR). pub struct UTXOView<'a> { pmmr: ReadonlyPMMR<'a, Output, PMMRBackend>, batch: &'a Batch<'a>, } impl<'a> UTXOView<'a> { /// Build a new UTXO view. pub fn new( pmmr: ReadonlyPMMR<'a, Output, PMMRBackend>, batch: &'a Batch<'_>, ) -> UTXOView<'a> { UTXOView { pmmr, batch } } /// Validate a block against the current UTXO set. /// Every input must spend an output that currently exists in the UTXO set. /// No duplicate outputs. pub fn validate_block(&self, block: &Block) -> Result<(), Error> { for output in block.outputs() { self.validate_output(output)?; } for input in block.inputs() { self.validate_input(input)?; } Ok(()) } /// Validate a transaction against the current UTXO set. /// Every input must spend an output that currently exists in the UTXO set. /// No duplicate outputs. pub fn validate_tx(&self, tx: &Transaction) -> Result<(), Error> { for output in tx.outputs() { self.validate_output(output)?; } for input in tx.inputs() { self.validate_input(input)?; } Ok(()) } // Input is valid if it is spending an (unspent) output // that currently exists in the output MMR. // Compare the hash in the output MMR at the expected pos. fn validate_input(&self, input: &Input) -> Result<(), Error> { if let Ok(pos) = self.batch.get_output_pos(&input.commitment()) { if let Some(hash) = self.pmmr.get_hash(pos) { if hash == input.hash_with_index(pos - 1) { return Ok(()); } } } Err(ErrorKind::AlreadySpent(input.commitment()).into()) } // Output is valid if it would not result in a duplicate commitment in the output MMR. fn validate_output(&self, output: &Output) -> Result<(), Error> { if let Ok(pos) = self.batch.get_output_pos(&output.commitment()) { if let Some(out_mmr) = self.pmmr.get_data(pos) { if out_mmr.commitment() == output.commitment() { return Err(ErrorKind::DuplicateCommitment(output.commitment()).into()); } } } Ok(()) } }