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

Unstable Feature Flags #5727

Closed
wants to merge 6 commits into from
Closed

Conversation

pradyunsg
Copy link
Member

This PR adds a helper and a general CLI option to provide functionality for feature flags. This was something @dstufft suggested in the same discussion where we decided to switch to CalVer.

Rationale

With the new release cycle and deprecation policy, it would be difficult to incrementally introduce new functionality while maintaining those guarantees. This is particularly true for big-ticket items like #988, #4575, #4659 and possibly PEP 517 too, which will make fairly extensive user-visible changes.

Feature Flags serve as a tool to enable incrementally implementing new functionality while not having to ensure backward compatibility of it with previous iterations of itself. Any user-visible behavior hidden behind a feature flag will not be subject to backward compatibility guarantees (and neither will the names of the feature flags).

Additional benefits include:

  • allowing users to use unstable features and provide feedback on it.
  • refactoring as a part of implementing new features get adopted immediately, even before the new feature is stablised

@pradyunsg pradyunsg added the type: maintenance Related to Development and Maintenance Processes label Aug 23, 2018
@pradyunsg
Copy link
Member Author

Should this get a .process NEWS fragment?

@pradyunsg
Copy link
Member Author

LOL. This unintentionally includes #5726. I'll rebase on master later today, if that PR isn't merged by then.

@pfmoore
Copy link
Member

pfmoore commented Aug 23, 2018

Probably needs something added to the docs (in the developer section) as I've no idea what this would mean or how I'd use it, as it stands. I don't expect to use it for PEP 517, in any case.

@pradyunsg
Copy link
Member Author

Probably needs something added to the docs (in the developer section) as I've no idea what this would mean or how I'd use it, as it stands. I don't expect to use it for PEP 517, in any case.

Yeah, I was adding that -- specifically in the deprecation policy portion. There's no good place to put it in for explaining how to use them though.

@pradyunsg pradyunsg force-pushed the unstable-feature-flags branch from 9bd7fb4 to f607ccd Compare August 23, 2018 08:42
Copy link
Member

@cjerdonek cjerdonek left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A few comments.

src/pip/_internal/utils/unstable.py Show resolved Hide resolved
src/pip/_internal/exceptions.py Show resolved Hide resolved
src/pip/_internal/utils/unstable.py Show resolved Hide resolved
@cjerdonek
Copy link
Member

cjerdonek commented Aug 24, 2018 via email

@pradyunsg pradyunsg force-pushed the unstable-feature-flags branch from afadf75 to 2ffe967 Compare August 26, 2018 07:14
@pradyunsg
Copy link
Member Author

You could still have an independent easily tested helper function that is called by the Option subclass. And parse_args() is easy to test on top of that to make sure the subclass is wired up and operating correctly.

I ended up with defining a new subclass that's being used in just one specific option. It'd be doing something that's equally well handled by a single line in an existing function and is tested using the exact same testing patterns (call a method and check some monkey-patched mock was called). Further, patching the helper is slightly more tedious in the case of using an Option subclass.

I feel it's better to just have the helper be called in main(). I'm open to someone improving this in a follow up PR though -- perhaps I'm missing something here or simply not thinking the same way as you are. :)

@pradyunsg pradyunsg requested review from a team and cjerdonek and removed request for dstufft, xavfernandez and pfmoore August 26, 2018 07:25
@pfmoore
Copy link
Member

pfmoore commented Aug 26, 2018

I still don't understand the point of this change. If I'm implementing a feature, how would I make it conditional on an "unstable feature flag" and why would doing so be any better than simply adding an --enable-my-feature command line option? This still feels to me like a solution looking for a problem.

IMO, this sounds like we're adding a policy about how we add new features, but I don't really understand what that policy is, or why we need it.

You say

With the new release cycle and deprecation policy, it would be difficult to incrementally introduce new functionality while maintaining those guarantees

but I don't understand this. To me, it's simple - if you want to introduce an incomplete function, either simply don't document it (hence there's no backward compatibility guarantee, as there's no supported behaviour) or document it as "subject to change". If you're breaking existing behaviour, you should wait until you have all the pieces in place before doing so. PEP 517 is a perfect case in point. I'm ensuring that whatever I release passes the full test suite before I merge it, and I'm holding off on documenting the new features or adding tests for new functionality until I have a solid backward compatible foundation (that's not always easy - ahem setuptools dependency 😄 - but it's possible, and IMO worthwhile).

@pradyunsg
Copy link
Member Author

pradyunsg commented Aug 26, 2018

If I'm implementing a feature, how would I make it conditional on an "unstable feature flag"

This should be put in the documentation somewhere...

  • do self.unstable.register("flag") in a command's __init__()
  • check self.unstable.is_enabled("flag") in run() to gate the functionality
    • is_enabled would return True if you pass -X flag on the CLI

why would doing so be any better than simply adding an --enable-my-feature command line option?

IMO, this provides a clearer sense of "don't consider this stable/complete" than a --enable-my-feature flag. Feature flags would let us make smaller PRs, incrementally implementing functionality and even get feedback on half-ready features from end users while having a clear way for telling them it's exempted from backwards compatibility guarantees.

If you're breaking existing behaviour, you should wait until you have all the pieces in place before doing so.

IMO big feature PRs aren't usually very easy to work with. They have more changes in a single go, almost always are harder to review and hold up work in the code being modified (or need to constantly keep syncing up).

IIRC during the discussion, the motivating case for this was PEP 518 -- we had a potentially "broken" implementation and we had to hold off the release till we fixed the issues on that front. Then, communicate effectively that the functionality wasn't complete. With date based releases, we can't really hold off for very long so having a mechanism to clearly mark functionality as "in-development" while still being able to iterate would be helpful. Additionally, this would serve as a much easier path to get new functionality in the hands of early adopters to get feedback, before switching the default for everyone.

IMO, this sounds like we're adding a policy about how we add new features, but I don't really understand what that policy is, or why we need it.

Not really. The change here is that for changes where we might want some additional testing "in the wild" or feedback from users about some new functionality/behavior, we now have an(other) established way to do so.

@pfmoore
Copy link
Member

pfmoore commented Aug 26, 2018

IIRC during the discussion, the motivating case for this was PEP 518

OK. I agree that PEP 518 was a problem because it blocked release for a long time. But IMO that was at least in part because of how it was managed. I don't want to rake over old discussions, but I do believe that it could have been handled better. I'm hoping that PEP 517 will be evidence that this is possible. Note that pretty much all of the preparatory work for PEP 517 has been merged into master and could be released at any time with no user-visible impact, and what remains is (hopefully!!!) relatively small changes to actually call the backend hooks.

Anyway, I'm not going to block this change just because I don't see the need for it. If people want it, that's fine - but I don't expect to use it myself.

@pradyunsg
Copy link
Member Author

I don't want to rake over old discussions, but I do believe that it could have been handled better.

Same and agreed.

@BrownTruck
Copy link
Contributor

Hello!

I am an automated bot and I have noticed that this pull request is not currently able to be merged. If you are able to either merge the master branch into this pull request or rebase this pull request against master then it will be eligible for code review and hopefully merging!

@BrownTruck BrownTruck added the needs rebase or merge PR has conflicts with current master label Oct 4, 2018
@pradyunsg
Copy link
Member Author

I'm gonna go ahead and close this since I don't have the time currently to work through all the social/technical implications of adding this.

@pradyunsg pradyunsg closed this Nov 8, 2018
@pradyunsg pradyunsg deleted the unstable-feature-flags branch November 8, 2018 10:24
@lock
Copy link

lock bot commented May 31, 2019

This thread has been automatically locked since there has not been any recent activity after it was closed. Please open a new issue for related bugs.

@lock lock bot added the auto-locked Outdated issues that have been locked by automation label May 31, 2019
@lock lock bot locked as resolved and limited conversation to collaborators May 31, 2019
@pradyunsg
Copy link
Member Author

This has come up again in the dependency resolution work and is being discussed on Zulip: https://python.zulipchat.com/#narrow/stream/218659-pip-development/topic/opt-in.20flag

@brainwane
Copy link
Contributor

I think we do need to talk about the possibility of doing this. @pradyunsg could I ask you to reopen?

@pradyunsg
Copy link
Member Author

@brainwane I think I deleted the corresponding branch, but it can be revived pretty easily. We did end up implementing a diluted variant of this stuff in a PR by @uranusjr though.

@uranusjr
Copy link
Member

uranusjr commented Apr 3, 2020

#7845 is a simplified version of this.

I thought about how we should eventually switch in the resolver yesterday, and it occured to me --unstable-feature does not offer a good transition plan. From what I can tell, we should do it in these phases:

  1. Introduce the new behaviour behind a flag, allow users to opt into it.
  2. Make the new behaviour the default, but allow users to opt out of it (back to the legacy behaviour).
  3. Potentially remove the legacy behaviour.

With the current flag setup, phase 1 is covered by pip install --unstable-feature=resolver. Now for phase 2, do we make --unstable-feature=resolver the default? (It sounds weird to make an unstable feature the default.) How should users opt out of it? (Technically it’s possible to do pip install --unstable-feature="" but that is strange in a lot of ways, and would be more awkward if we ever have two unstable features at the same time).

Even step 3 could be a problem. Say we go with the above route (setting resolver by default), and some people think it’s a good idea to always pass --unstable-feature=resolver for explicitness (this would work in phase 2 since that’s just repeating the default). Now in phase 3, the user would see a hard error if we simply remove resolver as a possible value. Or do we keep the value forever, just as a no-op?

@xavfernandez
Copy link
Member

From my point of view, phase 1) can be enabled with --unstable-feature while phases 2) & 3) can be carried out with another flag --resolver=legacy/modern.

@pfmoore
Copy link
Member

pfmoore commented Apr 3, 2020

Maybe we could have a --deprecated-version flag, where --deprecated-version=resolver says "use the old (deprecated) version of the resolver. Then the process for users would be:

  1. Opt into the new resolver via --unstable-feature=resolver.
  2. Everyone gets the new resolver by default, but users can opt out using --deprecated-version=resolver. We keep --unstable-feature=resolver available so early adopters have some time to change scripts, etc. back.
  3. The legacy resolver goes away, the two flags lose the "resolver" option, and everyone gets the new version.

This has the advantage (for me) of very clearly saying that people opting out of the new behaviour in (2) can't do so indefinitely, so we have a clear path to removing the old code.

@pradyunsg
Copy link
Member Author

pradyunsg commented Jul 2, 2020

Note that #8371 covers this. I'm gonna go ahead and re-lock this old issue now. We never unlocked this issue. :)

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
auto-locked Outdated issues that have been locked by automation needs rebase or merge PR has conflicts with current master type: maintenance Related to Development and Maintenance Processes
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants