Add error message for non-static source

This commit is contained in:
David Tolnay 2020-03-20 22:38:58 -07:00
parent db357fac83
commit a1ae05c954
No known key found for this signature in database
GPG key ID: F9BA143B95FF6D82
3 changed files with 46 additions and 1 deletions

View file

@ -2,7 +2,7 @@ use crate::ast::{Enum, Field, Input, Struct, Variant};
use crate::attr::Attrs;
use quote::ToTokens;
use std::collections::BTreeSet as Set;
use syn::{Error, Member, Result};
use syn::{Error, GenericArgument, Member, PathArguments, Result, Type};
impl Input<'_> {
pub(crate) fn validate(&self) -> Result<()> {
@ -181,6 +181,14 @@ fn check_field_attrs(fields: &[Field]) -> Result<()> {
));
}
}
if let Some(source_field) = source_field.or(from_field) {
if contains_non_static_lifetime(source_field) {
return Err(Error::new_spanned(
source_field.original,
"non-static lifetimes are not allowed in the source of an error",
));
}
}
Ok(())
}
@ -191,3 +199,22 @@ fn same_member(one: &Field, two: &Field) -> bool {
_ => unreachable!(),
}
}
fn contains_non_static_lifetime(field: &Field) -> bool {
let ty = match field.ty {
Type::Path(ty) => ty,
_ => return false, // maybe implement later if there are common other cases
};
let bracketed = match &ty.path.segments.last().unwrap().arguments {
PathArguments::AngleBracketed(bracketed) => bracketed,
_ => return false,
};
for arg in &bracketed.args {
if let GenericArgument::Lifetime(lifetime) = arg {
if lifetime.ident != "static" {
return true;
}
}
}
false
}

13
tests/ui/lifetime.rs Normal file
View file

@ -0,0 +1,13 @@
use thiserror::Error;
#[derive(Error, Debug)]
#[error("error")]
struct Error<'a>(#[from] Inner<'a>);
#[derive(Error, Debug)]
#[error("{0}")]
struct Inner<'a>(&'a str);
fn main() -> Result<(), Error<'static>> {
Err(Error(Inner("some text")))
}

5
tests/ui/lifetime.stderr Normal file
View file

@ -0,0 +1,5 @@
error: non-static lifetimes are not allowed in the source of an error
--> $DIR/lifetime.rs:5:18
|
5 | struct Error<'a>(#[from] Inner<'a>);
| ^^^^^^^^^^^^^^^^^