This repository has been archived by the owner on Jan 24, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 41
SEP 7 - Change requisites/excludes #10
Closed
Closed
Changes from all commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
- Feature Name: change-requisites-and-excludes | ||
- Start Date: 2019-04-09 | ||
- SEP Status: Draft | ||
- SEP PR: https://github.com/saltstack/salt-enhancement-proposals/pull/10 | ||
- Salt Issue: (leave this empty) | ||
|
||
|
||
# Summary | ||
[summary]: #summary | ||
|
||
We want to change how requisites and excludes work together to build the | ||
highstate. This proposal requests that when we have an exclude that would break | ||
a requisite, that the exclude take precedence over the requisite. | ||
|
||
# Motivation | ||
[motivation]: #motivation | ||
|
||
In the past we've had [at least a few | ||
people](https://github.com/saltstack/salt/issues/6237) expect that if they | ||
exclude a state, the require would go away. Because this behavior is not well | ||
documented and unexpected, it causes frustration for people. | ||
|
||
We want *happy* users. :slightly_smiling_face: | ||
|
||
# Design | ||
[design]: #detailed-design | ||
|
||
The expectation of the users on #6237 is that exclude simply plucks the states | ||
out of everywhere they're referenced - state files, requirements, etc. | ||
|
||
The proposal here is that we change Salt to line up with the user expectations. | ||
The way this could work is that during the `apply_exclude` process, anything | ||
that can be excluded will be removed, as if it never existed. Here is a simple | ||
example state that currently does not work: | ||
|
||
test1: | ||
test.nop | ||
|
||
test2: | ||
test.nop: | ||
- require: | ||
- test: test1 | ||
|
||
exclude: | ||
- id: test1 | ||
|
||
This fails because when it tries to run test2, the require still exists, but | ||
refers to a state that is no longer there. | ||
|
||
While there are some obvious ways to handle this case, a slightly more subtle | ||
example was also encountered: | ||
|
||
test1: | ||
test.nop | ||
|
||
test2: | ||
test.nop: | ||
- require_in: | ||
- test: test1 | ||
|
||
exclude: | ||
- id: test2 | ||
|
||
In this case, it feels quite odd if you're not super familiar with the | ||
requisites because you're excluding a state, but part of that state has already | ||
affected another one. | ||
|
||
In these cases, the behavior that Salt's users expect is that the exclude will | ||
take precedence, resulting in these states: | ||
|
||
test2: | ||
test.nop | ||
|
||
And | ||
|
||
test1: | ||
test.nop | ||
|
||
In order to preserve backwards compatibility, we [should make this | ||
configurable][1]. The following command line argument should be added to the | ||
salt/salt-call cli: | ||
|
||
--exclude-before-requisites | ||
|
||
We should also add the same setting to the config files: | ||
|
||
exclude_before_requisites: True | ||
|
||
When these flags are set to `True`, excludes should work as proposed. When the | ||
flags are set to `False`, current behavior should be kept, though we should add | ||
an explicit message that the exclude would remove a required state. | ||
|
||
At first glance, it's *possible* that this code change is as simple as moving | ||
the `apply_excludes` line in `state.py` a couple of lines earlier before the | ||
`requisite_in` calls. Further tests indicate there's more considerations. | ||
|
||
The test cases are pretty straight forward, though it's possible that it breaks | ||
some existing tests. | ||
|
||
Test cases: | ||
|
||
- when the flag is not set, excluding a requisite should produce a helpful | ||
error message (`'Warning: excluding {thing} that is required by | ||
{other_thing}'`, for example) | ||
- exclude with no requires still excludes | ||
- exclude with [each | ||
requisite][2] | ||
properly excludes when the required state, or the requiring state is excluded | ||
|
||
## Alternatives | ||
[alternatives]: #alternatives | ||
|
||
There are a couple of alternative approaches: | ||
|
||
1. Do nothing - everything is fine. | ||
2. Add documentation to either requisites and/or includes that mentions that | ||
this is considered an anti-pattern, and provide some patterns to follow for | ||
people who think they want to take this approach. | ||
3. Codify the existing behavior by adding a proper error/exception when exclude | ||
excludes something that was required (should still require documentation | ||
updates). | ||
|
||
## Unresolved questions | ||
[unresolved]: #unresolved-questions | ||
|
||
I know this applies to states - do we have similar problems when there are | ||
entire files? What about includes/requires across state files? Can | ||
pillars/grains/??? suffer from the same issues? | ||
|
||
Are there issues with require/exclude cycles? | ||
|
||
# Drawbacks | ||
[drawbacks]: #drawbacks | ||
|
||
- Implementation and opportunity cost of actually building/testing/etc. | ||
- <strike>This could cause *serious* issues if someone excluded something that | ||
ended out having a greater impact than they thought it should have. For | ||
example, it's possible that I exclude a state that looks simple, but it ends | ||
out to have a cascading effect, eliminating half of my states. That would be | ||
undesirable.</strike> By requiring opt-in to this new behavior, it will not | ||
break any existing setups. Since we'll already be introducing some changes, | ||
we can update the existing behavior to provide a more helpful error if there | ||
is a conflict between exclude and require. | ||
|
||
|
||
[1]: https://github.com/saltstack/salt/pull/51183#issuecomment-454646418 | ||
[2]: https://docs.saltstack.com/en/latest/ref/states/requisites.html#direct-requisite-and-requisite-in-types |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like the PR, but if we go down this path instead, I'd advocate a warning instead of an error. An error would be a breaking change if someone was relying on current behavior. If an error is really desired, then have one major release use a warning with a deprecation notice first, then make it an error.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's a great point all around. Definitely a warning+deprecation notice, e.g. "This will become an error and in release ". That would handle the current case, where it's implicit behavior, as well as provide a good path going forward.