forked from hyperledger-iroha/iroha
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Custom instructions in executor (hyperledger-iroha#4645)
* feat: Custom instructions in executor Signed-off-by: Dmitry Murzin <[email protected]>
- Loading branch information
Showing
25 changed files
with
826 additions
and
11 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
15 changes: 15 additions & 0 deletions
15
client/tests/integration/smartcontracts/executor_custom_data_model/Cargo.toml
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
[package] | ||
name = "executor_custom_data_model" | ||
|
||
edition.workspace = true | ||
version.workspace = true | ||
authors.workspace = true | ||
|
||
license.workspace = true | ||
|
||
[dependencies] | ||
iroha_data_model.workspace = true | ||
iroha_schema.workspace = true | ||
|
||
serde_json.workspace = true | ||
serde.workspace = true |
310 changes: 310 additions & 0 deletions
310
client/tests/integration/smartcontracts/executor_custom_data_model/src/complex.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,310 @@ | ||
//! Example of custom expression system. | ||
//! Only few expressions are implemented to show proof-of-concept. | ||
//! See `smartcontracts/executor_custom_instructions_complex`. | ||
//! This is simplified version of expression system from Iroha v2.0.0-pre-rc.20 | ||
pub use evaluate::*; | ||
pub use expression::*; | ||
pub use isi::*; | ||
|
||
mod isi { | ||
use alloc::{boxed::Box, format, string::String, vec::Vec}; | ||
|
||
use iroha_data_model::{ | ||
isi::{Custom, InstructionBox}, | ||
JsonString, | ||
}; | ||
use iroha_schema::IntoSchema; | ||
use serde::{Deserialize, Serialize}; | ||
|
||
use crate::complex::expression::EvaluatesTo; | ||
|
||
#[derive(Debug, Deserialize, Serialize, IntoSchema)] | ||
pub enum CustomInstructionExpr { | ||
Core(CoreExpr), | ||
If(Box<ConditionalExpr>), | ||
// Other custom instructions | ||
} | ||
|
||
impl From<CustomInstructionExpr> for Custom { | ||
fn from(isi: CustomInstructionExpr) -> Self { | ||
let payload = | ||
JsonString::serialize(&isi).expect("Couldn't serialize custom instruction"); | ||
Self::new(payload) | ||
} | ||
} | ||
|
||
impl CustomInstructionExpr { | ||
pub fn into_instruction(self) -> InstructionBox { | ||
InstructionBox::Custom(self.into()) | ||
} | ||
} | ||
|
||
impl TryFrom<&JsonString> for CustomInstructionExpr { | ||
type Error = serde_json::Error; | ||
|
||
fn try_from(payload: &JsonString) -> serde_json::Result<Self> { | ||
payload.deserialize() | ||
} | ||
} | ||
|
||
// Built-in instruction (can be evaluated based on query values, etc) | ||
#[derive(Debug, Deserialize, Serialize, IntoSchema)] | ||
pub struct CoreExpr { | ||
pub object: EvaluatesTo<InstructionBox>, | ||
} | ||
|
||
impl CoreExpr { | ||
pub fn new(object: impl Into<EvaluatesTo<InstructionBox>>) -> Self { | ||
Self { | ||
object: object.into(), | ||
} | ||
} | ||
} | ||
|
||
/// Composite instruction for a conditional execution of other instructions. | ||
#[derive(Debug, Deserialize, Serialize, IntoSchema)] | ||
pub struct ConditionalExpr { | ||
/// Condition to be checked. | ||
pub condition: EvaluatesTo<bool>, | ||
/// Instruction to be executed if condition pass. | ||
pub then: CustomInstructionExpr, | ||
} | ||
|
||
impl ConditionalExpr { | ||
pub fn new( | ||
condition: impl Into<EvaluatesTo<bool>>, | ||
then: impl Into<CustomInstructionExpr>, | ||
) -> Self { | ||
Self { | ||
condition: condition.into(), | ||
then: then.into(), | ||
} | ||
} | ||
} | ||
} | ||
|
||
mod expression { | ||
use alloc::{ | ||
boxed::Box, | ||
format, | ||
string::{String, ToString}, | ||
vec, | ||
vec::Vec, | ||
}; | ||
use core::marker::PhantomData; | ||
|
||
use iroha_data_model::{ | ||
isi::InstructionBox, | ||
prelude::{Numeric, QueryBox}, | ||
}; | ||
use iroha_schema::{IntoSchema, TypeId}; | ||
use serde::{Deserialize, Serialize}; | ||
|
||
/// Struct for type checking and converting expression results. | ||
#[derive(Debug, Deserialize, Serialize, TypeId)] | ||
pub struct EvaluatesTo<V> { | ||
#[serde(flatten)] | ||
pub(crate) expression: Box<Expression>, | ||
_value_type: PhantomData<V>, | ||
} | ||
|
||
impl<V> EvaluatesTo<V> { | ||
pub fn new_unchecked(expression: impl Into<Expression>) -> Self { | ||
Self { | ||
expression: Box::new(expression.into()), | ||
_value_type: PhantomData, | ||
} | ||
} | ||
} | ||
|
||
/// Represents all possible expressions. | ||
#[derive(Debug, Deserialize, Serialize, IntoSchema)] | ||
pub enum Expression { | ||
/// Raw value. | ||
Raw(Value), | ||
/// Greater expression. | ||
Greater(Greater), | ||
/// Query to Iroha state. | ||
Query(QueryBox), | ||
} | ||
|
||
/// Returns whether the `left` expression is greater than the `right`. | ||
#[derive(Debug, Deserialize, Serialize, IntoSchema)] | ||
pub struct Greater { | ||
pub left: EvaluatesTo<Numeric>, | ||
pub right: EvaluatesTo<Numeric>, | ||
} | ||
|
||
impl Greater { | ||
/// Construct new [`Greater`] expression | ||
pub fn new( | ||
left: impl Into<EvaluatesTo<Numeric>>, | ||
right: impl Into<EvaluatesTo<Numeric>>, | ||
) -> Self { | ||
Self { | ||
left: left.into(), | ||
right: right.into(), | ||
} | ||
} | ||
} | ||
|
||
impl From<Greater> for EvaluatesTo<bool> { | ||
fn from(expression: Greater) -> Self { | ||
let expression = Expression::Greater(expression); | ||
EvaluatesTo::new_unchecked(expression) | ||
} | ||
} | ||
|
||
/// Sized container for all possible values. | ||
#[derive(Debug, Clone, Deserialize, Serialize, IntoSchema)] | ||
pub enum Value { | ||
Bool(bool), | ||
Numeric(Numeric), | ||
InstructionBox(InstructionBox), | ||
} | ||
|
||
impl From<bool> for Value { | ||
fn from(value: bool) -> Self { | ||
Value::Bool(value) | ||
} | ||
} | ||
|
||
impl From<Numeric> for EvaluatesTo<Numeric> { | ||
fn from(value: Numeric) -> Self { | ||
let value = Value::Numeric(value); | ||
let expression = Expression::Raw(value); | ||
EvaluatesTo::new_unchecked(expression) | ||
} | ||
} | ||
|
||
impl From<InstructionBox> for EvaluatesTo<InstructionBox> { | ||
fn from(value: InstructionBox) -> Self { | ||
let value = Value::InstructionBox(value); | ||
let expression = Expression::Raw(value); | ||
EvaluatesTo::new_unchecked(expression) | ||
} | ||
} | ||
|
||
impl TryFrom<Value> for bool { | ||
type Error = String; | ||
|
||
fn try_from(value: Value) -> Result<Self, Self::Error> { | ||
match value { | ||
Value::Bool(value) => Ok(value), | ||
_ => Err("Expected bool".to_string()), | ||
} | ||
} | ||
} | ||
|
||
impl TryFrom<Value> for Numeric { | ||
type Error = String; | ||
|
||
fn try_from(value: Value) -> Result<Self, Self::Error> { | ||
match value { | ||
Value::Numeric(value) => Ok(value), | ||
_ => Err("Expected Numeric".to_string()), | ||
} | ||
} | ||
} | ||
|
||
impl TryFrom<Value> for InstructionBox { | ||
type Error = String; | ||
|
||
fn try_from(value: Value) -> Result<Self, Self::Error> { | ||
match value { | ||
Value::InstructionBox(value) => Ok(value), | ||
_ => Err("Expected InstructionBox".to_string()), | ||
} | ||
} | ||
} | ||
|
||
impl<V: TryFrom<Value> + IntoSchema> IntoSchema for EvaluatesTo<V> { | ||
fn type_name() -> String { | ||
format!("EvaluatesTo<{}>", V::type_name()) | ||
} | ||
fn update_schema_map(map: &mut iroha_schema::MetaMap) { | ||
const EXPRESSION: &str = "expression"; | ||
|
||
if !map.contains_key::<Self>() { | ||
map.insert::<Self>(iroha_schema::Metadata::Struct( | ||
iroha_schema::NamedFieldsMeta { | ||
declarations: vec![iroha_schema::Declaration { | ||
name: String::from(EXPRESSION), | ||
ty: core::any::TypeId::of::<Expression>(), | ||
}], | ||
}, | ||
)); | ||
|
||
Expression::update_schema_map(map); | ||
} | ||
} | ||
} | ||
} | ||
|
||
mod evaluate { | ||
use alloc::string::ToString; | ||
|
||
use iroha_data_model::{ | ||
isi::error::InstructionExecutionError, query::QueryBox, ValidationFail, | ||
}; | ||
|
||
use crate::complex::expression::{EvaluatesTo, Expression, Greater, Value}; | ||
|
||
pub trait Evaluate { | ||
/// The resulting type of the expression. | ||
type Value; | ||
|
||
/// Calculate result. | ||
fn evaluate(&self, context: &impl Context) -> Result<Self::Value, ValidationFail>; | ||
} | ||
|
||
pub trait Context { | ||
/// Execute query against the current state of `Iroha` | ||
fn query(&self, query: &QueryBox) -> Result<Value, ValidationFail>; | ||
} | ||
|
||
impl<V: TryFrom<Value>> Evaluate for EvaluatesTo<V> | ||
where | ||
V::Error: ToString, | ||
{ | ||
type Value = V; | ||
|
||
fn evaluate(&self, context: &impl Context) -> Result<Self::Value, ValidationFail> { | ||
let expr = self.expression.evaluate(context)?; | ||
V::try_from(expr) | ||
.map_err(|e| InstructionExecutionError::Conversion(e.to_string())) | ||
.map_err(ValidationFail::InstructionFailed) | ||
} | ||
} | ||
|
||
impl Evaluate for Expression { | ||
type Value = Value; | ||
|
||
fn evaluate(&self, context: &impl Context) -> Result<Self::Value, ValidationFail> { | ||
match self { | ||
Expression::Raw(value) => Ok(value.clone()), | ||
Expression::Greater(expr) => expr.evaluate(context).map(Into::into), | ||
Expression::Query(expr) => expr.evaluate(context), | ||
} | ||
} | ||
} | ||
|
||
impl Evaluate for Greater { | ||
type Value = bool; | ||
|
||
fn evaluate(&self, context: &impl Context) -> Result<Self::Value, ValidationFail> { | ||
let left = self.left.evaluate(context)?; | ||
let right = self.right.evaluate(context)?; | ||
Ok(left > right) | ||
} | ||
} | ||
|
||
impl Evaluate for QueryBox { | ||
type Value = Value; | ||
|
||
fn evaluate(&self, context: &impl Context) -> Result<Self::Value, ValidationFail> { | ||
context.query(self) | ||
} | ||
} | ||
} |
8 changes: 8 additions & 0 deletions
8
client/tests/integration/smartcontracts/executor_custom_data_model/src/lib.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
//! Example of custom instructions which can be used in custom executor | ||
#![no_std] | ||
|
||
extern crate alloc; | ||
|
||
pub mod complex; | ||
pub mod simple; |
Oops, something went wrong.