ink_codegen/lib.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//! The ink! code generation.
16//!
17//! This module defines everything the ink! procedural macro needs in order to
18//! generate Rust code for ink! smart contracts.
19//!
20//! We avoid structuring things via hidden or private modules, this
21//! was the ink! v1 way of doing code generation and it was bug prone
22//! and hard to maintain.
23//!
24//! What we do instead to generate code that is hidden from users is to
25//! pack it into anonymous `const` (so `const _: () = { … };`) and connect
26//! those to the outside world with private trait implementations.
27//!
28//! You can see how the generated code looks by installing
29//! [`cargo-expand`](https://github.com/dtolnay/cargo-expand)
30//! and executing `cargo expand --manifest-path ./integration-tests/flipper/Cargo.toml` in
31//! this repository.
32
33#![doc(
34 html_logo_url = "https://use.ink/img/crate-docs/logo.png",
35 html_favicon_url = "https://use.ink/crate-docs/favicon.png"
36)]
37
38pub use ink_primitives::reflect;
39mod enforced_error;
40mod generator;
41mod traits;
42
43pub use generator::generate_type_spec;
44
45use self::{
46 enforced_error::EnforcedErrors,
47 traits::{
48 GenerateCode,
49 GenerateCodeUsing,
50 },
51};
52use proc_macro2::TokenStream as TokenStream2;
53
54/// Types for which code can be generated by this crate.
55pub trait CodeGenerator: Sized {
56 /// The underlying generator generating the code.
57 type Generator: From<Self> + GenerateCode;
58}
59
60impl<'a> CodeGenerator for &'a ir::Contract {
61 type Generator = generator::Contract<'a>;
62}
63
64impl<'a> CodeGenerator for &'a ir::Event {
65 type Generator = generator::Event<'a>;
66}
67
68impl<'a> CodeGenerator for &'a ir::StorageItem {
69 type Generator = generator::StorageItem<'a>;
70}
71
72impl<'a> CodeGenerator for &'a ir::InkTraitDefinition {
73 type Generator = generator::TraitDefinition<'a>;
74}
75
76impl<'a> CodeGenerator for &'a ir::InkTest {
77 type Generator = generator::InkTest<'a>;
78}
79
80impl<'a> CodeGenerator for &'a ir::ChainExtension {
81 type Generator = generator::ChainExtension<'a>;
82}
83
84impl<'a> CodeGenerator for &'a ir::SelectorMacro<ir::marker::SelectorId> {
85 type Generator = generator::SelectorId<'a>;
86}
87
88impl<'a> CodeGenerator for &'a ir::SelectorMacro<ir::marker::SelectorBytes> {
89 type Generator = generator::SelectorBytes<'a>;
90}
91
92impl<'a> CodeGenerator for &'a ir::Blake2x256Macro {
93 type Generator = generator::Blake2x256<'a>;
94}
95
96/// Generates the entire code for the given ink! contract.
97pub fn generate_code<T>(entity: T) -> TokenStream2
98where
99 T: CodeGenerator,
100{
101 <T as CodeGenerator>::Generator::from(entity).generate_code()
102}