Merge pull request #2405 from dtolnay/syn

Update to syn 2
This commit is contained in:
David Tolnay
2023-03-17 17:23:20 -07:00
committed by GitHub
12 changed files with 199 additions and 199 deletions
+2 -2
View File
@@ -108,12 +108,12 @@ jobs:
- run: cd serde && cargo build - run: cd serde && cargo build
derive: derive:
name: Rust 1.31.0 name: Rust 1.56.0
runs-on: ubuntu-latest runs-on: ubuntu-latest
timeout-minutes: 45 timeout-minutes: 45
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v3
- uses: dtolnay/rust-toolchain@1.31.0 - uses: dtolnay/rust-toolchain@1.56.0
- run: cd serde && cargo check --no-default-features - run: cd serde && cargo check --no-default-features
- run: cd serde && cargo check - run: cd serde && cargo check
- run: cd serde_derive && cargo check - run: cd serde_derive && cargo check
+1 -1
View File
@@ -24,7 +24,7 @@ proc-macro = true
[dependencies] [dependencies]
proc-macro2 = "1.0" proc-macro2 = "1.0"
quote = "1.0" quote = "1.0"
syn = "1.0.104" syn = "2.0"
[dev-dependencies] [dev-dependencies]
serde = { version = "1.0", path = "../serde" } serde = { version = "1.0", path = "../serde" }
+13 -5
View File
@@ -200,10 +200,16 @@ pub fn with_bound(
for arg in &arguments.args { for arg in &arguments.args {
match arg { match arg {
syn::GenericArgument::Type(arg) => self.visit_type(arg), syn::GenericArgument::Type(arg) => self.visit_type(arg),
syn::GenericArgument::Binding(arg) => self.visit_type(&arg.ty), syn::GenericArgument::AssocType(arg) => self.visit_type(&arg.ty),
syn::GenericArgument::Lifetime(_) syn::GenericArgument::Lifetime(_)
| syn::GenericArgument::Constraint(_) | syn::GenericArgument::Const(_)
| syn::GenericArgument::Const(_) => {} | syn::GenericArgument::AssocConst(_)
| syn::GenericArgument::Constraint(_) => {}
#[cfg_attr(
all(test, exhaustive),
deny(non_exhaustive_omitted_patterns)
)]
_ => {}
} }
} }
} }
@@ -226,7 +232,9 @@ pub fn with_bound(
fn visit_type_param_bound(&mut self, bound: &'ast syn::TypeParamBound) { fn visit_type_param_bound(&mut self, bound: &'ast syn::TypeParamBound) {
match bound { match bound {
syn::TypeParamBound::Trait(bound) => self.visit_path(&bound.path), syn::TypeParamBound::Trait(bound) => self.visit_path(&bound.path),
syn::TypeParamBound::Lifetime(_) => {} syn::TypeParamBound::Lifetime(_) | syn::TypeParamBound::Verbatim(_) => {}
#[cfg_attr(all(test, exhaustive), deny(non_exhaustive_omitted_patterns))]
_ => {}
} }
} }
@@ -334,7 +342,7 @@ pub fn with_self_bound(
pub fn with_lifetime_bound(generics: &syn::Generics, lifetime: &str) -> syn::Generics { pub fn with_lifetime_bound(generics: &syn::Generics, lifetime: &str) -> syn::Generics {
let bound = syn::Lifetime::new(lifetime, Span::call_site()); let bound = syn::Lifetime::new(lifetime, Span::call_site());
let def = syn::LifetimeDef { let def = syn::LifetimeParam {
attrs: Vec::new(), attrs: Vec::new(),
lifetime: bound.clone(), lifetime: bound.clone(),
colon_token: None, colon_token: None,
+10 -10
View File
@@ -244,9 +244,9 @@ impl BorrowedLifetimes {
} }
} }
fn de_lifetime_def(&self) -> Option<syn::LifetimeDef> { fn de_lifetime_param(&self) -> Option<syn::LifetimeParam> {
match self { match self {
BorrowedLifetimes::Borrowed(bounds) => Some(syn::LifetimeDef { BorrowedLifetimes::Borrowed(bounds) => Some(syn::LifetimeParam {
attrs: Vec::new(), attrs: Vec::new(),
lifetime: syn::Lifetime::new("'de", Span::call_site()), lifetime: syn::Lifetime::new("'de", Span::call_site()),
colon_token: None, colon_token: None,
@@ -3011,7 +3011,7 @@ struct InPlaceImplGenerics<'a>(&'a Parameters);
impl<'a> ToTokens for DeImplGenerics<'a> { impl<'a> ToTokens for DeImplGenerics<'a> {
fn to_tokens(&self, tokens: &mut TokenStream) { fn to_tokens(&self, tokens: &mut TokenStream) {
let mut generics = self.0.generics.clone(); let mut generics = self.0.generics.clone();
if let Some(de_lifetime) = self.0.borrowed.de_lifetime_def() { if let Some(de_lifetime) = self.0.borrowed.de_lifetime_param() {
generics.params = Some(syn::GenericParam::Lifetime(de_lifetime)) generics.params = Some(syn::GenericParam::Lifetime(de_lifetime))
.into_iter() .into_iter()
.chain(generics.params) .chain(generics.params)
@@ -3046,7 +3046,7 @@ impl<'a> ToTokens for InPlaceImplGenerics<'a> {
.into_iter() .into_iter()
.chain(generics.params) .chain(generics.params)
.collect(); .collect();
if let Some(de_lifetime) = self.0.borrowed.de_lifetime_def() { if let Some(de_lifetime) = self.0.borrowed.de_lifetime_param() {
generics.params = Some(syn::GenericParam::Lifetime(de_lifetime)) generics.params = Some(syn::GenericParam::Lifetime(de_lifetime))
.into_iter() .into_iter()
.chain(generics.params) .chain(generics.params)
@@ -3071,8 +3071,8 @@ struct InPlaceTypeGenerics<'a>(&'a Parameters);
impl<'a> ToTokens for DeTypeGenerics<'a> { impl<'a> ToTokens for DeTypeGenerics<'a> {
fn to_tokens(&self, tokens: &mut TokenStream) { fn to_tokens(&self, tokens: &mut TokenStream) {
let mut generics = self.0.generics.clone(); let mut generics = self.0.generics.clone();
if self.0.borrowed.de_lifetime_def().is_some() { if self.0.borrowed.de_lifetime_param().is_some() {
let def = syn::LifetimeDef { let def = syn::LifetimeParam {
attrs: Vec::new(), attrs: Vec::new(),
lifetime: syn::Lifetime::new("'de", Span::call_site()), lifetime: syn::Lifetime::new("'de", Span::call_site()),
colon_token: None, colon_token: None,
@@ -3097,8 +3097,8 @@ impl<'a> ToTokens for InPlaceTypeGenerics<'a> {
.chain(generics.params) .chain(generics.params)
.collect(); .collect();
if self.0.borrowed.de_lifetime_def().is_some() { if self.0.borrowed.de_lifetime_param().is_some() {
let def = syn::LifetimeDef { let def = syn::LifetimeParam {
attrs: Vec::new(), attrs: Vec::new(),
lifetime: syn::Lifetime::new("'de", Span::call_site()), lifetime: syn::Lifetime::new("'de", Span::call_site()),
colon_token: None, colon_token: None,
@@ -3122,8 +3122,8 @@ impl<'a> DeTypeGenerics<'a> {
} }
#[cfg(feature = "deserialize_in_place")] #[cfg(feature = "deserialize_in_place")]
fn place_lifetime() -> syn::LifetimeDef { fn place_lifetime() -> syn::LifetimeParam {
syn::LifetimeDef { syn::LifetimeParam {
attrs: Vec::new(), attrs: Vec::new(),
lifetime: syn::Lifetime::new("'place", Span::call_site()), lifetime: syn::Lifetime::new("'place", Span::call_site()),
colon_token: None, colon_token: None,
+148 -163
View File
@@ -8,9 +8,7 @@ use std::iter::FromIterator;
use syn; use syn;
use syn::parse::ParseStream; use syn::parse::ParseStream;
use syn::punctuated::Punctuated; use syn::punctuated::Punctuated;
use syn::Meta::{List, NameValue, Path}; use syn::{Expr, Ident, Lifetime, Meta, MetaList};
use syn::NestedMeta::{Lit, Meta};
use syn::{Ident, Lifetime};
// This module handles parsing of `#[serde(...)]` attributes. The entrypoints // This module handles parsing of `#[serde(...)]` attributes. The entrypoints
// are `attr::Container::from_ast`, `attr::Variant::from_ast`, and // are `attr::Container::from_ast`, `attr::Variant::from_ast`, and
@@ -313,24 +311,24 @@ impl Container {
{ {
match &meta_item { match &meta_item {
// Parse `#[serde(rename = "foo")]` // Parse `#[serde(rename = "foo")]`
Meta(NameValue(m)) if m.path == RENAME => { Meta::NameValue(m) if m.path == RENAME => {
if let Ok(s) = get_lit_str(cx, RENAME, &m.lit) { if let Ok(s) = get_lit_str(cx, RENAME, &m.value) {
ser_name.set(&m.path, s.value()); ser_name.set(&m.path, s.value());
de_name.set(&m.path, s.value()); de_name.set(&m.path, s.value());
} }
} }
// Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]` // Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]`
Meta(List(m)) if m.path == RENAME => { Meta::List(m) if m.path == RENAME => {
if let Ok((ser, de)) = get_renames(cx, &m.nested) { if let Ok((ser, de)) = get_renames(cx, m) {
ser_name.set_opt(&m.path, ser.map(syn::LitStr::value)); ser_name.set_opt(&m.path, ser.as_ref().map(syn::LitStr::value));
de_name.set_opt(&m.path, de.map(syn::LitStr::value)); de_name.set_opt(&m.path, de.as_ref().map(syn::LitStr::value));
} }
} }
// Parse `#[serde(rename_all = "foo")]` // Parse `#[serde(rename_all = "foo")]`
Meta(NameValue(m)) if m.path == RENAME_ALL => { Meta::NameValue(m) if m.path == RENAME_ALL => {
if let Ok(s) = get_lit_str(cx, RENAME_ALL, &m.lit) { if let Ok(s) = get_lit_str(cx, RENAME_ALL, &m.value) {
match RenameRule::from_str(&s.value()) { match RenameRule::from_str(&s.value()) {
Ok(rename_rule) => { Ok(rename_rule) => {
rename_all_ser_rule.set(&m.path, rename_rule); rename_all_ser_rule.set(&m.path, rename_rule);
@@ -342,8 +340,8 @@ impl Container {
} }
// Parse `#[serde(rename_all(serialize = "foo", deserialize = "bar"))]` // Parse `#[serde(rename_all(serialize = "foo", deserialize = "bar"))]`
Meta(List(m)) if m.path == RENAME_ALL => { Meta::List(m) if m.path == RENAME_ALL => {
if let Ok((ser, de)) = get_renames(cx, &m.nested) { if let Ok((ser, de)) = get_renames(cx, m) {
if let Some(ser) = ser { if let Some(ser) = ser {
match RenameRule::from_str(&ser.value()) { match RenameRule::from_str(&ser.value()) {
Ok(rename_rule) => rename_all_ser_rule.set(&m.path, rename_rule), Ok(rename_rule) => rename_all_ser_rule.set(&m.path, rename_rule),
@@ -360,17 +358,17 @@ impl Container {
} }
// Parse `#[serde(transparent)]` // Parse `#[serde(transparent)]`
Meta(Path(word)) if word == TRANSPARENT => { Meta::Path(word) if word == TRANSPARENT => {
transparent.set_true(word); transparent.set_true(word);
} }
// Parse `#[serde(deny_unknown_fields)]` // Parse `#[serde(deny_unknown_fields)]`
Meta(Path(word)) if word == DENY_UNKNOWN_FIELDS => { Meta::Path(word) if word == DENY_UNKNOWN_FIELDS => {
deny_unknown_fields.set_true(word); deny_unknown_fields.set_true(word);
} }
// Parse `#[serde(default)]` // Parse `#[serde(default)]`
Meta(Path(word)) if word == DEFAULT => match &item.data { Meta::Path(word) if word == DEFAULT => match &item.data {
syn::Data::Struct(syn::DataStruct { fields, .. }) => match fields { syn::Data::Struct(syn::DataStruct { fields, .. }) => match fields {
syn::Fields::Named(_) => { syn::Fields::Named(_) => {
default.set(word, Default::Default); default.set(word, Default::Default);
@@ -392,8 +390,8 @@ impl Container {
}, },
// Parse `#[serde(default = "...")]` // Parse `#[serde(default = "...")]`
Meta(NameValue(m)) if m.path == DEFAULT => { Meta::NameValue(m) if m.path == DEFAULT => {
if let Ok(path) = parse_lit_into_expr_path(cx, DEFAULT, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, DEFAULT, &m.value) {
match &item.data { match &item.data {
syn::Data::Struct(syn::DataStruct { fields, .. }) => match fields { syn::Data::Struct(syn::DataStruct { fields, .. }) => match fields {
syn::Fields::Named(_) => { syn::Fields::Named(_) => {
@@ -417,23 +415,23 @@ impl Container {
} }
// Parse `#[serde(bound = "T: SomeBound")]` // Parse `#[serde(bound = "T: SomeBound")]`
Meta(NameValue(m)) if m.path == BOUND => { Meta::NameValue(m) if m.path == BOUND => {
if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.lit) { if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.value) {
ser_bound.set(&m.path, where_predicates.clone()); ser_bound.set(&m.path, where_predicates.clone());
de_bound.set(&m.path, where_predicates); de_bound.set(&m.path, where_predicates);
} }
} }
// Parse `#[serde(bound(serialize = "...", deserialize = "..."))]` // Parse `#[serde(bound(serialize = "...", deserialize = "..."))]`
Meta(List(m)) if m.path == BOUND => { Meta::List(m) if m.path == BOUND => {
if let Ok((ser, de)) = get_where_predicates(cx, &m.nested) { if let Ok((ser, de)) = get_where_predicates(cx, m) {
ser_bound.set_opt(&m.path, ser); ser_bound.set_opt(&m.path, ser);
de_bound.set_opt(&m.path, de); de_bound.set_opt(&m.path, de);
} }
} }
// Parse `#[serde(untagged)]` // Parse `#[serde(untagged)]`
Meta(Path(word)) if word == UNTAGGED => match item.data { Meta::Path(word) if word == UNTAGGED => match item.data {
syn::Data::Enum(_) => { syn::Data::Enum(_) => {
untagged.set_true(word); untagged.set_true(word);
} }
@@ -448,8 +446,8 @@ impl Container {
}, },
// Parse `#[serde(tag = "type")]` // Parse `#[serde(tag = "type")]`
Meta(NameValue(m)) if m.path == TAG => { Meta::NameValue(m) if m.path == TAG => {
if let Ok(s) = get_lit_str(cx, TAG, &m.lit) { if let Ok(s) = get_lit_str(cx, TAG, &m.value) {
match &item.data { match &item.data {
syn::Data::Enum(_) => { syn::Data::Enum(_) => {
internal_tag.set(&m.path, s.value()); internal_tag.set(&m.path, s.value());
@@ -472,8 +470,8 @@ impl Container {
} }
// Parse `#[serde(content = "c")]` // Parse `#[serde(content = "c")]`
Meta(NameValue(m)) if m.path == CONTENT => { Meta::NameValue(m) if m.path == CONTENT => {
if let Ok(s) = get_lit_str(cx, CONTENT, &m.lit) { if let Ok(s) = get_lit_str(cx, CONTENT, &m.value) {
match &item.data { match &item.data {
syn::Data::Enum(_) => { syn::Data::Enum(_) => {
content.set(&m.path, s.value()); content.set(&m.path, s.value());
@@ -491,29 +489,29 @@ impl Container {
} }
// Parse `#[serde(from = "Type")]` // Parse `#[serde(from = "Type")]`
Meta(NameValue(m)) if m.path == FROM => { Meta::NameValue(m) if m.path == FROM => {
if let Ok(from_ty) = parse_lit_into_ty(cx, FROM, &m.lit) { if let Ok(from_ty) = parse_lit_into_ty(cx, FROM, &m.value) {
type_from.set_opt(&m.path, Some(from_ty)); type_from.set_opt(&m.path, Some(from_ty));
} }
} }
// Parse `#[serde(try_from = "Type")]` // Parse `#[serde(try_from = "Type")]`
Meta(NameValue(m)) if m.path == TRY_FROM => { Meta::NameValue(m) if m.path == TRY_FROM => {
if let Ok(try_from_ty) = parse_lit_into_ty(cx, TRY_FROM, &m.lit) { if let Ok(try_from_ty) = parse_lit_into_ty(cx, TRY_FROM, &m.value) {
type_try_from.set_opt(&m.path, Some(try_from_ty)); type_try_from.set_opt(&m.path, Some(try_from_ty));
} }
} }
// Parse `#[serde(into = "Type")]` // Parse `#[serde(into = "Type")]`
Meta(NameValue(m)) if m.path == INTO => { Meta::NameValue(m) if m.path == INTO => {
if let Ok(into_ty) = parse_lit_into_ty(cx, INTO, &m.lit) { if let Ok(into_ty) = parse_lit_into_ty(cx, INTO, &m.value) {
type_into.set_opt(&m.path, Some(into_ty)); type_into.set_opt(&m.path, Some(into_ty));
} }
} }
// Parse `#[serde(remote = "...")]` // Parse `#[serde(remote = "...")]`
Meta(NameValue(m)) if m.path == REMOTE => { Meta::NameValue(m) if m.path == REMOTE => {
if let Ok(path) = parse_lit_into_path(cx, REMOTE, &m.lit) { if let Ok(path) = parse_lit_into_path(cx, REMOTE, &m.value) {
if is_primitive_path(&path, "Self") { if is_primitive_path(&path, "Self") {
remote.set(&m.path, item.ident.clone().into()); remote.set(&m.path, item.ident.clone().into());
} else { } else {
@@ -523,30 +521,30 @@ impl Container {
} }
// Parse `#[serde(field_identifier)]` // Parse `#[serde(field_identifier)]`
Meta(Path(word)) if word == FIELD_IDENTIFIER => { Meta::Path(word) if word == FIELD_IDENTIFIER => {
field_identifier.set_true(word); field_identifier.set_true(word);
} }
// Parse `#[serde(variant_identifier)]` // Parse `#[serde(variant_identifier)]`
Meta(Path(word)) if word == VARIANT_IDENTIFIER => { Meta::Path(word) if word == VARIANT_IDENTIFIER => {
variant_identifier.set_true(word); variant_identifier.set_true(word);
} }
// Parse `#[serde(crate = "foo")]` // Parse `#[serde(crate = "foo")]`
Meta(NameValue(m)) if m.path == CRATE => { Meta::NameValue(m) if m.path == CRATE => {
if let Ok(path) = parse_lit_into_path(cx, CRATE, &m.lit) { if let Ok(path) = parse_lit_into_path(cx, CRATE, &m.value) {
serde_path.set(&m.path, path); serde_path.set(&m.path, path);
} }
} }
// Parse `#[serde(expecting = "a message")]` // Parse `#[serde(expecting = "a message")]`
Meta(NameValue(m)) if m.path == EXPECTING => { Meta::NameValue(m) if m.path == EXPECTING => {
if let Ok(s) = get_lit_str(cx, EXPECTING, &m.lit) { if let Ok(s) = get_lit_str(cx, EXPECTING, &m.value) {
expecting.set(&m.path, s.value()); expecting.set(&m.path, s.value());
} }
} }
Meta(meta_item) => { meta_item => {
let path = meta_item let path = meta_item
.path() .path()
.into_token_stream() .into_token_stream()
@@ -555,17 +553,12 @@ impl Container {
let msg = format!("unknown serde container attribute `{}`", path); let msg = format!("unknown serde container attribute `{}`", path);
cx.error_spanned_by(meta_item.path(), msg); cx.error_spanned_by(meta_item.path(), msg);
} }
Lit(lit) => {
let msg = "unexpected literal in serde container attribute";
cx.error_spanned_by(lit, msg);
}
} }
} }
let mut is_packed = false; let mut is_packed = false;
for attr in &item.attrs { for attr in &item.attrs {
if attr.path == REPR { if attr.path() == REPR {
let _ = attr.parse_args_with(|input: ParseStream| { let _ = attr.parse_args_with(|input: ParseStream| {
while let Some(token) = input.parse()? { while let Some(token) = input.parse()? {
if let TokenTree::Ident(ident) = token { if let TokenTree::Ident(ident) = token {
@@ -829,8 +822,8 @@ impl Variant {
{ {
match &meta_item { match &meta_item {
// Parse `#[serde(rename = "foo")]` // Parse `#[serde(rename = "foo")]`
Meta(NameValue(m)) if m.path == RENAME => { Meta::NameValue(m) if m.path == RENAME => {
if let Ok(s) = get_lit_str(cx, RENAME, &m.lit) { if let Ok(s) = get_lit_str(cx, RENAME, &m.value) {
ser_name.set(&m.path, s.value()); ser_name.set(&m.path, s.value());
de_name.set_if_none(s.value()); de_name.set_if_none(s.value());
de_aliases.insert(&m.path, s.value()); de_aliases.insert(&m.path, s.value());
@@ -838,9 +831,9 @@ impl Variant {
} }
// Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]` // Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]`
Meta(List(m)) if m.path == RENAME => { Meta::List(m) if m.path == RENAME => {
if let Ok((ser, de)) = get_multiple_renames(cx, &m.nested) { if let Ok((ser, de)) = get_multiple_renames(cx, m) {
ser_name.set_opt(&m.path, ser.map(syn::LitStr::value)); ser_name.set_opt(&m.path, ser.as_ref().map(syn::LitStr::value));
for de_value in de { for de_value in de {
de_name.set_if_none(de_value.value()); de_name.set_if_none(de_value.value());
de_aliases.insert(&m.path, de_value.value()); de_aliases.insert(&m.path, de_value.value());
@@ -849,15 +842,15 @@ impl Variant {
} }
// Parse `#[serde(alias = "foo")]` // Parse `#[serde(alias = "foo")]`
Meta(NameValue(m)) if m.path == ALIAS => { Meta::NameValue(m) if m.path == ALIAS => {
if let Ok(s) = get_lit_str(cx, ALIAS, &m.lit) { if let Ok(s) = get_lit_str(cx, ALIAS, &m.value) {
de_aliases.insert(&m.path, s.value()); de_aliases.insert(&m.path, s.value());
} }
} }
// Parse `#[serde(rename_all = "foo")]` // Parse `#[serde(rename_all = "foo")]`
Meta(NameValue(m)) if m.path == RENAME_ALL => { Meta::NameValue(m) if m.path == RENAME_ALL => {
if let Ok(s) = get_lit_str(cx, RENAME_ALL, &m.lit) { if let Ok(s) = get_lit_str(cx, RENAME_ALL, &m.value) {
match RenameRule::from_str(&s.value()) { match RenameRule::from_str(&s.value()) {
Ok(rename_rule) => { Ok(rename_rule) => {
rename_all_ser_rule.set(&m.path, rename_rule); rename_all_ser_rule.set(&m.path, rename_rule);
@@ -869,8 +862,8 @@ impl Variant {
} }
// Parse `#[serde(rename_all(serialize = "foo", deserialize = "bar"))]` // Parse `#[serde(rename_all(serialize = "foo", deserialize = "bar"))]`
Meta(List(m)) if m.path == RENAME_ALL => { Meta::List(m) if m.path == RENAME_ALL => {
if let Ok((ser, de)) = get_renames(cx, &m.nested) { if let Ok((ser, de)) = get_renames(cx, m) {
if let Some(ser) = ser { if let Some(ser) = ser {
match RenameRule::from_str(&ser.value()) { match RenameRule::from_str(&ser.value()) {
Ok(rename_rule) => rename_all_ser_rule.set(&m.path, rename_rule), Ok(rename_rule) => rename_all_ser_rule.set(&m.path, rename_rule),
@@ -887,45 +880,45 @@ impl Variant {
} }
// Parse `#[serde(skip)]` // Parse `#[serde(skip)]`
Meta(Path(word)) if word == SKIP => { Meta::Path(word) if word == SKIP => {
skip_serializing.set_true(word); skip_serializing.set_true(word);
skip_deserializing.set_true(word); skip_deserializing.set_true(word);
} }
// Parse `#[serde(skip_deserializing)]` // Parse `#[serde(skip_deserializing)]`
Meta(Path(word)) if word == SKIP_DESERIALIZING => { Meta::Path(word) if word == SKIP_DESERIALIZING => {
skip_deserializing.set_true(word); skip_deserializing.set_true(word);
} }
// Parse `#[serde(skip_serializing)]` // Parse `#[serde(skip_serializing)]`
Meta(Path(word)) if word == SKIP_SERIALIZING => { Meta::Path(word) if word == SKIP_SERIALIZING => {
skip_serializing.set_true(word); skip_serializing.set_true(word);
} }
// Parse `#[serde(other)]` // Parse `#[serde(other)]`
Meta(Path(word)) if word == OTHER => { Meta::Path(word) if word == OTHER => {
other.set_true(word); other.set_true(word);
} }
// Parse `#[serde(bound = "T: SomeBound")]` // Parse `#[serde(bound = "T: SomeBound")]`
Meta(NameValue(m)) if m.path == BOUND => { Meta::NameValue(m) if m.path == BOUND => {
if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.lit) { if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.value) {
ser_bound.set(&m.path, where_predicates.clone()); ser_bound.set(&m.path, where_predicates.clone());
de_bound.set(&m.path, where_predicates); de_bound.set(&m.path, where_predicates);
} }
} }
// Parse `#[serde(bound(serialize = "...", deserialize = "..."))]` // Parse `#[serde(bound(serialize = "...", deserialize = "..."))]`
Meta(List(m)) if m.path == BOUND => { Meta::List(m) if m.path == BOUND => {
if let Ok((ser, de)) = get_where_predicates(cx, &m.nested) { if let Ok((ser, de)) = get_where_predicates(cx, m) {
ser_bound.set_opt(&m.path, ser); ser_bound.set_opt(&m.path, ser);
de_bound.set_opt(&m.path, de); de_bound.set_opt(&m.path, de);
} }
} }
// Parse `#[serde(with = "...")]` // Parse `#[serde(with = "...")]`
Meta(NameValue(m)) if m.path == WITH => { Meta::NameValue(m) if m.path == WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, WITH, &m.value) {
let mut ser_path = path.clone(); let mut ser_path = path.clone();
ser_path ser_path
.path .path
@@ -942,21 +935,21 @@ impl Variant {
} }
// Parse `#[serde(serialize_with = "...")]` // Parse `#[serde(serialize_with = "...")]`
Meta(NameValue(m)) if m.path == SERIALIZE_WITH => { Meta::NameValue(m) if m.path == SERIALIZE_WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, SERIALIZE_WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, SERIALIZE_WITH, &m.value) {
serialize_with.set(&m.path, path); serialize_with.set(&m.path, path);
} }
} }
// Parse `#[serde(deserialize_with = "...")]` // Parse `#[serde(deserialize_with = "...")]`
Meta(NameValue(m)) if m.path == DESERIALIZE_WITH => { Meta::NameValue(m) if m.path == DESERIALIZE_WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, DESERIALIZE_WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, DESERIALIZE_WITH, &m.value) {
deserialize_with.set(&m.path, path); deserialize_with.set(&m.path, path);
} }
} }
// Parse `#[serde(borrow)]` // Parse `#[serde(borrow)]`
Meta(Path(word)) if word == BORROW => match &variant.fields { Meta::Path(word) if word == BORROW => match &variant.fields {
syn::Fields::Unnamed(fields) if fields.unnamed.len() == 1 => { syn::Fields::Unnamed(fields) if fields.unnamed.len() == 1 => {
borrow.set( borrow.set(
word, word,
@@ -973,9 +966,9 @@ impl Variant {
}, },
// Parse `#[serde(borrow = "'a + 'b")]` // Parse `#[serde(borrow = "'a + 'b")]`
Meta(NameValue(m)) if m.path == BORROW => match &variant.fields { Meta::NameValue(m) if m.path == BORROW => match &variant.fields {
syn::Fields::Unnamed(fields) if fields.unnamed.len() == 1 => { syn::Fields::Unnamed(fields) if fields.unnamed.len() == 1 => {
if let Ok(lifetimes) = parse_lit_into_lifetimes(cx, &m.lit) { if let Ok(lifetimes) = parse_lit_into_lifetimes(cx, &m.value) {
borrow.set( borrow.set(
&m.path, &m.path,
BorrowAttribute { BorrowAttribute {
@@ -991,7 +984,7 @@ impl Variant {
} }
}, },
Meta(meta_item) => { meta_item => {
let path = meta_item let path = meta_item
.path() .path()
.into_token_stream() .into_token_stream()
@@ -1000,11 +993,6 @@ impl Variant {
let msg = format!("unknown serde variant attribute `{}`", path); let msg = format!("unknown serde variant attribute `{}`", path);
cx.error_spanned_by(meta_item.path(), msg); cx.error_spanned_by(meta_item.path(), msg);
} }
Lit(lit) => {
let msg = "unexpected literal in serde variant attribute";
cx.error_spanned_by(lit, msg);
}
} }
} }
@@ -1165,8 +1153,8 @@ impl Field {
{ {
match &meta_item { match &meta_item {
// Parse `#[serde(rename = "foo")]` // Parse `#[serde(rename = "foo")]`
Meta(NameValue(m)) if m.path == RENAME => { Meta::NameValue(m) if m.path == RENAME => {
if let Ok(s) = get_lit_str(cx, RENAME, &m.lit) { if let Ok(s) = get_lit_str(cx, RENAME, &m.value) {
ser_name.set(&m.path, s.value()); ser_name.set(&m.path, s.value());
de_name.set_if_none(s.value()); de_name.set_if_none(s.value());
de_aliases.insert(&m.path, s.value()); de_aliases.insert(&m.path, s.value());
@@ -1174,9 +1162,9 @@ impl Field {
} }
// Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]` // Parse `#[serde(rename(serialize = "foo", deserialize = "bar"))]`
Meta(List(m)) if m.path == RENAME => { Meta::List(m) if m.path == RENAME => {
if let Ok((ser, de)) = get_multiple_renames(cx, &m.nested) { if let Ok((ser, de)) = get_multiple_renames(cx, m) {
ser_name.set_opt(&m.path, ser.map(syn::LitStr::value)); ser_name.set_opt(&m.path, ser.as_ref().map(syn::LitStr::value));
for de_value in de { for de_value in de {
de_name.set_if_none(de_value.value()); de_name.set_if_none(de_value.value());
de_aliases.insert(&m.path, de_value.value()); de_aliases.insert(&m.path, de_value.value());
@@ -1185,64 +1173,64 @@ impl Field {
} }
// Parse `#[serde(alias = "foo")]` // Parse `#[serde(alias = "foo")]`
Meta(NameValue(m)) if m.path == ALIAS => { Meta::NameValue(m) if m.path == ALIAS => {
if let Ok(s) = get_lit_str(cx, ALIAS, &m.lit) { if let Ok(s) = get_lit_str(cx, ALIAS, &m.value) {
de_aliases.insert(&m.path, s.value()); de_aliases.insert(&m.path, s.value());
} }
} }
// Parse `#[serde(default)]` // Parse `#[serde(default)]`
Meta(Path(word)) if word == DEFAULT => { Meta::Path(word) if word == DEFAULT => {
default.set(word, Default::Default); default.set(word, Default::Default);
} }
// Parse `#[serde(default = "...")]` // Parse `#[serde(default = "...")]`
Meta(NameValue(m)) if m.path == DEFAULT => { Meta::NameValue(m) if m.path == DEFAULT => {
if let Ok(path) = parse_lit_into_expr_path(cx, DEFAULT, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, DEFAULT, &m.value) {
default.set(&m.path, Default::Path(path)); default.set(&m.path, Default::Path(path));
} }
} }
// Parse `#[serde(skip_serializing)]` // Parse `#[serde(skip_serializing)]`
Meta(Path(word)) if word == SKIP_SERIALIZING => { Meta::Path(word) if word == SKIP_SERIALIZING => {
skip_serializing.set_true(word); skip_serializing.set_true(word);
} }
// Parse `#[serde(skip_deserializing)]` // Parse `#[serde(skip_deserializing)]`
Meta(Path(word)) if word == SKIP_DESERIALIZING => { Meta::Path(word) if word == SKIP_DESERIALIZING => {
skip_deserializing.set_true(word); skip_deserializing.set_true(word);
} }
// Parse `#[serde(skip)]` // Parse `#[serde(skip)]`
Meta(Path(word)) if word == SKIP => { Meta::Path(word) if word == SKIP => {
skip_serializing.set_true(word); skip_serializing.set_true(word);
skip_deserializing.set_true(word); skip_deserializing.set_true(word);
} }
// Parse `#[serde(skip_serializing_if = "...")]` // Parse `#[serde(skip_serializing_if = "...")]`
Meta(NameValue(m)) if m.path == SKIP_SERIALIZING_IF => { Meta::NameValue(m) if m.path == SKIP_SERIALIZING_IF => {
if let Ok(path) = parse_lit_into_expr_path(cx, SKIP_SERIALIZING_IF, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, SKIP_SERIALIZING_IF, &m.value) {
skip_serializing_if.set(&m.path, path); skip_serializing_if.set(&m.path, path);
} }
} }
// Parse `#[serde(serialize_with = "...")]` // Parse `#[serde(serialize_with = "...")]`
Meta(NameValue(m)) if m.path == SERIALIZE_WITH => { Meta::NameValue(m) if m.path == SERIALIZE_WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, SERIALIZE_WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, SERIALIZE_WITH, &m.value) {
serialize_with.set(&m.path, path); serialize_with.set(&m.path, path);
} }
} }
// Parse `#[serde(deserialize_with = "...")]` // Parse `#[serde(deserialize_with = "...")]`
Meta(NameValue(m)) if m.path == DESERIALIZE_WITH => { Meta::NameValue(m) if m.path == DESERIALIZE_WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, DESERIALIZE_WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, DESERIALIZE_WITH, &m.value) {
deserialize_with.set(&m.path, path); deserialize_with.set(&m.path, path);
} }
} }
// Parse `#[serde(with = "...")]` // Parse `#[serde(with = "...")]`
Meta(NameValue(m)) if m.path == WITH => { Meta::NameValue(m) if m.path == WITH => {
if let Ok(path) = parse_lit_into_expr_path(cx, WITH, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, WITH, &m.value) {
let mut ser_path = path.clone(); let mut ser_path = path.clone();
ser_path ser_path
.path .path
@@ -1259,31 +1247,31 @@ impl Field {
} }
// Parse `#[serde(bound = "T: SomeBound")]` // Parse `#[serde(bound = "T: SomeBound")]`
Meta(NameValue(m)) if m.path == BOUND => { Meta::NameValue(m) if m.path == BOUND => {
if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.lit) { if let Ok(where_predicates) = parse_lit_into_where(cx, BOUND, BOUND, &m.value) {
ser_bound.set(&m.path, where_predicates.clone()); ser_bound.set(&m.path, where_predicates.clone());
de_bound.set(&m.path, where_predicates); de_bound.set(&m.path, where_predicates);
} }
} }
// Parse `#[serde(bound(serialize = "...", deserialize = "..."))]` // Parse `#[serde(bound(serialize = "...", deserialize = "..."))]`
Meta(List(m)) if m.path == BOUND => { Meta::List(m) if m.path == BOUND => {
if let Ok((ser, de)) = get_where_predicates(cx, &m.nested) { if let Ok((ser, de)) = get_where_predicates(cx, m) {
ser_bound.set_opt(&m.path, ser); ser_bound.set_opt(&m.path, ser);
de_bound.set_opt(&m.path, de); de_bound.set_opt(&m.path, de);
} }
} }
// Parse `#[serde(borrow)]` // Parse `#[serde(borrow)]`
Meta(Path(word)) if word == BORROW => { Meta::Path(word) if word == BORROW => {
if let Ok(borrowable) = borrowable_lifetimes(cx, &ident, field) { if let Ok(borrowable) = borrowable_lifetimes(cx, &ident, field) {
borrowed_lifetimes.set(word, borrowable); borrowed_lifetimes.set(word, borrowable);
} }
} }
// Parse `#[serde(borrow = "'a + 'b")]` // Parse `#[serde(borrow = "'a + 'b")]`
Meta(NameValue(m)) if m.path == BORROW => { Meta::NameValue(m) if m.path == BORROW => {
if let Ok(lifetimes) = parse_lit_into_lifetimes(cx, &m.lit) { if let Ok(lifetimes) = parse_lit_into_lifetimes(cx, &m.value) {
if let Ok(borrowable) = borrowable_lifetimes(cx, &ident, field) { if let Ok(borrowable) = borrowable_lifetimes(cx, &ident, field) {
for lifetime in &lifetimes { for lifetime in &lifetimes {
if !borrowable.contains(lifetime) { if !borrowable.contains(lifetime) {
@@ -1300,18 +1288,18 @@ impl Field {
} }
// Parse `#[serde(getter = "...")]` // Parse `#[serde(getter = "...")]`
Meta(NameValue(m)) if m.path == GETTER => { Meta::NameValue(m) if m.path == GETTER => {
if let Ok(path) = parse_lit_into_expr_path(cx, GETTER, &m.lit) { if let Ok(path) = parse_lit_into_expr_path(cx, GETTER, &m.value) {
getter.set(&m.path, path); getter.set(&m.path, path);
} }
} }
// Parse `#[serde(flatten)]` // Parse `#[serde(flatten)]`
Meta(Path(word)) if word == FLATTEN => { Meta::Path(word) if word == FLATTEN => {
flatten.set_true(word); flatten.set_true(word);
} }
Meta(meta_item) => { meta_item => {
let path = meta_item let path = meta_item
.path() .path()
.into_token_stream() .into_token_stream()
@@ -1320,11 +1308,6 @@ impl Field {
let msg = format!("unknown serde field attribute `{}`", path); let msg = format!("unknown serde field attribute `{}`", path);
cx.error_spanned_by(meta_item.path(), msg); cx.error_spanned_by(meta_item.path(), msg);
} }
Lit(lit) => {
let msg = "unexpected literal in serde field attribute";
cx.error_spanned_by(lit, msg);
}
} }
} }
@@ -1478,29 +1461,36 @@ impl Field {
type SerAndDe<T> = (Option<T>, Option<T>); type SerAndDe<T> = (Option<T>, Option<T>);
fn get_ser_and_de<'a, 'b, T, F>( fn get_ser_and_de<'c, T, F>(
cx: &'b Ctxt, cx: &'c Ctxt,
attr_name: Symbol, attr_name: Symbol,
metas: &'a Punctuated<syn::NestedMeta, Token![,]>, metas: &MetaList,
f: F, f: F,
) -> Result<(VecAttr<'b, T>, VecAttr<'b, T>), ()> ) -> Result<(VecAttr<'c, T>, VecAttr<'c, T>), ()>
where where
T: 'a, F: Fn(&Ctxt, Symbol, Symbol, &syn::Expr) -> Result<T, ()>,
F: Fn(&Ctxt, Symbol, Symbol, &'a syn::Lit) -> Result<T, ()>,
{ {
let mut ser_meta = VecAttr::none(cx, attr_name); let mut ser_meta = VecAttr::none(cx, attr_name);
let mut de_meta = VecAttr::none(cx, attr_name); let mut de_meta = VecAttr::none(cx, attr_name);
for meta in metas { let nested = match metas.parse_args_with(Punctuated::<Meta, Token![,]>::parse_terminated) {
Ok(nested) => nested,
Err(err) => {
cx.syn_error(err);
return Err(());
}
};
for meta in nested {
match meta { match meta {
Meta(NameValue(meta)) if meta.path == SERIALIZE => { Meta::NameValue(meta) if meta.path == SERIALIZE => {
if let Ok(v) = f(cx, attr_name, SERIALIZE, &meta.lit) { if let Ok(v) = f(cx, attr_name, SERIALIZE, &meta.value) {
ser_meta.insert(&meta.path, v); ser_meta.insert(&meta.path, v);
} }
} }
Meta(NameValue(meta)) if meta.path == DESERIALIZE => { Meta::NameValue(meta) if meta.path == DESERIALIZE => {
if let Ok(v) = f(cx, attr_name, DESERIALIZE, &meta.lit) { if let Ok(v) = f(cx, attr_name, DESERIALIZE, &meta.value) {
de_meta.insert(&meta.path, v); de_meta.insert(&meta.path, v);
} }
} }
@@ -1519,41 +1509,36 @@ where
Ok((ser_meta, de_meta)) Ok((ser_meta, de_meta))
} }
fn get_renames<'a>( fn get_renames(cx: &Ctxt, items: &MetaList) -> Result<SerAndDe<syn::LitStr>, ()> {
cx: &Ctxt,
items: &'a Punctuated<syn::NestedMeta, Token![,]>,
) -> Result<SerAndDe<&'a syn::LitStr>, ()> {
let (ser, de) = get_ser_and_de(cx, RENAME, items, get_lit_str2)?; let (ser, de) = get_ser_and_de(cx, RENAME, items, get_lit_str2)?;
Ok((ser.at_most_one()?, de.at_most_one()?)) Ok((ser.at_most_one()?, de.at_most_one()?))
} }
fn get_multiple_renames<'a>( fn get_multiple_renames(
cx: &Ctxt, cx: &Ctxt,
items: &'a Punctuated<syn::NestedMeta, Token![,]>, items: &MetaList,
) -> Result<(Option<&'a syn::LitStr>, Vec<&'a syn::LitStr>), ()> { ) -> Result<(Option<syn::LitStr>, Vec<syn::LitStr>), ()> {
let (ser, de) = get_ser_and_de(cx, RENAME, items, get_lit_str2)?; let (ser, de) = get_ser_and_de(cx, RENAME, items, get_lit_str2)?;
Ok((ser.at_most_one()?, de.get())) Ok((ser.at_most_one()?, de.get()))
} }
fn get_where_predicates( fn get_where_predicates(
cx: &Ctxt, cx: &Ctxt,
items: &Punctuated<syn::NestedMeta, Token![,]>, items: &MetaList,
) -> Result<SerAndDe<Vec<syn::WherePredicate>>, ()> { ) -> Result<SerAndDe<Vec<syn::WherePredicate>>, ()> {
let (ser, de) = get_ser_and_de(cx, BOUND, items, parse_lit_into_where)?; let (ser, de) = get_ser_and_de(cx, BOUND, items, parse_lit_into_where)?;
Ok((ser.at_most_one()?, de.at_most_one()?)) Ok((ser.at_most_one()?, de.at_most_one()?))
} }
pub fn get_serde_meta_items(cx: &Ctxt, attr: &syn::Attribute) -> Result<Vec<syn::NestedMeta>, ()> { pub fn get_serde_meta_items(cx: &Ctxt, attr: &syn::Attribute) -> Result<Vec<Meta>, ()> {
if attr.path != SERDE { if attr.path() != SERDE {
return Ok(Vec::new()); return Ok(Vec::new());
} }
match attr.parse_meta() { let nested = attr.parse_args_with(Punctuated::<Meta, Token![,]>::parse_terminated);
Ok(List(meta)) => Ok(meta.nested.into_iter().collect()),
Ok(other) => { match nested {
cx.error_spanned_by(other, "expected #[serde(...)]"); Ok(nested) => Ok(nested.into_iter().collect()),
Err(())
}
Err(err) => { Err(err) => {
cx.syn_error(err); cx.syn_error(err);
Err(()) Err(())
@@ -1561,29 +1546,30 @@ pub fn get_serde_meta_items(cx: &Ctxt, attr: &syn::Attribute) -> Result<Vec<syn:
} }
} }
fn get_lit_str<'a>(cx: &Ctxt, attr_name: Symbol, lit: &'a syn::Lit) -> Result<&'a syn::LitStr, ()> { fn get_lit_str(cx: &Ctxt, attr_name: Symbol, lit: &syn::Expr) -> Result<syn::LitStr, ()> {
get_lit_str2(cx, attr_name, attr_name, lit) get_lit_str2(cx, attr_name, attr_name, lit)
} }
fn get_lit_str2<'a>( fn get_lit_str2(
cx: &Ctxt, cx: &Ctxt,
attr_name: Symbol, attr_name: Symbol,
meta_item_name: Symbol, meta_item_name: Symbol,
lit: &'a syn::Lit, expr: &syn::Expr,
) -> Result<&'a syn::LitStr, ()> { ) -> Result<syn::LitStr, ()> {
if let syn::Lit::Str(lit) = lit { if let Expr::Lit(expr) = expr {
Ok(lit) if let syn::Lit::Str(lit) = &expr.lit {
} else { return Ok(lit.clone());
}
}
let msg = format!( let msg = format!(
"expected serde {} attribute to be a string: `{} = \"...\"`", "expected serde {} attribute to be a string: `{} = \"...\"`",
attr_name, meta_item_name, attr_name, meta_item_name,
); );
cx.error_spanned_by(lit, msg); cx.error_spanned_by(expr, msg);
Err(()) Err(())
} }
}
fn parse_lit_into_path(cx: &Ctxt, attr_name: Symbol, lit: &syn::Lit) -> Result<syn::Path, ()> { fn parse_lit_into_path(cx: &Ctxt, attr_name: Symbol, lit: &syn::Expr) -> Result<syn::Path, ()> {
let string = get_lit_str(cx, attr_name, lit)?; let string = get_lit_str(cx, attr_name, lit)?;
string.parse().map_err(|_| { string.parse().map_err(|_| {
let msg = format!("failed to parse path: {:?}", string.value()); let msg = format!("failed to parse path: {:?}", string.value());
@@ -1594,7 +1580,7 @@ fn parse_lit_into_path(cx: &Ctxt, attr_name: Symbol, lit: &syn::Lit) -> Result<s
fn parse_lit_into_expr_path( fn parse_lit_into_expr_path(
cx: &Ctxt, cx: &Ctxt,
attr_name: Symbol, attr_name: Symbol,
lit: &syn::Lit, lit: &syn::Expr,
) -> Result<syn::ExprPath, ()> { ) -> Result<syn::ExprPath, ()> {
let string = get_lit_str(cx, attr_name, lit)?; let string = get_lit_str(cx, attr_name, lit)?;
string.parse().map_err(|_| { string.parse().map_err(|_| {
@@ -1607,7 +1593,7 @@ fn parse_lit_into_where(
cx: &Ctxt, cx: &Ctxt,
attr_name: Symbol, attr_name: Symbol,
meta_item_name: Symbol, meta_item_name: Symbol,
lit: &syn::Lit, lit: &syn::Expr,
) -> Result<Vec<syn::WherePredicate>, ()> { ) -> Result<Vec<syn::WherePredicate>, ()> {
let string = get_lit_str2(cx, attr_name, meta_item_name, lit)?; let string = get_lit_str2(cx, attr_name, meta_item_name, lit)?;
@@ -1617,7 +1603,7 @@ fn parse_lit_into_where(
.map_err(|err| cx.error_spanned_by(lit, err)) .map_err(|err| cx.error_spanned_by(lit, err))
} }
fn parse_lit_into_ty(cx: &Ctxt, attr_name: Symbol, lit: &syn::Lit) -> Result<syn::Type, ()> { fn parse_lit_into_ty(cx: &Ctxt, attr_name: Symbol, lit: &syn::Expr) -> Result<syn::Type, ()> {
let string = get_lit_str(cx, attr_name, lit)?; let string = get_lit_str(cx, attr_name, lit)?;
string.parse().map_err(|_| { string.parse().map_err(|_| {
@@ -1628,7 +1614,7 @@ fn parse_lit_into_ty(cx: &Ctxt, attr_name: Symbol, lit: &syn::Lit) -> Result<syn
// Parses a string literal like "'a + 'b + 'c" containing a nonempty list of // Parses a string literal like "'a + 'b + 'c" containing a nonempty list of
// lifetimes separated by `+`. // lifetimes separated by `+`.
fn parse_lit_into_lifetimes(cx: &Ctxt, lit: &syn::Lit) -> Result<BTreeSet<syn::Lifetime>, ()> { fn parse_lit_into_lifetimes(cx: &Ctxt, lit: &syn::Expr) -> Result<BTreeSet<syn::Lifetime>, ()> {
let string = get_lit_str(cx, BORROW, lit)?; let string = get_lit_str(cx, BORROW, lit)?;
if let Ok(lifetimes) = string.parse_with(|input: ParseStream| { if let Ok(lifetimes) = string.parse_with(|input: ParseStream| {
@@ -1853,11 +1839,10 @@ fn collect_lifetimes(ty: &syn::Type, out: &mut BTreeSet<syn::Lifetime>) {
syn::GenericArgument::Type(ty) => { syn::GenericArgument::Type(ty) => {
collect_lifetimes(ty, out); collect_lifetimes(ty, out);
} }
syn::GenericArgument::Binding(binding) => { syn::GenericArgument::AssocType(binding) => {
collect_lifetimes(&binding.ty, out); collect_lifetimes(&binding.ty, out);
} }
syn::GenericArgument::Constraint(_) _ => {}
| syn::GenericArgument::Const(_) => {}
} }
} }
} }
+12 -5
View File
@@ -179,10 +179,13 @@ impl ReplaceReceiver<'_> {
for arg in &mut arguments.args { for arg in &mut arguments.args {
match arg { match arg {
GenericArgument::Type(arg) => self.visit_type_mut(arg), GenericArgument::Type(arg) => self.visit_type_mut(arg),
GenericArgument::Binding(arg) => self.visit_type_mut(&mut arg.ty), GenericArgument::AssocType(arg) => self.visit_type_mut(&mut arg.ty),
GenericArgument::Lifetime(_) GenericArgument::Lifetime(_)
| GenericArgument::Constraint(_) | GenericArgument::Const(_)
| GenericArgument::Const(_) => {} | GenericArgument::AssocConst(_)
| GenericArgument::Constraint(_) => {}
#[cfg_attr(all(test, exhaustive), deny(non_exhaustive_omitted_patterns))]
_ => {}
} }
} }
} }
@@ -205,7 +208,9 @@ impl ReplaceReceiver<'_> {
fn visit_type_param_bound_mut(&mut self, bound: &mut TypeParamBound) { fn visit_type_param_bound_mut(&mut self, bound: &mut TypeParamBound) {
match bound { match bound {
TypeParamBound::Trait(bound) => self.visit_path_mut(&mut bound.path), TypeParamBound::Trait(bound) => self.visit_path_mut(&mut bound.path),
TypeParamBound::Lifetime(_) => {} TypeParamBound::Lifetime(_) | TypeParamBound::Verbatim(_) => {}
#[cfg_attr(all(test, exhaustive), deny(non_exhaustive_omitted_patterns))]
_ => {}
} }
} }
@@ -229,7 +234,9 @@ impl ReplaceReceiver<'_> {
self.visit_type_param_bound_mut(bound); self.visit_type_param_bound_mut(bound);
} }
} }
WherePredicate::Lifetime(_) | WherePredicate::Eq(_) => {} WherePredicate::Lifetime(_) => {}
#[cfg_attr(all(test, exhaustive), deny(non_exhaustive_omitted_patterns))]
_ => {}
} }
} }
} }
+1 -1
View File
@@ -17,7 +17,7 @@ path = "lib.rs"
[dependencies] [dependencies]
proc-macro2 = "1.0" proc-macro2 = "1.0"
quote = "1.0" quote = "1.0"
syn = { version = "1.0.104", default-features = false, features = ["derive", "parsing", "printing", "clone-impls"] } syn = { version = "2.0", default-features = false, features = ["derive", "parsing", "printing", "clone-impls"] }
[package.metadata.docs.rs] [package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"] targets = ["x86_64-unknown-linux-gnu"]
+1 -1
View File
@@ -1,4 +1,4 @@
error: unexpected end of input, expected literal error: unexpected end of input, expected an expression
--> tests/ui/malformed/cut_off.rs:4:17 --> tests/ui/malformed/cut_off.rs:4:17
| |
4 | #[serde(rename =)] 4 | #[serde(rename =)]
@@ -1,11 +1,11 @@
error: expected #[serde(...)] error: expected attribute arguments in parentheses: #[serde(...)]
--> tests/ui/malformed/not_list.rs:4:3 --> tests/ui/malformed/not_list.rs:4:3
| |
4 | #[serde] 4 | #[serde]
| ^^^^^ | ^^^^^
error: expected #[serde(...)] error: expected parentheses: #[serde(...)]
--> tests/ui/malformed/not_list.rs:5:3 --> tests/ui/malformed/not_list.rs:5:9
| |
5 | #[serde = "?"] 5 | #[serde = "?"]
| ^^^^^^^^^^^ | ^
@@ -1,4 +1,4 @@
error: unexpected literal in serde container attribute error: expected path
--> tests/ui/unexpected-literal/container.rs:4:9 --> tests/ui/unexpected-literal/container.rs:4:9
| |
4 | #[serde("literal")] 4 | #[serde("literal")]
@@ -1,4 +1,4 @@
error: unexpected literal in serde field attribute error: expected path
--> tests/ui/unexpected-literal/field.rs:5:13 --> tests/ui/unexpected-literal/field.rs:5:13
| |
5 | #[serde("literal")] 5 | #[serde("literal")]
@@ -1,4 +1,4 @@
error: unexpected literal in serde variant attribute error: expected path
--> tests/ui/unexpected-literal/variant.rs:5:13 --> tests/ui/unexpected-literal/variant.rs:5:13
| |
5 | #[serde("literal")] 5 | #[serde("literal")]