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

Extend the grammar/semantics of default: to support boolean expressions #5328

Open
hvr opened this issue May 16, 2018 · 7 comments
Open

Extend the grammar/semantics of default: to support boolean expressions #5328

hvr opened this issue May 16, 2018 · 7 comments
Labels

Comments

@hvr
Copy link
Member

hvr commented May 16, 2018

This is expanded version of this comment:

Currently, the grammar for the default field is limited to

default: <bool-value>

This is perfectly sufficient for properly used automatic flags (i.e. manual: False and used in a way which allows the solver to unique determine their value -- there's a separate feature request to have cabal detect "non-functional" uses of automatic flags); however, for manual flags the limitation of allowing only a constant boolean literal in default: is rather limiting.

Instead, starting with e.g. cabal-version:3.0, we should allow an enhanced version of default

default: <boolean-expression>

and where <boolean-expression> would be the expression we already support in if clauses which currently supports the

  • the literals True/0 and False/0,
  • binary operator &&/2, ||/2,
  • the unary operator !/1,
  • grouping via ( and ), and the
  • builtin predicates:
    • impl/1
    • flag/1
    • arch/1
    • os/1

Such expression can be statically determined, as the impl/1, arch/1, and os/1 predicates are globally constant for a solver session.

The flag/1 predicate however requires us to impose a limitation though, in order to keep the implementation simple (allow for referential checks during single-pass parsing) and avoid the issue of undecidable flags (e.g. via cyclic references). To that end, the following simple well-formedness rule needs to be added to the specification of flag/1

  • flag(<flagname>) may only refer to a <flagname> which is lexically visible, i.e. completely defined before their use-site (i.e. only after its Flag stanza ends, a flag is considered completely defined; this prefents direct self-referential uses of flag/1 from within the same Flag stanza)

  • Furthermore (as pointed out in Extend the grammar/semantics of default: to support boolean expressions #5328 (comment)), a manual flag's default expression must not be refer to an automatic flag.

@phadej
Copy link
Collaborator

phadej commented May 16, 2018

Can/cannot manual flag default value depend on automatic flag value?

flag old-time
  manual : False
  default: False

flag other-flag
  manual :True
  default: flag(old-time)

Concrete motivation for proposal would be good to have too. What existing problems it would solve?

@hvr
Copy link
Member Author

hvr commented May 16, 2018

@phadej that's a very good point; I've augmented the spec; a more drastic way would be to simply disallow any use of flag() within default expressions.

My main use-case for this would be architectural conditionals; e.g. defaults for a manual flag inferred based on architecture or ghc versions (simplest case: a flag portable which enables a pure Haskell version of an implementation avoiding any platform/compiler specific features), but with the ability for a user to override this default. Another use-case is a flag to enable pkgconfig-based discoverability; you'd want this default on for some platforms like os(linux), but not necessarily on os(windows); otoh, some users may still prefer to turn on pkgconfig support on windows regardless. I stumble over this kind of situation every now and then, and keep wishing this had been already implemented...

@juhp
Copy link
Collaborator

juhp commented May 16, 2018

Yes my use-case is basically ghc version.

@grayjay
Copy link
Collaborator

grayjay commented Jun 11, 2018

#2397 and #2429 are related, but they use an if-else around the default field.

@avh4
Copy link

avh4 commented Aug 14, 2022

My use-case for this is that my project has multiple frontend drivers, including a TUI driver, a gtk+ driver, and a Cocoa driver (and possibly others in the future). The gtk+ driver can in theory build on MacOS, but doing so requires a bunch of native library setup that normally won't be wanted; whereas gtk+ will be the preferred frontend for Linux builds.

I'd like to be able to have a "gtk" flag that defaults to False when os(darwin), defaults to True otherwise, and can be manually overridden in either case.

@ulysses4ever ulysses4ever added the re: flag Concerning user-defined flags in cabal files label Aug 14, 2022
@michaelpj
Copy link
Collaborator

Here's a concrete usecase based on GHC versions.

HLS has a bunch of cabal flags determining whether to try and build in support for some tool or other, say stan. Now, often we know that stan will only build on certain GHC versions. We basically have two options:

  1. haskell-language-server +stan means "build HLS with stan support, possibly getting build failures if it doesn't work on your GHC version". In this case we probably need to have the default value of the stan flag be false, so that HLS works when built from Hackage, which means that users need to manually set a bunch of flags to true.
  2. haskell-language-server +stan means "build HLS with stan support, if that will work". In this case we can set the default value of the flag to "true", and put conditional logic in the cabal file to ignore if it we're on a non-working GHC version. Then users don't have to set flags manually, but they might be surprised that the flag is on and they don't get the feature!

The ideal situation would be 1, but we can change the default value of the flag based on the GHC version. Then users will get sensible behaviour in all cases, but the flag continues to mean "definitely try and include this functionality".

@phadej
Copy link
Collaborator

phadej commented Jun 14, 2024

@michaelpj fwiw, the #3526 would also solve that in a convenient way. You could have a three way flag, True, False, and Auto.

FWIW, you can encode that today too with two flags, but that's not very user friendly.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

7 participants