Compare commits

..

No commits in common. "master" and "2.0.4" have entirely different histories.

24 changed files with 92 additions and 249 deletions

View file

@ -41,7 +41,6 @@ jobs:
- run: cargo test --workspace --exclude thiserror_no_std_test
- run: cargo test --manifest-path tests/no-std/Cargo.toml
if: matrix.rust != '1.70.0'
- run: cargo test --no-default-features
- uses: actions/upload-artifact@v4
if: matrix.rust == 'nightly' && always()
with:

5
.gitignore vendored
View file

@ -1,2 +1,3 @@
/target/
/Cargo.lock
/target
**/*.rs.bk
Cargo.lock

View file

@ -1,6 +1,6 @@
[package]
name = "thiserror"
version = "2.0.12"
version = "2.0.4"
authors = ["David Tolnay <dtolnay@gmail.com>"]
categories = ["rust-patterns"]
description = "derive(Error)"
@ -28,7 +28,7 @@ default = ["std"]
std = []
[dependencies]
thiserror-impl = { version = "=2.0.12", path = "impl" }
thiserror-impl = { version = "=2.0.4", path = "impl" }
[dev-dependencies]
anyhow = "1.0.73"
@ -41,9 +41,4 @@ members = ["impl", "tests/no-std"]
[package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"]
rustdoc-args = [
"--generate-link-to-definition",
"--extern-html-root-url=core=https://doc.rust-lang.org",
"--extern-html-root-url=alloc=https://doc.rust-lang.org",
"--extern-html-root-url=std=https://doc.rust-lang.org",
]
rustdoc-args = ["--generate-link-to-definition"]

View file

@ -1,6 +1,6 @@
[package]
name = "thiserror-impl"
version = "2.0.12"
version = "2.0.4"
authors = ["David Tolnay <dtolnay@gmail.com>"]
description = "Implementation detail of the `thiserror` crate"
edition = "2021"
@ -18,10 +18,4 @@ syn = "2.0.87"
[package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"]
rustdoc-args = [
"--generate-link-to-definition",
"--extern-html-root-url=core=https://doc.rust-lang.org",
"--extern-html-root-url=alloc=https://doc.rust-lang.org",
"--extern-html-root-url=std=https://doc.rust-lang.org",
"--extern-html-root-url=proc_macro=https://doc.rust-lang.org",
]
rustdoc-args = ["--generate-link-to-definition"]

View file

@ -1,9 +1,8 @@
use crate::ast::{Enum, Field, Input, Struct};
use crate::attr::Trait;
use crate::fallback;
use crate::generics::InferredBounds;
use crate::unraw::MemberUnraw;
use proc_macro2::{Ident, Span, TokenStream};
use proc_macro2::{Ident, TokenStream};
use quote::{format_ident, quote, quote_spanned, ToTokens};
use std::collections::BTreeSet as Set;
use syn::{DeriveInput, GenericArgument, PathArguments, Result, Token, Type};
@ -12,9 +11,9 @@ pub fn derive(input: &DeriveInput) -> TokenStream {
match try_expand(input) {
Ok(expanded) => expanded,
// If there are invalid attributes in the input, expand to an Error impl
// anyway to minimize spurious secondary errors in other code that uses
// anyway to minimize spurious knock-on errors in other code that uses
// this type as an Error.
Err(error) => fallback::expand(input, error),
Err(error) => fallback(input, error),
}
}
@ -27,8 +26,36 @@ fn try_expand(input: &DeriveInput) -> Result<TokenStream> {
})
}
fn fallback(input: &DeriveInput, error: syn::Error) -> TokenStream {
let ty = &input.ident;
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
let error = error.to_compile_error();
quote! {
#error
#[allow(unused_qualifications)]
#[automatically_derived]
impl #impl_generics ::thiserror::__private::Error for #ty #ty_generics #where_clause
where
// Work around trivial bounds being unstable.
// https://github.com/rust-lang/rust/issues/48214
for<'workaround> #ty #ty_generics: ::core::fmt::Debug,
{}
#[allow(unused_qualifications)]
#[automatically_derived]
impl #impl_generics ::core::fmt::Display for #ty #ty_generics #where_clause {
fn fmt(&self, __formatter: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::unreachable!()
}
}
}
}
fn impl_struct(input: Struct) -> TokenStream {
let ty = call_site_ident(&input.ident);
let ty = &input.ident;
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
let mut error_inferred_bounds = InferredBounds::new();
@ -169,26 +196,16 @@ fn impl_struct(input: Struct) -> TokenStream {
let from = unoptional_type(from_field.ty);
let source_var = Ident::new("source", span);
let body = from_initializer(from_field, backtrace_field, &source_var);
let from_function = quote! {
fn from(#source_var: #from) -> Self {
#ty #body
}
};
let from_impl = quote_spanned! {span=>
quote_spanned! {span=>
#[allow(unused_qualifications, clippy::needless_lifetimes)]
#[automatically_derived]
impl #impl_generics ::core::convert::From<#from> for #ty #ty_generics #where_clause {
#from_function
#[allow(deprecated)]
fn from(#source_var: #from) -> Self {
#ty #body
}
}
};
Some(quote! {
#[allow(
deprecated,
unused_qualifications,
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
)]
#from_impl
})
}
});
if input.generics.type_params().next().is_some() {
@ -211,7 +228,7 @@ fn impl_struct(input: Struct) -> TokenStream {
}
fn impl_enum(input: Enum) -> TokenStream {
let ty = call_site_ident(&input.ident);
let ty = &input.ident;
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
let mut error_inferred_bounds = InferredBounds::new();
@ -444,25 +461,15 @@ fn impl_enum(input: Enum) -> TokenStream {
let from = unoptional_type(from_field.ty);
let source_var = Ident::new("source", span);
let body = from_initializer(from_field, backtrace_field, &source_var);
let from_function = quote! {
fn from(#source_var: #from) -> Self {
#ty::#variant #body
}
};
let from_impl = quote_spanned! {span=>
Some(quote_spanned! {span=>
#[allow(unused_qualifications, clippy::needless_lifetimes)]
#[automatically_derived]
impl #impl_generics ::core::convert::From<#from> for #ty #ty_generics #where_clause {
#from_function
#[allow(deprecated)]
fn from(#source_var: #from) -> Self {
#ty::#variant #body
}
}
};
Some(quote! {
#[allow(
deprecated,
unused_qualifications,
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
)]
#from_impl
})
});
@ -485,14 +492,6 @@ fn impl_enum(input: Enum) -> TokenStream {
}
}
// Create an ident with which we can expand `impl Trait for #ident {}` on a
// deprecated type without triggering deprecation warning on the generated impl.
pub(crate) fn call_site_ident(ident: &Ident) -> Ident {
let mut ident = ident.clone();
ident.set_span(ident.span().resolved_at(Span::call_site()));
ident
}
fn fields_pat(fields: &[Field]) -> TokenStream {
let mut members = fields.iter().map(|field| &field.member).peekable();
match members.peek() {

View file

@ -1,32 +0,0 @@
use crate::expand::call_site_ident;
use proc_macro2::TokenStream;
use quote::quote;
use syn::DeriveInput;
pub(crate) fn expand(input: &DeriveInput, error: syn::Error) -> TokenStream {
let ty = call_site_ident(&input.ident);
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
let error = error.to_compile_error();
quote! {
#error
#[allow(unused_qualifications)]
#[automatically_derived]
impl #impl_generics ::thiserror::__private::Error for #ty #ty_generics #where_clause
where
// Work around trivial bounds being unstable.
// https://github.com/rust-lang/rust/issues/48214
for<'workaround> #ty #ty_generics: ::core::fmt::Debug,
{}
#[allow(unused_qualifications)]
#[automatically_derived]
impl #impl_generics ::core::fmt::Display for #ty #ty_generics #where_clause {
fn fmt(&self, __formatter: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::unreachable!()
}
}
}
}

View file

@ -107,14 +107,12 @@ impl Display<'_> {
}
};
infinite_recursive |= member == *"self" && bound == Trait::Display;
let field = match member_index.get(&member) {
Some(&field) => field,
None => {
out += &member.to_string();
continue;
}
};
implied_bounds.insert((field, bound));
if let Some(&field) = member_index.get(&member) {
implied_bounds.insert((field, bound));
} else {
out += &member.to_string();
continue;
}
let formatvar_prefix = if bonus_display {
"__display"
} else if bound == Trait::Pointer {
@ -131,17 +129,15 @@ impl Display<'_> {
while user_named_args.contains(&formatvar) {
formatvar = IdentUnraw::new(format_ident!("_{}", formatvar.to_string()));
}
formatvar.set_span(span);
out += &formatvar.to_string();
if !macro_named_args.insert(formatvar.clone()) {
// Already added to bindings by a previous use.
continue;
}
let mut binding_value = match &member {
let binding_value = match &member {
MemberUnraw::Unnamed(index) => format_ident!("_{}", index),
MemberUnraw::Named(ident) => ident.to_local(),
};
binding_value.set_span(span.resolved_at(fields[field].member.span()));
let wrapped_binding_value = if bonus_display {
quote_spanned!(span=> #binding_value.as_display())
} else if bound == Trait::Pointer {

View file

@ -25,12 +25,11 @@ impl<'a> ParamsInScope<'a> {
fn crawl(in_scope: &ParamsInScope, ty: &Type, found: &mut bool) {
if let Type::Path(ty) = ty {
if let Some(qself) = &ty.qself {
crawl(in_scope, &qself.ty, found);
} else {
let front = ty.path.segments.first().unwrap();
if front.arguments.is_none() && in_scope.names.contains(&front.ident) {
*found = true;
if ty.qself.is_none() {
if let Some(ident) = ty.path.get_ident() {
if in_scope.names.contains(ident) {
*found = true;
}
}
}
for segment in &ty.path.segments {

View file

@ -21,7 +21,6 @@ extern crate proc_macro;
mod ast;
mod attr;
mod expand;
mod fallback;
mod fmt;
mod generics;
mod prop;

View file

@ -28,10 +28,6 @@ impl IdentUnraw {
}
unraw
}
pub fn set_span(&mut self, span: Span) {
self.0.set_span(span);
}
}
impl Display for IdentUnraw {

View file

@ -44,7 +44,7 @@ impl<'a> AsDynError<'a> for dyn Error + Send + Sync + UnwindSafe + 'a {
#[doc(hidden)]
pub trait Sealed {}
impl<T: Error> Sealed for T {}
impl Sealed for dyn Error + '_ {}
impl Sealed for dyn Error + Send + '_ {}
impl Sealed for dyn Error + Send + Sync + '_ {}
impl Sealed for dyn Error + Send + Sync + UnwindSafe + '_ {}
impl<'a> Sealed for dyn Error + 'a {}
impl<'a> Sealed for dyn Error + Send + 'a {}
impl<'a> Sealed for dyn Error + Send + Sync + 'a {}
impl<'a> Sealed for dyn Error + Send + Sync + UnwindSafe + 'a {}

View file

@ -9,6 +9,8 @@
//! This library provides a convenient derive macro for the standard library's
//! [`std::error::Error`] trait.
//!
//! [`std::error::Error`]: https://doc.rust-lang.org/std/error/trait.Error.html
//!
//! <br>
//!
//! # Example
@ -257,9 +259,8 @@
//! [`anyhow`]: https://github.com/dtolnay/anyhow
#![no_std]
#![doc(html_root_url = "https://docs.rs/thiserror/2.0.12")]
#![doc(html_root_url = "https://docs.rs/thiserror/2.0.4")]
#![allow(
clippy::elidable_lifetime_names,
clippy::module_name_repetitions,
clippy::needless_lifetimes,
clippy::return_self_not_must_use,

View file

@ -1,4 +1,3 @@
#![cfg(feature = "std")]
#![cfg_attr(thiserror_nightly_testing, feature(error_generic_member_access))]
use thiserror::Error;

10
tests/test_deprecated.rs Normal file
View file

@ -0,0 +1,10 @@
#![deny(deprecated, clippy::all, clippy::pedantic)]
use thiserror::Error;
#[derive(Error, Debug)]
pub enum Error {
#[deprecated]
#[error("...")]
Deprecated,
}

View file

@ -1,5 +1,4 @@
#![allow(
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
clippy::needless_raw_string_hashes,
clippy::trivially_copy_pass_by_ref,
@ -267,18 +266,18 @@ fn test_pointer() {
}
#[test]
fn test_macro_rules_variant_from_call_site() {
fn test_macro_rules() {
// Regression test for https://github.com/dtolnay/thiserror/issues/86
macro_rules! decl_error {
($variant:ident($value:ident)) => {
#[derive(Error, Debug)]
#[derive(Debug, Error)]
pub enum Error0 {
#[error("{0:?}")]
$variant($value),
}
#[derive(Error, Debug)]
#[derive(Debug, Error)]
#[error("{0:?}")]
pub enum Error1 {
$variant($value),
@ -292,30 +291,6 @@ fn test_macro_rules_variant_from_call_site() {
assert("0", Error1::Repro(0));
}
#[test]
fn test_macro_rules_message_from_call_site() {
// Regression test for https://github.com/dtolnay/thiserror/issues/398
macro_rules! decl_error {
($($errors:tt)*) => {
#[derive(Error, Debug)]
pub enum Error {
$($errors)*
}
};
}
decl_error! {
#[error("{0}")]
Unnamed(u8),
#[error("{x}")]
Named { x: u8 },
}
assert("0", Error::Unnamed(0));
assert("0", Error::Named { x: 0 });
}
#[test]
fn test_raw() {
#[derive(Error, Debug)]

View file

@ -1,7 +1,6 @@
#![allow(clippy::iter_cloned_collect, clippy::uninlined_format_args)]
use core::fmt::Display;
#[cfg(feature = "std")]
use std::path::PathBuf;
use thiserror::Error;
@ -91,7 +90,6 @@ fn test_rustup() {
}
// Regression test for https://github.com/dtolnay/thiserror/issues/335
#[cfg(feature = "std")]
#[test]
#[allow(non_snake_case)]
fn test_assoc_type_equality_constraint() {

View file

@ -1,7 +1,6 @@
#![allow(clippy::needless_late_init, clippy::uninlined_format_args)]
use core::fmt::{self, Debug, Display};
use core::str::FromStr;
use thiserror::Error;
pub struct NoFormat;
@ -161,24 +160,6 @@ pub struct StructFromGeneric<E> {
#[error(transparent)]
pub struct StructTransparentGeneric<E>(pub E);
// Should expand to:
//
// impl<T: FromStr> Display for AssociatedTypeError<T>
// where
// T::Err: Display;
//
// impl<T: FromStr> Error for AssociatedTypeError<T>
// where
// Self: Debug + Display;
//
#[derive(Error, Debug)]
pub enum AssociatedTypeError<T: FromStr> {
#[error("couldn't parse matrix")]
Other,
#[error("couldn't parse entry: {0}")]
EntryParseError(T::Err),
}
// Regression test for https://github.com/dtolnay/thiserror/issues/345
#[test]
fn test_no_bound_on_named_fmt() {

View file

@ -4,17 +4,6 @@ use thiserror::Error;
pub use std::error::Error;
#[test]
fn test_allow_attributes() {
#![deny(clippy::allow_attributes)]
#[derive(Error, Debug)]
#[error("...")]
pub struct MyError(#[from] anyhow::Error);
let _: MyError;
}
#[test]
fn test_unused_qualifications() {
#![deny(unused_qualifications)]
@ -23,7 +12,7 @@ fn test_unused_qualifications() {
// std::error::Error is already imported in the caller's scope so it must
// suppress unused_qualifications.
#[derive(Error, Debug)]
#[derive(Debug, Error)]
#[error("...")]
pub struct MyError;
@ -33,9 +22,9 @@ fn test_unused_qualifications() {
#[test]
fn test_needless_lifetimes() {
#![allow(dead_code)]
#![deny(clippy::elidable_lifetime_names, clippy::needless_lifetimes)]
#![deny(clippy::needless_lifetimes)]
#[derive(Error, Debug)]
#[derive(Debug, Error)]
#[error("...")]
pub enum MyError<'a> {
A(#[from] std::io::Error),
@ -44,53 +33,3 @@ fn test_needless_lifetimes() {
let _: MyError;
}
#[test]
fn test_deprecated() {
#![deny(deprecated)]
#[derive(Error, Debug)]
#[deprecated]
#[error("...")]
pub struct DeprecatedStruct;
#[derive(Error, Debug)]
#[error("{message} {}", .message)]
pub struct DeprecatedStructField {
#[deprecated]
message: String,
}
#[derive(Error, Debug)]
#[deprecated]
pub enum DeprecatedEnum {
#[error("...")]
Variant,
}
#[derive(Error, Debug)]
pub enum DeprecatedVariant {
#[deprecated]
#[error("...")]
Variant,
}
#[derive(Error, Debug)]
pub enum DeprecatedFrom {
#[error(transparent)]
Variant(
#[from]
#[allow(deprecated)]
DeprecatedStruct,
),
}
#[allow(deprecated)]
let _: DeprecatedStruct;
#[allow(deprecated)]
let _: DeprecatedStructField;
#[allow(deprecated)]
let _ = DeprecatedEnum::Variant;
#[allow(deprecated)]
let _ = DeprecatedVariant::Variant;
}

View file

@ -1,4 +1,3 @@
#![cfg(feature = "std")]
#![cfg_attr(thiserror_nightly_testing, feature(error_generic_member_access))]
#[cfg(thiserror_nightly_testing)]

View file

@ -1,5 +1,3 @@
#![cfg(feature = "std")]
use core::fmt::Display;
use ref_cast::RefCast;
use std::path::{Path, PathBuf};

View file

@ -1,6 +1,6 @@
use thiserror::Error;
#[derive(Error, Debug)]
#[derive(Debug, Error)]
pub struct Error {
#[source]
source: std::io::Error,

View file

@ -1,8 +1,6 @@
error[E0277]: `MyError` doesn't implement `std::fmt::Display`
--> tests/ui/missing-display.rs:4:10
|
3 | #[derive(Error, Debug)]
| ----- in this derive macro expansion
4 | pub enum MyError {
| ^^^^^^^ `MyError` cannot be formatted with the default formatter
|
@ -13,4 +11,3 @@ note: required by a bound in `std::error::Error`
|
| pub trait Error: Debug + Display {
| ^^^^^^^ required by this bound in `Error`
= note: this error originates in the derive macro `Error` (in Nightly builds, run with -Z macro-backtrace for more info)

View file

@ -5,7 +5,7 @@ pub enum Error {
#[error("failed to open")]
OpenFile(#[from] std::io::Error),
#[error("failed to close")]
CloseFile(#[from] std::io::Error),
CloseFIle(#[from] std::io::Error),
}
fn main() {}

View file

@ -4,5 +4,5 @@ error[E0119]: conflicting implementations of trait `From<std::io::Error>` for ty
6 | OpenFile(#[from] std::io::Error),
| ------- first implementation here
7 | #[error("failed to close")]
8 | CloseFile(#[from] std::io::Error),
8 | CloseFIle(#[from] std::io::Error),
| ^^^^^^^ conflicting implementation for `Error`