Files
pixiv_downloader/proc_macros/proc_macros.rs
2022-07-08 04:43:29 +00:00

261 lines
8.3 KiB
Rust

use proc_macro::TokenStream;
use quote::quote;
use syn::parse::Parse;
use syn::parse_macro_input;
use syn::token;
use syn::Block;
use syn::Expr;
use syn::Ident;
use syn::ItemFn;
use syn::Lit;
use syn::LitInt;
use syn::LitStr;
#[proc_macro]
pub fn define_struct_reader_fn(item: TokenStream) -> TokenStream {
let i = parse_macro_input!(item as Ident);
let lefname = format!("read_le_{}", i);
let lefname = Ident::new(&lefname, i.span());
let befname = format!("read_be_{}", i);
let befname = Ident::new(&befname, i.span());
let stream = quote! {
#[doc = concat!("Read [", stringify!(#i), "] in little endian.")]
fn #lefname(&mut self) -> Result<#i, Self::Error>;
#[doc = concat!("Read [", stringify!(#i), "] in big endian.")]
fn #befname(&mut self) -> Result<#i, Self::Error>;
};
stream.into()
}
#[proc_macro]
pub fn impl_struct_reader_read(item: TokenStream) -> TokenStream {
let i = parse_macro_input!(item as Ident);
let lefname = format!("read_le_{}", i);
let lefname = Ident::new(&lefname, i.span());
let befname = format!("read_be_{}", i);
let befname = Ident::new(&befname, i.span());
let stream = quote! {
fn #lefname(&mut self) -> Result<#i, Self::Error> {
let mut buf = [0u8; std::mem::size_of::<#i>()];
self.read_exact(&mut buf)?;
Ok(<#i>::from_le_bytes(buf))
}
fn #befname(&mut self) -> Result<#i, Self::Error> {
let mut buf = [0u8; std::mem::size_of::<#i>()];
self.read_exact(&mut buf)?;
Ok(<#i>::from_be_bytes(buf))
}
};
stream.into()
}
#[proc_macro]
pub fn define_struct_writer_fn(item: TokenStream) -> TokenStream {
let i = parse_macro_input!(item as Ident);
let lefname = format!("write_le_{}", i);
let lefname = Ident::new(&lefname, i.span());
let befname = format!("write_be_{}", i);
let befname = Ident::new(&befname, i.span());
let stream = quote! {
#[doc = concat!("Write [", stringify!(#i), "] in little endian.")]
fn #lefname(&mut self, data: #i) -> Result<(), Self::Error>;
#[doc = concat!("Write [", stringify!(#i), "] in big endian.")]
fn #befname(&mut self, data: #i) -> Result<(), Self::Error>;
};
stream.into()
}
#[proc_macro]
pub fn impl_struct_writer_write(item: TokenStream) -> TokenStream {
let i = parse_macro_input!(item as Ident);
let lefname = format!("write_le_{}", i);
let lefname = Ident::new(&lefname, i.span());
let befname = format!("write_be_{}", i);
let befname = Ident::new(&befname, i.span());
let stream = quote! {
#[inline]
fn #lefname(&mut self, data: #i) -> Result<(), Self::Error> {
self.write_all(&data.to_le_bytes())
}
#[inline]
fn #befname(&mut self, data: #i) -> Result<(), Self::Error> {
self.write_all(&data.to_be_bytes())
}
};
stream.into()
}
#[proc_macro_attribute]
pub fn async_timeout_test(attr: TokenStream, item: TokenStream) -> TokenStream {
let ItemFn {
attrs,
vis,
sig,
block,
} = parse_macro_input!(item as ItemFn);
let dura = parse_duration::parse(attr.to_string().as_str()).unwrap();
let secs = LitInt::new(format!("{}", dura.as_secs()).as_str(), sig.ident.span());
let nanos = LitInt::new(
format!("{}", dura.subsec_nanos()).as_str(),
sig.ident.span(),
);
let stmts = &block.stmts;
let stream = quote! {
#(#attrs)* #vis #sig {
let dura = std::time::Duration::new(#secs, #nanos);
let f = async {
#(#stmts)*
};
tokio::time::timeout(dura, f).await.unwrap();
}
};
stream.into()
}
struct FanboxApiTest {
pub name: Ident,
pub block: Block,
}
impl Parse for FanboxApiTest {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let name = Ident::parse(input)?;
token::Comma::parse(input)?;
let block = syn::Block::parse(input)?;
Ok(Self { name, block })
}
}
#[proc_macro]
pub fn fanbox_api_test(item: TokenStream) -> TokenStream {
let FanboxApiTest { name, block } = parse_macro_input!(item as FanboxApiTest);
let stmts = block.stmts;
let stream = quote! {
#[proc_macros::async_timeout_test(120s)]
#[tokio::test(flavor = "multi_thread")]
async fn #name() {
match std::env::var("FANBOX_COOKIES_FILE") {
Ok(path) => {
let client = crate::fanbox_api::FanboxClient::new();
if !client.init(Some(path)) {
panic!("Failed to initiailze the client.");
}
if !client.check_login().await {
println!("The client is not logined. Skip test.");
return;
}
#(#stmts)*
}
Err(_) => {
println!("No cookies files specified, skip test.")
}
}
}
};
stream.into()
}
struct FanboxApiQuickTest {
pub name: Ident,
pub expr: Expr,
pub errmsg: LitStr,
}
impl Parse for FanboxApiQuickTest {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let name = Ident::parse(input)?;
token::Comma::parse(input)?;
let expr = Expr::parse(input)?;
token::Comma::parse(input)?;
let errmsg = Lit::parse(input)?;
match errmsg {
Lit::Str(errmsg) => Ok(Self { name, expr, errmsg }),
_ => Err(syn::Error::new(errmsg.span(), "Failed to parse string.")),
}
}
}
#[proc_macro]
pub fn fanbox_api_quick_test(item: TokenStream) -> TokenStream {
let FanboxApiQuickTest { name, expr, errmsg } = parse_macro_input!(item as FanboxApiQuickTest);
let stream = quote! {
#[proc_macros::async_timeout_test(120s)]
#[tokio::test(flavor = "multi_thread")]
async fn #name() {
match std::env::var("FANBOX_COOKIES_FILE") {
Ok(path) => {
let client = crate::fanbox_api::FanboxClient::new();
if !client.init(Some(path)) {
panic!("Failed to initiailze the client.");
}
if !client.check_login().await {
println!("The client is not logined. Skip test.");
return;
}
match #expr.await {
Some(data) => {
println!("{:?}", data);
}
None => {
panic!("{}", #errmsg);
}
}
}
Err(_) => {
println!("No cookies files specified, skip test.")
}
}
}
};
stream.into()
}
struct FilterHttpMethods {
pub req: Ident,
pub typ: Expr,
pub methods: Vec<Ident>,
}
impl Parse for FilterHttpMethods {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let req = Ident::parse(input)?;
token::Comma::parse(input)?;
let typ = Expr::parse(input)?;
let mut methods = Vec::new();
loop {
if input.cursor().eof() {
break;
}
token::Comma::parse(input)?;
let method = Ident::parse(input)?;
methods.push(method);
}
Ok(Self { req, typ, methods })
}
}
/// Filter http methods.
///
/// `request, 405 body, [method [, method [, method ...]]]`
#[proc_macro]
pub fn filter_http_methods(item: TokenStream) -> TokenStream {
let FilterHttpMethods { req, typ, methods } = parse_macro_input!(item as FilterHttpMethods);
let mut header_value = Vec::new();
let mut streams = Vec::new();
for method in methods {
header_value.push(method.to_string());
streams.push(quote!(&hyper::Method::#method => {}));
}
let allow_header = header_value.join(", ");
let allow_header = LitStr::new(allow_header.as_str(), req.span());
let stream = quote! {
match #req.method() {
#(#streams)*
_ => {
return Ok(hyper::Response::builder().status(405).header("Allow", #allow_header).body(#typ).unwrap())
}
}
};
stream.into()
}