Compare commits

...

35 commits

Author SHA1 Message Date
David Tolnay
95a5126693
Release 2.0.12 2025-03-02 20:24:05 -08:00
David Tolnay
76490f743e
Merge pull request #413 from dtolnay/elidablelifetime
Clippy elidable_lifetime_names lint
2025-03-02 23:23:21 -05:00
David Tolnay
9f27b766f5
Ignore elidable_lifetime_names pedantic clippy lint
warning: the following explicit lifetimes could be elided: 'a
     --> src/var.rs:5:6
      |
    5 | impl<'a, T: Pointer + ?Sized> Pointer for Var<'a, T> {
      |      ^^                                       ^^
      |
      = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
      = note: `-W clippy::elidable-lifetime-names` implied by `-W clippy::pedantic`
      = help: to override `-W clippy::pedantic` add `#[allow(clippy::elidable_lifetime_names)]`
    help: elide the lifetimes
      |
    5 - impl<'a, T: Pointer + ?Sized> Pointer for Var<'a, T> {
    5 + impl<T: Pointer + ?Sized> Pointer for Var<'_, T> {
      |

    warning: the following explicit lifetimes could be elided: 'a
      --> tests/test_lints.rs:40:22
       |
    40 |     pub enum MyError<'a> {
       |                      ^^
       |
       = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
       = note: `-W clippy::elidable-lifetime-names` implied by `-W clippy::pedantic`
       = help: to override `-W clippy::pedantic` add `#[allow(clippy::elidable_lifetime_names)]`
    help: elide the lifetimes
       |
    40 -     pub enum MyError<'a> {
    40 +     pub enum MyError'_> {
       |

    warning: the following explicit lifetimes could be elided: 'a
       --> tests/test_display.rs:157:14
        |
    157 |         impl<'a> Display for Msg<'a> {
        |              ^^                  ^^
        |
        = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
        = note: `-W clippy::elidable-lifetime-names` implied by `-W clippy::pedantic`
        = help: to override `-W clippy::pedantic` add `#[allow(clippy::elidable_lifetime_names)]`
    help: elide the lifetimes
        |
    157 -         impl<'a> Display for Msg<'a> {
    157 +         impl Display for Msg<'_> {
        |
2025-03-02 19:56:43 -08:00
David Tolnay
daf2a6f36e
Resolve some elidable_lifetime_names pedantic clippy lint
warning: the following explicit lifetimes could be elided: 'a
      --> src/aserror.rs:47:6
       |
    47 | impl<'a> Sealed for dyn Error + 'a {}
       |      ^^                         ^^
       |
       = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
       = note: `-W clippy::elidable-lifetime-names` implied by `-W clippy::pedantic`
       = help: to override `-W clippy::pedantic` add `#[allow(clippy::elidable_lifetime_names)]`
    help: elide the lifetimes
       |
    47 - impl<'a> Sealed for dyn Error + 'a {}
    47 + impl Sealed for dyn Error + '_ {}
       |

    warning: the following explicit lifetimes could be elided: 'a
      --> src/aserror.rs:48:6
       |
    48 | impl<'a> Sealed for dyn Error + Send + 'a {}
       |      ^^                                ^^
       |
       = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
    help: elide the lifetimes
       |
    48 - impl<'a> Sealed for dyn Error + Send + 'a {}
    48 + impl Sealed for dyn Error + Send + '_ {}
       |

    warning: the following explicit lifetimes could be elided: 'a
      --> src/aserror.rs:49:6
       |
    49 | impl<'a> Sealed for dyn Error + Send + Sync + 'a {}
       |      ^^                                       ^^
       |
       = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
    help: elide the lifetimes
       |
    49 - impl<'a> Sealed for dyn Error + Send + Sync + 'a {}
    49 + impl Sealed for dyn Error + Send + Sync + '_ {}
       |

    warning: the following explicit lifetimes could be elided: 'a
      --> src/aserror.rs:50:6
       |
    50 | impl<'a> Sealed for dyn Error + Send + Sync + UnwindSafe + 'a {}
       |      ^^                                                    ^^
       |
       = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#elidable_lifetime_names
    help: elide the lifetimes
       |
    50 - impl<'a> Sealed for dyn Error + Send + Sync + UnwindSafe + 'a {}
    50 + impl Sealed for dyn Error + Send + Sync + UnwindSafe + '_ {}
       |
2025-03-02 19:52:49 -08:00
David Tolnay
5f07160c35
Point standard library links to stable 2025-02-20 05:24:47 -08:00
David Tolnay
6706a5121b
Convert html links to intra-doc links 2025-02-19 22:30:39 -08:00
David Tolnay
2706873a04
More precise gitignore patterns 2025-01-23 01:36:08 -08:00
David Tolnay
70bc20d848
Remove **/*.rs.bk from project-specific gitignore
Cargo stopped generating this in its project template 5 years ago. It
would belong in a global gitignore instead.
2025-01-22 19:30:31 -08:00
David Tolnay
0f532e326e
Release 2.0.11 2025-01-10 10:03:49 -08:00
David Tolnay
3d15543a91
Merge pull request #410 from dtolnay/testnostd
Add CI step to test with "std" disabled
2025-01-10 10:03:23 -08:00
David Tolnay
1a226ae42c
Disable two more integration tests in no-std mode 2025-01-10 10:00:21 -08:00
David Tolnay
8b5f2d78f0
Fix unused import in test when built without std
warning: unused import: `std::path::PathBuf`
     --> tests/test_expr.rs:4:5
      |
    4 | use std::path::PathBuf;
      |     ^^^^^^^^^^^^^^^^^^
      |
      = note: `#[warn(unused_imports)]` on by default
2025-01-10 09:56:35 -08:00
David Tolnay
eecd247cdf
Add CI step to test with "std" disabled 2025-01-10 09:55:47 -08:00
David Tolnay
8f2a76b4ba
Merge pull request #409 from Maytha8/std-tests
Add feature gate to tests that use std
2025-01-10 05:30:24 -08:00
Maytham Alsudany
693a6cddad
Add feature gate to tests that use std 2025-01-10 18:43:36 +08:00
David Tolnay
349f6960ff
Release 2.0.10 2025-01-08 11:46:25 -08:00
David Tolnay
6cd87bc228
Merge pull request #408 from dtolnay/assoctype
Generate trait bounds on associated types
2025-01-08 11:45:42 -08:00
David Tolnay
6b3e1e50b2
Generate trait bounds on associated types 2025-01-08 11:43:29 -08:00
David Tolnay
136859154b
Add regression test for issue 405
error[E0599]: the method `as_display` exists for reference `&<T as FromStr>::Err`, but its trait bounds were not satisfied
       --> tests/test_generics.rs:178:13
        |
    178 |     #[error("couldn't parse entry: {0}")]
        |             ^^^^^^^^^^^^^^^^^^^^^^^^^^^ method cannot be called on `&<T as FromStr>::Err` due to unsatisfied trait bounds
        |
        = note: the following trait bounds were not satisfied:
                `<T as FromStr>::Err: std::fmt::Display`
                which is required by `&<T as FromStr>::Err: AsDisplay<'_>`
2025-01-08 11:43:18 -08:00
David Tolnay
c535cecb6f
Release 2.0.9 2024-12-21 10:26:04 -08:00
David Tolnay
0a0516db73
Merge pull request #404 from dtolnay/fromfn
Unspan From impl contents
2024-12-21 10:25:40 -08:00
David Tolnay
e5169bb127
Unspan From impl contents 2024-12-21 10:22:33 -08:00
David Tolnay
c008375268
FIx typo in ui test 2024-12-21 10:20:21 -08:00
David Tolnay
2bd29821f4
Release 2.0.8 2024-12-17 19:18:03 -08:00
David Tolnay
a7de3ab22d
Merge pull request #399 from dtolnay/respan
Fix spans on macro-generated bindings and format variables
2024-12-17 19:17:30 -08:00
David Tolnay
f1243a0ceb
Fix spans on macro-generated bindings and format variables 2024-12-17 19:14:10 -08:00
David Tolnay
6a07345135
Add regression test for issue 398
error[E0425]: cannot find value `_0` in this scope
       --> tests/test_display.rs:308:17
        |
    308 |         #[error("{0}")]
        |                 ^^^^^ not found in this scope

    error[E0425]: cannot find value `__display_x` in this scope
       --> tests/test_display.rs:310:17
        |
    310 |         #[error("{x}")]
        |                 ^^^^^ not found in this scope
2024-12-17 19:14:03 -08:00
David Tolnay
9c0f2d230d
Release 2.0.7 2024-12-13 14:42:57 -08:00
David Tolnay
2deec96fc0
Merge pull request 397 from zertosh/from_allow_expect 2024-12-13 14:40:50 -08:00
Andres Suarez
100d9164f2 Avoid associating #[from] with lint allow 2024-12-13 14:06:02 -08:00
David Tolnay
485c2b7eed
Reword spurious errors comment 2024-12-08 11:28:58 -08:00
David Tolnay
2075e87257
Release 2.0.6 2024-12-08 10:39:40 -08:00
David Tolnay
e9a9085150
Merge pull request #396 from dtolnay/deprecatedfrom
Suppress deprecation warning on generated From impls
2024-12-08 10:39:10 -08:00
David Tolnay
6e8c7244c9
Suppress deprecation warning on generated From impls 2024-12-08 10:37:43 -08:00
David Tolnay
caf585c978
Add test of deprecated type in From impl
error: use of deprecated struct `test_deprecated::DeprecatedStruct`
      --> tests/test_lints.rs:73:13
       |
    73 |             DeprecatedStruct,
       |             ^^^^^^^^^^^^^^^^
       |
    note: the lint level is defined here
      --> tests/test_lints.rs:39:13
       |
    39 |     #![deny(deprecated)]
       |             ^^^^^^^^^^
2024-12-08 10:36:57 -08:00
19 changed files with 155 additions and 45 deletions

View file

@ -41,6 +41,7 @@ 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,3 +1,2 @@
/target
**/*.rs.bk
Cargo.lock
/target/
/Cargo.lock

View file

@ -1,6 +1,6 @@
[package]
name = "thiserror"
version = "2.0.5"
version = "2.0.12"
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.5", path = "impl" }
thiserror-impl = { version = "=2.0.12", path = "impl" }
[dev-dependencies]
anyhow = "1.0.73"
@ -41,4 +41,9 @@ members = ["impl", "tests/no-std"]
[package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"]
rustdoc-args = ["--generate-link-to-definition"]
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",
]

View file

@ -1,6 +1,6 @@
[package]
name = "thiserror-impl"
version = "2.0.5"
version = "2.0.12"
authors = ["David Tolnay <dtolnay@gmail.com>"]
description = "Implementation detail of the `thiserror` crate"
edition = "2021"
@ -18,4 +18,10 @@ syn = "2.0.87"
[package.metadata.docs.rs]
targets = ["x86_64-unknown-linux-gnu"]
rustdoc-args = ["--generate-link-to-definition"]
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",
]

View file

@ -12,7 +12,7 @@ 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 knock-on errors in other code that uses
// anyway to minimize spurious secondary errors in other code that uses
// this type as an Error.
Err(error) => fallback::expand(input, error),
}
@ -169,16 +169,26 @@ 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);
quote_spanned! {span=>
#[allow(unused_qualifications, clippy::needless_lifetimes)]
let from_function = quote! {
fn from(#source_var: #from) -> Self {
#ty #body
}
};
let from_impl = quote_spanned! {span=>
#[automatically_derived]
impl #impl_generics ::core::convert::From<#from> for #ty #ty_generics #where_clause {
#[allow(deprecated)]
fn from(#source_var: #from) -> Self {
#ty #body
}
#from_function
}
}
};
Some(quote! {
#[allow(
deprecated,
unused_qualifications,
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
)]
#from_impl
})
});
if input.generics.type_params().next().is_some() {
@ -434,15 +444,25 @@ 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);
Some(quote_spanned! {span=>
#[allow(unused_qualifications, clippy::needless_lifetimes)]
let from_function = quote! {
fn from(#source_var: #from) -> Self {
#ty::#variant #body
}
};
let from_impl = quote_spanned! {span=>
#[automatically_derived]
impl #impl_generics ::core::convert::From<#from> for #ty #ty_generics #where_clause {
#[allow(deprecated)]
fn from(#source_var: #from) -> Self {
#ty::#variant #body
}
#from_function
}
};
Some(quote! {
#[allow(
deprecated,
unused_qualifications,
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
)]
#from_impl
})
});

View file

@ -107,12 +107,14 @@ impl Display<'_> {
}
};
infinite_recursive |= member == *"self" && bound == Trait::Display;
if let Some(&field) = member_index.get(&member) {
implied_bounds.insert((field, bound));
} else {
out += &member.to_string();
continue;
}
let field = match member_index.get(&member) {
Some(&field) => field,
None => {
out += &member.to_string();
continue;
}
};
implied_bounds.insert((field, bound));
let formatvar_prefix = if bonus_display {
"__display"
} else if bound == Trait::Pointer {
@ -129,15 +131,17 @@ 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 binding_value = match &member {
let mut 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,11 +25,12 @@ impl<'a> ParamsInScope<'a> {
fn crawl(in_scope: &ParamsInScope, ty: &Type, found: &mut bool) {
if let Type::Path(ty) = ty {
if ty.qself.is_none() {
if let Some(ident) = ty.path.get_ident() {
if in_scope.names.contains(ident) {
*found = true;
}
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;
}
}
for segment in &ty.path.segments {

View file

@ -28,6 +28,10 @@ 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<'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 {}
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 + '_ {}

View file

@ -9,8 +9,6 @@
//! 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
@ -259,8 +257,9 @@
//! [`anyhow`]: https://github.com/dtolnay/anyhow
#![no_std]
#![doc(html_root_url = "https://docs.rs/thiserror/2.0.5")]
#![doc(html_root_url = "https://docs.rs/thiserror/2.0.12")]
#![allow(
clippy::elidable_lifetime_names,
clippy::module_name_repetitions,
clippy::needless_lifetimes,
clippy::return_self_not_must_use,

View file

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

View file

@ -1,4 +1,5 @@
#![allow(
clippy::elidable_lifetime_names,
clippy::needless_lifetimes,
clippy::needless_raw_string_hashes,
clippy::trivially_copy_pass_by_ref,
@ -266,7 +267,7 @@ fn test_pointer() {
}
#[test]
fn test_macro_rules() {
fn test_macro_rules_variant_from_call_site() {
// Regression test for https://github.com/dtolnay/thiserror/issues/86
macro_rules! decl_error {
@ -291,6 +292,30 @@ fn test_macro_rules() {
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,6 +1,7 @@
#![allow(clippy::iter_cloned_collect, clippy::uninlined_format_args)]
use core::fmt::Display;
#[cfg(feature = "std")]
use std::path::PathBuf;
use thiserror::Error;
@ -90,6 +91,7 @@ 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,6 +1,7 @@
#![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;
@ -160,6 +161,24 @@ 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,6 +4,17 @@ 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)]
@ -22,7 +33,7 @@ fn test_unused_qualifications() {
#[test]
fn test_needless_lifetimes() {
#![allow(dead_code)]
#![deny(clippy::needless_lifetimes)]
#![deny(clippy::elidable_lifetime_names, clippy::needless_lifetimes)]
#[derive(Error, Debug)]
#[error("...")]
@ -64,6 +75,16 @@ fn test_deprecated() {
Variant,
}
#[derive(Error, Debug)]
pub enum DeprecatedFrom {
#[error(transparent)]
Variant(
#[from]
#[allow(deprecated)]
DeprecatedStruct,
),
}
#[allow(deprecated)]
let _: DeprecatedStruct;
#[allow(deprecated)]

View file

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

View file

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

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`