* Contains should take three arguments (root, key, value) * Add a test for frontend Dictionaries * Separate frontend and middleware operations * Make tests pass: add arg to contains * Cargo fmt * Merkleproof verify circuit (#143) * merkletree: add keypath circuit * merkletree-circuit: implement proof of existence verification in-circuit * parametrize max_depth at the tree circuit * Constrain selectors in-circuit * implement merketree nonexistence proof circuit, and add edgecase tests * add non-existence proofs documentation in the mdbook, mv EMPTY->EMPTY_VALUE & NULL->EMPTY_HASH, dependency clean and public exposure methods * review comments, some extra polishing and add a test that expects wrong proofs to fail * Add circuit to check only merkleproofs-of-existence With this, the merkletree_circuit module offers two different circuits: - `MerkleProofCircuit`: allows to verify both proofs of existence and proofs non-existence with the same circuit. - `MerkleProofExistenceCircuit`: allows to verify proofs of existence only. In this way, if only proofs of existence are needed, `MerkleProofExistenceCircuit` should be used, which requires less amount of constraints than `MerkleProofCircuit`. * Code review --------- Co-authored-by: Ahmad <root@ahmadafuni.com> * Towards Contains/NotContains in middleware and backend * Fix build * Adding error handling to deal with op compile introduce extra ops * Incorporate Merkle proofs into MockMainPod * Merkleproof verify circuit (#143) * merkletree: add keypath circuit * merkletree-circuit: implement proof of existence verification in-circuit * parametrize max_depth at the tree circuit * Constrain selectors in-circuit * implement merketree nonexistence proof circuit, and add edgecase tests * add non-existence proofs documentation in the mdbook, mv EMPTY->EMPTY_VALUE & NULL->EMPTY_HASH, dependency clean and public exposure methods * review comments, some extra polishing and add a test that expects wrong proofs to fail * Add circuit to check only merkleproofs-of-existence With this, the merkletree_circuit module offers two different circuits: - `MerkleProofCircuit`: allows to verify both proofs of existence and proofs non-existence with the same circuit. - `MerkleProofExistenceCircuit`: allows to verify proofs of existence only. In this way, if only proofs of existence are needed, `MerkleProofExistenceCircuit` should be used, which requires less amount of constraints than `MerkleProofCircuit`. * Code review --------- Co-authored-by: Ahmad <root@ahmadafuni.com> * Towards Contains/NotContains in middleware and backend * Frontend compound types -- allow one frontend operation to produce multiple middleware statements (in progress) * Incorporate Merkle proofs into MockMainPod * Incorporate Merkle proof op arg into frontend * Compile one statement to many, in progress * Fix remaining tests * Minor clean-up * Oops I did a bunch of work in the middle of a rebase, committing * Incorporate Merkle proof op arg into frontend * still working on frontend compound types, refactor compile() to output multiple statements * Contains statements for frontend types: code compiles * Tests pass * Examples use front-end compound types * Remove old Contains and NotContains from frontend * Add nin to typos * Code review --------- Co-authored-by: arnaucube <git@arnaucube.com> Co-authored-by: Ahmad <root@ahmadafuni.com>
499 lines
21 KiB
Rust
499 lines
21 KiB
Rust
use anyhow::{anyhow, Result};
|
|
use log::error;
|
|
use plonky2::field::types::Field;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::fmt;
|
|
use std::iter;
|
|
|
|
use super::{CustomPredicateRef, NativePredicate, Statement, StatementArg, ToFields, F};
|
|
use crate::{
|
|
backends::plonky2::primitives::merkletree::{MerkleProof, MerkleTree},
|
|
middleware::{AnchoredKey, Params, Predicate, Value, SELF},
|
|
};
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
|
|
pub enum OperationType {
|
|
Native(NativeOperation),
|
|
Custom(CustomPredicateRef),
|
|
}
|
|
|
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
pub enum OperationAux {
|
|
None,
|
|
MerkleProof(MerkleProof),
|
|
}
|
|
|
|
impl fmt::Display for OperationAux {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
match self {
|
|
Self::None => write!(f, "<no aux>")?,
|
|
Self::MerkleProof(pf) => write!(f, "merkle_proof({})", pf)?,
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl ToFields for OperationType {
|
|
fn to_fields(&self, params: &Params) -> Vec<F> {
|
|
let mut fields: Vec<F> = match self {
|
|
Self::Native(p) => iter::once(F::from_canonical_u64(1))
|
|
.chain(p.to_fields(params))
|
|
.collect(),
|
|
Self::Custom(CustomPredicateRef(pb, i)) => iter::once(F::from_canonical_u64(3))
|
|
.chain(pb.hash(params).0)
|
|
.chain(iter::once(F::from_canonical_usize(*i)))
|
|
.collect(),
|
|
};
|
|
fields.resize_with(Params::operation_type_size(), || F::from_canonical_u64(0));
|
|
fields
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)]
|
|
pub enum NativeOperation {
|
|
None = 0,
|
|
NewEntry = 1,
|
|
CopyStatement = 2,
|
|
EqualFromEntries = 3,
|
|
NotEqualFromEntries = 4,
|
|
GtFromEntries = 5,
|
|
LtFromEntries = 6,
|
|
TransitiveEqualFromStatements = 7,
|
|
GtToNotEqual = 8,
|
|
LtToNotEqual = 9,
|
|
ContainsFromEntries = 10,
|
|
NotContainsFromEntries = 11,
|
|
SumOf = 13,
|
|
ProductOf = 14,
|
|
MaxOf = 15,
|
|
}
|
|
|
|
impl ToFields for NativeOperation {
|
|
fn to_fields(&self, _params: &Params) -> Vec<F> {
|
|
vec![F::from_canonical_u64(*self as u64)]
|
|
}
|
|
}
|
|
|
|
impl OperationType {
|
|
/// Gives the type of predicate that the operation will output, if known.
|
|
/// CopyStatement may output any predicate (it will match the statement copied),
|
|
/// so output_predicate returns None on CopyStatement.
|
|
pub fn output_predicate(&self) -> Option<Predicate> {
|
|
match self {
|
|
OperationType::Native(native_op) => match native_op {
|
|
NativeOperation::None => Some(Predicate::Native(NativePredicate::None)),
|
|
NativeOperation::NewEntry => Some(Predicate::Native(NativePredicate::ValueOf)),
|
|
NativeOperation::CopyStatement => None,
|
|
NativeOperation::EqualFromEntries => {
|
|
Some(Predicate::Native(NativePredicate::Equal))
|
|
}
|
|
NativeOperation::NotEqualFromEntries => {
|
|
Some(Predicate::Native(NativePredicate::NotEqual))
|
|
}
|
|
NativeOperation::GtFromEntries => Some(Predicate::Native(NativePredicate::Gt)),
|
|
NativeOperation::LtFromEntries => Some(Predicate::Native(NativePredicate::Lt)),
|
|
NativeOperation::TransitiveEqualFromStatements => {
|
|
Some(Predicate::Native(NativePredicate::Equal))
|
|
}
|
|
NativeOperation::GtToNotEqual => Some(Predicate::Native(NativePredicate::NotEqual)),
|
|
NativeOperation::LtToNotEqual => Some(Predicate::Native(NativePredicate::NotEqual)),
|
|
NativeOperation::ContainsFromEntries => {
|
|
Some(Predicate::Native(NativePredicate::Contains))
|
|
}
|
|
NativeOperation::NotContainsFromEntries => {
|
|
Some(Predicate::Native(NativePredicate::NotContains))
|
|
}
|
|
NativeOperation::SumOf => Some(Predicate::Native(NativePredicate::SumOf)),
|
|
NativeOperation::ProductOf => Some(Predicate::Native(NativePredicate::ProductOf)),
|
|
NativeOperation::MaxOf => Some(Predicate::Native(NativePredicate::MaxOf)),
|
|
},
|
|
OperationType::Custom(cpr) => Some(Predicate::Custom(cpr.clone())),
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: Refine this enum.
|
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
pub enum Operation {
|
|
None,
|
|
NewEntry,
|
|
CopyStatement(Statement),
|
|
EqualFromEntries(Statement, Statement),
|
|
NotEqualFromEntries(Statement, Statement),
|
|
GtFromEntries(Statement, Statement),
|
|
LtFromEntries(Statement, Statement),
|
|
TransitiveEqualFromStatements(Statement, Statement),
|
|
GtToNotEqual(Statement),
|
|
LtToNotEqual(Statement),
|
|
ContainsFromEntries(
|
|
/* root */ Statement,
|
|
/* key */ Statement,
|
|
/* value */ Statement,
|
|
/* proof */ MerkleProof,
|
|
),
|
|
NotContainsFromEntries(
|
|
/* root */ Statement,
|
|
/* key */ Statement,
|
|
/* proof */ MerkleProof,
|
|
),
|
|
SumOf(Statement, Statement, Statement),
|
|
ProductOf(Statement, Statement, Statement),
|
|
MaxOf(Statement, Statement, Statement),
|
|
Custom(CustomPredicateRef, Vec<Statement>),
|
|
}
|
|
|
|
impl Operation {
|
|
pub fn op_type(&self) -> OperationType {
|
|
type OT = OperationType;
|
|
use NativeOperation::*;
|
|
match self {
|
|
Self::None => OT::Native(None),
|
|
Self::NewEntry => OT::Native(NewEntry),
|
|
Self::CopyStatement(_) => OT::Native(CopyStatement),
|
|
Self::EqualFromEntries(_, _) => OT::Native(EqualFromEntries),
|
|
Self::NotEqualFromEntries(_, _) => OT::Native(NotEqualFromEntries),
|
|
Self::GtFromEntries(_, _) => OT::Native(GtFromEntries),
|
|
Self::LtFromEntries(_, _) => OT::Native(LtFromEntries),
|
|
Self::TransitiveEqualFromStatements(_, _) => OT::Native(TransitiveEqualFromStatements),
|
|
Self::GtToNotEqual(_) => OT::Native(GtToNotEqual),
|
|
Self::LtToNotEqual(_) => OT::Native(LtToNotEqual),
|
|
Self::ContainsFromEntries(_, _, _, _) => OT::Native(ContainsFromEntries),
|
|
Self::NotContainsFromEntries(_, _, _) => OT::Native(NotContainsFromEntries),
|
|
Self::SumOf(_, _, _) => OT::Native(SumOf),
|
|
Self::ProductOf(_, _, _) => OT::Native(ProductOf),
|
|
Self::MaxOf(_, _, _) => OT::Native(MaxOf),
|
|
Self::Custom(cpr, _) => OT::Custom(cpr.clone()),
|
|
}
|
|
}
|
|
|
|
pub fn args(&self) -> Vec<Statement> {
|
|
match self.clone() {
|
|
Self::None => vec![],
|
|
Self::NewEntry => vec![],
|
|
Self::CopyStatement(s) => vec![s],
|
|
Self::EqualFromEntries(s1, s2) => vec![s1, s2],
|
|
Self::NotEqualFromEntries(s1, s2) => vec![s1, s2],
|
|
Self::GtFromEntries(s1, s2) => vec![s1, s2],
|
|
Self::LtFromEntries(s1, s2) => vec![s1, s2],
|
|
Self::TransitiveEqualFromStatements(s1, s2) => vec![s1, s2],
|
|
Self::GtToNotEqual(s) => vec![s],
|
|
Self::LtToNotEqual(s) => vec![s],
|
|
Self::ContainsFromEntries(s1, s2, s3, pf) => vec![s1, s2, s3],
|
|
Self::NotContainsFromEntries(s1, s2, pf) => vec![s1, s2],
|
|
Self::SumOf(s1, s2, s3) => vec![s1, s2, s3],
|
|
Self::ProductOf(s1, s2, s3) => vec![s1, s2, s3],
|
|
Self::MaxOf(s1, s2, s3) => vec![s1, s2, s3],
|
|
Self::Custom(_, args) => args,
|
|
}
|
|
}
|
|
|
|
/// Extracts auxiliary data from operation.
|
|
pub fn aux(&self) -> OperationAux {
|
|
match self {
|
|
Self::ContainsFromEntries(_, _, _, mp) => OperationAux::MerkleProof(mp.clone()),
|
|
Self::NotContainsFromEntries(_, _, mp) => OperationAux::MerkleProof(mp.clone()),
|
|
_ => OperationAux::None,
|
|
}
|
|
}
|
|
|
|
/// Forms operation from op-code and arguments.
|
|
pub fn op(op_code: OperationType, args: &[Statement], aux: &OperationAux) -> Result<Self> {
|
|
type OA = OperationAux;
|
|
type NO = NativeOperation;
|
|
let arg_tup = (
|
|
args.first().cloned(),
|
|
args.get(1).cloned(),
|
|
args.get(2).cloned(),
|
|
);
|
|
Ok(match op_code {
|
|
OperationType::Native(o) => match (o, arg_tup, aux.clone(), args.len()) {
|
|
(NO::None, (None, None, None), OA::None, 0) => Self::None,
|
|
(NO::NewEntry, (None, None, None), OA::None, 0) => Self::NewEntry,
|
|
(NO::CopyStatement, (Some(s), None, None), OA::None, 1) => Self::CopyStatement(s),
|
|
(NO::EqualFromEntries, (Some(s1), Some(s2), None), OA::None, 2) => {
|
|
Self::EqualFromEntries(s1, s2)
|
|
}
|
|
(NO::NotEqualFromEntries, (Some(s1), Some(s2), None), OA::None, 2) => {
|
|
Self::NotEqualFromEntries(s1, s2)
|
|
}
|
|
(NO::GtFromEntries, (Some(s1), Some(s2), None), OA::None, 2) => {
|
|
Self::GtFromEntries(s1, s2)
|
|
}
|
|
(NO::LtFromEntries, (Some(s1), Some(s2), None), OA::None, 2) => {
|
|
Self::LtFromEntries(s1, s2)
|
|
}
|
|
(
|
|
NO::ContainsFromEntries,
|
|
(Some(s1), Some(s2), Some(s3)),
|
|
OA::MerkleProof(pf),
|
|
3,
|
|
) => Self::ContainsFromEntries(s1, s2, s3, pf),
|
|
(
|
|
NO::NotContainsFromEntries,
|
|
(Some(s1), Some(s2), None),
|
|
OA::MerkleProof(pf),
|
|
2,
|
|
) => Self::NotContainsFromEntries(s1, s2, pf),
|
|
(NO::SumOf, (Some(s1), Some(s2), Some(s3)), OA::None, 3) => Self::SumOf(s1, s2, s3),
|
|
(NO::ProductOf, (Some(s1), Some(s2), Some(s3)), OA::None, 3) => {
|
|
Self::ProductOf(s1, s2, s3)
|
|
}
|
|
(NO::MaxOf, (Some(s1), Some(s2), Some(s3)), OA::None, 3) => Self::MaxOf(s1, s2, s3),
|
|
_ => Err(anyhow!(
|
|
"Ill-formed operation {:?} with arguments {:?}.",
|
|
op_code,
|
|
args
|
|
))?,
|
|
},
|
|
OperationType::Custom(cpr) => Self::Custom(cpr, args.to_vec()),
|
|
})
|
|
}
|
|
/// Gives the output statement of the given operation, where determined
|
|
/// A ValueOf statement is not determined by the NewEntry operation, so returns Ok(None)
|
|
/// The outer Result is error handling
|
|
pub fn output_statement(&self) -> Result<Option<Statement>> {
|
|
use Statement::*;
|
|
let pred: Option<Predicate> = self.op_type().output_predicate();
|
|
|
|
let st_args: Option<Vec<StatementArg>> = match self {
|
|
Self::None => Some(vec![]),
|
|
Self::NewEntry => Option::None,
|
|
Self::CopyStatement(s1) => Some(s1.args()),
|
|
Self::EqualFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)) => {
|
|
if v1 == v2 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::EqualFromEntries(_, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::NotEqualFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)) => {
|
|
if v1 != v2 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::NotEqualFromEntries(_, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::GtFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)) => {
|
|
if v1 > v2 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::GtFromEntries(_, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::LtFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)) => {
|
|
if v1 < v2 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::LtFromEntries(_, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::TransitiveEqualFromStatements(Equal(ak1, ak2), Equal(ak3, ak4)) => {
|
|
if ak2 == ak3 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak3)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::TransitiveEqualFromStatements(_, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::GtToNotEqual(Gt(ak1, ak2)) => {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
}
|
|
Self::GtToNotEqual(_) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::LtToNotEqual(Gt(ak1, ak2)) => {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
}
|
|
Self::LtToNotEqual(_) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::ContainsFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2), ValueOf(ak3, v3), pf)
|
|
if MerkleTree::verify(pf.siblings.len(), (*v1).into(), &pf, v2, v3)? == () =>
|
|
{
|
|
Some(vec![
|
|
StatementArg::Key(*ak1),
|
|
StatementArg::Key(*ak2),
|
|
StatementArg::Key(*ak3),
|
|
])
|
|
}
|
|
Self::ContainsFromEntries(_, _, _, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::NotContainsFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2), pf)
|
|
if MerkleTree::verify_nonexistence(pf.siblings.len(), (*v1).into(), &pf, v2)?
|
|
== () =>
|
|
{
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
}
|
|
Self::NotContainsFromEntries(_, _, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::SumOf(ValueOf(ak1, v1), ValueOf(ak2, v2), ValueOf(ak3, v3)) => {
|
|
let v1: i64 = (*v1).try_into()?;
|
|
let v2: i64 = (*v2).try_into()?;
|
|
let v3: i64 = (*v3).try_into()?;
|
|
if v1 == v2 + v3 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::SumOf(_, _, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::ProductOf(ValueOf(ak1, v1), ValueOf(ak2, v2), ValueOf(ak3, v3)) => {
|
|
let v1: i64 = (*v1).try_into()?;
|
|
let v2: i64 = (*v2).try_into()?;
|
|
let v3: i64 = (*v3).try_into()?;
|
|
if v1 == v2 * v3 {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::ProductOf(_, _, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::MaxOf(ValueOf(ak1, v1), ValueOf(ak2, v2), ValueOf(ak3, v3)) => {
|
|
let v1: i64 = (*v1).try_into()?;
|
|
let v2: i64 = (*v2).try_into()?;
|
|
let v3: i64 = (*v3).try_into()?;
|
|
if v1 == std::cmp::max(v2, v3) {
|
|
Some(vec![StatementArg::Key(*ak1), StatementArg::Key(*ak2)])
|
|
} else {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
}
|
|
Self::MaxOf(_, _, _) => {
|
|
return Err(anyhow!("Invalid operation"));
|
|
}
|
|
Self::Custom(_, _) => todo!(),
|
|
};
|
|
|
|
let x: Option<Result<Statement>> = pred
|
|
.zip(st_args)
|
|
.map(|(pred, st_args)| Statement::from_args(pred, st_args));
|
|
x.transpose()
|
|
}
|
|
/// Checks the given operation against a statement, and prints information if the check does not pass
|
|
pub fn check_and_log(&self, params: &Params, output_statement: &Statement) -> Result<bool> {
|
|
let valid: bool = self.check(params, output_statement)?;
|
|
if !valid {
|
|
error!("Check failed on the following statement");
|
|
error!("{}", output_statement);
|
|
}
|
|
Ok(valid)
|
|
}
|
|
/// Checks the given operation against a statement.
|
|
pub fn check(&self, _params: &Params, output_statement: &Statement) -> Result<bool> {
|
|
use Statement::*;
|
|
match (self, output_statement) {
|
|
(Self::None, None) => Ok(true),
|
|
(Self::NewEntry, ValueOf(AnchoredKey(pod_id, _), _)) => Ok(pod_id == &SELF),
|
|
(Self::CopyStatement(s1), s2) => Ok(s1 == s2),
|
|
(Self::EqualFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)), Equal(ak3, ak4)) => {
|
|
Ok(v1 == v2 && ak3 == ak1 && ak4 == ak2)
|
|
}
|
|
(Self::NotEqualFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)), NotEqual(ak3, ak4)) => {
|
|
Ok(v1 != v2 && ak3 == ak1 && ak4 == ak2)
|
|
}
|
|
(Self::GtFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)), Gt(ak3, ak4)) => {
|
|
Ok(v1 > v2 && ak3 == ak1 && ak4 == ak2)
|
|
}
|
|
(Self::LtFromEntries(ValueOf(ak1, v1), ValueOf(ak2, v2)), Lt(ak3, ak4)) => {
|
|
Ok(v1 < v2 && ak3 == ak1 && ak4 == ak2)
|
|
}
|
|
(Self::ContainsFromEntries(_, _, _, _), Contains(_, _, _)) =>
|
|
/* TODO */
|
|
{
|
|
Ok(true)
|
|
}
|
|
(Self::NotContainsFromEntries(_, _, _), NotContains(_, _)) =>
|
|
/* TODO */
|
|
{
|
|
Ok(true)
|
|
}
|
|
(
|
|
Self::TransitiveEqualFromStatements(Equal(ak1, ak2), Equal(ak3, ak4)),
|
|
Equal(ak5, ak6),
|
|
) => Ok(ak2 == ak3 && ak5 == ak1 && ak6 == ak4),
|
|
(Self::GtToNotEqual(Gt(ak1, ak2)), NotEqual(ak3, ak4)) => Ok(ak1 == ak3 && ak2 == ak4),
|
|
(Self::LtToNotEqual(Lt(ak1, ak2)), NotEqual(ak3, ak4)) => Ok(ak1 == ak3 && ak2 == ak4),
|
|
(
|
|
Self::SumOf(ValueOf(ak1, v1), ValueOf(ak2, v2), ValueOf(ak3, v3)),
|
|
SumOf(ak4, ak5, ak6),
|
|
) => {
|
|
let v1: i64 = (*v1).try_into()?;
|
|
let v2: i64 = (*v2).try_into()?;
|
|
let v3: i64 = (*v3).try_into()?;
|
|
Ok((v1 == v2 + v3) && ak4 == ak1 && ak5 == ak2 && ak6 == ak3)
|
|
}
|
|
(Self::Custom(CustomPredicateRef(cpb, i), args), Custom(cpr, s_args))
|
|
if cpb == &cpr.0 && i == &cpr.1 =>
|
|
{
|
|
// Bind according to custom predicate pattern match against arg list.
|
|
let bindings = cpr.match_against(args)?;
|
|
// Check arg length
|
|
let arg_len = cpr.arg_len();
|
|
if arg_len != 2 * s_args.len() {
|
|
Err(anyhow!("Custom predicate arg list {:?} must have {} arguments after destructuring.", s_args, arg_len))
|
|
} else {
|
|
let bound_args = (0..arg_len)
|
|
.map(|i| {
|
|
bindings.get(&i).cloned().ok_or(anyhow!(
|
|
"Wildcard {} of custom predicate {:?} is unbound.",
|
|
i,
|
|
cpr
|
|
))
|
|
})
|
|
.collect::<Result<Vec<_>>>()?;
|
|
let s_args = s_args
|
|
.iter()
|
|
.flat_map(|AnchoredKey(o, k)| [Value::from(o.0), (*k).into()])
|
|
.collect::<Vec<_>>();
|
|
if bound_args != s_args {
|
|
Err(anyhow!("Arguments to output statement {} do not match those implied by operation {:?}", output_statement,self))
|
|
} else {
|
|
Ok(true)
|
|
}
|
|
}
|
|
}
|
|
_ => Err(anyhow!(
|
|
"Invalid deduction: {:?} ⇏ {:#}",
|
|
self,
|
|
output_statement
|
|
)),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ToFields for Operation {
|
|
fn to_fields(&self, params: &Params) -> Vec<F> {
|
|
todo!()
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for Operation {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
writeln!(f, "middleware::Operation:")?;
|
|
writeln!(f, " {:?} ", self.op_type())?;
|
|
for arg in self.args().iter() {
|
|
writeln!(f, " {}", arg)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|