Skip to content
This repository has been archived by the owner on Jan 1, 2022. It is now read-only.

Deprecate AppSettings / ArgSettings in favor of binary methods #200

Open
epage opened this issue Dec 6, 2021 · 7 comments
Open

Deprecate AppSettings / ArgSettings in favor of binary methods #200

epage opened this issue Dec 6, 2021 · 7 comments
Milestone

Comments

@epage
Copy link
Owner

epage commented Dec 6, 2021

Issue by epage
Wednesday Aug 18, 2021 at 15:21 GMT
Originally opened as clap-rs/clap#2717


Since I was already in a conversation with kbknapp, I figured I might as well ask him for the reason for the transitio to ArgSettings early in clap3.

In part, it was primarily the fn(bool) functions because at the time (perhaps still so?) the Rust API guidelines suggested taking a bool is not as good as an enum variant which is more clear. It was also more in line with App structs which are primarily configured via AppSettings and don't have fn(bool) methods.

However, I also didn't want two ways to do certain settings (i.e. both setting(ArgSetting) and fn(bool)), so I picked enum variants for those bool-ish settings.

The guideline in question: Arguments convey meaning through types, not bool or Option (C-CUSTOM-TYPE)

The motivations given are:

  • it is not immediately clear what true and false are conveying without looking up the argument names
  • Using custom types makes it easier to expand the options later on, for example by adding an ExtraLarge variant.

This motivation doesn't quite line up with clap because the arguments are being used in a builder position which automatically names things. Its fairly common in builders to still use bools. Its also hard to predict what the expansion should be for what to name enum variants. Plus, they would balloon the API.

Proposal

We emphasize the methods, restoring any inferring, and work to deprecate the ArgSettings / AppSettings. Besides the improvements below, this will make the clap2->clap3 transition easier

This is only affecting the API and is independent of how we store these settings.

Problems with Settings

User convenience of inferring settings

Unnecessary verbosity: #[clap(multiple_occurrences)] vs #[clap(setting(clap::ArgSettings::MultipleOccurrences)]

Interferes with actually following the API guidelines (see the individual Color settings vs having an enum

Disjoint API leaking implementation details: Related to the above, we have to distinct ways of doing things, based on the conceptual data type. This makes the API more confusing and increases the barrier to change, including to follow the guidelines this was done for (again, see above with Color)

Keeping both means we are maintaining two distinct APIs for modifying the builders

kbknapp's thoughts

While kbknapp's word isn't the only consideration here, I thought I'd add his thoughts

Regarding Settings

This has one bigger downside of you can no longer easily abstract over a CLI setting dynamically, such as

let should_be_required: bool = /*.. some method of determining this outside of clap */; arg.required(should_be_required);

But I think that's a rare enough use case that if one wanted they could manually unset_setting.

Another downside, or at least it could be argued as a downside, is for documenting, functions are better than enum variants. But this to me is somewhat moot, as there will always been some settings as ArgSettings and some as fn(...) at least non-bool methods.

As far pure keyboard typing goes, the methods are easier to type and subjectively to me more visually pleasing than the settting(Foo).

As for which direction to go, he originally stated:

I'm not sold either way though and could be swayed as to which method is "better"

and later

I agree with everything you said. To be clear, I'm definitely not against the bools and part of the reason I pulled back a little on it.

Originally posted by @epage in clap-rs/clap#2607 (reply in thread)

@epage epage added this to the 3.0 milestone Dec 6, 2021
@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by epage
Wednesday Aug 18, 2021 at 15:25 GMT


As we resolve this, we'll need to create issues for

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by epage
Wednesday Aug 18, 2021 at 15:39 GMT


@pksunkara I think this should be tagged for the 3.0 release as it will lower the barrier for transition from clap2->3. Note that this is only for deprecating and not removing settings. Removing the settings would definitely for a 4.0 thing

EDIT: And will resolve documentation regressions

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by pksunkara
Thursday Sep 23, 2021 at 11:27 GMT


However, I also didn't want two ways to do certain settings (i.e. both setting(ArgSetting) and fn(bool)), so I picked enum variants for those bool-ish settings.

To confirm, the settings approach was picked instead of the methods approach first. But then, we started adding convenience helper methods (Ex: require_delimiter in 920b559 and 09d4d0a). As you can see from that code, these helper methods are not always simply toggling the setting, but rather toggling one or more related settings when the method is called.

Basically, the methods we added are builder methods, but they are not 1:1 with the settings. They are helper methods which manipulate one or more settings.

Early in clap3 history, it looks like the plan was to deprecate binary methods on args (e.g. arg.takes_value(true)) in favor of arg settings (e.g. arg.setting(ArgSettings::TakesValue)) in clap2 with them removed in clap3 (see 6fc70d8 and #1037). This was reverted in 82ffb82 to ease the transition.

We had to revert the removing of methods because of the above reason. They are helpers which change one or more settings and I don't think kbknapp realised that.

Interferes with actually following the API guidelines (see the individual Color settings vs having an enum

Color settings should never have been there in the first place. I agree that it should be a separate enum and should be used by a method which takes that enum. Let's fix this for 3.0 by creating an issue "Migrate color control to an enum, from distinct binary flags"

Disjoint API leaking implementation details: Related to the above, we have to distinct ways of doing things, based on the conceptual data type. This makes the API more confusing and increases the barrier to change, including to follow the guidelines this was done for (again, see above with Color)

I am not sure what exactly you mean by this, but I assume it's an issue with the color related settings only. Thus, this would be fixed by removing them as proposed above.

We emphasize the methods, restoring any inferring, and work to deprecate the ArgSettings / AppSettings. Besides the improvements below, this will make the clap2->clap3 transition easier

I would argue that the transition is harder compared to the reverse.

Keeping both means we are maintaining two distinct APIs for modifying the builders

I'm not sold either way though and could be swayed as to which method is "better"

What I would propose instead is to remove (Deprecation discussion in #2617) the binary methods we currently have for Arg and go with settings enum everywhere (examples, tests, docs etc..).

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by epage
Friday Sep 24, 2021 at 15:06 GMT


I'm a bit surprised, your response feels like a 180 but maybe I misinterpreted us moving this from discussion to Issue. I had thought that we were past the high level discussion and good with the general direction.

Whatever the motivation was, if we still have this much to discuss, this would have best left to discussions because of threading so we can more easily keep the topics focused.

I'm also confused by parts of your response.

To confirm, the settings approach was picked instead of the methods approach first. But then, we started adding convenience helper methods (Ex: require_delimiter in 920b559 and 09d4d0a). As you can see from that code, these helper methods are not always simply toggling the setting, but rather toggling one or more related settings when the method is called.

Basically, the methods we added are builder methods, but they are not 1:1 with the settings. They are helper methods which manipulate one or more settings.

Clap started only with the methods, for example required in clap v1

Not digging through it all and taking what kbnapp said, the Settings were then added in an attempt to comply with the API guidelines.

We had to revert the removing of methods because of the above reason. They are helpers which change one or more settings and I don't think kbknapp realised that.

So maybe there were multiple reasons or different angles on the same but clap-rs/clap@82ffb82 explicitly says the reason was to smooth the upgrade process.

Regardless of any of the above, I don't see how this impacts my proposal. Is there something that the user can do today through Settings that they can't do through methods that is a motivating use case to focus on Settings?

Disjoint API leaking implementation details: Related to the above, we have to distinct ways of doing things, based on the conceptual data type. This makes the API more confusing and increases the barrier to change, including to follow the guidelines this was done for (again, see above with Color)

I am not sure what exactly you mean by this, but I assume it's an issue with the color related settings only. Thus, this would be fixed by removing them as proposed above.

Let me re-phrase. We have parallel APIs for building Args and Apps, settings and methods.

  • Not all methods can be expressed via settings but all settings can be expressed in methods. By supporting Settings, we are forcing the user to use both of the parallel APIs that feel foreign to each other
  • Settings in the API is the low friction path and encourages developers to favor it, even if it isn't always in the best interest. I used Color as an example. Yes, we can fix color but that still leaves the environment that led to the creation of the color settings.
  • The Settings API exists more as a reflection that we implement an optimization for storing a lot configuration as bit flags. Focusing API details on implementation leads to a brittle, obtuse API.

I would argue that the transition is harder compared to the reverse.

  • App doesn't have any of those methods and all the settings are currently being done by *_setting()

My proposal was to start with deprecating settings, not obsoleting / removing them. Also, from my other investigations, there are generally 0-3 App Settings while there are a lot of arg methods.

  • Arg's helper methods are converted to binary methods already so they don't exactly maintain backward compatibility.

I'm not sure what you mean by "helper" vs "binary" method and what I'm supposed to notice is different in the linked functions.

What I would propose instead is to remove (Deprecation discussion in #2617) the binary methods we currently have for Arg and go with settings enum everywhere (examples, tests, docs etc..).

So if I'm understanding, it sounds like you are in favor of resuming kbnapp's efforts to remove the methods, in favor of settings? Under what pretense? I did not see any justification given for them and yet there are many flaws with the settings API, as has been previously enumerated.

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by pksunkara
Friday Sep 24, 2021 at 15:36 GMT


I'm a bit surprised, your response feels like a 180 but maybe I misinterpreted us moving this from discussion to Issue. I had thought that we were past the high level discussion and good with the general direction.

I am sorry, it was my mistake in misunderstanding. When you asked the below question in that discussion:

Are we agreed on this for it to be worth creating an Issue so this can be tracked as part of the 4.0 milestone

I thought you wanted to discuss this more in this issue to finalize the design. I was under the assumption that we were using the discussions threads as a maybe idea and issues as todo idea. Which is why I answered yes to create an issue as an agreement that we need to resolve this.

Am I correct in assuming that you want to move the whole design finalization to discussions?

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by epage
Friday Sep 24, 2021 at 16:02 GMT


I thought you wanted to discuss this more in this issue to finalize the design. I was under the assumption that we were using the discussions threads as a maybe idea and issues as todo idea. Which is why I answered yes to create an issue as an agreement that we need to resolve this.

Am I correct in assuming that you want to move the whole design finalization to discussions?

I'll break this down into

  • Problem
  • Solution (focused on requirements, use cases, and workflows)
  • Implementation (focused on code)

With each being broken down into high level and detailed.

We discussed the Problem and Solution and I thought we were agreed and ready to move on to Implementation. However, if we are still working out Settings vs Methods, then we have not agreed on a Solution.

Where it lives relative to where we are at in the process doesn't matter too much except for how much discussion it generates. Issues being single threaded make it hard to break down and discuss individual points.

@epage
Copy link
Owner Author

epage commented Dec 6, 2021

Comment by pksunkara
Friday Oct 01, 2021 at 00:21 GMT


We discussed the Problem and Solution and I thought we were agreed and ready to move on to Implementation. However, if we are still working out Settings vs Methods, then we have not agreed on a Solution.

Yeah, we haven't agreed on a solution. As I said, it was a miscommunication that gave the impression that we did.

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

No branches or pull requests

1 participant