#![allow(
clippy::missing_const_for_fn,
clippy::std_instead_of_core,
clippy::std_instead_of_alloc,
clippy::alloc_instead_of_core,
reason = "irrelevant for proc macros"
)]
#![allow(
clippy::missing_docs_in_private_items,
missing_docs,
reason = "may be removed eventually"
)]
#[allow(
unused_macros,
reason = "may not be used for all feature flag combinations"
)]
macro_rules! bug {
() => { compile_error!("provide an error message to help fix a possible bug") };
($descr:literal $($rest:tt)?) => {
unreachable!(concat!("internal error: ", $descr) $($rest)?)
}
}
#[macro_use]
mod quote;
mod date;
mod datetime;
mod error;
#[cfg(any(feature = "formatting", feature = "parsing"))]
mod format_description;
mod helpers;
mod offset;
#[cfg(all(feature = "serde", any(feature = "formatting", feature = "parsing")))]
mod serde_format_description;
mod time;
mod to_tokens;
mod utc_datetime;
#[cfg(any(feature = "formatting", feature = "parsing"))]
use std::iter::Peekable;
#[cfg(all(feature = "serde", any(feature = "formatting", feature = "parsing")))]
use proc_macro::Delimiter;
use proc_macro::TokenStream;
#[cfg(any(feature = "formatting", feature = "parsing"))]
use proc_macro::TokenTree;
use self::error::Error;
macro_rules! impl_macros {
($($name:ident)*) => {$(
#[proc_macro]
pub fn $name(input: TokenStream) -> TokenStream {
use crate::to_tokens::ToTokenStream;
let mut iter = input.into_iter().peekable();
match $name::parse(&mut iter) {
Ok(value) => match iter.peek() {
Some(tree) => Error::UnexpectedToken { tree: tree.clone() }.to_compile_error(),
None => quote_! { const { #S(value.into_token_stream()) } },
},
Err(err) => err.to_compile_error(),
}
}
)*};
}
impl_macros![date datetime utc_datetime offset time];
#[cfg(any(feature = "formatting", feature = "parsing"))]
type PeekableTokenStreamIter = Peekable<proc_macro::token_stream::IntoIter>;
#[cfg(any(feature = "formatting", feature = "parsing"))]
#[derive(Clone, Copy)]
enum FormatDescriptionVersion {
V1,
V2,
V3,
}
#[cfg(any(feature = "formatting", feature = "parsing"))]
impl FormatDescriptionVersion {
fn is_v1(self) -> bool {
match self {
Self::V1 => true,
Self::V2 | Self::V3 => false,
}
}
fn is_at_most_v2(self) -> bool {
match self {
Self::V1 | Self::V2 => true,
Self::V3 => false,
}
}
fn is_at_least_v2(self) -> bool {
match self {
Self::V1 => false,
Self::V2 | Self::V3 => true,
}
}
fn is_at_least_v3(self) -> bool {
match self {
Self::V1 | Self::V2 => false,
Self::V3 => true,
}
}
}
#[cfg(any(feature = "formatting", feature = "parsing"))]
fn parse_format_description_version<const NO_EQUALS_IS_MOD_NAME: bool>(
iter: &mut PeekableTokenStreamIter,
) -> Result<Option<FormatDescriptionVersion>, Error> {
let end_of_input_err = || {
if NO_EQUALS_IS_MOD_NAME {
Error::UnexpectedEndOfInput
} else {
Error::ExpectedString {
span_start: None,
span_end: None,
}
}
};
let version_ident = match iter.peek().ok_or_else(end_of_input_err)? {
version @ TokenTree::Ident(ident) if ident.to_string() == "version" => {
let version_ident = version.clone();
iter.next(); version_ident
}
_ => return Ok(None),
};
match iter.peek() {
Some(TokenTree::Punct(punct)) if punct.as_char() == '=' => iter.next(),
_ if NO_EQUALS_IS_MOD_NAME => {
*iter = std::iter::once(version_ident)
.chain(iter.clone())
.collect::<TokenStream>()
.into_iter()
.peekable();
return Ok(None);
}
Some(token) => {
return Err(Error::Custom {
message: "expected `=`".into(),
span_start: Some(token.span()),
span_end: Some(token.span()),
});
}
None => {
return Err(Error::Custom {
message: "expected `=`".into(),
span_start: None,
span_end: None,
});
}
};
let version_literal = match iter.next() {
Some(TokenTree::Literal(literal)) => literal,
Some(token) => {
return Err(Error::Custom {
message: "expected 1, 2, or 3".into(),
span_start: Some(token.span()),
span_end: Some(token.span()),
});
}
None => {
return Err(Error::Custom {
message: "expected 1, 2, or 3".into(),
span_start: None,
span_end: None,
});
}
};
let version = match version_literal.to_string().as_str() {
"1" => FormatDescriptionVersion::V1,
"2" => FormatDescriptionVersion::V2,
"3" => FormatDescriptionVersion::V3,
_ => {
return Err(Error::Custom {
message: "invalid format description version".into(),
span_start: Some(version_literal.span()),
span_end: Some(version_literal.span()),
});
}
};
helpers::consume_punct(',', iter)?;
Ok(Some(version))
}
#[cfg(all(feature = "serde", any(feature = "formatting", feature = "parsing")))]
fn parse_visibility(iter: &mut PeekableTokenStreamIter) -> Result<TokenStream, Error> {
let mut visibility = match iter.peek().ok_or(Error::UnexpectedEndOfInput)? {
pub_ident @ TokenTree::Ident(ident) if ident.to_string() == "pub" => {
let visibility = quote_! { #(pub_ident.clone()) };
iter.next(); visibility
}
_ => return Ok(quote_! {}),
};
match iter.peek().ok_or(Error::UnexpectedEndOfInput)? {
group @ TokenTree::Group(path) if path.delimiter() == Delimiter::Parenthesis => {
visibility.extend(std::iter::once(group.clone()));
iter.next(); }
_ => {}
}
Ok(visibility)
}
#[cfg(any(feature = "formatting", feature = "parsing"))]
fn expand_format_description(item: format_description::public::OwnedFormatItem) -> TokenStream {
match item.version {
FormatDescriptionVersion::V1 | FormatDescriptionVersion::V2 => match &item.inner {
format_description::public::OwnedFormatItemInner::Compound(items) => {
let items = items
.iter()
.map(|inner_item| {
quote_! {
#S(format_description::public::OwnedFormatItem {
version: item.version,
inner: inner_item.clone(),
}),
}
})
.collect::<TokenStream>();
quote_! {
const {
use ::time::format_description::{*, modifier::*};
&[#S(items)] as StaticFormatDescription
}
}
}
_ => quote_! {
const {
use ::time::format_description::{*, modifier::*};
&[#S(item)] as StaticFormatDescription
}
},
},
FormatDescriptionVersion::V3 => quote_! {
const {
use ::time::format_description::__private::*;
use ::time::format_description::modifier::*;
#S(item).into_opaque()
}
},
}
}
#[cfg(any(feature = "formatting", feature = "parsing"))]
#[proc_macro]
pub fn format_description(input: TokenStream) -> TokenStream {
(|| {
let mut input = input.into_iter().peekable();
let version = parse_format_description_version::<false>(&mut input)?
.unwrap_or(FormatDescriptionVersion::V1);
let (span, string) = helpers::get_string_literal(version.is_at_most_v2(), input)?;
let items = format_description::parse_with_version(version, &string, span)?;
Ok(expand_format_description(items))
})()
.unwrap_or_else(|err: Error| err.to_compile_error())
}
#[cfg(all(feature = "serde", any(feature = "formatting", feature = "parsing")))]
#[proc_macro]
pub fn serde_format_description(input: TokenStream) -> TokenStream {
(|| {
let mut tokens = input.into_iter().peekable();
let maybe_version = parse_format_description_version::<true>(&mut tokens)?;
let visibility = parse_visibility(&mut tokens)?;
let version = if let Some(version) = maybe_version {
version
} else if maybe_version.is_none()
&& let Some(TokenTree::Ident(ident)) = tokens.peek()
&& ident.to_string() == "mod"
{
FormatDescriptionVersion::V3
} else {
FormatDescriptionVersion::V1
};
let (mod_name, ty) = match version {
FormatDescriptionVersion::V1 | FormatDescriptionVersion::V2 => {
let mod_name = match tokens.next() {
Some(TokenTree::Ident(ident)) => Ok(ident),
Some(tree) => Err(Error::UnexpectedToken { tree }),
None => Err(Error::UnexpectedEndOfInput),
}?;
helpers::consume_punct(',', &mut tokens)?;
let ty = match tokens.next() {
Some(tree @ TokenTree::Ident(_)) => Ok(tree.into()),
Some(tree) => Err(Error::UnexpectedToken { tree }),
None => Err(Error::UnexpectedEndOfInput),
}?;
helpers::consume_punct(',', &mut tokens)?;
(mod_name, ty)
}
FormatDescriptionVersion::V3 => {
match tokens.next() {
Some(TokenTree::Ident(ident)) if ident.to_string() == "mod" => {}
Some(tree) => {
return Err(Error::Custom {
message: "expected `mod`".into(),
span_start: Some(tree.span().start()),
span_end: Some(tree.span().end()),
});
}
None => return Err(Error::UnexpectedEndOfInput),
};
let mod_name = match tokens.next() {
Some(TokenTree::Ident(ident)) => Ok(ident),
Some(tree) => Err(Error::UnexpectedToken { tree }),
None => Err(Error::UnexpectedEndOfInput),
}?;
let ty = match tokens.next() {
Some(TokenTree::Group(group)) if group.delimiter() == Delimiter::Bracket => {
Ok(group.stream())
}
Some(tree) => Err(Error::UnexpectedToken { tree }),
None => Err(Error::UnexpectedEndOfInput),
}?;
helpers::consume_punct('=', &mut tokens)?;
(mod_name, ty)
}
};
let (format, format_description_display) = match tokens.peek() {
Some(TokenTree::Literal(_)) => {
let (span, format_string) =
helpers::get_string_literal(version.is_at_most_v2(), tokens)?;
let items = format_description::parse_with_version(version, &format_string, span)?;
let items = expand_format_description(items);
(items, String::from_utf8_lossy(&format_string).into_owned())
}
Some(_) => {
let tokens = tokens.collect::<TokenStream>();
let tokens_string = tokens.to_string();
(tokens, tokens_string)
}
None => return Err(Error::UnexpectedEndOfInput),
};
Ok(serde_format_description::build(
version,
visibility,
mod_name,
ty,
format,
format_description_display,
))
})()
.unwrap_or_else(|err: Error| err.to_compile_error_standalone())
}