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
use proc_macro2::{Span, TokenStream as TokenStream2};
use quote::{quote, ToTokens, TokenStreamExt};
use syn::{
parse::{Parse, ParseStream},
DeriveInput, Generics, ItemEnum,
};
use syn::{punctuated::Punctuated, Ident, Token};
#[derive(Debug, Clone)]
pub struct PostgresEnum {
name: Ident,
generics: Generics,
variants: Punctuated<syn::Variant, Token![,]>,
}
impl PostgresEnum {
pub fn new(
name: Ident,
generics: Generics,
variants: Punctuated<syn::Variant, Token![,]>,
) -> Self {
Self {
name,
generics,
variants,
}
}
pub fn from_derive_input(derive_input: DeriveInput) -> Result<Self, syn::Error> {
let data_enum = match derive_input.data {
syn::Data::Enum(data_enum) => data_enum,
syn::Data::Union(_) | syn::Data::Struct(_) => {
return Err(syn::Error::new(derive_input.ident.span(), "expected enum"))
}
};
Ok(Self::new(
derive_input.ident,
derive_input.generics,
data_enum.variants,
))
}
}
impl Parse for PostgresEnum {
fn parse(input: ParseStream) -> Result<Self, syn::Error> {
let parsed: ItemEnum = input.parse()?;
Ok(Self::new(parsed.ident, parsed.generics, parsed.variants))
}
}
impl ToTokens for PostgresEnum {
fn to_tokens(&self, tokens: &mut TokenStream2) {
let name = self.name.clone();
let mut static_generics = self.generics.clone();
for lifetime in static_generics.lifetimes_mut() {
lifetime.lifetime.ident = Ident::new("static", Span::call_site());
}
let (_impl_generics, ty_generics, _where_clauses) = static_generics.split_for_impl();
let variants = self.variants.iter();
let sql_graph_entity_fn_name =
syn::Ident::new(&format!("__pgx_internals_enum_{}", name), Span::call_site());
let inv = quote! {
#[no_mangle]
pub extern "C" fn #sql_graph_entity_fn_name() -> pgx::datum::sql_entity_graph::SqlGraphEntity {
let mut mappings = Default::default();
<#name #ty_generics as pgx::datum::WithTypeIds>::register_with_refs(&mut mappings, stringify!(#name).to_string());
pgx::datum::WithSizedTypeIds::<#name #ty_generics>::register_sized_with_refs(&mut mappings, stringify!(#name).to_string());
pgx::datum::WithArrayTypeIds::<#name #ty_generics>::register_array_with_refs(&mut mappings, stringify!(#name).to_string());
pgx::datum::WithVarlenaTypeIds::<#name #ty_generics>::register_varlena_with_refs(&mut mappings, stringify!(#name).to_string());
let submission = pgx::datum::sql_entity_graph::PostgresEnumEntity {
name: stringify!(#name),
file: file!(),
line: line!(),
module_path: module_path!(),
full_path: core::any::type_name::<#name #ty_generics>(),
mappings,
variants: vec![ #( stringify!(#variants) ),* ],
};
pgx::datum::sql_entity_graph::SqlGraphEntity::Enum(submission)
}
};
tokens.append_all(inv);
}
}