ink_storage_traits/storage.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 ink_primitives::Key;
16
17/// Trait for representing types which can be read and written to storage.
18///
19/// This trait is not the same as the [`scale::Codec`]. Each type that implements
20/// [`scale::Codec`] are storable by default and transferable between contracts.
21/// But not each storable type is transferable.
22pub trait Storable: Sized {
23 /// Convert self to a slice and append it to the destination.
24 fn encode<T: scale::Output + ?Sized>(&self, dest: &mut T);
25
26 /// Attempt to deserialize the value from input.
27 fn decode<I: scale::Input>(input: &mut I) -> Result<Self, scale::Error>;
28
29 /// The exact number of bytes this type consumes in the encoded form.
30 fn encoded_size(&self) -> usize;
31}
32
33/// Types which implement `scale::Encode` and `scale::Decode` are `Storable` by default
34/// because they can be written directly into the storage cell.
35impl<P> Storable for P
36where
37 P: scale::Codec,
38{
39 #[inline]
40 fn encode<T: scale::Output + ?Sized>(&self, dest: &mut T) {
41 scale::Encode::encode_to(self, dest)
42 }
43
44 #[inline]
45 fn decode<I: scale::Input>(input: &mut I) -> Result<Self, scale::Error> {
46 scale::Decode::decode(input)
47 }
48
49 #[inline]
50 fn encoded_size(&self) -> usize {
51 <P as scale::Encode>::encoded_size(self)
52 }
53}
54
55/// Decode and consume all of the given input data.
56///
57/// If not all data is consumed, an error is returned.
58pub fn decode_all<T: Storable>(input: &mut &[u8]) -> Result<T, scale::Error> {
59 let res = <T as Storable>::decode(input)?;
60
61 if input.is_empty() {
62 Ok(res)
63 } else {
64 Err("Input buffer has still data left after decoding!".into())
65 }
66}
67
68pub(crate) mod private {
69 /// Seals the implementation of `Packed`.
70 pub trait Sealed {}
71}
72
73/// Trait for describing types that can be read and written to storage while all fields
74/// occupy only a single storage cell.
75///
76/// If at least one of the fields in the type occupies its own storage cell, this type
77/// is considered non-packed.
78///
79/// # Note
80///
81/// The trait is automatically implemented for types that implement [`scale::Codec`]
82/// via blanket implementation.
83pub trait Packed: Storable + scale::Codec + private::Sealed {}
84
85/// Holds storage key for the type.
86///
87/// # Note
88///
89/// The trait is automatically implemented for [`Packed`] types
90/// via blanket implementation.
91pub trait StorageKey {
92 /// Storage key of the type.
93 const KEY: Key;
94
95 /// Returns the storage key.
96 fn key(&self) -> Key {
97 Self::KEY
98 }
99}
100
101/// Describes the type that should be used for storing the value and preferred storage
102/// key.
103///
104/// # Note
105///
106/// The trait is automatically implemented for [`Packed`] types
107/// via blanket implementation.
108pub trait StorableHint<Key: StorageKey> {
109 /// Storable type with storage key inside.
110 type Type: Storable;
111 /// The storage key that the type prefers. It can be overwritten by an auto-generated
112 /// storage key.
113 type PreferredKey: StorageKey;
114}
115
116/// Automatically returns the type that should be used for storing the value.
117///
118/// The trait is used by codegen to determine which storage key the type should have.
119pub trait AutoStorableHint<Key: StorageKey> {
120 /// Storable type with storage key inside.
121 type Type: Storable;
122}