Skip to content

Commit

Permalink
feat: Custom instructions in executor (hyperledger-iroha#4645)
Browse files Browse the repository at this point in the history
* feat: Custom instructions in executor

Signed-off-by: Dmitry Murzin <[email protected]>
  • Loading branch information
dima74 authored Jun 10, 2024
1 parent f5537fd commit 6cf6378
Show file tree
Hide file tree
Showing 25 changed files with 826 additions and 11 deletions.
11 changes: 11 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,7 @@ irohad = { workspace = true }
iroha_wasm_builder = { workspace = true }
iroha_genesis = { workspace = true }
test_network = { workspace = true }
executor_custom_data_model = { path = "tests/integration/smartcontracts/executor_custom_data_model" }

tokio = { workspace = true, features = ["rt-multi-thread"] }
criterion = { workspace = true, features = ["html_reports"] }
Expand Down
7 changes: 6 additions & 1 deletion client/tests/integration/smartcontracts/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,9 @@ members = [
"executor_with_custom_permission",
"executor_remove_permission",
"executor_with_migration_fail",
"executor_custom_instructions_simple",
"executor_custom_instructions_complex",
"executor_custom_data_model",
"query_assets_and_save_cursor",
"smart_contract_can_filter_queries",
]
Expand All @@ -32,8 +35,10 @@ codegen-units = 1 # Further reduces binary size but increases compilation time
[workspace.dependencies]
iroha_smart_contract = { version = "=2.0.0-pre-rc.21", path = "../../../../smart_contract", features = ["debug"] }
iroha_trigger = { version = "=2.0.0-pre-rc.21", path = "../../../../smart_contract/trigger", features = ["debug"] }
iroha_executor = { version = "=2.0.0-pre-rc.21", path = "../../../../smart_contract/executor" }
iroha_executor = { version = "=2.0.0-pre-rc.21", path = "../../../../smart_contract/executor", features = ["debug"] }
iroha_schema = { version = "=2.0.0-pre-rc.21", path = "../../../../schema" }
iroha_data_model = { version = "=2.0.0-pre-rc.21", path = "../../../../data_model", default-features = false }
executor_custom_data_model = { version = "=2.0.0-pre-rc.21", path = "executor_custom_data_model" }

parity-scale-codec = { version = "3.2.1", default-features = false }
anyhow = { version = "1.0.71", default-features = false }
Expand Down
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
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)
}
}
}
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;
Loading

0 comments on commit 6cf6378

Please sign in to comment.