Discussão original: https://users.rust-lang.org/t/unexpected-behaviors-of-trait-bounds/12286/10
Parece que está relacionado a este problema antigo: https://github.com/rust-lang/rust/issues/29859
Basicamente, tendo trait Complete
require trait Partial
, e trait Partial
require trait PartialEq
(ou qualquer outra característica), um tipo ( struct TypeB
) com impl Complete
não precisa satisfazer impl PartialEq
, que apenas se parece com um bug.
Curiosamente, sempre que <TypeB as Partial>
é visto, a verificação é executada e o erro apropriado é gerado.
Aqui está uma reprodução:
pub trait Partial: PartialEq {
fn foo() -> Option<bool>;
}
pub trait Complete: Partial {
fn foo() -> bool;
}
impl<T> Partial for T
where
T: Complete,
{
fn foo() -> Option<bool> {
Some(<Self as Complete>::foo())
}
}
// ----
#[derive(PartialEq)]
pub struct TypeA {}
impl Partial for TypeA {
fn foo() -> Option<bool> {
None
}
}
// ----
// BUG: No compile warning about `PartialEq` not being implemented
// #[derive(PartialEq)]
pub struct TypeB {}
impl Complete for TypeB {
fn foo() -> bool {
true
}
}
// ----
pub fn main() {
println!("{:?}", TypeA::foo());
// This works, but shouldn't!
println!("{:?}", <TypeB as Complete>::foo());
// This would trigger the issue, though.
//println!("{:?}", <TypeB as Partial>::foo());
/* Result in:
error[E0277]: the trait bound `TypeB: std::cmp::PartialEq` is not satisfied
--> src/bin/cyclic_traits.rs:48:22
|
48 | println!("{:?}", <TypeB as Partial>::foo());
| ^^^^^^^^^^^^^^^^^^^^^^^ can't compare `TypeB` with `TypeB`
|
= help: the trait `std::cmp::PartialEq` is not implemented for `TypeB`
= note: required by `Partial::foo`
*/
}
Aqui está uma reprodução minimizada no formato ICE:
trait Base {
fn base() {}
}
trait Partial: Base {}
trait Complete: Partial {}
impl<T: Complete> Partial for T {}
struct TypeB;
impl Complete for TypeB {}
fn main() {
ice::<TypeB>();
}
fn ice<P: Partial>() {
P::base();
}
error: internal compiler error: /checkout/src/librustc/traits/trans/mod.rs:75: Encountered error `Unimplemented` selecting `Binder(<TypeB as Base>)` during trans
note: the compiler unexpectedly panicked. this is a bug.
note: we would appreciate a bug report: https://github.com/rust-lang/rust/blob/master/CONTRIBUTING.md#bug-reports
note: rustc 1.21.0-nightly (cbbe17aa7 2017-08-07) running on x86_64-unknown-linux-gnu
Parece que @ arielb1 também abriu seu próprio problema (# 43784) para isso. De qualquer forma, isso foi corrigido (bem deveria ser) por # 43786.
Eu acredito que este é um dup de # 43784, encerrando.
Comentários muito úteis
Aqui está uma reprodução minimizada no formato ICE: