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
extern crate quote;
extern crate syn;
extern crate proc_macro;
extern crate deq_core;
use proc_macro::TokenStream;
use quote::quote;
use syn::{parse_macro_input, DeriveInput, parse::Parser};
#[proc_macro_derive(Transaction)]
pub fn transaction_macro_derive(input: TokenStream) -> TokenStream {
let ast = syn::parse(input).unwrap();
impl_transaction_macro(&ast)
}
#[proc_macro_attribute]
pub fn transaction_fields(_args: TokenStream, input: TokenStream) -> TokenStream {
let mut ast = parse_macro_input!(input as DeriveInput);
match &mut ast.data {
syn::Data::Struct(ref mut struct_data) => {
match &mut struct_data.fields {
syn::Fields::Named(fields) => {
fields
.named
.push(syn::Field::parse_named.parse2(
quote! { pub transaction_data: TransactionData<Self> }).unwrap());
}
_ => {
()
}
}
return quote! {
#ast
}.into();
}
_ => panic!("`transactions_fields` must be used with structs")
}
}
fn impl_transaction_macro(ast: &syn::DeriveInput) -> TokenStream {
let name = &ast.ident;
let gen = quote! {
impl Transaction for #name {
fn begin(&mut self) {
self.transaction_data.t.push(self.clone());
}
fn commit(&mut self) -> Result<(), TransactionError> {
match self.transaction_data.t.pop() {
Some(_) => Ok(()),
None => Err(TransactionError::new(TransactionErrorType::TransactionNotStarted))
}
}
fn revert(&mut self) -> Result<(), TransactionError> {
match self.transaction_data.t.pop() {
Some(prev) => {
*self = prev;
return Ok(());
},
None => Err(TransactionError::new(TransactionErrorType::TransactionNotStarted))
}
}
fn len(&self) -> usize {
self.transaction_data.t.len()
}
}
};
gen.into()
}