pgrx_sql_entity_graph/pg_extern/
mod.rs1mod argument;
19mod attribute;
20mod cast;
21pub mod entity;
22mod operator;
23mod returning;
24mod search_path;
25
26pub use argument::PgExternArgument;
27pub(crate) use attribute::Attribute;
28pub use cast::PgCast;
29pub use operator::PgOperator;
30pub use returning::NameMacro;
31use syn::token::Comma;
32
33use self::returning::Returning;
34use super::UsedType;
35use crate::enrich::{CodeEnrichment, ToEntityGraphTokens, ToRustCodeTokens};
36use crate::finfo::{finfo_v1_extern_c, finfo_v1_tokens};
37use crate::fmt::ErrHarder;
38use crate::ToSqlConfig;
39use operator::{PgrxOperatorAttributeWithIdent, PgrxOperatorOpName};
40use search_path::SearchPathList;
41
42use proc_macro2::{Ident, Span, TokenStream as TokenStream2};
43use quote::quote;
44use syn::parse::{Parse, ParseStream, Parser};
45use syn::punctuated::Punctuated;
46use syn::spanned::Spanned;
47use syn::{Meta, Token};
48
49macro_rules! quote_spanned {
50 ($span:expr=> $($expansion:tt)*) => {
51 {
52 let synthetic = Span::mixed_site();
53 let synthetic = synthetic.located_at($span);
54 quote::quote_spanned! {synthetic=> $($expansion)* }
55 }
56 };
57}
58
59macro_rules! format_ident {
60 ($s:literal, $e:expr) => {{
61 let mut synthetic = $e.clone();
62 synthetic.set_span(Span::call_site().located_at($e.span()));
63 quote::format_ident!($s, synthetic)
64 }};
65}
66
67#[derive(Debug, Clone)]
90pub struct PgExtern {
91 attrs: Vec<Attribute>,
92 func: syn::ItemFn,
93 to_sql_config: ToSqlConfig,
94 operator: Option<PgOperator>,
95 cast: Option<PgCast>,
96 search_path: Option<SearchPathList>,
97 inputs: Vec<PgExternArgument>,
98 input_types: Vec<syn::Type>,
99 returns: Returning,
100}
101
102impl PgExtern {
103 #[track_caller]
104 pub fn new(attr: TokenStream2, item: TokenStream2) -> Result<CodeEnrichment<Self>, syn::Error> {
105 let mut attrs = Vec::new();
106 let mut to_sql_config: Option<ToSqlConfig> = None;
107
108 let parser = Punctuated::<Attribute, Token![,]>::parse_terminated;
109 let attr_ts = attr.clone();
110 let punctuated_attrs = parser
111 .parse2(attr)
112 .more_error(lazy_err!(attr_ts, "failed parsing pg_extern arguments"))?;
113 for pair in punctuated_attrs.into_pairs() {
114 match pair.into_value() {
115 Attribute::Sql(config) => to_sql_config = to_sql_config.or(Some(config)),
116 attr => attrs.push(attr),
117 }
118 }
119
120 let mut to_sql_config = to_sql_config.unwrap_or_default();
121
122 let func = syn::parse2::<syn::ItemFn>(item)?;
123
124 if let Some(ref mut content) = to_sql_config.content {
125 let value = content.value();
126 let span = content.span();
128 let updated_value =
129 value.replace("@FUNCTION_NAME@", &(func.sig.ident.to_string() + "_wrapper")) + "\n";
130 *content = syn::LitStr::new(&updated_value, span);
131 }
132
133 if !to_sql_config.overrides_default() {
134 crate::ident_is_acceptable_to_postgres(&func.sig.ident)?;
135 }
136 let operator = Self::operator(&func)?;
137 let search_path = Self::search_path(&func)?;
138 let inputs = Self::inputs(&func)?;
139 let input_types = Self::input_types(&func)?;
140 let returns = Returning::try_from(&func.sig.output)?;
141 Ok(CodeEnrichment(Self {
142 attrs,
143 func,
144 to_sql_config,
145 operator,
146 cast: None,
147 search_path,
148 inputs,
149 input_types,
150 returns,
151 }))
152 }
153
154 pub fn as_cast(&self, pg_cast: PgCast) -> PgExtern {
156 let mut result = self.clone();
157 result.cast = Some(pg_cast);
158 result
159 }
160
161 #[track_caller]
162 fn input_types(func: &syn::ItemFn) -> syn::Result<Vec<syn::Type>> {
163 func.sig
164 .inputs
165 .iter()
166 .filter_map(|v| -> Option<syn::Result<syn::Type>> {
167 match v {
168 syn::FnArg::Receiver(_) => None,
169 syn::FnArg::Typed(pat_ty) => {
170 let ty = match UsedType::new(*pat_ty.ty.clone()) {
171 Ok(v) => v.resolved_ty,
172 Err(e) => return Some(Err(e)),
173 };
174 Some(Ok(ty))
175 }
176 }
177 })
178 .collect()
179 }
180
181 fn name(&self) -> String {
182 self.attrs
183 .iter()
184 .find_map(|a| match a {
185 Attribute::Name(name) => Some(name.value()),
186 _ => None,
187 })
188 .unwrap_or_else(|| self.func.sig.ident.to_string())
189 }
190
191 fn schema(&self) -> Option<String> {
192 self.attrs.iter().find_map(|a| match a {
193 Attribute::Schema(name) => Some(name.value()),
194 _ => None,
195 })
196 }
197
198 pub fn extern_attrs(&self) -> &[Attribute] {
199 self.attrs.as_slice()
200 }
201
202 #[track_caller]
203 fn overridden(&self) -> Option<syn::LitStr> {
204 let mut span = None;
205 let mut retval = None;
206 let mut in_commented_sql_block = false;
207 for meta in self.func.attrs.iter().filter_map(|attr| {
208 if attr.meta.path().is_ident("doc") {
209 Some(attr.meta.clone())
210 } else {
211 None
212 }
213 }) {
214 let Meta::NameValue(syn::MetaNameValue { ref value, .. }) = meta else { continue };
215 let syn::Expr::Lit(syn::ExprLit { lit: syn::Lit::Str(inner), .. }) = value else {
216 continue;
217 };
218 span.get_or_insert(value.span());
219 if !in_commented_sql_block && inner.value().trim() == "```pgrxsql" {
220 in_commented_sql_block = true;
221 } else if in_commented_sql_block && inner.value().trim() == "```" {
222 in_commented_sql_block = false;
223 } else if in_commented_sql_block {
224 let line = inner
225 .value()
226 .trim_start()
227 .replace("@FUNCTION_NAME@", &(self.func.sig.ident.to_string() + "_wrapper"))
228 + "\n";
229 retval.get_or_insert_with(String::default).push_str(&line);
230 }
231 }
232 retval.map(|s| syn::LitStr::new(s.as_ref(), span.unwrap()))
233 }
234
235 #[track_caller]
236 fn operator(func: &syn::ItemFn) -> syn::Result<Option<PgOperator>> {
237 let mut skel = Option::<PgOperator>::default();
238 for attr in &func.attrs {
239 let last_segment = attr.path().segments.last().unwrap();
240 match last_segment.ident.to_string().as_str() {
241 s @ "opname" => {
242 let attr = attr
244 .parse_args::<PgrxOperatorOpName>()
245 .more_error(lazy_err!(attr.clone(), "bad parse of {s}?"))?;
246 skel.get_or_insert_with(Default::default).opname.get_or_insert(attr);
247 }
248 s @ "commutator" => {
249 let attr: PgrxOperatorAttributeWithIdent = attr
250 .parse_args()
251 .more_error(lazy_err!(attr.clone(), "bad parse of {s}?"))?;
252
253 skel.get_or_insert_with(Default::default).commutator.get_or_insert(attr);
254 }
255 s @ "negator" => {
256 let attr: PgrxOperatorAttributeWithIdent = attr
257 .parse_args()
258 .more_error(lazy_err!(attr.clone(), "bad parse of {s}?"))?;
259 skel.get_or_insert_with(Default::default).negator.get_or_insert(attr);
260 }
261 s @ "join" => {
262 let attr: PgrxOperatorAttributeWithIdent = attr
263 .parse_args()
264 .more_error(lazy_err!(attr.clone(), "bad parse of {s}?"))?;
265
266 skel.get_or_insert_with(Default::default).join.get_or_insert(attr);
267 }
268 s @ "restrict" => {
269 let attr: PgrxOperatorAttributeWithIdent = attr
270 .parse_args()
271 .more_error(lazy_err!(attr.clone(), "bad parse of {s}?"))?;
272
273 skel.get_or_insert_with(Default::default).restrict.get_or_insert(attr);
274 }
275 "hashes" => {
276 skel.get_or_insert_with(Default::default).hashes = true;
277 }
278 "merges" => {
279 skel.get_or_insert_with(Default::default).merges = true;
280 }
281 _ => (),
282 }
283 }
284 Ok(skel)
285 }
286
287 fn search_path(func: &syn::ItemFn) -> syn::Result<Option<SearchPathList>> {
288 func.attrs
289 .iter()
290 .find(|f| {
291 f.path()
292 .segments
293 .first()
294 .map(|f| f.ident == Ident::new("search_path", func.span()))
296 .unwrap_or_default()
297 })
298 .map(|attr| attr.parse_args::<SearchPathList>())
299 .transpose()
300 }
301
302 fn inputs(func: &syn::ItemFn) -> syn::Result<Vec<PgExternArgument>> {
303 let mut args = Vec::default();
304 for input in &func.sig.inputs {
305 let arg = PgExternArgument::build(input.clone())?;
306 args.push(arg);
307 }
308 Ok(args)
309 }
310
311 fn entity_tokens(&self) -> TokenStream2 {
312 let ident = &self.func.sig.ident;
313 let name = self.name();
314 let unsafety = &self.func.sig.unsafety;
315 let schema = self.schema();
316 let schema_iter = schema.iter();
317 let extern_attrs = self
318 .attrs
319 .iter()
320 .map(|attr| attr.to_sql_entity_graph_tokens())
321 .collect::<Punctuated<_, Token![,]>>();
322 let search_path = self.search_path.clone().into_iter();
323 let inputs = &self.inputs;
324 let inputs_iter = inputs.iter().map(|v| v.entity_tokens());
325
326 let input_types = self.input_types.iter().cloned();
327
328 let returns = &self.returns;
329
330 let return_type = match &self.func.sig.output {
331 syn::ReturnType::Default => None,
332 syn::ReturnType::Type(arrow, ty) => {
333 let ty = ty.clone();
334 Some(syn::ReturnType::Type(*arrow, ty))
335 }
336 };
337
338 let operator = self.operator.clone().into_iter();
339 let cast = self.cast.clone().into_iter();
340 let to_sql_config = match self.overridden() {
341 None => self.to_sql_config.clone(),
342 Some(content) => ToSqlConfig { content: Some(content), ..self.to_sql_config.clone() },
343 };
344
345 let lifetimes = self
346 .func
347 .sig
348 .generics
349 .params
350 .iter()
351 .filter_map(|generic| match generic {
352 syn::GenericParam::Lifetime(lt) => Some(lt),
354 _ => None, })
356 .collect::<Vec<_>>();
357 let hrtb = if lifetimes.is_empty() { None } else { Some(quote! { for<#(#lifetimes),*> }) };
358
359 let sql_graph_entity_fn_name = format_ident!("__pgrx_internals_fn_{}", ident);
360 quote_spanned! { self.func.sig.span() =>
361 #[no_mangle]
362 #[doc(hidden)]
363 #[allow(unknown_lints, clippy::no_mangle_with_rust_abi)]
364 pub extern "Rust" fn #sql_graph_entity_fn_name() -> ::pgrx::pgrx_sql_entity_graph::SqlGraphEntity {
365 extern crate alloc;
366 #[allow(unused_imports)]
367 use alloc::{vec, vec::Vec};
368 type FunctionPointer = #hrtb #unsafety fn(#( #input_types ),*) #return_type;
369 let submission = ::pgrx::pgrx_sql_entity_graph::PgExternEntity {
370 name: #name,
371 unaliased_name: stringify!(#ident),
372 module_path: core::module_path!(),
373 full_path: concat!(core::module_path!(), "::", stringify!(#ident)),
374 metadata: <FunctionPointer as ::pgrx::pgrx_sql_entity_graph::metadata::FunctionMetadata::<_>>::entity(),
375 fn_args: vec![#(#inputs_iter),*],
376 fn_return: #returns,
377 #[allow(clippy::or_fun_call)]
378 schema: None #( .unwrap_or_else(|| Some(#schema_iter)) )*,
379 file: file!(),
380 line: line!(),
381 extern_attrs: vec![#extern_attrs],
382 #[allow(clippy::or_fun_call)]
383 search_path: None #( .unwrap_or_else(|| Some(vec![#search_path])) )*,
384 #[allow(clippy::or_fun_call)]
385 operator: None #( .unwrap_or_else(|| Some(#operator)) )*,
386 cast: None #( .unwrap_or_else(|| Some(#cast)) )*,
387 to_sql_config: #to_sql_config,
388 };
389 ::pgrx::pgrx_sql_entity_graph::SqlGraphEntity::Function(submission)
390 }
391 }
392 }
393
394 pub fn wrapper_func(&self) -> Result<syn::ItemFn, syn::Error> {
395 let signature = &self.func.sig;
396 let func_name = &signature.ident;
397 let synthetic_ident_span = Span::mixed_site().located_at(signature.ident.span());
399 let fcinfo_ident = syn::Ident::new("fcinfo", synthetic_ident_span);
400 let mut lifetimes = signature
401 .generics
402 .lifetimes()
403 .cloned()
404 .collect::<syn::punctuated::Punctuated<_, Comma>>();
405 let fc_lt = lifetimes
408 .first()
409 .map(|lt_p| lt_p.lifetime.clone())
410 .filter(|lt| lt.ident != "static")
411 .unwrap_or(syn::Lifetime::new("'fcx", Span::mixed_site()));
412 let fc_ltparam = syn::LifetimeParam::new(fc_lt.clone());
414 if lifetimes.first() != Some(&fc_ltparam) {
415 lifetimes.insert(0, fc_ltparam)
416 }
417
418 let args = &self.inputs;
419 let arg_pats = args.iter().map(|v| format_ident!("{}_", &v.pat)).collect::<Vec<_>>();
421 let args_ident = proc_macro2::Ident::new("_args", Span::call_site());
422 let arg_fetches = arg_pats.iter().map(|pat| {
423 quote_spanned!{ pat.span() =>
424 let #pat = #args_ident.next_arg_unchecked().unwrap_or_else(|| panic!("unboxing {} argument failed", stringify!(#pat)));
425 }
426 }
427 );
428
429 match &self.returns {
430 Returning::None
431 | Returning::Type(_)
432 | Returning::SetOf { .. }
433 | Returning::Iterated { .. } => {
434 let ret_ty = match &signature.output {
435 syn::ReturnType::Default => syn::parse_quote! { () },
436 syn::ReturnType::Type(_, ret_ty) => ret_ty.clone(),
437 };
438 let wrapper_code = quote_spanned! { self.func.block.span() =>
439 fn _internal_wrapper<#lifetimes>(fcinfo: &mut ::pgrx::callconv::FcInfo<#fc_lt>) -> ::pgrx::datum::Datum<#fc_lt> {
440 #[allow(unused_unsafe)]
441 unsafe {
442 let call_flow = <#ret_ty as ::pgrx::callconv::RetAbi>::check_and_prepare(fcinfo);
443 let result = match call_flow {
444 ::pgrx::callconv::CallCx::WrappedFn(mcx) => {
445 let mut mcx = ::pgrx::PgMemoryContexts::For(mcx);
446 let #args_ident = &mut fcinfo.args();
447 let call_result = mcx.switch_to(|_| {
448 #(#arg_fetches)*
449 #func_name( #(#arg_pats),* )
450 });
451 ::pgrx::callconv::RetAbi::to_ret(call_result)
452 }
453 ::pgrx::callconv::CallCx::RestoreCx => <#ret_ty as ::pgrx::callconv::RetAbi>::ret_from_fcx(fcinfo),
454 };
455 unsafe { <#ret_ty as ::pgrx::callconv::RetAbi>::box_ret_in(fcinfo, result) }
456 }
457 }
458 let datum = unsafe {
460 ::pgrx::pg_sys::submodules::panic::pgrx_extern_c_guard(|| {
461 let mut fcinfo = ::pgrx::callconv::FcInfo::from_ptr(#fcinfo_ident);
462 _internal_wrapper(&mut fcinfo)
463 })
464 };
465 datum.sans_lifetime()
466 };
467 finfo_v1_extern_c(&self.func, fcinfo_ident, wrapper_code)
468 }
469 }
470 }
471}
472
473trait LastIdent {
474 fn filter_last_ident(&self, id: &str) -> Option<&syn::PathSegment>;
475 #[inline]
476 fn last_ident_is(&self, id: &str) -> bool {
477 self.filter_last_ident(id).is_some()
478 }
479}
480
481impl LastIdent for syn::Type {
482 #[inline]
483 fn filter_last_ident(&self, id: &str) -> Option<&syn::PathSegment> {
484 let syn::Type::Path(syn::TypePath { path, .. }) = self else { return None };
485 path.filter_last_ident(id)
486 }
487}
488impl LastIdent for syn::TypePath {
489 #[inline]
490 fn filter_last_ident(&self, id: &str) -> Option<&syn::PathSegment> {
491 self.path.filter_last_ident(id)
492 }
493}
494impl LastIdent for syn::Path {
495 #[inline]
496 fn filter_last_ident(&self, id: &str) -> Option<&syn::PathSegment> {
497 self.segments.filter_last_ident(id)
498 }
499}
500impl<P> LastIdent for Punctuated<syn::PathSegment, P> {
501 #[inline]
502 fn filter_last_ident(&self, id: &str) -> Option<&syn::PathSegment> {
503 self.last().filter(|segment| segment.ident == id)
504 }
505}
506
507impl ToEntityGraphTokens for PgExtern {
508 fn to_entity_graph_tokens(&self) -> TokenStream2 {
509 self.entity_tokens()
510 }
511}
512
513impl ToRustCodeTokens for PgExtern {
514 fn to_rust_code_tokens(&self) -> TokenStream2 {
515 let original_func = &self.func;
516 let wrapper_func = self.wrapper_func().unwrap();
517 let finfo_tokens = finfo_v1_tokens(wrapper_func.sig.ident.clone()).unwrap();
518
519 quote_spanned! { self.func.sig.span() =>
520 #original_func
521 #wrapper_func
522 #finfo_tokens
523 }
524 }
525}
526
527impl Parse for CodeEnrichment<PgExtern> {
528 fn parse(input: ParseStream) -> Result<Self, syn::Error> {
529 let parser = Punctuated::<Attribute, Token![,]>::parse_terminated;
530 let punctuated_attrs = input.call(parser).ok().unwrap_or_default();
531 let attrs = punctuated_attrs.into_pairs().map(|pair| pair.into_value());
532 PgExtern::new(quote! {#(#attrs)*}, input.parse()?)
533 }
534}