2021-10-17 17:40:22 +00:00
|
|
|
// Copyright 2021 Oxide Computer Company
|
|
|
|
|
|
|
|
use std::path::Path;
|
|
|
|
|
|
|
|
use openapiv3::OpenAPI;
|
|
|
|
use proc_macro::TokenStream;
|
|
|
|
use progenitor_impl::Generator;
|
2021-11-17 02:00:55 +00:00
|
|
|
use quote::{quote, ToTokens};
|
2021-10-29 14:16:39 +00:00
|
|
|
use syn::{
|
|
|
|
parse::{Parse, ParseStream},
|
|
|
|
ExprClosure, LitStr, Token,
|
|
|
|
};
|
2021-10-17 17:40:22 +00:00
|
|
|
|
|
|
|
#[proc_macro]
|
|
|
|
pub fn generate_api(item: TokenStream) -> TokenStream {
|
|
|
|
match do_generate_api(item) {
|
|
|
|
Err(err) => err.to_compile_error().into(),
|
|
|
|
Ok(out) => out,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-29 14:16:39 +00:00
|
|
|
struct Settings {
|
|
|
|
file: LitStr,
|
|
|
|
inner: Option<proc_macro2::TokenStream>,
|
|
|
|
pre: Option<proc_macro2::TokenStream>,
|
|
|
|
post: Option<proc_macro2::TokenStream>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Parse for Settings {
|
|
|
|
fn parse(input: ParseStream) -> Result<Self, syn::Error> {
|
|
|
|
let file = input.parse::<LitStr>()?;
|
|
|
|
let inner = parse_inner(input)?;
|
|
|
|
let pre = parse_hook(input)?;
|
|
|
|
let post = parse_hook(input)?;
|
|
|
|
|
|
|
|
// Optional trailing comma.
|
|
|
|
if input.peek(Token!(,)) {
|
|
|
|
let _ = input.parse::<Token!(,)>();
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Settings {
|
|
|
|
file,
|
|
|
|
inner,
|
|
|
|
pre,
|
|
|
|
post,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_inner(
|
|
|
|
input: ParseStream,
|
|
|
|
) -> Result<Option<proc_macro2::TokenStream>, syn::Error> {
|
|
|
|
if input.is_empty() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
let _: Token!(,) = input.parse()?;
|
|
|
|
if input.is_empty() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
Ok(Some(input.parse::<syn::Type>()?.to_token_stream()))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_hook(
|
|
|
|
input: ParseStream,
|
|
|
|
) -> Result<Option<proc_macro2::TokenStream>, syn::Error> {
|
|
|
|
if input.is_empty() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
let _: Token!(,) = input.parse()?;
|
|
|
|
if input.is_empty() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
if let Ok(closure) = input.parse::<ExprClosure>() {
|
|
|
|
Ok(Some(closure.to_token_stream()))
|
|
|
|
} else {
|
|
|
|
Ok(Some(input.parse::<syn::Path>()?.to_token_stream()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-17 17:40:22 +00:00
|
|
|
fn do_generate_api(item: TokenStream) -> Result<TokenStream, syn::Error> {
|
2021-10-29 14:16:39 +00:00
|
|
|
let Settings {
|
|
|
|
file,
|
|
|
|
inner,
|
|
|
|
pre,
|
|
|
|
post,
|
|
|
|
} = syn::parse::<Settings>(item)?;
|
2021-10-17 17:40:22 +00:00
|
|
|
let dir = std::env::var("CARGO_MANIFEST_DIR").map_or_else(
|
|
|
|
|_| std::env::current_dir().unwrap(),
|
|
|
|
|s| Path::new(&s).to_path_buf(),
|
|
|
|
);
|
|
|
|
|
2021-10-29 14:16:39 +00:00
|
|
|
let path = dir.join(file.value());
|
2021-11-17 02:00:55 +00:00
|
|
|
let path_str = path.to_string_lossy();
|
2021-10-17 17:40:22 +00:00
|
|
|
|
2021-11-17 02:00:55 +00:00
|
|
|
let spec: OpenAPI =
|
|
|
|
serde_json::from_reader(std::fs::File::open(&path).map_err(|e| {
|
|
|
|
syn::Error::new(
|
|
|
|
file.span(),
|
|
|
|
format!("couldn't read file {}: {}", path_str, e.to_string()),
|
|
|
|
)
|
|
|
|
})?)
|
|
|
|
.map_err(|e| {
|
|
|
|
syn::Error::new(
|
|
|
|
file.span(),
|
|
|
|
format!("failed to parse {}: {}", path_str, e.to_string()),
|
|
|
|
)
|
|
|
|
})?;
|
2021-10-17 17:40:22 +00:00
|
|
|
|
|
|
|
let mut builder = Generator::new();
|
2021-10-29 14:16:39 +00:00
|
|
|
inner.map(|inner_type| builder.with_inner_type(inner_type));
|
|
|
|
pre.map(|pre_hook| builder.with_pre_hook(pre_hook));
|
|
|
|
post.map(|post_hook| builder.with_post_hook(post_hook));
|
2021-11-17 02:00:55 +00:00
|
|
|
let code = builder.generate_tokens(&spec).map_err(|e| {
|
2021-10-17 17:40:22 +00:00
|
|
|
syn::Error::new(
|
2021-10-29 14:16:39 +00:00
|
|
|
file.span(),
|
|
|
|
format!("generation error for {}: {}", file.value(), e.to_string()),
|
2021-10-17 17:40:22 +00:00
|
|
|
)
|
|
|
|
})?;
|
|
|
|
|
2021-11-17 02:00:55 +00:00
|
|
|
let output = quote! {
|
|
|
|
#code
|
|
|
|
|
|
|
|
// Force a rebuild when the given file is modified.
|
|
|
|
const _: &str = include_str!(#path_str);
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(output.into())
|
2021-10-17 17:40:22 +00:00
|
|
|
}
|