#![warn(clippy::pedantic)]
#![warn(rust_2018_idioms)]
#![allow(clippy::expl_impl_clone_on_copy)]
#![cfg_attr(nightly, feature(proc_macro_span))]
#[cfg(nightly)]
mod backtrace;
mod binread;
mod binrw_attr;
mod binwrite;
mod codegen;
mod named_args;
mod parser;
use crate::{
codegen::typed_builder::{Builder, BuilderField, BuilderFieldKind},
named_args::NamedArgAttr,
};
use proc_macro::TokenStream;
use syn::{parse_macro_input, spanned::Spanned, DeriveInput};
#[proc_macro_derive(BinRead, attributes(binread, br, brw))]
#[cfg(not(tarpaulin_include))]
pub fn derive_binread_trait(input: TokenStream) -> TokenStream {
binread::derive_from_input(&parse_macro_input!(input as DeriveInput), true)
.1
.into()
}
#[proc_macro_attribute]
#[cfg(not(tarpaulin_include))]
pub fn binread(_: TokenStream, input: TokenStream) -> TokenStream {
binread::derive_from_attribute(parse_macro_input!(input as DeriveInput)).into()
}
#[proc_macro_derive(BinWrite, attributes(binwrite, bw, brw))]
#[cfg(not(tarpaulin_include))]
pub fn derive_binwrite_trait(input: TokenStream) -> TokenStream {
binwrite::derive_from_input(&parse_macro_input!(input as DeriveInput), true)
.1
.into()
}
#[proc_macro_attribute]
#[cfg(not(tarpaulin_include))]
pub fn binwrite(_: TokenStream, input: TokenStream) -> TokenStream {
binwrite::derive_from_attribute(parse_macro_input!(input as DeriveInput)).into()
}
#[proc_macro_attribute]
#[cfg(not(tarpaulin_include))]
pub fn binrw(_: TokenStream, input: TokenStream) -> TokenStream {
binrw_attr::derive_from_attribute(parse_macro_input!(input as DeriveInput)).into()
}
fn binrw_named_args(input: DeriveInput) -> proc_macro2::TokenStream {
let fields = match match input.data {
syn::Data::Struct(s) => s
.fields
.iter()
.map(|field| {
let attrs: Vec<NamedArgAttr> = field
.attrs
.iter()
.filter_map(|attr| {
let is_named_args = attr
.path
.get_ident()
.map_or(false, |ident| ident == "named_args");
if is_named_args {
attr.parse_args().ok()
} else {
None
}
})
.collect();
let kind = if attrs
.iter()
.any(|attr| matches!(attr, NamedArgAttr::TryOptional))
{
BuilderFieldKind::TryOptional
} else if let Some(NamedArgAttr::Default(default)) = attrs
.iter()
.find(|attr| matches!(attr, NamedArgAttr::Default(_)))
{
BuilderFieldKind::Optional {
default: Box::new(default.clone()),
}
} else {
BuilderFieldKind::Required
};
Ok(BuilderField {
kind,
name: match field.ident.as_ref() {
Some(ident) => ident.clone(),
None => {
return Err(syn::Error::new(
field.span(),
"must not be a tuple-style field",
))
}
},
ty: field.ty.clone(),
})
})
.collect::<Result<Vec<_>, syn::Error>>(),
_ => return syn::Error::new(input.span(), "only structs are supported").to_compile_error(),
} {
Ok(fields) => fields,
Err(err) => return err.into_compile_error(),
};
let generics: Vec<_> = input.generics.params.iter().cloned().collect();
Builder {
result_name: &input.ident,
builder_name: "e::format_ident!("{}Builder", input.ident),
fields: &fields,
generics: &generics,
vis: &input.vis,
}
.generate(false)
}
#[cfg(not(tarpaulin_include))]
#[proc_macro_derive(BinrwNamedArgs, attributes(named_args))]
pub fn derive_binrw_named_args(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as DeriveInput);
binrw_named_args(input).into()
}
#[cfg(test)]
#[cfg(tarpaulin)]
#[test]
fn derive_code_coverage_for_tarpaulin() {
use runtime_macros_derive::emulate_derive_expansion_fallible;
use std::{env, fs};
let derive_tests_folder = env::current_dir()
.unwrap()
.join("..")
.join("binrw")
.join("tests")
.join("derive");
let mut run_success = true;
for entry in fs::read_dir(derive_tests_folder).unwrap() {
let entry = entry.unwrap();
if entry.file_type().unwrap().is_file() {
let file = fs::File::open(entry.path()).unwrap();
if emulate_derive_expansion_fallible(file, "BinRead", |input| {
binread::derive_from_input(&input, true).1
})
.is_err()
{
run_success = false;
}
}
}
assert!(run_success)
}
#[cfg(test)]
#[cfg(tarpaulin)]
#[test]
fn derive_binwrite_code_coverage_for_tarpaulin() {
use runtime_macros_derive::emulate_derive_expansion_fallible;
use std::{env, fs};
let derive_tests_folder = env::current_dir()
.unwrap()
.join("..")
.join("binrw/tests/derive/write");
let mut run_success = true;
for entry in fs::read_dir(derive_tests_folder).unwrap() {
let entry = entry.unwrap();
if entry.file_type().unwrap().is_file() {
let file = fs::File::open(entry.path()).unwrap();
if emulate_derive_expansion_fallible(file, "BinWrite", |input| {
binwrite::derive_from_input(&input, true).1
})
.is_err()
{
run_success = false;
}
}
}
assert!(run_success)
}
#[cfg(test)]
#[cfg(tarpaulin)]
#[test]
fn derive_named_args_code_coverage_for_tarpaulin() {
use runtime_macros_derive::emulate_derive_expansion_fallible;
use std::fs;
let file = fs::File::open("../binrw/tests/builder.rs").unwrap();
emulate_derive_expansion_fallible(file, "BinrwNamedArgs", |input| binrw_named_args(input))
.unwrap();
}