cfg-if icon indicating copy to clipboard operation
cfg-if copied to clipboard

else can be used without if

Open hellow554 opened this issue 6 years ago • 12 comments

TIL: https://lukaslueg.github.io/macro_railroad_wasm_demo

It clearly shows that an else can be used solely without a leading `if. A quick jump into the compiler shows it as well:

use cfg_if::cfg_if;

fn main() { 
    cfg_if! {
        else {
            fn a() {
                println!("Foo");
            }
        }
    }
    a();
}

Is this intended? If yes, it should be documented, if not, it should be prohibited.

hellow554 avatar Aug 05 '19 11:08 hellow554

I think this is just a mistake of the macro! It's not an intended use case but I don't really think it's that big of an issue to either document or prohibit. It seems like sort of a weird quirk at best?

alexcrichton avatar Aug 05 '19 15:08 alexcrichton

My gut reaction is that you shouldn't allow something like this because quirks like that that you aren't expecting could lead to bugs. This would be an opportunity to accidentally forget the if of the statement which would be a logic error in ( I think ) every instance.

I mean, imagine if rustc let us have else's without if's.

If it is a big pain to fix from an implementation perspective, though, then I think that documenting the quirk would be reasonable enough.

zicklag avatar Mar 29 '20 00:03 zicklag

I agree with @alexcrichton. But I would add that I think an else clause by itself is kinda cute.

camsteffen avatar Oct 08 '20 18:10 camsteffen

This is fixed on the current master, appears to be changed in https://github.com/alexcrichton/cfg-if/commit/349913689695130c2c63d1075e57b0917c98ee6e

Any hope this gets released? The commit is one year old.

I'd happily take this. I'm using cfg_if quite heavily for architectural dependencies and I've already had a confusing bug whose root cause was a missing if due to some copy-paste error. There's no use case for allowing this syntax, I'd prefer it to be rejected.

V0ldek avatar Jan 11 '22 15:01 V0ldek

I don’t think there’s no use case. I can imagine realistic code depending on this behaviour. Something along these lines:

use cfg_if::cfg_if;

// Several different implementations of a thing
#[derive(Debug)] struct Windows;
#[derive(Debug)] struct Unix;
#[derive(Debug)] struct Fallback;

macro_rules! demo {
    ($($x:ident => $y:expr),*) => {{
        cfg_if! {
            $(
                if #[cfg($x)] {
                    $y
                }
            )else*
            else {
                Fallback
            }
        }
    }}
}

fn main() {
    println!("{:?}", demo!());
    println!("{:?}", demo!(unix => Unix, windows => Windows));
}

That single-character change of * to + is a breaking change; although I doubt that anyone is depending on it, I reckon it should either be reverted, or the next release called 2.0.0.

(It’s pure coincidence that I’m commenting less than a day after you: I just happen to be starting doing detailed code review of crates starting with this one today.)

chris-morgan avatar Jan 12 '22 13:01 chris-morgan

Yes, this is technically a breaking change, so semver requires that release be 2.0.0. That's fine by me, I'd just like to see this released, it's frozen on master which is simply a waste :)

V0ldek avatar Jan 14 '22 12:01 V0ldek

@rust-lang/crate-maintainers Any thoughts on this? I think releasing it as 2.0.0 makes some sense but also concerned that would affect a lot of crates. The decision here seems important.

JohnTitor avatar Apr 23 '23 01:04 JohnTitor

I don't think this alone is worth a release, given that it would introduce a transition period where crates would be using both for a while. If we have other features motivating a major release then we could include this as well.

But personally, I would love to see this included in the standard library instead. If people are going to go through a transition, let's have them go through a transition to the version in the standard library.

joshtriplett avatar Apr 23 '23 02:04 joshtriplett

That sounds great, thank you! Then I'm going to do the following if no one raises any objections in a few days:

  • revert else+ to else* in this crate
  • submit a PR as a proposal to include it in the stdlib

JohnTitor avatar Apr 23 '23 03:04 JohnTitor

@JohnTitor There's already an internal copy in the stdlib; I put it there. The PR would just need to propose stabilizing it.

(Also, you might add a comment next to the version number in Cargo.toml saying if this ever needs bumping to a new major version then we should fix this.)

joshtriplett avatar Apr 23 '23 05:04 joshtriplett

If we're considering moving this to the standard library (which I wholly support!) then we should perhaps consider whether to introduce language-level syntax for this considering how widely used cfg_if is. However this discussion is starting to be off-topic for this issue.

Regarding the issue itself, I agree that we should not do a 2.0.0 release just for this.

Amanieu avatar Apr 23 '23 05:04 Amanieu

Any progress on moving this to the standard library?

photino avatar Jul 22 '23 04:07 photino