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

Ensure controls are applied #243

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,12 @@

[Unreleased]: https://github.com/chaostoolkit/chaostoolkit-lib/compare/1.22.1...HEAD

### Changed

- Ensure controls are properly applied and none are missed [#242][242]

[242]: https://github.com/chaostoolkit/chaostoolkit-lib/issues/242

## [1.22.1][] - 2021-10-04


Expand Down
54 changes: 40 additions & 14 deletions chaoslib/control/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,9 @@ def initialize_controls(
exc_info=True,
)

logger.debug(
"Controls have been loaded from the experiment: {}".format(controls))


def cleanup_controls(experiment: Experiment):
"""
Expand Down Expand Up @@ -167,6 +170,10 @@ def initialize_global_controls(
controls.remove(control)
set_global_controls(controls)

logger.debug(
"Global controls have been loaded from the experiment: {}".format(
controls))


def load_global_controls(settings: Settings):
"""
Expand Down Expand Up @@ -334,31 +341,50 @@ def get_context_controls(
if not experiment:
return glbl_controls

top_level_controls = experiment.get("controls", [])
experiment_top_level_controls = experiment.get("controls", [])
controls = copy(context.get("controls", []))
controls.extend(glbl_controls)

# do we even have something at the top level to be merged?
if not top_level_controls:
if not experiment_top_level_controls:
# then return only any global controls
return controls

if not controls:
return [deepcopy(c) for c in top_level_controls if c.get("automatic", True)]

# method and rollbacks are specials and they require we inject any controls
# automatically to them as they don't really have a structure where the
# user could have defined them
if level in ["method", "rollback"]:
return [deepcopy(c) for c in top_level_controls if c.get("automatic", True)]
return [
deepcopy(c)
for c in experiment_top_level_controls if c.get("automatic", True)
]

for c in controls:
# no global controls? let's return all experiment level controls directly
if not controls:
return [
deepcopy(c)
for c in experiment_top_level_controls if c.get("automatic", True)
]

# At this stage, controls is made of the global controls and the
# current context's controls. When the current context is experiment,
# we shouldn't merge anything since they are already there.
if level == "experiment":
return [deepcopy(c) for c in controls]

# now we have global controls and experiment level controls while being
# at a more nested level like activity, hypothesis, etc...
# Let's merge them unless one of them automatic is set to False on one of
# the experiment top level control, indicating they should not be applied
# down the tree
for c in controls[:]:
if "ref" in c:
for top_level_control in top_level_controls:
if c["ref"] == top_level_control["name"]:
controls.append(deepcopy(top_level_control))
for tc in experiment_top_level_controls:
if c["ref"] == tc["name"]:
controls.append(deepcopy(tc))
break
else:
for tc in top_level_controls:
if c.get("name") == tc.get("name"):
break
else:
for tc in experiment_top_level_controls:
if tc.get("automatic", True):
controls.append(deepcopy(tc))

Expand Down