ink_engine/
chain_extension.rs

1// Copyright (C) Use Ink (UK) Ltd.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15use super::Error;
16use derive_more::From;
17use std::collections::{
18    hash_map::Entry,
19    HashMap,
20};
21
22/// Chain extension registry.
23///
24/// Allows to register chain extension methods and call them.
25pub struct ChainExtensionHandler {
26    /// The currently registered runtime call handler.
27    registered: HashMap<ExtensionId, Box<dyn ChainExtension>>,
28    /// The output buffer used and reused for chain extension method call results.
29    output: Vec<u8>,
30}
31
32/// The unique ID of the registered chain extension.
33#[derive(
34    Debug, From, scale::Encode, scale::Decode, PartialEq, Eq, PartialOrd, Ord, Hash,
35)]
36pub struct ExtensionId(u16);
37
38/// Types implementing this trait can be used as chain extensions.
39///
40/// This trait is only useful for testing contract via the off-chain environment.
41pub trait ChainExtension {
42    /// The static ID of the chain extension.
43    ///
44    /// # Note
45    ///
46    /// This is expected to return a constant value.
47    fn ext_id(&self) -> u16;
48
49    /// Calls the chain extension with the given input.
50    ///
51    /// Returns an error code and may fill the `output` buffer with a SCALE encoded
52    /// result.
53    #[allow(clippy::ptr_arg)]
54    fn call(&mut self, func_id: u16, input: &[u8], output: &mut Vec<u8>) -> u32;
55}
56
57impl Default for ChainExtensionHandler {
58    fn default() -> Self {
59        ChainExtensionHandler::new()
60    }
61}
62
63impl ChainExtensionHandler {
64    /// Creates a new chain extension handler.
65    ///
66    /// Initialized with an empty set of chain extensions.
67    pub fn new() -> Self {
68        Self {
69            registered: HashMap::new(),
70            output: Vec::new(),
71        }
72    }
73
74    /// Resets the chain extension handler to uninitialized state.
75    pub fn reset(&mut self) {
76        self.registered.clear();
77        self.output.clear();
78    }
79
80    /// Register a new chain extension.
81    pub fn register(&mut self, extension: Box<dyn ChainExtension>) {
82        let ext_id = extension.ext_id();
83        self.registered.insert(ExtensionId::from(ext_id), extension);
84    }
85
86    /// Evaluates the chain extension with the given parameters.
87    ///
88    /// Upon success returns the values returned by the evaluated chain extension.
89    pub fn eval(&mut self, id: u32, input: &[u8]) -> Result<(u32, &[u8]), Error> {
90        self.output.clear();
91
92        let func_id = (id & 0x0000FFFF) as u16;
93        let ext_id = (id >> 16) as u16;
94
95        let extension_id = ExtensionId::from(ext_id);
96        match self.registered.entry(extension_id) {
97            Entry::Occupied(occupied) => {
98                let status_code =
99                    occupied.into_mut().call(func_id, input, &mut self.output);
100                Ok((status_code, &mut self.output))
101            }
102            Entry::Vacant(_vacant) => Err(Error::UnregisteredChainExtension),
103        }
104    }
105}