ink_primitives/reflect/trait_def/info.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
15/// Stores information for every ink! trait message of an ink! trait definition.
16///
17/// This information includes if the ink! trait message is payable
18/// as well as its derived or manually specified selector.
19///
20/// In the future this info trait might be extended to contain
21/// more information about a single ink! trait message.
22///
23/// The information provided through this trait can be used on the
24/// implementer side of an ink! trait to check and guard certain
25/// properties on a Rust type system level. This is important since
26/// ink! cannot be guaranteed to have both the ink! trait definition
27/// and all of its implementers under its scope and radar.
28///
29/// # Note
30///
31/// - The `TraitMessageInfo<LOCAL_ID>` is implemented by the automatically generated ink!
32/// trait definition information object associated to the ink! trait definition at hand.
33/// - For every ink! trait message defined by the ink! trait definition the associated
34/// ink! trait definition information object implements this trait given the
35/// `TRAIT_LOCAL_MESSAGE_ID` of each ink! trait message respectively.
36/// - The local IDs uniquely identifying all the ink! trait messages of the ink! trait
37/// definition are computed solely using the Rust identifier of the ink! trait message
38/// which can be derived from ink! implementation blocks in order to query the
39/// information stored by this ink! trait information object trait implementation.
40///
41/// # Usage
42///
43/// ```
44/// #[ink::trait_definition]
45/// pub trait InkTrait {
46/// #[ink(message)]
47/// fn trait_message_1(&self);
48///
49/// #[ink(message, payable, selector = 0xC0DECAFE)]
50/// fn trait_message_2(&mut self);
51/// }
52///
53/// #[ink::trait_definition(namespace = "foo")]
54/// pub trait InkTrait2 {
55/// #[ink(message)]
56/// fn trait_message(&self);
57/// }
58///
59/// #[ink::contract]
60/// pub mod contract {
61/// use super::{InkTrait, InkTrait2};
62///
63/// #[ink(storage)]
64/// pub struct Contract {}
65///
66/// impl Contract {
67/// #[ink(constructor)]
68/// pub fn constructor() -> Self { Contract {} }
69/// }
70///
71/// impl InkTrait for Contract {
72/// #[ink(message)]
73/// fn trait_message_1(&self) {}
74///
75/// #[ink(message)]
76/// fn trait_message_2(&mut self) {}
77/// }
78///
79/// impl InkTrait2 for Contract {
80/// #[ink(message)]
81/// fn trait_message(&self) {}
82/// }
83/// }
84///
85/// # use ink::reflect::TraitDefinitionRegistry;
86/// # use ink::reflect::TraitMessageInfo;
87/// # use ink_env::DefaultEnvironment;
88/// # use ink::{selector_id, selector_bytes};
89///
90/// fn main() {
91/// assert_eq!(
92/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait>::__ink_TraitInfo
93/// as TraitMessageInfo<{selector_id!("trait_message_1")}>>::PAYABLE,
94/// false,
95/// );
96/// assert_eq!(
97/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait>::__ink_TraitInfo
98/// as TraitMessageInfo<{selector_id!("trait_message_2")}>>::PAYABLE,
99/// true,
100/// );
101/// assert_eq!(
102/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait2>::__ink_TraitInfo
103/// as TraitMessageInfo<{selector_id!("trait_message")}>>::PAYABLE,
104/// false,
105/// );
106/// assert_eq!(
107/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait>::__ink_TraitInfo
108/// as TraitMessageInfo<{selector_id!("trait_message_1")}>>::SELECTOR,
109/// selector_bytes!("InkTrait::trait_message_1")
110/// );
111/// assert_eq!(
112/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait>::__ink_TraitInfo
113/// as TraitMessageInfo<{selector_id!("trait_message_2")}>>::SELECTOR,
114/// [0xC0, 0xDE, 0xCA, 0xFE]
115/// );
116/// assert_eq!(
117/// <<TraitDefinitionRegistry<DefaultEnvironment> as InkTrait2>::__ink_TraitInfo
118/// as TraitMessageInfo<{selector_id!("trait_message")}>>::SELECTOR,
119/// selector_bytes!("foo::InkTrait2::trait_message")
120/// );
121/// }
122/// ```
123pub trait TraitMessageInfo<const TRAIT_LOCAL_MESSAGE_ID: u32> {
124 /// Is `true` if the ink! trait message has been annotated with `#[ink(payable)]`.
125 const PAYABLE: bool;
126
127 /// The unique selector of the ink! trait message.
128 ///
129 /// This might have been adjusted using `#[ink(selector = N:u32)]` at the
130 /// ink! trait definition site.
131 const SELECTOR: [u8; 4];
132}
133
134/// Captures info about an ink! trait definition.
135pub trait TraitInfo {
136 /// The unique id of the ink! trait definition.
137 const ID: u32;
138
139 /// The module path of the ink! trait definition.
140 ///
141 /// This is equivalent to Rust's builtin `module_path!` macro
142 /// invocation at the definition site of the ink! trait.
143 const PATH: &'static str;
144
145 /// The name of the ink! trait.
146 ///
147 /// This is just for convenience.
148 const NAME: &'static str;
149}