use syn::Token;
use syn::parse::{Parse, ParseStream};
use syn::punctuated::Punctuated;
pub struct PgrxAttribute {
pub args: Vec<PgrxArg>,
}
impl Parse for PgrxAttribute {
fn parse(input: ParseStream<'_>) -> syn::Result<Self> {
let parser = Punctuated::<PgrxArg, Token![,]>::parse_terminated;
let punctuated = input.call(parser)?;
let args = punctuated.into_pairs().map(|p| p.into_value()).collect::<Vec<_>>();
Ok(Self { args })
}
}
#[derive(Debug)]
pub enum PgrxArg {
NameValue(NameValueArg),
}
impl Parse for PgrxArg {
#[track_caller]
fn parse(input: ParseStream<'_>) -> syn::Result<Self> {
let path = input.parse::<syn::Path>()?;
if input.parse::<Token![=]>().is_ok() {
Ok(Self::NameValue(NameValueArg { path, value: input.parse()? }))
} else {
Err(input.error("unsupported argument to #[pgrx] in this context"))
}
}
}
#[derive(Debug)]
pub struct NameValueArg {
pub path: syn::Path,
pub value: ArgValue,
}
#[derive(Debug)]
pub enum ArgValue {
Path(syn::Path),
Lit(syn::Lit),
}
impl Parse for ArgValue {
fn parse(input: ParseStream<'_>) -> syn::Result<Self> {
if input.peek(syn::Lit) {
return Ok(Self::Lit(input.parse()?));
}
Ok(Self::Path(input.parse()?))
}
}