1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
//! [`contracttrait`] macro for reusable contract interfaces.
//!
//! _Note: This feature was released in v23.4.0 but is being included in the migration notes for the
//! next major version, v25._
//!
//! The [`contracttrait`] macro enables defining reusable contract interfaces as Rust traits with
//! default implementations. Contracts can implement these traits, and the default implementations
//! are automatically exported as contract functions.
//!
//! ## Generated Functionality
//!
//! When applied to a trait, [`contracttrait`] generates:
//!
//! - `{TraitName}Client` - A client for invoking the trait's functions on a contract
//! - `{TraitName}Args` - An enum of function arguments for the trait's functions
//! - `{TraitName}Spec` - The contract specification for the trait's functions
//!
//! These names can be customized using macro arguments (e.g., `client_name`, `args_name`,
//! `spec_name`).
//!
//! ## When to Use
//!
//! Use [`contracttrait`] when you want the trait to represent a contract interface.
//!
//! The [`contracttrait`] will make it possible to:
//! - Access a generated client for any contract implementing the interface.
//! - Automatically export default implementations that contracts can optionally override
//! - Share common functionality across contracts
//!
//! ## How It Works
//!
//! 1. **Define a trait** with [`contracttrait`]
//!
//! 2. **Implement the trait** with [`contractimpl`], including the `contracttrait` option:
//! `#[contractimpl(contracttrait)]`
//!
//! 3. **Override default functions as needed** - Contracts can provide their own implementations of
//! any function with default implementations.
//!
//! ## Patterns For Use
//!
//! Place traits that use [`contracttrait`] into a library crate, to share and make those traits
//! available to other crates and developers.
//!
//! [`contracttrait`]: crate::contracttrait
//! [`contractimpl`]: crate::contractimpl
//!
//! ## Example: Defining and Implementing a Trait
//!
//! ```
//! use soroban_sdk::{contract, contractimpl, contracttrait, Address, Env};
//!
//! // A regular trait for admin access control - not exported as contract functions
//! pub trait RequireAuthForPause {
//! fn require_auth_for_pause(env: &Env);
//! }
//!
//! // Define a contracttrait with default implementations that require RequireAuthForPause
//! #[contracttrait]
//! pub trait Pausable: RequireAuthForPause {
//! fn is_paused(env: &Env) -> bool {
//! env.storage().instance().has(&"paused")
//! }
//!
//! fn pause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().set(&"paused", &true);
//! }
//!
//! fn unpause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().remove(&"paused");
//! }
//! }
//!
//! #[contract]
//! pub struct MyContract;
//!
//! impl RequireAuthForPause for MyContract {
//! fn require_auth_for_pause(env: &Env) {
//! let admin: Address = env.storage().instance().get(&"admin").unwrap();
//! admin.require_auth();
//! }
//! }
//!
//! // Implement the trait - default functions are automatically exported
//! #[contractimpl(contracttrait)]
//! impl Pausable for MyContract {}
//!
//! #[contractimpl]
//! impl MyContract {
//! pub fn __constructor(env: &Env, admin: Address) {
//! env.storage().instance().set(&"admin", &admin);
//! }
//!
//! pub fn do_something(env: &Env) {
//! if Self::is_paused(env) {
//! panic!("contract is paused");
//! }
//! // ... rest of the function
//! }
//! }
//!
//! #[test]
//! fn test() {
//! # }
//! # #[cfg(feature = "testutils")]
//! # fn main() {
//! use soroban_sdk::{testutils::{Address as _, MockAuth, MockAuthInvoke}, IntoVal};
//! let env = Env::default();
//! let admin = Address::generate(&env);
//! let contract_id = env.register(MyContract, (&admin,));
//! let client = PausableClient::new(&env, &contract_id);
//!
//! assert!(!client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "pause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).pause();
//! assert!(client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "unpause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).unpause();
//! assert!(!client.is_paused());
//! }
//! # #[cfg(not(feature = "testutils"))]
//! # fn main() { }
//! ```
//!
//! ## Example: Overriding Default Implementations
//!
//! Contracts can override specific functions while keeping the defaults for others:
//!
//! ```
//! use soroban_sdk::{contract, contractimpl, contracttrait, Address, Env};
//!
//! // A regular trait for admin access control - not exported as contract functions
//! pub trait RequireAuthForPause {
//! fn require_auth_for_pause(env: &Env);
//! }
//!
//! // Define a contracttrait with default implementations that require RequireAuthForPause
//! #[contracttrait]
//! pub trait Pausable: RequireAuthForPause {
//! fn is_paused(env: &Env) -> bool {
//! env.storage().instance().has(&"paused")
//! }
//!
//! fn pause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().set(&"paused", &true);
//! }
//!
//! fn unpause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().remove(&"paused");
//! }
//! }
//!
//! #[contract]
//! pub struct MyContract;
//!
//! impl RequireAuthForPause for MyContract {
//! fn require_auth_for_pause(env: &Env) {
//! let admin: Address = env.storage().instance().get(&"admin").unwrap();
//! admin.require_auth();
//! }
//! }
//!
//! // Implement the trait - override default implementations as needed
//! #[contractimpl(contracttrait)]
//! impl Pausable for MyContract {
//! // Override is_paused with custom logic that returns false when not set
//! fn is_paused(env: &Env) -> bool {
//! env.storage().instance().get(&"paused").unwrap_or(false)
//! }
//! // pause() and unpause() use the default implementations
//! }
//!
//! #[contractimpl]
//! impl MyContract {
//! pub fn __constructor(env: &Env, admin: Address) {
//! env.storage().instance().set(&"admin", &admin);
//! }
//!
//! pub fn do_something(env: &Env) {
//! if Self::is_paused(env) {
//! panic!("contract is paused");
//! }
//! // ... rest of the function
//! }
//! }
//!
//! #[test]
//! fn test() {
//! # }
//! # #[cfg(feature = "testutils")]
//! # fn main() {
//! use soroban_sdk::{testutils::{Address as _, MockAuth, MockAuthInvoke}, IntoVal};
//! let env = Env::default();
//! let admin = Address::generate(&env);
//! let contract_id = env.register(MyContract, (&admin,));
//! let client = PausableClient::new(&env, &contract_id);
//!
//! assert!(!client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "pause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).pause();
//! assert!(client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "unpause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).unpause();
//! assert!(!client.is_paused());
//! }
//! # #[cfg(not(feature = "testutils"))]
//! # fn main() { }
//! ```
//!
//! ## Example: Using the Generated Client
//!
//! The generated `{TraitName}Client` can be used to call any contract that implements the trait:
//!
//! ```
//! use soroban_sdk::{contract, contractimpl, contracttrait, Address, Env};
//!
//! // A regular trait for admin access control - not exported as contract functions
//! pub trait RequireAuthForPause {
//! fn require_auth_for_pause(env: &Env);
//! }
//!
//! // Define a contracttrait with default implementations that require RequireAuthForPause
//! #[contracttrait]
//! pub trait Pausable: RequireAuthForPause {
//! fn is_paused(env: &Env) -> bool {
//! env.storage().instance().has(&"paused")
//! }
//!
//! fn pause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().set(&"paused", &true);
//! }
//!
//! fn unpause(env: &Env) {
//! Self::require_auth_for_pause(env);
//! env.storage().instance().remove(&"paused");
//! }
//! }
//!
//! #[contract]
//! pub struct MyContract;
//!
//! impl RequireAuthForPause for MyContract {
//! fn require_auth_for_pause(env: &Env) {
//! let admin: Address = env.storage().instance().get(&"admin").unwrap();
//! admin.require_auth();
//! }
//! }
//!
//! // Implement the trait - default functions are automatically exported
//! #[contractimpl(contracttrait)]
//! impl Pausable for MyContract {}
//!
//! #[contractimpl]
//! impl MyContract {
//! pub fn __constructor(env: &Env, admin: Address) {
//! env.storage().instance().set(&"admin", &admin);
//! }
//!
//! pub fn do_something(env: &Env) {
//! if Self::is_paused(env) {
//! panic!("contract is paused");
//! }
//! // ... rest of the function
//! }
//! }
//!
//! #[test]
//! fn test() {
//! # }
//! # #[cfg(feature = "testutils")]
//! # fn main() {
//! use soroban_sdk::{testutils::{Address as _, MockAuth, MockAuthInvoke}, IntoVal};
//! let env = Env::default();
//! let admin = Address::generate(&env);
//! let contract_id = env.register(MyContract, (&admin,));
//! let client = PausableClient::new(&env, &contract_id);
//!
//! assert!(!client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "pause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).pause();
//! assert!(client.is_paused());
//! client.mock_auths(&[MockAuth {
//! address: &admin,
//! invoke: &MockAuthInvoke {
//! contract: &contract_id,
//! fn_name: "unpause",
//! args: ().into_val(&env),
//! sub_invokes: &[],
//! },
//! }]).unpause();
//! assert!(!client.is_paused());
//! }
//! # #[cfg(not(feature = "testutils"))]
//! # fn main() { }
//! ```