bok/bok-macro/src/pkgs.rs

884 lines
31 KiB
Rust
Raw Normal View History

/*
* Copyright 2024 Luca Fulchir <luca.fulchir@runesauth.com>
*
* Licensed under the Apache License, Version 2.0 with LLVM exception (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License and of the exception at
*
* http://www.apache.org/licenses/LICENSE-2.0
* https://spdx.org/licenses/LLVM-exception.html
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
use ::proc_macro::TokenStream;
use ::quote::quote;
use ::syn::{parse::Parser, parse_macro_input, DeriveInput};
use syn::{parse_quote, spanned::Spanned};
pub(crate) fn pkg_fn_to_code(
_attrs: TokenStream,
input: TokenStream,
) -> TokenStream {
let ast_orig: ::syn::ItemFn = parse_macro_input!(input as ::syn::ItemFn);
let mut ast_new = ast_orig.clone();
let new_fn_name = quote::format_ident!("{}_code", ast_orig.sig.ident);
ast_new.sig.ident = quote::format_ident!("{}_code", new_fn_name);
use ::quote::ToTokens;
let fn_block = ast_orig.block.to_token_stream();
let asdf = quote! {
#ast_orig
fn #new_fn_name (&self) -> ::proc_macro2::TokenStream {
::quote::quote! #fn_block
}
};
asdf.into()
}
pub(crate) fn package(
attrs: TokenStream,
input: TokenStream,
__source_path: TokenStream,
) -> TokenStream {
use ::syn::{spanned::Spanned, Fields, ItemStruct};
let local = parse_macro_input!(input as ItemStruct);
let local_span = local.span();
let Fields::Named(local_fields) = local.fields else {
use ::syn::spanned::Spanned;
return ::syn::Error::new(
local.fields.span(),
"unnamed fields are not supported",
)
.to_compile_error()
.into();
};
let local_attrs = local.attrs.iter();
let generics = local.generics;
let ident = local.ident;
let vis = local.vis;
let source_path = parse_macro_input!(__source_path as ::syn::Path);
let base = parse_macro_input!(attrs as ItemStruct);
let Fields::Named(base_fields) = &base.fields else {
use ::syn::spanned::Spanned;
return ::syn::Error::new(
base.fields.span(),
"unnamed fields are not supported",
)
.to_compile_error()
.into();
};
// make sure the 'version' field is the first after the user's fields
// we will use as a separator from the other imported fields
let mut base_fields_extra = Vec::with_capacity(base.fields.len());
match base_fields.named.iter().find(|&x| match &x.ident {
Some(id_local) => id_local == "version",
_ => false,
}) {
Some(_) => {
let pkg_version = ::syn::Field::parse_named
.parse2(quote! {
version: ::bok::Version
})
.unwrap();
let base_marker = ::syn::Field::parse_named
.parse2(quote! {
_bok_base: ::std::marker::PhantomData<#source_path>
})
.unwrap();
base_fields_extra.push(pkg_version);
base_fields_extra.push(base_marker);
}
None => {
return ::syn::Error::new(
local_span,
"pkg: \"".to_owned()
+ &ident.to_string()
+ "\" base package \""
+ &base.ident.to_string()
+ "\" does not have 'version' field",
)
.to_compile_error()
.into();
}
}
for f in base_fields.named.iter() {
let Some(id) = &f.ident else { continue };
// don't add 'version' again
if id.to_string() == "version" || id.to_string() == "_bok_base" {
continue;
}
// compiler error if you try to extend a package while
// reusing its fields names
if let Some(repeated) = local_fields.named.iter().find(|&x| {
let Some(id_local) = &x.ident else {
return false;
};
id_local.to_string() == id.to_string()
}) {
return ::syn::Error::new(
local_span,
"pkg: \"".to_owned()
+ &ident.to_string()
+ "\" field \""
+ &repeated.ident.as_ref().unwrap().to_string()
+ "\" is already in package "
+ &base.ident.to_string()
+ "\" or its base",
)
.to_compile_error()
.into();
}
base_fields_extra.push(f.clone());
}
let mut all_fields = Vec::new();
all_fields.extend(local_fields.named.iter());
all_fields.extend(base_fields_extra.iter());
quote! {
#(#local_attrs)
*
#[::macro_magic::export_tokens]
#[derive(::bok::Package, ::std::fmt::Debug, Clone)]
#vis struct #ident<#generics> {
#(#all_fields),
*
}
}
.into()
}
pub(crate) fn deps_build(
attrs: TokenStream,
input: TokenStream,
) -> TokenStream {
let packages = parse_macro_input!(attrs as crate::PathList);
let local = parse_macro_input!(input as ::syn::ItemStruct);
let ::syn::Fields::Named(local_fields) = local.fields else {
use ::syn::spanned::Spanned;
return ::syn::Error::new(
local.fields.span(),
"unnamed fields are not supported",
)
.to_compile_error()
.into();
};
let local_attrs = local.attrs.iter();
let local_fields = local_fields.named.iter();
let generics = local.generics;
let ident = local.ident;
let vis = local.vis;
let pkg_trait = quote::format_ident!("BokDeps{}", ident);
let deps = packages
.0
.iter()
.map(|x| x.segments.last().unwrap().ident.clone())
.collect::<Vec<::syn::Ident>>();
quote! {
#(#local_attrs)
*
#vis struct #ident<#generics> {
#(#local_fields),
*
}
#[::macro_magic::export_tokens(#pkg_trait)]
pub trait #pkg_trait {
#(fn #deps(&self) -> ::std::boxed::Box<dyn ::bok::Pkg>;)
*
}
}
.into()
}
// package_impl has 3 stages, and uses macro_magic import for the last two:
// * called without parameters => add default ::bok::Pkg functions, then add the
// same macro with the package type as argument
// * called with the package type as argument => get the base type, then add the
// same macro with the base type IMPL as argument
// * called with the a package impl as argument, copy the non-implemented
// functions from the base
pub(crate) fn package_impl(
_attrs: TokenStream,
input: TokenStream,
) -> TokenStream {
let ast = parse_macro_input!(input as ::syn::ItemImpl);
let name_pkg = match &*ast.self_ty {
::syn::Type::Path(tp) => match tp.path.get_ident() {
Some(id) => id.clone(),
_ => {
return ::syn::Error::new(
ast.span(),
"package_impl: type::Path but no ident",
)
.to_compile_error()
.into()
}
},
_ => {
return ::syn::Error::new(
ast.span(),
"package_impl: not a type::Path",
)
.to_compile_error()
.into()
}
};
let full_path_name = proc_macro2::Ident::new(
&("_bok_pkg_".to_owned() + &name_pkg.to_string()),
proc_macro2::Span::call_site(),
);
quote! {
use ::bok_macro::package_impl_base;
#[::bok_macro::package_impl_base(#name_pkg)]
#[::macro_magic::export_tokens(#full_path_name)]
#ast
}
.into()
}
// get the actual base type and re-call the macro with the impl symbols
pub(crate) fn package_impl_base(
attrs: TokenStream,
input: TokenStream,
) -> TokenStream {
// This is a two-phase macro
// * The first time this is called the argument will be the same path as the
// struct that implements ::bok::Pkg
// * The macro will be rewritten with the base package name that is extended
// by the package
// And now that we have the proper base name we can add the missing
// methods from the base
let mut ast = parse_macro_input!(input as ::syn::ItemImpl);
let name_pkg = match &*ast.self_ty {
::syn::Type::Path(tp) => match tp.path.get_ident() {
Some(id) => id.clone(),
_ => panic!("::bok::package_impl expected path ident"),
},
_ => panic!("::bok::package_impl expected path"),
};
let Ok(syntax_tree) = syn::parse::<::syn::ItemStruct>(attrs.clone()) else {
return package_impl_base_add(attrs, &mut ast);
};
// we were passed an exported struct. first phase.
if syntax_tree.ident.to_string() != name_pkg.to_string() {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" with wrong initial name"
+ &syntax_tree.ident.to_string()
+ "\" ",
)
.to_compile_error()
.into();
}
// go through the struct definition, extract the base type
// from the `_bok_base` phantom data marker
let base_path = match syntax_tree.fields.iter().find(|&x| {
let Some(id) = &x.ident else {
return false;
};
return id.to_string() == "_bok_base";
}) {
Some(field) => match &field.ty {
::syn::Type::Path(tp) => {
if tp.path.segments.len() != 3
|| tp.path.segments[0].ident.to_string() != "std"
|| tp.path.segments[1].ident.to_string() != "marker"
|| tp.path.segments[2].ident.to_string() != "PhantomData"
{
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" base is not PhantomData, "
+ "what are you doing?",
)
.to_compile_error()
.into();
}
let ::syn::PathArguments::AngleBracketed(psa) =
&tp.path.segments[2].arguments
else {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" base PhantomData has no arguments "
+ "what are you doing?",
)
.to_compile_error()
.into();
};
if psa.args.len() != 1 {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" base PhantomData has too many arguments "
+ "what are you doing?",
)
.to_compile_error()
.into();
}
let ::syn::GenericArgument::Type(::syn::Type::Path(
base_pkg_path,
)) = &psa.args[0]
else {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" base PhantomData has wrong argument type "
+ "what are you doing?",
)
.to_compile_error()
.into();
};
base_pkg_path.path.clone()
}
_ => {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" base of unknown type",
)
.to_compile_error()
.into();
}
},
None => {
return ::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ &name_pkg.to_string()
+ "\" can't find base",
)
.to_compile_error()
.into();
}
};
// now, `base_path` is an absolute path.
// but it does not mean that it is in our crate
// we need to maintain the whole path and change only the last ident,
// to the one generated by us instead of by macro_magic
// now `base_type` is the base struct. but what we actually want is
// to import the `impl ::bok::Pkg` for `base_type`
let mut exported_impl_name = base_path.clone();
let last_segment_n = exported_impl_name.segments.len() - 1;
let last_segment = &mut exported_impl_name.segments[last_segment_n];
last_segment.ident =
::quote::format_ident!("_bok_pkg_{}", last_segment.ident);
return quote! {
#[::bok_macro::package_impl_base(#exported_impl_name)]
#ast
}
.into();
}
// implements ::bok::Pkg:
// * add base methods
// * copy the non-implemented methods from the base
pub(crate) fn package_impl_base_add(
attrs: TokenStream,
ast: &mut ::syn::ItemImpl,
) -> TokenStream {
let name_pkg = match &*ast.self_ty {
::syn::Type::Path(tp) => match tp.path.get_ident() {
Some(id) => id.clone(),
_ => panic!("::bok::package_impl expected path ident"),
},
_ => panic!("::bok::package_impl expected path"),
};
let base_ast = parse_macro_input!(attrs as ::syn::ItemImpl);
let name: ::syn::ImplItem = ::syn::parse_quote! {
fn name(&self) -> ::bok::PkgName {
(module_path!().to_owned() + "::" + ::std::stringify!(#name_pkg)).as_str().into()
}
};
let path: ::syn::ImplItem = ::syn::parse_quote! {
fn path(&self) -> ::bok::Path<::bok::PkgName> {
(module_path!().to_owned() + "::" + ::std::stringify!(#name_pkg)).as_str().into()
}
};
let version: ::syn::ImplItem = ::syn::parse_quote! {
fn version(&self) -> ::bok::Version {
self.version.clone()
}
};
let any: ::syn::ImplItem = ::syn::parse_quote! {
fn as_any(&self) -> &dyn ::std::any::Any {
self
}
};
let any_mut: ::syn::ImplItem = ::syn::parse_quote! {
fn as_any_mut(&mut self) -> &mut dyn ::std::any::Any {
self
}
};
let hash: ::syn::ImplItem = ::syn::parse_quote! {
fn hash(&self) -> ::bok::Hash {
use ::sha3::{Digest};
let mut hasher = ::sha3::Sha3_256::new();
let mut pkg_vars = String::new();
use ::std::fmt::Write;
write!(&mut pkg_vars,
"{:?}",
self,
).ok();
hasher.update(&pkg_vars);
hasher.update(self.to_string());
let hash = hasher.finalize();
::bok::Hash::Sha3(hash)
}
};
let hash_code: ::syn::ImplItem = ::syn::parse_quote! {
fn hash_code(&self) -> ::bok::Hash {
use ::sha3::{Digest};
let mut hasher = ::sha3::Sha3_256::new();
hasher.update(self.to_string());
let hash = hasher.finalize();
::bok::Hash::Sha3(hash)
}
};
for mut impl_item in ast.items.iter_mut() {
if let ::syn::ImplItem::Fn(ref mut fn_impl) = &mut impl_item {
if ["prepare", "configure", "build", "check", "install"]
.iter()
.find(|&f| *f == fn_impl.sig.ident.to_string().as_str())
.is_none()
{
continue;
}
let fn_with_code: ::syn::ImplItemFn = ::syn::parse_quote! {
#[::bok::pkg_fn_to_code]
#fn_impl
};
*fn_impl = fn_with_code;
} else {
panic!("wtf");
}
}
ast.items.push(name);
ast.items.push(path);
ast.items.push(version);
ast.items.push(any);
ast.items.push(any_mut);
ast.items.push(hash);
ast.items.push(hash_code);
fn maybe_add_fn(
name_pkg: &str,
ast: &mut ::syn::ItemImpl,
base_ast: &::syn::ItemImpl,
name: &str,
) -> Result<(), TokenStream> {
let Some(base_fn) = base_ast.items.iter().find(|&f| {
let ::syn::ImplItem::Fn(func) = f else {
return false;
};
return func.sig.ident.to_string() == name;
}) else {
return Err(::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ name_pkg
+ "\" maybe_add_fn can't find "
+ name
+ " on base",
)
.to_compile_error()
.into());
};
let ::syn::ImplItem::Fn(mut base_fn_no_code) = base_fn.clone() else {
return Err(::syn::Error::new(
ast.span(),
"package_impl: called on \"".to_owned()
+ name_pkg
+ "\" what was Fn is not anymore",
)
.to_compile_error()
.into());
};
base_fn_no_code.attrs.retain_mut(|a| {
let ::syn::Meta::Path(mp) = &a.meta else {
return true;
};
if mp.segments.len() == 2
&& mp.segments[0].ident.to_string() == "bok"
&& mp.segments[1].ident.to_string() == "pkg_fn_to_code"
{
return false;
}
return true;
});
if ast
.items
.iter()
.find(|&f| {
let ::syn::ImplItem::Fn(func) = f else {
return false;
};
return func.sig.ident.to_string() == name;
})
.is_none()
{
let new_fn: ::syn::ImplItem = parse_quote!(
#[::bok::pkg_fn_to_code]
#base_fn_no_code
);
ast.items.push(new_fn);
}
Ok(())
}
for name in
["prepare", "configure", "build", "check", "install"].into_iter()
{
let Err(e) = maybe_add_fn(&name_pkg.to_string(), ast, &base_ast, name)
else {
continue;
};
return e;
}
quote! {
#ast
}
.into()
}
pub(crate) fn derive_package(input: TokenStream) -> TokenStream {
let mut input = parse_macro_input!(input as DeriveInput);
let name = input.ident.clone();
let name_builder = quote::format_ident!("{name}Builder");
let name_builder2 = name_builder.clone();
let name_builder3 = name_builder.clone();
let elements = match &input.data {
::syn::Data::Struct(s) => match &s.fields {
syn::Fields::Named(n) => n.named.clone(),
_ => panic!("only named supported"),
},
_ => panic!("only struct allowed"),
};
let all_fields = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() != "version" && id.to_string() != "_bok_base" {
return Some(&field.ident);
}
}
None
});
let all_fields2 = all_fields.clone();
let all_fields3 = all_fields.clone();
let all_fields_mut = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() != "version" && id.to_string() != "_bok_base" {
return Some(quote::format_ident!("{}_mut", id.to_string()));
}
}
None
});
let all_types = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() != "version" && id.to_string() != "_bok_base" {
return Some(&field.ty);
}
}
None
});
let all_types2 = all_types.clone();
let non_opt_fields = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() == "version" || id.to_string() == "_bok_base" {
return None;
}
}
match &field.ty {
syn::Type::Path(pth) => {
let first_path = pth.path.segments.first().unwrap();
if first_path.ident == "Option" {
None
} else {
let id = &field.ident;
Some(quote! {#id})
}
}
t => Some(quote! {#t}),
}
});
let opt_fields = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() == "version" || id.to_string() == "_bok_base" {
return None;
}
}
match &field.ty {
syn::Type::Path(pth) => {
let first_path = pth.path.segments.first().unwrap();
if first_path.ident == "Option" {
let id = &field.ident;
Some(quote! {#id})
} else {
None
}
}
_ => None,
}
});
let non_opt_types = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() == "version" || id.to_string() == "_bok_base" {
return None;
}
}
match &field.ty {
syn::Type::Path(pth) => {
let first_path = pth.path.segments.first().unwrap();
if first_path.ident == "Option" {
None
} else {
let t = &field.ty;
Some(quote! {#t})
}
}
_ => None,
}
});
let opt_types = elements.iter().filter_map(|field| {
if let Some(id) = field.ident.clone() {
if id.to_string() == "version" || id.to_string() == "_bok_base" {
return None;
}
}
if let ::syn::Type::Path(pth) = &field.ty {
let first_path = pth.path.segments.first().unwrap();
if first_path.ident == "Option" {
if let syn::PathArguments::AngleBracketed(
syn::AngleBracketedGenericArguments { args, .. },
) = &first_path.arguments
{
if let Some(syn::GenericArgument::Type(syn::Type::Path(
p,
))) = args.first()
{
let id = &p.path.segments.first().unwrap().ident;
return Some(quote! {#id});
}
}
}
}
None
});
let non_opt_fields2 = non_opt_fields.clone();
let non_opt_fields3 = non_opt_fields.clone();
let non_opt_fields4 = non_opt_fields.clone();
let non_opt_fields5 = non_opt_fields.clone();
let opt_fields2 = opt_fields.clone();
let opt_fields3 = opt_fields.clone();
let opt_types2 = opt_types.clone();
let non_opt_types2 = non_opt_types.clone();
{
let ::syn::Data::Struct(ref mut ds) = input.data else {
panic!("::bok::package expected a struct");
};
let ::syn::Fields::Named(ref mut _old_nf) = ds.fields else {
panic!("::bok::package expected a named field");
};
}
let pkg_struct = quote! {
#input
};
let mut pkg_struct_str = String::new();
{
use ::core::fmt::Write;
write!(&mut pkg_struct_str, "{}", pkg_struct)
.expect("can't write package struct into string");
}
let expanded = quote! {
impl ::core::fmt::Display for #name {
fn fmt(&self, f: &mut ::core::fmt::Formatter<'_>)
->::core::fmt::Result
{
<#name as ::bok::PkgCode>::fmt(self, f)
}
}
impl ::bok::PkgCode for #name {
fn fmt(&self, f: &mut ::core::fmt::Formatter<'_>)
->::core::fmt::Result
{
// FIXME: this sounds convoluted.
// first we turn everything into a string, then we
// parse the string into tokenstream again to format it
// reason: the `prettyplease` works on `::syn::File`
// and I can't find an easy translation
// `TokenStream2 -> TokenStream`
use ::bok::Pkg;
// don't add a method if it is just the base Trait implementation
fn maybe_add_fn(out: &mut String, name: &str, fn_str: &str) -> ::core::fmt::Result {
//FIXME: check with the code exported from macro_magic base
let str_base = "self . base () . unwrap () . ".to_owned() + name + " ()";
if fn_str.trim() != str_base {
use ::core::fmt::Write;
write!(out,
"\
fn {}(&self) -> Result<(), ()> {{\n\
{}\n\
}}\n",
name,
fn_str,
)
} else {
::core::fmt::Result::Ok(())
}
};
use ::core::fmt::Write;
let pkg_empty = ::bok::PkgEmpty::default();
let mut pkg_string = String::new();
// FIXME: filter out the things added from base pkg
write!(&mut pkg_string,
"#[::bok::package(::bok::PkgEmpty)]\n\
{}",
#pkg_struct_str)?;
write!(&mut pkg_string,
"#[::bok::package_impl]\n\
impl ::bok::Pkg for {} {{\n",
::std::stringify!(#name)
)?;
maybe_add_fn(&mut pkg_string, "prepare", &self.prepare_code().to_string())?;
maybe_add_fn(&mut pkg_string, "configure", &self.configure_code().to_string())?;
maybe_add_fn(&mut pkg_string, "build", &self.build_code().to_string())?;
maybe_add_fn(&mut pkg_string, "check", &self.check_code().to_string())?;
maybe_add_fn(&mut pkg_string, "install", &self.install_code().to_string())?;
write!(&mut pkg_string, "}}\n")?;
let re_parsed = ::syn::parse_file(&pkg_string).unwrap();
let formatted = prettyplease::unparse(&re_parsed);
f.write_str(&formatted)
}
}
impl #name {
pub fn as_pkg(&self) -> &dyn ::bok::Pkg {
self
}
pub fn as_pkg_mut(&mut self) -> &mut dyn ::bok::Pkg {
self
}
pub fn builder() -> #name_builder {
#name_builder::default()
}
#(pub fn #all_fields2(&self) -> &#all_types {
&self.#all_fields2
})*
#(pub fn #all_fields_mut(&mut self) -> &#all_types2 {
&mut self.#all_fields3
})*
}
#[derive(::std::default::Default, ::std::fmt::Debug)]
pub struct #name_builder {
#(#non_opt_fields: ::std::option::Option<#non_opt_types>,)*
#(#opt_fields: ::std::option::Option<#opt_types>,)*
}
impl ::bok::PkgBuilder for #name_builder2 {
fn name(&self) -> ::bok::PkgName {
use ::bok::Pkg;
#name::default().name()
}
fn path(&self) -> ::bok::Path<::bok::PkgName> {
use ::bok::Pkg;
#name::default().path()
}
fn version(&self) -> ::bok::Version {
use ::bok::Pkg;
#name::default().version()
}
fn default_unused(&mut self) -> &mut dyn ::bok::PkgBuilder {
let def = #name::default();
#(if self.#non_opt_fields5.is_none() {
self.#non_opt_fields5 = Some(def.#non_opt_fields5);
})*
self
}
fn build(&mut self) -> Result<Box<dyn ::bok::Pkg>, ::std::boxed::Box<dyn ::std::error::Error>> {
#(if self.#non_opt_fields2.is_none() {
return ::std::result::Result::Err("unset field".into());
})*
Ok(
Box::new(#name {
_bok_base: ::std::marker::PhantomData::default(),
version: #name::default().version,
#(#non_opt_fields3 : self.#non_opt_fields3.clone().unwrap(),)*
#(#opt_fields2 : self.#opt_fields2.clone(),)*
})
)
}
}
impl #name_builder3 {
pub fn as_any(&self) -> &dyn ::std::any::Any {
self
}
pub fn as_any_mut(&mut self) -> &mut dyn ::std::any::Any {
self
}
pub fn as_builder(&self) -> &dyn ::bok::PkgBuilder {
self
}
pub fn as_builder_mut(&mut self) -> &mut dyn ::bok::PkgBuilder {
self
}
#(pub fn #non_opt_fields4 (&mut self, val : #non_opt_types2) -> &mut Self {
if self.#non_opt_fields4 != None &&
self.#non_opt_fields4 != Some(val) {
panic!("Package \"{}\": mandatory attribute set multiple times: \"{}\"",
::std::stringify!(#name),
::std::stringify!(#non_opt_fields4));
}
self.#non_opt_fields4 = Some(val);
self
})*
#(pub fn #opt_fields3 (&mut self, val : #opt_types2) -> &mut Self {
if self.#opt_fields3 != None &&
self.#opt_fields3 != Some(val) {
panic!("Package \"{}\": optional attribute set multiple times: \"{}\"",
::std::stringify!(#name),
::std::stringify!(#opt_fields3));
}
self.#opt_fields3 = Some(val);
self
})*
}
};
TokenStream::from(expanded)
}