Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

assert_eq!(a,b) fails to compile for slices while assert!(a == b) works fine #51206

Open
gnzlbg opened this issue May 30, 2018 · 9 comments
Open
Labels
C-enhancement Category: An issue proposing an enhancement or a PR with one. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue.

Comments

@gnzlbg
Copy link
Contributor

gnzlbg commented May 30, 2018

Example (playground):

pub fn foo(xs: &[u32], k: u32) -> (&[u32], u32) { (xs, 0) }
fn main() {
    assert!(foo(&[], 10) == (&[], 0));  // OK
    assert_eq!(foo(&[], 10), (&[], 0));  // FAILS
}
@shepmaster
Copy link
Member

&[] isn't a slice, it's a reference to an array:

  = note: expected type `(&[u32], u32)`
             found type `(&[_; 0], {integer})`

@gnzlbg
Copy link
Contributor Author

gnzlbg commented May 30, 2018

Yes, but assert! succeeds because there is a PartialEq implementation for slices and arrays. For some reason assert_eq! is not invoking that one here (and per the docs it says it does).

@shepmaster
Copy link
Member

and per the docs it says it does

It does, but it does other things first. Macro expanded and then hand-cleaned:

fn main() {
    if !(foo(&[], 10) == (&[], 0)) {
        panic!();
    };

    match (&foo(&[], 10), &(&[], 0)) {
        (left_val, right_val) => {
            if !(*left_val == *right_val) {
                panic!()
            }
        }
    };
}

@kennytm
Copy link
Member

kennytm commented May 30, 2018

@gnzlbg There is PartialEq implementation between &[T] and &[T; n], but not (&[T], u32) and (&[T; n], u32). The PartialEq impl for tuples are like:

impl<A, B, C, D, E, F, G, H> PartialEq<(A, B, C, D, E, F, G, H)> for (A, B, C, D, E, F, G, H) where
    A: PartialEq<A>,
    B: PartialEq<B>,
    C: PartialEq<C>,
    D: PartialEq<D>,
    E: PartialEq<E>,
    F: PartialEq<F>,
    G: PartialEq<G>,
    H: PartialEq<H> + ?Sized, 

note that the LHS and RHS are equal types.

@kennytm kennytm added the T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. label May 30, 2018
@gnzlbg
Copy link
Contributor Author

gnzlbg commented May 30, 2018

@kennytm but how come that assert!(foo(&[], 10) == (&[], 0)); works then? I am confused 😕

@kennytm
Copy link
Member

kennytm commented May 30, 2018

@gnzlbg assert_eq expands to

match (&left, &right) {
    (l, r) => {
        if !(*l == *r) {
            panic!("...");
        }
    }
}

Probably the match has changed some inference behavior making &[] not able to coerce from &[_; 0] to an &[u32].

@eddyb
Copy link
Member

eddyb commented May 30, 2018

This isn't a regression, is it? Pretty sure this never worked.

As for why, the macro needs to bind the two expressions to variables (by reference) to be able to not only compare but also print them (in case of a mismatch) - this is why assert_eq is a separate macro, after all - and this means coercion information can't flow through ==.

OTOH, in foo(&[], 10) == (&[], 0), the &[] on the RHS of == gets &[u32] as "expected type".

@eddyb
Copy link
Member

eddyb commented May 30, 2018

We can probably make assert_eq do coercions just like == like so (untested!):

// perma-unstable
pub fn check_assert_eq<'a, A, B>(a: &'a A, b: &'a B) -> Result<(), (&'a A, &'a B)>
    where A: PartialEq<B>
{
    if a == b { Ok(()) } else { Err((a, b)) }
}
macro_rules! assert_eq {
    ($left:expr, $right:expr) => ({
        if let Err((left_val, right_val)) = check_assert_eq(&$left, &$right) {
            panic!(r#"assertion failed: `(left == right)`
  left: `{:?}`,
 right: `{:?}`"#, left_val, right_val);
        }
    });
}

@kennytm
Copy link
Member

kennytm commented May 30, 2018

cc #44838.

@XAMPPRocky XAMPPRocky added the C-enhancement Category: An issue proposing an enhancement or a PR with one. label Aug 23, 2019
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
C-enhancement Category: An issue proposing an enhancement or a PR with one. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue.
Projects
None yet
Development

No branches or pull requests

5 participants