diff --git a/.coverage.data b/.coverage.data new file mode 100644 index 000000000..2079bf72e Binary files /dev/null and b/.coverage.data differ diff --git a/ops/framework.py b/ops/framework.py index 3246799b5..1d497704c 100644 --- a/ops/framework.py +++ b/ops/framework.py @@ -851,6 +851,9 @@ def _event_is_in_storage( if ( existing_observer_path != observer_path or existing_method_name != method_name + # The notices all have paths that include [id] at the end. If one + # was somehow missing, then the split would be the empty string and + # match anyway. or existing_event_path.split('[')[0] != event_path.split('[')[0] ): continue diff --git a/testing/build/lib/scenario/__init__.py b/testing/build/lib/scenario/__init__.py new file mode 100644 index 000000000..2cf042cc6 --- /dev/null +++ b/testing/build/lib/scenario/__init__.py @@ -0,0 +1,150 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Charm state-transition testing SDK for Ops charms. + +Write tests that declaratively define the Juju state all at once, define the +Juju context against which to test the charm, and fire a single event on the +charm to execute its logic. The tests can then assert that the Juju state has +changed as expected. + +These tests are 'state-transition' tests, a way to test isolated units of charm +functionality (how the state changes in reaction to events). They are not +necessarily tests of individual methods or functions (but might be, depending on +the charm's event observers); they are testing the 'contract' of the charm: given +a certain state, when a certain event happens, the charm should transition to a +certain (likely different) state. They do not test against a real Juju +controller and model, and focus on a single Juju unit, unlike integration tests. +For simplicity, we refer to them as 'unit' tests in the charm context. + +Writing these tests should nudge you into thinking of a charm as a black-box +input->output function. The input is the union of an `Event` (why am I, charm, +being executed), a `State` (am I leader? what is my integration data? what is my +config?...) and the charm's execution `Context` (what integrations can I have? +what containers can I have?...). The output is another `State`: the state after +the charm has had a chance to interact with the mocked Juju model and affect the +state. + +.. image:: https://raw.githubusercontent.com/canonical/ops-scenario/main/resources/state-transition-model.png + :alt: Transition diagram, with the input state and event on the left, the context including the charm in the centre, and the state out on the right + +Writing unit tests for a charm, then, means verifying that: + +- the charm does not raise uncaught exceptions while handling the event +- the output state (as compared with the input state) is as expected. + +A test consists of three broad steps: + +- **Arrange**: + - declare the context + - declare the input state +- **Act**: + - select an event to fire + - run the context (i.e. obtain the output state, given the input state and the event) +- **Assert**: + - verify that the output state (as compared with the input state) is how you expect it to be + - verify that the charm has seen a certain sequence of statuses, events, and `juju-log` calls + - optionally, you can use a context manager to get a hold of the charm instance and run + assertions on APIs and state internal to it. + +The most basic scenario is one in which all is defaulted and barely any data is +available. The charm has no config, no integrations, no leadership, and its +status is `unknown`. With that, we can write the simplest possible test: + +.. code-block:: python + + def test_base(): + ctx = Context(MyCharm) + out = ctx.run(ctx.on.start(), State()) + assert out.unit_status == UnknownStatus() +""" + +from ops._private.harness import ActionFailed # For backwards compatibility. + +from .context import CharmEvents, Context, Manager +from .errors import StateValidationError # For backwards compatibility. +from .state import ( + ActiveStatus, + Address, + AnyJson, + BindAddress, + BlockedStatus, + CharmType, + CheckInfo, + CloudCredential, + CloudSpec, + Container, + DeferredEvent, + ErrorStatus, + Exec, + ICMPPort, + JujuLogLine, + MaintenanceStatus, + Model, + Mount, + Network, + Notice, + PeerRelation, + Port, + RawDataBagContents, + RawSecretRevisionContents, + Relation, + RelationBase, + Resource, + Secret, + State, + Storage, + StoredState, + SubordinateRelation, + TCPPort, + UDPPort, + UnitID, + UnknownStatus, + WaitingStatus, +) + +__all__ = [ + "ActionFailed", + "ActiveStatus", + "Address", + "AnyJson", + "BindAddress", + "BlockedStatus", + "CharmEvents", + "CharmType", + "CheckInfo", + "CloudCredential", + "CloudSpec", + "Container", + "Context", + "DeferredEvent", + "ErrorStatus", + "Exec", + "ICMPPort", + "JujuLogLine", + "MaintenanceStatus", + "Manager", + "Model", + "Mount", + "Network", + "Notice", + "PeerRelation", + "Port", + "RawDataBagContents", + "RawSecretRevisionContents", + "Relation", + "RelationBase", + "Resource", + "Secret", + "State", + "StateValidationError", + "Storage", + "StoredState", + "SubordinateRelation", + "TCPPort", + "UDPPort", + "UnitID", + "UnknownStatus", + "WaitingStatus", +] diff --git a/testing/build/lib/scenario/_consistency_checker.py b/testing/build/lib/scenario/_consistency_checker.py new file mode 100644 index 000000000..1d21a60ac --- /dev/null +++ b/testing/build/lib/scenario/_consistency_checker.py @@ -0,0 +1,720 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +""" +The :meth:`check_consistency` function is the primary entry point for the +consistency checks. Calling it ensures that the :class:`State` and the event, +in combination with the ``Context``, is viable in Juju. For example, Juju can't +emit a ``foo-relation-changed`` event on your charm unless your charm has +declared a ``foo`` relation endpoint in its metadata. + +Normally, there is no need to explicitly call this function; that happens +automatically behind the scenes whenever you trigger an event. + +If you have a clear false negative, are explicitly testing 'edge', +inconsistent situations, or for whatever reason the checker is in your way, you +can set the ``SCENARIO_SKIP_CONSISTENCY_CHECKS`` environment variable and skip +it altogether. +""" + +import marshal +import os +import re +from collections import Counter, defaultdict +from collections.abc import Sequence +from numbers import Number +from typing import ( + TYPE_CHECKING, + Any, + Callable, + Dict, + Iterable, + List, + NamedTuple, + Optional, + Set, + Tuple, + Union, +) + +from .errors import InconsistentScenarioError +from .runtime import logger as scenario_logger +from .state import ( + CharmType, + PeerRelation, + SubordinateRelation, + _Action, + _CharmSpec, + _normalise_name, +) + +if TYPE_CHECKING: # pragma: no cover + from .state import State, _Event + +logger = scenario_logger.getChild("consistency_checker") + + +class Results(NamedTuple): + """Consistency checker return type. + + Each consistency check function returns a ``Results`` instance with the + warnings and errors found during the check. + """ + + errors: Iterable[str] + warnings: Iterable[str] + + +def check_consistency( + state: "State", + event: "_Event", + charm_spec: "_CharmSpec[Any]", + juju_version: str, +): + """Validate the combination of a state, an event, a charm spec, and a Juju version. + + When invoked, it performs a series of checks that validate that the state is + consistent with itself, with the event being emitted, the charm metadata, + and so on. + + This function performs some basic validation of the combination of inputs + that goes into a test and determines if the scenario is a + realistic/plausible/consistent one. + + A scenario is inconsistent if it can practically never occur because it + contradicts the Juju model. For example: Juju guarantees that upon calling + ``config-get``, a charm will only ever get the keys it declared in its + config metadata, so a :class:`scenario.State` declaring some config keys + that are not in the charm's ``charmcraft.yaml`` is nonsense, and the + combination of the two is inconsistent. + """ + juju_version_: Tuple[int, ...] = tuple(map(int, juju_version.split("."))) + + if os.getenv("SCENARIO_SKIP_CONSISTENCY_CHECKS"): + logger.info("skipping consistency checks.") + return + + errors: List[str] = [] + warnings: List[str] = [] + + checks: Tuple[Callable[..., Results]] = ( + check_containers_consistency, + check_config_consistency, + check_resource_consistency, + check_event_consistency, + check_secrets_consistency, + check_storages_consistency, + check_relation_consistency, + check_network_consistency, + check_cloudspec_consistency, + check_storedstate_consistency, + ) # type: ignore + for check in checks: + results = check( + state=state, + event=event, + charm_spec=charm_spec, + juju_version=juju_version_, + ) + errors.extend(results.errors) + warnings.extend(results.warnings) + + if errors: + err_fmt = "\n".join(errors) + raise InconsistentScenarioError( + f"Inconsistent scenario. The following errors were found: {err_fmt}", + ) + if warnings: + err_fmt = "\n".join(warnings) + logger.warning( + f"This scenario is probably inconsistent. Double check, and ignore this " + f"warning if you're sure. " + f"The following warnings were found: {err_fmt}", + ) + + +def check_resource_consistency( + *, + state: "State", + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the internal consistency of the resources from metadata and in :class:`scenario.State`.""" + errors: List[str] = [] + warnings: List[str] = [] + + resources_from_meta = set(charm_spec.meta.get("resources", {})) + resources_from_state = {resource.name for resource in state.resources} + if not resources_from_meta.issuperset(resources_from_state): + errors.append( + f"any and all resources passed to State.resources need to have been defined in " + f"metadata.yaml. Metadata resources: {resources_from_meta}; " + f"State.resources: {resources_from_state}.", + ) + return Results(errors, warnings) + + +def check_event_consistency( + *, + event: "_Event", + charm_spec: "_CharmSpec[CharmType]", + state: "State", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the internal consistency of the ``_Event`` data structure. + + For example, it checks that a relation event has a relation instance, and that + the relation endpoint name matches the event prefix. + """ + errors: List[str] = [] + warnings: List[str] = [] + + if not event._is_builtin_event(charm_spec): + # This is a custom event - we can't make assumptions about its name and + # semantics. It doesn't really make sense to do checks that are designed + # for relations, workloads, and so on - most likely those will end up + # with false positives. Realistically, we can't know about what the + # requirements for the custom event are (in terms of the state), so we + # skip everything here. Perhaps in the future, custom events could + # optionally include some sort of state metadata that made testing + # consistency possible? + return Results(errors, warnings) + + if event._is_relation_event: + _check_relation_event(charm_spec, event, state, errors, warnings) + + if event._is_workload_event: + _check_workload_event(charm_spec, event, state, errors, warnings) + + if event._is_action_event: + _check_action_event(charm_spec, event, state, errors, warnings) + + if event._is_storage_event: + _check_storage_event(charm_spec, event, state, errors, warnings) + + return Results(errors, warnings) + + +def _check_relation_event( + charm_spec: _CharmSpec[CharmType], # noqa: U100 + event: "_Event", + state: "State", + errors: List[str], + warnings: List[str], # noqa: U100 +): + if not event.relation: + errors.append( + "cannot construct a relation event without the relation instance. " + "Please pass one.", + ) + else: + if not event.name.startswith(_normalise_name(event.relation.endpoint)): + errors.append( + f"relation event should start with relation endpoint name. {event.name} does " + f"not start with {event.relation.endpoint}.", + ) + if event.relation not in state.relations: + errors.append( + f"cannot emit {event.name} because relation {event.relation.id} is not in the " + f"state (a relation with the same ID is not sufficient - you must " + f"pass the object in the state to the event).", + ) + + +def _check_workload_event( + charm_spec: _CharmSpec[CharmType], # noqa: U100 + event: "_Event", + state: "State", + errors: List[str], + warnings: List[str], +): + if not event.container: + errors.append( + "cannot construct a workload event without the container instance. " + "Please pass one.", + ) + else: + if not event.name.startswith(_normalise_name(event.container.name)): + errors.append( + f"workload event should start with container name. {event.name} does " + f"not start with {event.container.name}.", + ) + if event.container not in state.containers: + errors.append( + f"cannot emit {event.name} because container {event.container.name} " + f"is not in the state (a container with the same name is not " + f"sufficient - you must pass the object in the state to the event).", + ) + if not event.container.can_connect: + warnings.append( + "you **can** fire fire pebble-ready while the container cannot connect, " + "but that's most likely not what you want.", + ) + names = Counter(exec.command_prefix for exec in event.container.execs) + if dupes := [n for n in names if names[n] > 1]: + errors.append( + f"container {event.container.name} has duplicate command prefixes: {dupes}", + ) + + +def _check_action_event( + charm_spec: _CharmSpec[CharmType], + event: "_Event", + state: "State", # noqa: U100 + errors: List[str], + warnings: List[str], +): + action = event.action + if not action: + errors.append( + "cannot construct a workload event without the container instance. " + "Please pass one.", + ) + return + + elif not event.name.startswith(_normalise_name(action.name)): + errors.append( + f"action event should start with action name. {event.name} does " + f"not start with {action.name}.", + ) + if action.name not in (charm_spec.actions or ()): + errors.append( + f"action event {event.name} refers to action {action.name} " + f"which is not declared in the charm metadata (actions.yaml).", + ) + return + + _check_action_param_types(charm_spec, action, errors, warnings) + + +def _check_storage_event( + charm_spec: _CharmSpec[CharmType], + event: "_Event", + state: "State", + errors: List[str], + warnings: List[str], # noqa: U100 +): + storage = event.storage + meta = charm_spec.meta + + if not storage: + errors.append( + "cannot construct a storage event without the Storage instance. " + "Please pass one.", + ) + elif not event.name.startswith(_normalise_name(storage.name)): + errors.append( + f"storage event should start with storage name. {event.name} does " + f"not start with {storage.name}.", + ) + elif storage.name not in meta["storage"]: + errors.append( + f"storage event {event.name} refers to storage {storage.name} " + f"which is not declared in the charm metadata (metadata.yaml) under 'storage'.", + ) + elif storage not in state.storages: + errors.append( + f"cannot emit {event.name} because storage {storage.name} " + f"is not in the state (an object with the same name and index is not " + f"sufficient - you must pass the object in the state to the event).", + ) + + +def _check_action_param_types( + charm_spec: _CharmSpec[CharmType], + action: _Action, + errors: List[str], + warnings: List[str], +): + actions = charm_spec.actions + if not actions: + return + + to_python_type = { + "string": str, + "boolean": bool, + "integer": int, + "number": Number, + "array": Sequence, + "object": dict, + } + expected_param_type: Dict[str, Any] = {} + for par_name, par_spec in actions[action.name].get("params", {}).items(): + value = par_spec.get("type") + if not value: + errors.append( + f"action parameter {par_name} has no type. " + f"Charmcraft will be unhappy about this. ", + ) + continue + + try: + expected_param_type[par_name] = to_python_type[value] + except KeyError: + warnings.append( + f"unknown data type declared for parameter {par_name}: type={value}. " + f"Cannot consistency-check.", + ) + + for provided_param_name, provided_param_value in action.params.items(): + expected_type = expected_param_type.get(provided_param_name) + if not expected_type: + errors.append( + f"param {provided_param_name} is not a valid parameter for {action.name}: " + "missing from action specification", + ) + continue + if not isinstance(provided_param_value, expected_type): + errors.append( + f"param {provided_param_name} is of type {type(provided_param_value)}: " + f"expecting {expected_type}", + ) + + +def check_storages_consistency( + *, + state: "State", + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of the :class:`scenario.State` storages with the charm_spec metadata.""" + state_storage = state.storages + meta_storage = (charm_spec.meta or {}).get("storage", {}) + errors: List[str] = [] + + if missing := {s.name for s in state_storage}.difference( + set(meta_storage.keys()), + ): + errors.append( + f"some storages passed to State were not defined in metadata.yaml: {missing}", + ) + + seen: List[Tuple[str, int]] = [] + for s in state_storage: + tag = (s.name, s.index) + if tag in seen: + errors.append( + f"duplicate storage in State: storage {s.name} with index {s.index} " + f"occurs multiple times in State.storages.", + ) + seen.append(tag) + + return Results(errors, []) + + +def _is_secret_identifier(value: Union[str, int, float, bool]) -> bool: + """Return true iff the value is in the form `secret:{secret id}`.""" + # cf. https://github.com/juju/juju/blob/13eb9df3df16a84fd471af8a3c95ddbd04389b71/core/secrets/secret.go#L48 + return bool(re.match(r"secret:[0-9a-z]{20}$", str(value))) + + +def check_config_consistency( + *, + state: "State", + charm_spec: "_CharmSpec[CharmType]", + juju_version: Tuple[int, ...], + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of the :class:`scenario.State` config with the charm_spec config.""" + state_config = state.config + meta_config = (charm_spec.config or {}).get("options", {}) + errors: List[str] = [] + + for key, value in state_config.items(): + if key not in meta_config: + errors.append( + f"config option {key!r} in state.config but not specified in config.yaml or " + f"charmcraft.yaml.", + ) + continue + + converters = { + "string": str, + "int": int, + "float": float, + "boolean": bool, + } + if juju_version >= (3, 4): + converters["secret"] = str + + validators = { + "secret": _is_secret_identifier, + } + + expected_type_name = meta_config[key].get("type", None) + if not expected_type_name: + errors.append(f"config.yaml invalid; option {key!r} has no 'type'.") + continue + validator = validators.get(expected_type_name) + + expected_type = converters.get(expected_type_name) + if not expected_type: + errors.append( + f"config invalid for option {key!r}: 'type' {expected_type_name} unknown", + ) + + elif not isinstance(value, expected_type): + errors.append( + f"config invalid; option {key!r} should be of type {expected_type} " + f"but is of type {type(value)}.", + ) + + elif validator and not validator(value): + errors.append( + f"config invalid: option {key!r} value {value!r} is not valid.", + ) + + return Results(errors, []) + + +def check_secrets_consistency( + *, + event: "_Event", + state: "State", + juju_version: Tuple[int, ...], + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of any :class:`scenario.Secret` in the :class:`scenario.State`.""" + errors: List[str] = [] + if not event._is_secret_event: + return Results(errors, []) + + assert event.secret is not None + if event.secret not in state.secrets: + secret_key = event.secret.id if event.secret.id else event.secret.label + errors.append( + f"cannot emit {event.name} because secret {secret_key} is not in the state " + f"(a secret with the same ID is not sufficient - you must pass the object " + f"in the state to the event).", + ) + elif juju_version < (3,): + errors.append( + f"secrets are not supported in the specified juju version {juju_version}. " + f"Should be at least 3.0.", + ) + + return Results(errors, []) + + +def check_network_consistency( + *, + state: "State", + event: "_Event", # noqa: U100 + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of any :class:`scenario.Network` in the :class:`scenario.State`.""" + errors: List[str] = [] + + meta_bindings = set(charm_spec.meta.get("extra-bindings", ())) + # add the implicit juju-info binding so we can override its network without + # having to declare a relation for it in metadata + implicit_bindings = {"juju-info"} + all_relations = charm_spec.get_all_relations() + non_sub_relations = { + endpoint + for endpoint, metadata in all_relations + if metadata.get("scope") != "container" # mark of a sub + } + + state_bindings = {network.binding_name for network in state.networks} + if diff := state_bindings.difference( + meta_bindings.union(non_sub_relations).union(implicit_bindings), + ): + errors.append( + f"Some network bindings defined in State are not in the metadata: {diff}.", + ) + + endpoints = {endpoint for endpoint, _ in all_relations} + if collisions := endpoints.intersection(meta_bindings): + errors.append( + f"Extra bindings and integration endpoints cannot share the same name: {collisions}.", + ) + + return Results(errors, []) + + +def check_relation_consistency( + *, + state: "State", + event: "_Event", # noqa: U100 + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of any relations in the :class:`scenario.State`.""" + errors: List[str] = [] + + peer_relations_meta = charm_spec.meta.get("peers", {}).items() + all_relations_meta = charm_spec.get_all_relations() + + def _get_relations(r: str): + try: + return state.get_relations(r) + except ValueError: + return () + + # check relation types + for endpoint, _ in peer_relations_meta: + for relation in _get_relations(endpoint): + if not isinstance(relation, PeerRelation): + errors.append( + f"endpoint {endpoint} is a peer relation; " + f"expecting relation to be of type PeerRelation, got {type(relation)}", + ) + + known_endpoints = [a[0] for a in all_relations_meta] + for relation in state.relations: + if (ep := relation.endpoint) not in known_endpoints: + errors.append(f"relation endpoint {ep} is not declared in metadata.") + + seen_ids: Set[int] = set() + for endpoint, relation_meta in all_relations_meta: + expected_sub = relation_meta.get("scope", "") == "container" + relations = _get_relations(endpoint) + for relation in relations: + if relation.id in seen_ids: + errors.append( + f"duplicate relation ID: {relation.id} is claimed " + f"by multiple Relation instances", + ) + + seen_ids.add(relation.id) + is_sub = isinstance(relation, SubordinateRelation) + if is_sub and not expected_sub: + errors.append( + f"endpoint {endpoint} is not a subordinate relation; " + f"expecting relation to be of type Relation, " + f"got {type(relation)}", + ) + if expected_sub and not is_sub: + errors.append( + f"endpoint {endpoint} is not a subordinate relation; " + f"expecting relation to be of type SubordinateRelation, " + f"got {type(relation)}", + ) + + # check for duplicate endpoint names + seen_endpoints: Set[str] = set() + for endpoint, _ in all_relations_meta: + if endpoint in seen_endpoints: + errors.append("duplicate endpoint name in metadata.") + break + seen_endpoints.add(endpoint) + + return Results(errors, []) + + +def check_containers_consistency( + *, + state: "State", + event: "_Event", + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the consistency of :class:`scenario.State` containers with the charm_spec metadata.""" + + # event names will be normalized; need to compare against normalized container names. + meta = charm_spec.meta + meta_containers = list(map(_normalise_name, meta.get("containers", {}))) + state_containers = [_normalise_name(c.name) for c in state.containers] + all_notices = {notice.id for c in state.containers for notice in c.notices} + all_checks = { + (c.name, check.name) for c in state.containers for check in c.check_infos + } + errors: List[str] = [] + + # it's fine if you have containers in meta that are not in state.containers (yet), but it's + # not fine if: + # - you're processing a Pebble event and that container is not in state.containers or + # meta.containers + if event._is_workload_event: + evt_container_name = event.name.split("_pebble_")[0] + if evt_container_name not in meta_containers: + errors.append( + f"the event being processed concerns container {evt_container_name!r}, but a " + f"container with that name is not declared in the charm metadata", + ) + if evt_container_name not in state_containers: + errors.append( + f"the event being processed concerns container {evt_container_name!r}, but a " + f"container with that name is not present in the state. It's odd, but " + f"consistent, if it cannot connect; but it should at least be there.", + ) + # - you're processing a Notice event and that notice is not in any of the containers + if event.notice and event.notice.id not in all_notices: + errors.append( + f"the event being processed concerns notice {event.notice!r}, but that " + "notice is not in any of the containers present in the state.", + ) + # - you're processing a Check event and that check is not in the check's container + if ( + event.check_info + and (evt_container_name, event.check_info.name) not in all_checks + ): + errors.append( + f"the event being processed concerns check {event.check_info.name}, but that " + f"check is not in the {evt_container_name} container.", + ) + + # - a container in state.containers is not in meta.containers + if diff := (set(state_containers).difference(set(meta_containers))): + errors.append( + f"some containers declared in the state are not specified in metadata. " + f"That's not possible. " + f"Missing from metadata: {diff}.", + ) + + return Results(errors, []) + + +def check_cloudspec_consistency( + *, + state: "State", + event: "_Event", + charm_spec: "_CharmSpec[CharmType]", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check that Kubernetes models don't have :attr:`scenario.State.cloud_spec` set.""" + + errors: List[str] = [] + warnings: List[str] = [] + + if state.model.type == "kubernetes" and state.model.cloud_spec: + errors.append( + "CloudSpec is only available for machine charms, not Kubernetes charms. " + "Simulate a machine substrate with: `State(..., model=Model(type='lxd'))`.", + ) + + return Results(errors, warnings) + + +def check_storedstate_consistency( + *, + state: "State", + **_kwargs: Any, # noqa: U101 +) -> Results: + """Check the internal consistency of any :class:`scenario.StoredState` in the :class:`scenario.State`.""" + errors: List[str] = [] + + # Attribute names must be unique on each object. + names: defaultdict[Optional[str], List[str]] = defaultdict(list) + for ss in state.stored_states: + names[ss.owner_path].append(ss.name) + for owner, owner_names in names.items(): + if len(owner_names) != len(set(owner_names)): + errors.append( + f"{owner} has multiple StoredState objects with the same name.", + ) + + # The content must be marshallable. + for ss in state.stored_states: + # We don't need the marshalled state, just to know that it can be. + # This is the same "only simple types" check that ops does. + try: + marshal.dumps(ss.content) + except ValueError: + errors.append( + f"The StoredState object {ss.owner_path}.{ss.name} should contain only simple types.", + ) + return Results(errors, []) diff --git a/testing/build/lib/scenario/_ops_main_mock.py b/testing/build/lib/scenario/_ops_main_mock.py new file mode 100644 index 000000000..87f2ec1ab --- /dev/null +++ b/testing/build/lib/scenario/_ops_main_mock.py @@ -0,0 +1,197 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +import dataclasses +import marshal +import re +import sys +from typing import TYPE_CHECKING, Any, Dict, FrozenSet, List, Sequence, Set + +import ops +import ops.jujucontext +import ops.storage + +from ops.framework import _event_regex +from ops._main import _Dispatcher, _Manager +from ops._main import logger as ops_logger + +from .errors import BadOwnerPath, NoObserverError +from .logger import logger as scenario_logger +from .mocking import _MockModelBackend +from .state import DeferredEvent, StoredState + +if TYPE_CHECKING: # pragma: no cover + from .context import Context + from .state import CharmType, State, _CharmSpec, _Event + +EVENT_REGEX = re.compile(_event_regex) +STORED_STATE_REGEX = re.compile( + r"((?P.*)\/)?(?P<_data_type_name>\D+)\[(?P.*)\]", +) + +logger = scenario_logger.getChild("ops_main_mock") + +# pyright: reportPrivateUsage=false + + +class UnitStateDB: + """Wraps the unit-state database with convenience methods for adjusting the state.""" + + def __init__(self, underlying_store: ops.storage.SQLiteStorage): + self._db = underlying_store + + def get_stored_states(self) -> FrozenSet["StoredState"]: + """Load any StoredState data structures from the db.""" + db = self._db + stored_states: Set[StoredState] = set() + for handle_path in db.list_snapshots(): + if not EVENT_REGEX.match(handle_path) and ( + match := STORED_STATE_REGEX.match(handle_path) + ): + stored_state_snapshot = db.load_snapshot(handle_path) + kwargs = match.groupdict() + sst = StoredState(content=stored_state_snapshot, **kwargs) + stored_states.add(sst) + + return frozenset(stored_states) + + def get_deferred_events(self) -> List["DeferredEvent"]: + """Load any DeferredEvent data structures from the db.""" + db = self._db + deferred: List[DeferredEvent] = [] + for handle_path in db.list_snapshots(): + if EVENT_REGEX.match(handle_path): + notices = db.notices(handle_path) + for handle, owner, observer in notices: + try: + snapshot_data = db.load_snapshot(handle) + except ops.storage.NoSnapshotError: + snapshot_data: Dict[str, Any] = {} + + event = DeferredEvent( + handle_path=handle, + owner=owner, + observer=observer, + snapshot_data=snapshot_data, + ) + deferred.append(event) + + return deferred + + def apply_state(self, state: "State"): + """Add DeferredEvent and StoredState from this State instance to the storage.""" + db = self._db + for event in state.deferred: + db.save_notice(event.handle_path, event.owner, event.observer) + try: + marshal.dumps(event.snapshot_data) + except ValueError as e: + raise ValueError( + f"unable to save the data for {event}, it must contain only simple types.", + ) from e + db.save_snapshot(event.handle_path, event.snapshot_data) + + for stored_state in state.stored_states: + db.save_snapshot(stored_state._handle_path, stored_state.content) + + +class Ops(_Manager): + """Class to manage stepping through ops setup, event emission and framework commit.""" + + def __init__( + self, + state: "State", + event: "_Event", + context: "Context", + charm_spec: "_CharmSpec[CharmType]", + juju_context: ops.jujucontext._JujuContext, + ): + self.state = state + self.event = event + self.context = context + self.charm_spec = charm_spec + self.store = None + + model_backend = _MockModelBackend( + state=state, + event=event, + context=context, + charm_spec=charm_spec, + juju_context=juju_context, + ) + + super().__init__( + self.charm_spec.charm_type, model_backend, juju_context=juju_context + ) + + def _load_charm_meta(self): + metadata = (self._charm_root / "metadata.yaml").read_text() + actions_meta = self._charm_root / "actions.yaml" + if actions_meta.exists(): + actions_metadata = actions_meta.read_text() + else: + actions_metadata = None + + return ops.CharmMeta.from_yaml(metadata, actions_metadata) + + def _setup_root_logging(self): + # Ops sets sys.excepthook to go to Juju's debug-log, but that's not + # useful in a testing context, so we reset it here. + super()._setup_root_logging() + sys.excepthook = sys.__excepthook__ + + def _make_storage(self, _: _Dispatcher): + # TODO: add use_juju_for_storage support + # TODO: Pass a charm_state_path that is ':memory:' when appropriate. + charm_state_path = self._charm_root / self._charm_state_path + storage = ops.storage.SQLiteStorage(charm_state_path) + logger.info("Copying input state to storage.") + self.store = UnitStateDB(storage) + self.store.apply_state(self.state) + return storage + + def _get_event_to_emit(self, event_name: str): + owner = ( + self._get_owner(self.charm, self.event.owner_path) + if self.event + else self.charm.on + ) + + try: + event_to_emit = getattr(owner, event_name) + except AttributeError: + ops_logger.debug("Event %s not defined for %s.", event_name, self.charm) + raise NoObserverError( + f"Cannot fire {event_name!r} on {owner}: " + f"invalid event (not on charm.on).", + ) + return event_to_emit + + @staticmethod + def _get_owner(root: Any, path: Sequence[str]) -> ops.ObjectEvents: + """Walk path on root to an ObjectEvents instance.""" + obj = root + for step in path: + try: + obj = getattr(obj, step) + except AttributeError: + raise BadOwnerPath( + f"event_owner_path {path!r} invalid: {step!r} leads to nowhere.", + ) + if not isinstance(obj, ops.ObjectEvents): + raise BadOwnerPath( + f"event_owner_path {path!r} invalid: does not lead to " + f"an ObjectEvents instance.", + ) + return obj + + def _close(self): + """Now that we're done processing this event, read the charm state and expose it.""" + logger.info("Copying storage to output state.") + assert self.store is not None + deferred = self.store.get_deferred_events() + stored_state = self.store.get_stored_states() + self.state = dataclasses.replace( + self.state, deferred=deferred, stored_states=stored_state + ) diff --git a/testing/build/lib/scenario/_runtime.py b/testing/build/lib/scenario/_runtime.py new file mode 100644 index 000000000..889a2d96a --- /dev/null +++ b/testing/build/lib/scenario/_runtime.py @@ -0,0 +1,438 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +import copy +import dataclasses +import tempfile +import typing +from contextlib import contextmanager +from pathlib import Path +from typing import ( + TYPE_CHECKING, + Dict, + List, + Optional, + Type, + TypeVar, + Union, +) + +import yaml +from ops import ( + CollectStatusEvent, + pebble, + CommitEvent, + EventBase, + Framework, + Handle, + NoTypeError, + PreCommitEvent, +) +from ops.jujucontext import _JujuContext +from ops._private.harness import ActionFailed + +from .errors import NoObserverError, UncaughtCharmError +from .logger import logger as scenario_logger +from .state import ( + PeerRelation, + Relation, + SubordinateRelation, +) + +if TYPE_CHECKING: # pragma: no cover + from .context import Context + from .state import CharmType, State, _CharmSpec, _Event + +logger = scenario_logger.getChild("runtime") + +RUNTIME_MODULE = Path(__file__).parent + + +class Runtime: + """Charm runtime wrapper. + + This object bridges a local environment and a charm artifact. + """ + + def __init__( + self, + charm_spec: "_CharmSpec[CharmType]", + charm_root: Optional[Union[str, Path]] = None, + juju_version: str = "3.0.0", + app_name: Optional[str] = None, + unit_id: Optional[int] = 0, + ): + self._charm_spec = charm_spec + self._juju_version = juju_version + self._charm_root = charm_root + + app_name = app_name or self._charm_spec.meta.get("name") + if not app_name: + raise ValueError('invalid metadata: mandatory "name" field is missing.') + + self._app_name = app_name + self._unit_id = unit_id + + def _get_event_env(self, state: "State", event: "_Event", charm_root: Path): + """Build the simulated environment the operator framework expects.""" + env = { + "JUJU_VERSION": self._juju_version, + "JUJU_UNIT_NAME": f"{self._app_name}/{self._unit_id}", + "_": "./dispatch", + "JUJU_DISPATCH_PATH": f"hooks/{event.name}", + "JUJU_MODEL_NAME": state.model.name, + "JUJU_MODEL_UUID": state.model.uuid, + "JUJU_CHARM_DIR": str(charm_root.absolute()), + # todo consider setting pwd, (python)path + } + + if event._is_action_event and (action := event.action): + env.update( + { + # TODO: we should check we're doing the right thing here. + "JUJU_ACTION_NAME": action.name.replace("_", "-"), + "JUJU_ACTION_UUID": action.id, + }, + ) + + if event._is_relation_event and (relation := event.relation): + if isinstance(relation, PeerRelation): + remote_app_name = self._app_name + elif isinstance(relation, (Relation, SubordinateRelation)): + remote_app_name = relation.remote_app_name + else: + raise ValueError(f"Unknown relation type: {relation}") + env.update( + { + "JUJU_RELATION": relation.endpoint, + "JUJU_RELATION_ID": str(relation.id), + "JUJU_REMOTE_APP": remote_app_name, + }, + ) + + remote_unit_id = event.relation_remote_unit_id + + # don't check truthiness because remote_unit_id could be 0 + if remote_unit_id is None and not event.name.endswith( + ("_relation_created", "relation_broken"), + ): + remote_unit_ids = relation._remote_unit_ids + + if len(remote_unit_ids) == 1: + remote_unit_id = remote_unit_ids[0] + logger.info( + "there's only one remote unit, so we set JUJU_REMOTE_UNIT to it, " + "but you probably should be parametrizing the event with `remote_unit_id` " + "to be explicit.", + ) + elif len(remote_unit_ids) > 1: + remote_unit_id = remote_unit_ids[0] + logger.warning( + "remote unit ID unset, and multiple remote unit IDs are present; " + "We will pick the first one and hope for the best. You should be passing " + "`remote_unit_id` to the Event constructor.", + ) + else: + logger.warning( + "remote unit ID unset; no remote unit data present. " + "Is this a realistic scenario?", # TODO: is it? + ) + + if remote_unit_id is not None: + remote_unit = f"{remote_app_name}/{remote_unit_id}" + env["JUJU_REMOTE_UNIT"] = remote_unit + if event.name.endswith("_relation_departed"): + if event.relation_departed_unit_id: + env["JUJU_DEPARTING_UNIT"] = ( + f"{remote_app_name}/{event.relation_departed_unit_id}" + ) + else: + env["JUJU_DEPARTING_UNIT"] = remote_unit + + if container := event.container: + env.update({"JUJU_WORKLOAD_NAME": container.name}) + + if notice := event.notice: + if hasattr(notice.type, "value"): + notice_type = typing.cast(pebble.NoticeType, notice.type).value + else: + notice_type = str(notice.type) + env.update( + { + "JUJU_NOTICE_ID": notice.id, + "JUJU_NOTICE_TYPE": notice_type, + "JUJU_NOTICE_KEY": notice.key, + }, + ) + + if check_info := event.check_info: + env["JUJU_PEBBLE_CHECK_NAME"] = check_info.name + + if storage := event.storage: + env.update({"JUJU_STORAGE_ID": f"{storage.name}/{storage.index}"}) + + if secret := event.secret: + env.update( + { + "JUJU_SECRET_ID": secret.id, + "JUJU_SECRET_LABEL": secret.label or "", + }, + ) + # Don't check truthiness because revision could be 0. + if event.secret_revision is not None: + env["JUJU_SECRET_REVISION"] = str(event.secret_revision) + + return env + + @staticmethod + def _wrap(charm_type: Type["CharmType"]) -> Type["CharmType"]: + # dark sorcery to work around framework using class attrs to hold on to event sources + # todo this should only be needed if we call play multiple times on the same runtime. + # can we avoid it? + class WrappedEvents(charm_type.on.__class__): + pass + + WrappedEvents.__name__ = charm_type.on.__class__.__name__ + + class WrappedCharm(charm_type): + on = WrappedEvents() + + WrappedCharm.__name__ = charm_type.__name__ + return typing.cast(Type["CharmType"], WrappedCharm) + + @contextmanager + def _virtual_charm_root(self): + # If we are using runtime on a real charm, we can make some assumptions about the + # directory structure we are going to find. + # If we're, say, dynamically defining charm types and doing tests on them, we'll have to + # generate the metadata files ourselves. To be sure, we ALWAYS use a tempdir. Ground truth + # is what the user passed via the CharmSpec + spec = self._charm_spec + + if charm_virtual_root := self._charm_root: + charm_virtual_root_is_custom = True + virtual_charm_root = Path(charm_virtual_root) + else: + charm_virtual_root = tempfile.TemporaryDirectory() + virtual_charm_root = Path(charm_virtual_root.name) + charm_virtual_root_is_custom = False + + metadata_yaml = virtual_charm_root / "metadata.yaml" + config_yaml = virtual_charm_root / "config.yaml" + actions_yaml = virtual_charm_root / "actions.yaml" + + metadata_files_present: Dict[Path, Optional[str]] = { + file: file.read_text() if file.exists() else None + for file in (metadata_yaml, config_yaml, actions_yaml) + } + + any_metadata_files_present_in_charm_virtual_root = any( + v is not None for v in metadata_files_present.values() + ) + + if spec.is_autoloaded and charm_virtual_root_is_custom: + # since the spec is autoloaded, in theory the metadata contents won't differ, so we can + # overwrite away even if the custom vroot is the real charm root (the local repo). + # Still, log it for clarity. + if any_metadata_files_present_in_charm_virtual_root: + logger.debug( + f"metadata files found in custom charm_root {charm_virtual_root}. " + f"The spec was autoloaded so the contents should be identical. " + f"Proceeding...", + ) + + elif ( + not spec.is_autoloaded and any_metadata_files_present_in_charm_virtual_root + ): + logger.warning( + f"Some metadata files found in custom user-provided charm_root " + f"{charm_virtual_root} while you have passed meta, config or actions to " + f"Context.run(). " + "Single source of truth are the arguments passed to Context.run(). " + "charm_root metadata files will be overwritten for the " + "duration of this test, and restored afterwards. " + "To avoid this, clean any metadata files from the charm_root before calling run.", + ) + + metadata_yaml.write_text(yaml.safe_dump(spec.meta)) + config_yaml.write_text(yaml.safe_dump(spec.config or {})) + actions_yaml.write_text(yaml.safe_dump(spec.actions or {})) + + yield virtual_charm_root + + if charm_virtual_root_is_custom: + for file, previous_content in metadata_files_present.items(): + if previous_content is None: # None == file did not exist before + file.unlink() + else: + file.write_text(previous_content) + + else: + # charm_virtual_root is a tempdir + typing.cast(tempfile.TemporaryDirectory, charm_virtual_root).cleanup() # type: ignore + + @contextmanager + def _exec_ctx(self, ctx: "Context"): + """python 3.8 compatibility shim""" + with self._virtual_charm_root() as temporary_charm_root: + # TODO: allow customising capture_events + with capture_events( + include_deferred=ctx.capture_deferred_events, + include_framework=ctx.capture_framework_events, + ) as captured: + yield (temporary_charm_root, captured) + + @contextmanager + def exec( + self, + state: "State", + event: "_Event", + context: "Context", + ): + """Runs an event with this state as initial state on a charm. + + Returns the 'output state', that is, the state as mutated by the charm during the + event handling. + + This will set the environment up and call ops.main(). + After that it's up to ops. + """ + # todo consider forking out a real subprocess and do the mocking by + # mocking hook tool executables + + from ._consistency_checker import check_consistency # avoid cycles + + check_consistency(state, event, self._charm_spec, self._juju_version) + + charm_type = self._charm_spec.charm_type + logger.info(f"Preparing to fire {event.name} on {charm_type.__name__}") + + # we make a copy to avoid mutating the input state + output_state = copy.deepcopy(state) + + logger.info(" - generating virtual charm root") + with self._exec_ctx(context) as (temporary_charm_root, captured): + logger.info(" - preparing env") + env = self._get_event_env( + state=state, + event=event, + charm_root=temporary_charm_root, + ) + juju_context = _JujuContext.from_dict(env) + + logger.info(" - entering ops.main (mocked)") + from ._ops_main_mock import Ops # noqa: F811 + + try: + ops = Ops( + state=output_state, + event=event, + context=context, + charm_spec=dataclasses.replace( + self._charm_spec, + charm_type=self._wrap(charm_type), + ), + juju_context=juju_context, + ) + + yield ops + + except (NoObserverError, ActionFailed): + raise # propagate along + except Exception as e: + raise UncaughtCharmError( + f"Uncaught exception ({type(e)}) in operator/charm code: {e!r}", + ) from e + + finally: + logger.info(" - exited ops.main") + + context.emitted_events.extend(captured) + logger.info("event dispatched. done.") + context._set_output_state(ops.state) + + +_T = TypeVar("_T", bound=EventBase) + + +@contextmanager +def capture_events( + *types: Type[EventBase], + include_framework: bool = False, + include_deferred: bool = True, +): + """Capture all events of type `*types` (using instance checks). + + Arguments exposed so that you can define your own fixtures if you want to. + + Example:: + >>> from ops import StartEvent + >>> from scenario import Event, State + >>> from charm import MyCustomEvent, MyCharm # noqa + >>> + >>> def test_my_event(): + >>> with capture_events(StartEvent, MyCustomEvent) as captured: + >>> trigger(State(), ("start", MyCharm, meta=MyCharm.META) + >>> + >>> assert len(captured) == 2 + >>> e1, e2 = captured + >>> assert isinstance(e2, MyCustomEvent) + >>> assert e2.custom_attr == 'foo' + """ + allowed_types = types or (EventBase,) + + captured: List[EventBase] = [] + _real_emit = Framework._emit + _real_reemit = Framework.reemit + + def _wrapped_emit(self: Framework, evt: EventBase): + if not include_framework and isinstance( + evt, + (PreCommitEvent, CommitEvent, CollectStatusEvent), + ): + return _real_emit(self, evt) + + if isinstance(evt, allowed_types): + # dump/undump the event to ensure any custom attributes are (re)set by restore() + evt.restore(evt.snapshot()) + captured.append(evt) + + return _real_emit(self, evt) + + def _wrapped_reemit(self: Framework): + # Framework calls reemit() before emitting the main juju event. We intercept that call + # and capture all events in storage. + + if not include_deferred: + return _real_reemit(self) + + # load all notices from storage as events. + for event_path, _, _ in self._storage.notices(): + event_handle = Handle.from_path(event_path) + try: + event = self.load_snapshot(event_handle) + except NoTypeError: + continue + event = typing.cast(EventBase, event) + event.deferred = False + self._forget(event) # prevent tracking conflicts + + if not include_framework and isinstance( + event, + (PreCommitEvent, CommitEvent), + ): + continue + + if isinstance(event, allowed_types): + captured.append(event) + + return _real_reemit(self) + + Framework._emit = _wrapped_emit # type: ignore + Framework.reemit = _wrapped_reemit + + yield captured + + Framework._emit = _real_emit + Framework.reemit = _real_reemit diff --git a/testing/build/lib/scenario/context.py b/testing/build/lib/scenario/context.py new file mode 100644 index 000000000..8087480f2 --- /dev/null +++ b/testing/build/lib/scenario/context.py @@ -0,0 +1,692 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Test Context + +The test `Context` object provides the context of the wider Juju system that the +specific `State` exists in, and the events that can be executed on that `State`. +""" + +from __future__ import annotations + +import functools +import tempfile +from contextlib import contextmanager +from pathlib import Path +from typing import ( + Generic, + TYPE_CHECKING, + Any, + Callable, + Mapping, +) + +import ops +from ops._private.harness import ActionFailed + +from .errors import ( + AlreadyEmittedError, + ContextSetupError, + MetadataNotFoundError, +) +from .logger import logger as scenario_logger +from .runtime import Runtime +from .state import ( + CharmType, + CheckInfo, + Container, + Notice, + Secret, + Storage, + _Action, + _CharmSpec, + _Event, +) + +if TYPE_CHECKING: # pragma: no cover + from ops._private.harness import ExecArgs + from .ops_main_mock import Ops + from .state import ( + AnyJson, + JujuLogLine, + RelationBase, + State, + _EntityStatus, + ) + +logger = scenario_logger.getChild("runtime") + +_DEFAULT_JUJU_VERSION = "3.5" + + +class Manager(Generic[CharmType]): + """Context manager to offer test code some runtime charm object introspection. + + This class should not be instantiated directly: use a :class:`Context` + in a ``with`` statement instead, for example:: + + ctx = Context(MyCharm) + with ctx(ctx.on.start(), State()) as manager: + manager.charm.setup() + manager.run() + """ + + def __init__( + self, + ctx: Context[CharmType], + arg: _Event, + state_in: State, + ): + self._ctx = ctx + self._arg = arg + self._state_in = state_in + + self._emitted: bool = False + self._wrapped_ctx = None + + self.ops: Ops[CharmType] | None = None + + @property + def charm(self) -> CharmType: + """The charm object instantiated by ops to handle the event. + + The charm is only available during the context manager scope. + """ + if self.ops is None or self.ops.charm is None: + raise RuntimeError( + "you should __enter__ this context manager before accessing this", + ) + return self.ops.charm + + @property + def _runner(self): + return self._ctx._run # noqa + + def __enter__(self): + self._wrapped_ctx = wrapped_ctx = self._runner(self._arg, self._state_in) + self.ops = wrapped_ctx.__enter__() + return self + + def run(self) -> State: + """Emit the event and proceed with charm execution. + + This can only be done once. + """ + if self._emitted: + raise AlreadyEmittedError("Can only run once.") + self._emitted = True + + # wrap up Runtime.exec() so that we can gather the output state + assert self._wrapped_ctx is not None + self._wrapped_ctx.__exit__(None, None, None) + + assert self._ctx._output_state is not None + return self._ctx._output_state + + def __exit__(self, exc_type: Any, exc_val: Any, exc_tb: Any): # noqa: U100 + if not self._emitted: + logger.debug( + "user didn't emit the event within the context manager scope. Doing so implicitly upon exit...", + ) + self.run() + + +def _copy_doc(original_func: Callable[..., Any]): + """Copy the docstring from `original_func` to the wrapped function.""" + + def decorator(wrapper_func: Callable[..., Any]): + @functools.wraps(wrapper_func) + def wrapped(*args: Any, **kwargs: Any): + return wrapper_func(*args, **kwargs) + + wrapped.__doc__ = original_func.__doc__ + return wrapped + + return decorator + + +class CharmEvents: + """Events generated by Juju or ops pertaining to the application lifecycle. + + The events listed as attributes of this class should be accessed via the + :attr:`Context.on` attribute. For example:: + + ctx.run(ctx.on.config_changed(), state) + + This behaves similarly to the :class:`ops.CharmEvents` class but is much + simpler as there are no dynamically named attributes, and no ``__getattr__`` + version to get events. In addition, all of the attributes are methods, + which are used to connect the event to the specific object that they relate + to (or, for simpler events like "start" or "stop", take no arguments). + """ + + @staticmethod + @_copy_doc(ops.InstallEvent) + def install(): + return _Event("install") + + @staticmethod + @_copy_doc(ops.StartEvent) + def start(): + return _Event("start") + + @staticmethod + @_copy_doc(ops.StopEvent) + def stop(): + return _Event("stop") + + @staticmethod + @_copy_doc(ops.RemoveEvent) + def remove(): + return _Event("remove") + + @staticmethod + @_copy_doc(ops.UpdateStatusEvent) + def update_status(): + return _Event("update_status") + + @staticmethod + @_copy_doc(ops.ConfigChangedEvent) + def config_changed(): + return _Event("config_changed") + + @staticmethod + @_copy_doc(ops.UpgradeCharmEvent) + def upgrade_charm(): + return _Event("upgrade_charm") + + @staticmethod + @_copy_doc(ops.PreSeriesUpgradeEvent) + def pre_series_upgrade(): + return _Event("pre_series_upgrade") + + @staticmethod + @_copy_doc(ops.PostSeriesUpgradeEvent) + def post_series_upgrade(): + return _Event("post_series_upgrade") + + @staticmethod + @_copy_doc(ops.LeaderElectedEvent) + def leader_elected(): + return _Event("leader_elected") + + @staticmethod + @_copy_doc(ops.SecretChangedEvent) + def secret_changed(secret: Secret): + if secret.owner: + raise ValueError( + "This unit will never receive secret-changed for a secret it owns.", + ) + return _Event("secret_changed", secret=secret) + + @staticmethod + @_copy_doc(ops.SecretExpiredEvent) + def secret_expired(secret: Secret, *, revision: int): + if not secret.owner: + raise ValueError( + "This unit will never receive secret-expire for a secret it does not own.", + ) + return _Event("secret_expired", secret=secret, secret_revision=revision) + + @staticmethod + @_copy_doc(ops.SecretRotateEvent) + def secret_rotate(secret: Secret): + if not secret.owner: + raise ValueError( + "This unit will never receive secret-rotate for a secret it does not own.", + ) + return _Event("secret_rotate", secret=secret) + + @staticmethod + @_copy_doc(ops.SecretRemoveEvent) + def secret_remove(secret: Secret, *, revision: int): + if not secret.owner: + raise ValueError( + "This unit will never receive secret-removed for a secret it does not own.", + ) + return _Event("secret_remove", secret=secret, secret_revision=revision) + + @staticmethod + def collect_app_status(): + """Event triggered at the end of every hook to collect app statuses for evaluation""" + return _Event("collect_app_status") + + @staticmethod + def collect_unit_status(): + """Event triggered at the end of every hook to collect unit statuses for evaluation""" + return _Event("collect_unit_status") + + @staticmethod + @_copy_doc(ops.RelationCreatedEvent) + def relation_created(relation: RelationBase): + return _Event(f"{relation.endpoint}_relation_created", relation=relation) + + @staticmethod + @_copy_doc(ops.RelationJoinedEvent) + def relation_joined(relation: RelationBase, *, remote_unit: int | None = None): + return _Event( + f"{relation.endpoint}_relation_joined", + relation=relation, + relation_remote_unit_id=remote_unit, + ) + + @staticmethod + @_copy_doc(ops.RelationChangedEvent) + def relation_changed( + relation: RelationBase, + *, + remote_unit: int | None = None, + ): + return _Event( + f"{relation.endpoint}_relation_changed", + relation=relation, + relation_remote_unit_id=remote_unit, + ) + + @staticmethod + @_copy_doc(ops.RelationDepartedEvent) + def relation_departed( + relation: RelationBase, + *, + remote_unit: int | None = None, + departing_unit: int | None = None, + ): + return _Event( + f"{relation.endpoint}_relation_departed", + relation=relation, + relation_remote_unit_id=remote_unit, + relation_departed_unit_id=departing_unit, + ) + + @staticmethod + @_copy_doc(ops.RelationBrokenEvent) + def relation_broken(relation: RelationBase): + return _Event(f"{relation.endpoint}_relation_broken", relation=relation) + + @staticmethod + @_copy_doc(ops.StorageAttachedEvent) + def storage_attached(storage: Storage): + return _Event(f"{storage.name}_storage_attached", storage=storage) + + @staticmethod + @_copy_doc(ops.StorageDetachingEvent) + def storage_detaching(storage: Storage): + return _Event(f"{storage.name}_storage_detaching", storage=storage) + + @staticmethod + @_copy_doc(ops.PebbleReadyEvent) + def pebble_ready(container: Container): + return _Event(f"{container.name}_pebble_ready", container=container) + + @staticmethod + @_copy_doc(ops.PebbleCustomNoticeEvent) + def pebble_custom_notice(container: Container, notice: Notice): + return _Event( + f"{container.name}_pebble_custom_notice", + container=container, + notice=notice, + ) + + @staticmethod + @_copy_doc(ops.PebbleCheckFailedEvent) + def pebble_check_failed(container: Container, info: CheckInfo): + return _Event( + f"{container.name}_pebble_check_failed", + container=container, + check_info=info, + ) + + @staticmethod + @_copy_doc(ops.PebbleCheckRecoveredEvent) + def pebble_check_recovered(container: Container, info: CheckInfo): + return _Event( + f"{container.name}_pebble_check_recovered", + container=container, + check_info=info, + ) + + @staticmethod + @_copy_doc(ops.ActionEvent) + def action( + name: str, + params: Mapping[str, AnyJson] | None = None, + id: str | None = None, + ): + kwargs: dict[str, Any] = {} + if params: + kwargs["params"] = params + if id: + kwargs["id"] = id + return _Event(f"{name}_action", action=_Action(name, **kwargs)) + + +class Context(Generic[CharmType]): + """Represents a simulated charm's execution context. + + The main entry point to running a test. It contains: + + - the charm source code being executed + - the metadata files associated with it + - a charm project repository root + - the Juju version to be simulated + + After you have instantiated ``Context``, typically you will call :meth:`run()` to execute the + charm once, write any assertions you like on the output state returned by the call, write any + assertions you like on the ``Context`` attributes, then discard the ``Context``. + + Each ``Context`` instance is in principle designed to be single-use: + ``Context`` is not cleaned up automatically between charm runs. + + Any side effects generated by executing the charm, that are not rightful part of the + ``State``, are in fact stored in the ``Context``: + + - :attr:`juju_log` + - :attr:`app_status_history` + - :attr:`unit_status_history` + - :attr:`workload_version_history` + - :attr:`removed_secret_revisions` + - :attr:`requested_storages` + - :attr:`emitted_events` + - :attr:`action_logs` + - :attr:`action_results` + + This allows you to write assertions not only on the output state, but also, to some + extent, on the path the charm took to get there. + + A typical test will look like:: + + from charm import MyCharm, MyCustomEvent # noqa + + def test_foo(): + # Arrange: set the context up + ctx = Context(MyCharm) + # Act: prepare the state and emit an event + state_out = ctx.run(ctx.on.update_status(), State()) + # Assert: verify the output state is what you think it should be + assert state_out.unit_status == ActiveStatus('foobar') + # Assert: verify the Context contains what you think it should + assert len(c.emitted_events) == 4 + assert isinstance(c.emitted_events[3], MyCustomEvent) + + If you need access to the charm object that will handle the event, use the + class in a ``with`` statement, like:: + + def test_foo(): + ctx = Context(MyCharm) + with ctx(ctx.on.start(), State()) as manager: + manager.charm._some_private_setup() + manager.run() + """ + + juju_log: list[JujuLogLine] + """A record of what the charm has sent to juju-log""" + app_status_history: list[_EntityStatus] + """A record of the app statuses the charm has set""" + unit_status_history: list[_EntityStatus] + """A record of the unit statuses the charm has set""" + workload_version_history: list[str] + """A record of the workload versions the charm has set""" + removed_secret_revisions: list[int] + """A record of the secret revisions the charm has removed""" + emitted_events: list[ops.EventBase] + """A record of the events (including custom) that the charm has processed""" + requested_storages: dict[str, int] + """A record of the storages the charm has requested""" + action_logs: list[str] + """The logs associated with the action output, set by the charm with :meth:`ops.ActionEvent.log` + + This will be empty when handling a non-action event. + """ + action_results: dict[str, Any] | None + """A key-value mapping assigned by the charm as a result of the action. + + This will be ``None`` if the charm never calls :meth:`ops.ActionEvent.set_results` + """ + on: CharmEvents + """The events that this charm can respond to. + + Use this when calling :meth:`run` to specify the event to emit. + """ + + def __init__( + self, + charm_type: type[CharmType], + meta: dict[str, Any] | None = None, + *, + actions: dict[str, Any] | None = None, + config: dict[str, Any] | None = None, + charm_root: str | Path | None = None, + juju_version: str = _DEFAULT_JUJU_VERSION, + capture_deferred_events: bool = False, + capture_framework_events: bool = False, + app_name: str | None = None, + unit_id: int | None = 0, + app_trusted: bool = False, + ): + """Represents a simulated charm's execution context. + + If the charm, say, expects a ``./src/foo/bar.yaml`` file present relative to the + execution cwd, you need to use the ``charm_root`` argument. For example:: + + import tempfile + virtual_root = tempfile.TemporaryDirectory() + local_path = Path(local_path.name) + (local_path / 'foo').mkdir() + (local_path / 'foo' / 'bar.yaml').write_text('foo: bar') + Context(... charm_root=virtual_root).run(...) + + :arg charm_type: the :class:`ops.CharmBase` subclass to handle the event. + :arg meta: charm metadata to use. Needs to be a valid metadata.yaml format (as a dict). + If none is provided, we will search for a ``metadata.yaml`` file in the charm root. + :arg actions: charm actions to use. Needs to be a valid actions.yaml format (as a dict). + If none is provided, we will search for a ``actions.yaml`` file in the charm root. + :arg config: charm config to use. Needs to be a valid config.yaml format (as a dict). + If none is provided, we will search for a ``config.yaml`` file in the charm root. + :arg juju_version: Juju agent version to simulate. + :arg app_name: App name that this charm is deployed as. Defaults to the charm name as + defined in the metadata. + :arg unit_id: Unit ID that this charm is deployed as. + :arg app_trusted: whether the charm has Juju trust (deployed with ``--trust`` or added with + ``juju trust``). + :arg charm_root: virtual charm filesystem root the charm will be executed with. + """ + + if not any((meta, actions, config)): + logger.debug("Autoloading charmspec...") + try: + spec: _CharmSpec[CharmType] = _CharmSpec.autoload(charm_type) + except MetadataNotFoundError as e: + raise ContextSetupError( + f"Cannot setup scenario with `charm_type`={charm_type}. " + f"Did you forget to pass `meta` to this Context?", + ) from e + + else: + if not meta: + meta = {"name": str(charm_type.__name__)} + spec = _CharmSpec( + charm_type=charm_type, + meta=meta, + actions=actions, + config=config, + ) + + self.charm_spec = spec + self.charm_root = charm_root + self.juju_version = juju_version + if juju_version.split(".")[0] == "2": + logger.warning( + "Juju 2.x is closed and unsupported. You may encounter inconsistencies.", + ) + + self._app_name = app_name + self._unit_id = unit_id + self.app_trusted = app_trusted + self._tmp = tempfile.TemporaryDirectory() + + # config for what events to be captured in emitted_events. + self.capture_deferred_events = capture_deferred_events + self.capture_framework_events = capture_framework_events + + # streaming side effects from running an event + self.juju_log: list[JujuLogLine] = [] + self.app_status_history: list[_EntityStatus] = [] + self.unit_status_history: list[_EntityStatus] = [] + self.exec_history: dict[str, list[ExecArgs]] = {} + self.workload_version_history: list[str] = [] + self.removed_secret_revisions: list[int] = [] + self.emitted_events: list[ops.EventBase] = [] + self.requested_storages: dict[str, int] = {} + + # set by Runtime.exec() in self._run() + self._output_state: State | None = None + + # operations (and embedded tasks) from running actions + self.action_logs: list[str] = [] + self.action_results: dict[str, Any] | None = None + self._action_failure_message: str | None = None + + self.on = CharmEvents() + + def _set_output_state(self, output_state: State): + """Hook for Runtime to set the output state.""" + self._output_state = output_state + + def _get_container_root(self, container_name: str): + """Get the path to a tempdir where this container's simulated root will live.""" + return Path(self._tmp.name) / "containers" / container_name + + def _get_storage_root(self, name: str, index: int) -> Path: + """Get the path to a tempdir where this storage's simulated root will live.""" + storage_root = Path(self._tmp.name) / "storages" / f"{name}-{index}" + # in the case of _get_container_root, _MockPebbleClient will ensure the dir exists. + storage_root.mkdir(parents=True, exist_ok=True) + return storage_root + + def _record_status(self, state: State, is_app: bool): + """Record the previous status before a status change.""" + if is_app: + self.app_status_history.append(state.app_status) + else: + self.unit_status_history.append(state.unit_status) + + def __call__(self, event: _Event, state: State) -> Manager[CharmType]: + """Context manager to introspect live charm object before and after the event is emitted. + + Usage:: + + ctx = Context(MyCharm) + with ctx(ctx.on.start(), State()) as manager: + manager.charm._some_private_setup() + manager.run() # this will fire the event + assert manager.charm._some_private_attribute == "bar" # noqa + + Args: + event: the event that the charm will respond to. + state: the :class:`State` instance to use when handling the event. + """ + return Manager(self, event, state) + + def run_action(self, action: str, state: State): + """Use `run()` instead. + + :private: + """ + raise AttributeError( + f"call with `ctx.run`, like `ctx.run(ctx.on.action({action!r})` " + "and find the results in `ctx.action_results`", + ) + + def run(self, event: _Event, state: State) -> State: + """Trigger a charm execution with an event and a State. + + Calling this function will call ``ops.main`` and set up the context according to the + specified :class:`State`, then emit the event on the charm. + + :arg event: the event that the charm will respond to. Use the :attr:`on` attribute to + specify the event; for example: ``ctx.on.start()``. + :arg state: the :class:`State` instance to use as data source for the hook tool calls that + the charm will invoke when handling the event. + """ + # Help people transition from Scenario 6: + if isinstance(event, str): + event = event.replace("-", "_") # type: ignore + if event in ( + "install", + "start", + "stop", + "remove", + "update_status", + "config_changed", + "upgrade_charm", + "pre_series_upgrade", + "post_series_upgrade", + "leader_elected", + "collect_app_status", + "collect_unit_status", + ): # type: ignore + suggested = f"{event}()" + elif event in ("secret_changed", "secret_rotate"): # type: ignore + suggested = f"{event}(my_secret)" + elif event in ("secret_expired", "secret_remove"): # type: ignore + suggested = f"{event}(my_secret, revision=1)" + elif event in ( + "relation_created", + "relation_joined", + "relation_changed", + "relation_departed", + "relation_broken", + ): # type: ignore + suggested = f"{event}(my_relation)" + elif event in ("storage_attached", "storage_detaching"): # type: ignore + suggested = f"{event}(my_storage)" + elif event == "pebble_ready": + suggested = f"{event}(my_container)" + elif event == "pebble_custom_notice": + suggested = f"{event}(my_container, my_notice)" + else: + suggested = "event()" + raise TypeError( + f"call with an event from `ctx.on`, like `ctx.on.{suggested}`", + ) + if callable(event): + raise TypeError( + "You should call the event method. Did you forget to add parentheses?", + ) + + if event.action: + # Reset the logs, failure status, and results, in case the context + # is reused. + self.action_logs.clear() + if self.action_results is not None: + self.action_results.clear() + self._action_failure_message = None + with self._run(event=event, state=state) as manager: + manager.emit() + # We know that the output state will have been set by this point, + # so let the type checkers know that too. + assert self._output_state is not None + if event.action: + if self._action_failure_message is not None: + raise ActionFailed( + self._action_failure_message, + state=self._output_state, # type: ignore + ) + return self._output_state + + @contextmanager + def _run(self, event: _Event, state: State): + runtime = Runtime( + charm_spec=self.charm_spec, + juju_version=self.juju_version, + charm_root=self.charm_root, + app_name=self._app_name, + unit_id=self._unit_id, + ) + with runtime.exec( + state=state, + event=event, + context=self, # type: ignore + ) as ops: + yield ops diff --git a/testing/build/lib/scenario/errors.py b/testing/build/lib/scenario/errors.py new file mode 100644 index 000000000..81049eaa6 --- /dev/null +++ b/testing/build/lib/scenario/errors.py @@ -0,0 +1,56 @@ +#!/usr/bin/env python3 +# Copyright 2024 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Exceptions raised by the framework. + +Note that these exceptions are not meant to be caught by charm authors. They are +used by the framework to signal errors or inconsistencies in the charm tests +themselves. +""" + + +class ContextSetupError(RuntimeError): + """Raised by Context when setup fails.""" + + +class AlreadyEmittedError(RuntimeError): + """Raised when ``run()`` is called more than once.""" + + +class ScenarioRuntimeError(RuntimeError): + """Base class for exceptions raised by the runtime module.""" + + +class UncaughtCharmError(ScenarioRuntimeError): + """Error raised if the charm raises while handling the event being dispatched.""" + + +class InconsistentScenarioError(ScenarioRuntimeError): + """Error raised when the combination of state and event is inconsistent.""" + + +class StateValidationError(RuntimeError): + """Raised when individual parts of the State are inconsistent.""" + + # as opposed to InconsistentScenario error where the **combination** of + # several parts of the State are. + + +class MetadataNotFoundError(RuntimeError): + """Raised when a metadata file can't be found in the provided charm root.""" + + +class ActionMissingFromContextError(Exception): + """Raised when the user attempts to invoke action hook tools outside an action context.""" + + # This is not an ops error: in ops, you'd have to go exceptionally out of + # your way to trigger this flow. + + +class NoObserverError(RuntimeError): + """Error raised when the event being dispatched has no registered observers.""" + + +class BadOwnerPath(RuntimeError): + """Error raised when the owner path does not lead to a valid ObjectEvents instance.""" diff --git a/testing/build/lib/scenario/logger.py b/testing/build/lib/scenario/logger.py new file mode 100644 index 000000000..a7909f397 --- /dev/null +++ b/testing/build/lib/scenario/logger.py @@ -0,0 +1,11 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Test framework logger""" + +import logging +import os + +logger = logging.getLogger("ops-scenario") +logger.setLevel(os.getenv("OPS_SCENARIO_LOGGING", "WARNING")) diff --git a/testing/build/lib/scenario/mocking.py b/testing/build/lib/scenario/mocking.py new file mode 100644 index 000000000..b95b553be --- /dev/null +++ b/testing/build/lib/scenario/mocking.py @@ -0,0 +1,911 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Juju and Pebble mocking + +This module contains mocks for the Juju and Pebble APIs that are used by ops +to interact with the Juju controller and the Pebble service manager. +""" + +import datetime +import io +import shutil +from pathlib import Path +from typing import ( + TYPE_CHECKING, + Any, + Dict, + List, + Literal, + Mapping, + NoReturn, + Optional, + Set, + TextIO, + Tuple, + Union, + cast, + get_args, +) + +from ops import ( + JujuVersion, + pebble, + SecretInfo, + SecretNotFoundError, + RelationNotFoundError, + SecretRotate, + ModelError, +) +from ops._private.harness import ExecArgs, _TestingPebbleClient +from ops.jujucontext import _JujuContext +from ops.model import CloudSpec as CloudSpec_Ops +from ops.model import Port as Port_Ops +from ops.model import Secret as Secret_Ops # lol +from ops.model import ( + _format_action_result_dict, + _ModelBackend, + _SettableStatusName, +) +from ops.pebble import Client, ExecError + +from .errors import ActionMissingFromContextError +from .logger import logger as scenario_logger +from .state import ( + CharmType, + JujuLogLine, + Mount, + Network, + PeerRelation, + Relation, + RelationBase, + Storage, + SubordinateRelation, + _EntityStatus, + _port_cls_by_protocol, + _RawPortProtocolLiteral, +) + +if TYPE_CHECKING: # pragma: no cover + from .context import Context + from .state import Container as ContainerSpec + from .state import Exec, Secret, State, _CharmSpec, _Event + +logger = scenario_logger.getChild("mocking") + + +class _MockExecProcess: + def __init__( + self, + change_id: int, + args: ExecArgs, + return_code: int, + stdin: Optional[Union[TextIO, io.BytesIO]], + stdout: Optional[Union[TextIO, io.BytesIO]], + stderr: Optional[Union[TextIO, io.BytesIO]], + ): + self._change_id = change_id + self._args = args + self._return_code = return_code + self._waited = False + self.stdin = stdin + self.stdout = stdout + self.stderr = stderr + + def __del__(self): + if not self._waited: + self._close_stdin() + + def _close_stdin(self): + if self._args.stdin is None and self.stdin is not None: + self.stdin.seek(0) + self._args.stdin = self.stdin.read() + + def wait(self): + """Wait for the (mock) process to finish.""" + self._close_stdin() + self._waited = True + if self._return_code != 0: + raise ExecError(list(self._args.command), self._return_code, None, None) + + def wait_output(self): + """Wait for the (mock) process to finish and return tuple of (stdout, stderr).""" + self._close_stdin() + self._waited = True + stdout = self.stdout.read() if self.stdout is not None else None + stderr = self.stderr.read() if self.stderr is not None else None + if self._return_code != 0: + raise ExecError( + list(self._args.command), + self._return_code, + stdout, # type: ignore + stderr, # type: ignore + ) + return stdout, stderr + + def send_signal(self, sig: Union[int, str]) -> NoReturn: # noqa: U100 + """Send the given signal to the (mock) process.""" + raise NotImplementedError() + + +_NOT_GIVEN = object() # non-None default value sentinel + + +# pyright: reportIncompatibleMethodOverride=false +class _MockModelBackend(_ModelBackend): # type: ignore + def __init__( + self, + state: "State", + event: "_Event", + charm_spec: "_CharmSpec[CharmType]", + context: "Context", + juju_context: "_JujuContext", + ): + super().__init__(juju_context=juju_context) + self._state = state + self._event = event + self._context = context + self._charm_spec = charm_spec + + def opened_ports(self) -> Set[Port_Ops]: + return { + Port_Ops(protocol=port.protocol, port=port.port) + for port in self._state.opened_ports + } + + def open_port( + self, + protocol: "_RawPortProtocolLiteral", + port: Optional[int] = None, + ): + port_ = _port_cls_by_protocol[protocol](port=port) # type: ignore + ports = set(self._state.opened_ports) + if port_ not in ports: + ports.add(port_) + if ports != self._state.opened_ports: + self._state._update_opened_ports(frozenset(ports)) + + def close_port( + self, + protocol: "_RawPortProtocolLiteral", + port: Optional[int] = None, + ): + _port = _port_cls_by_protocol[protocol](port=port) # type: ignore + ports = set(self._state.opened_ports) + if _port in ports: + ports.remove(_port) + if ports != self._state.opened_ports: + self._state._update_opened_ports(frozenset(ports)) + + def get_pebble(self, socket_path: str) -> "Client": + container_name = socket_path.split("/")[ + 3 + ] # /charm/containers//pebble.socket + container_root = self._context._get_container_root(container_name) + try: + mounts = self._state.get_container(container_name).mounts + except KeyError: + # container not defined in state. + mounts = {} + + return cast( + Client, + _MockPebbleClient( + socket_path=socket_path, + container_root=container_root, + mounts=mounts, + state=self._state, + event=self._event, + charm_spec=self._charm_spec, + context=self._context, + container_name=container_name, + ), + ) + + def _get_relation_by_id(self, rel_id: int) -> "RelationBase": + try: + return self._state.get_relation(rel_id) + except KeyError: + raise RelationNotFoundError() from None + + def _get_secret(self, id: Optional[str] = None, label: Optional[str] = None): + if self._context.juju_version < "3.0.2": + raise ModelError( + "secrets are only available in juju >= 3.0.2." + "Set ``Context.juju_version`` to 3.0.2+ to use them.", + ) + + canonicalize_id = Secret_Ops._canonicalize_id + + if id: + # in scenario, you can create Secret(id="foo"), + # but ops.Secret will prepend a "secret:" prefix to that ID. + # we allow getting secret by either version. + secrets = [ + s + for s in self._state.secrets + if canonicalize_id(s.id, model_uuid=self._state.model.uuid) + == canonicalize_id(id, model_uuid=self._state.model.uuid) + ] + if not secrets: + raise SecretNotFoundError(id) + return secrets[0] + + if label: + try: + return self._state.get_secret(label=label) + except KeyError: + raise SecretNotFoundError(label) from None + + # if all goes well, this should never be reached. ops.model.Secret will check upon + # instantiation that either an id or a label are set, and raise a TypeError if not. + raise RuntimeError("need id or label.") + + def _check_app_data_access(self, is_app: bool): + if not isinstance(is_app, bool): + raise TypeError("is_app parameter to relation_get must be a boolean") + + if not is_app: + return + + version = JujuVersion(self._context.juju_version) + if not version.has_app_data(): + raise RuntimeError( + f"setting application data is not supported on Juju version {version}", + ) + + def relation_get(self, relation_id: int, member_name: str, is_app: bool): + self._check_app_data_access(is_app) + relation = self._get_relation_by_id(relation_id) + if is_app and member_name == self.app_name: + return relation.local_app_data + if is_app: + if isinstance(relation, PeerRelation): + return relation.local_app_data + if isinstance(relation, (Relation, SubordinateRelation)): + return relation.remote_app_data + raise TypeError("relation_get: unknown relation type") + if member_name == self.unit_name: + return relation.local_unit_data + + unit_id = int(member_name.split("/")[-1]) + return relation._get_databag_for_remote(unit_id) # noqa + + def is_leader(self): + return self._state.leader + + def status_get(self, *, is_app: bool = False): + status = self._state.app_status if is_app else self._state.unit_status + return {"status": status.name, "message": status.message} + + def relation_ids(self, relation_name: str): + return [ + rel.id for rel in self._state.relations if rel.endpoint == relation_name + ] + + def relation_list(self, relation_id: int) -> Tuple[str, ...]: + relation = self._get_relation_by_id(relation_id) + + if isinstance(relation, PeerRelation): + return tuple( + f"{self.app_name}/{unit_id}" for unit_id in relation.peers_data + ) + remote_name = self.relation_remote_app_name(relation_id) + return tuple( + f"{remote_name}/{unit_id}" for unit_id in relation._remote_unit_ids + ) + + def config_get(self): + state_config = self._state.config.copy() # dedup or we'll mutate the state! + + # add defaults + charm_config = self._charm_spec.config + if not charm_config: + return state_config + + for key, value in charm_config["options"].items(): + # if it has a default, and it's not overwritten from State, use it: + if key not in state_config: + default_value = value.get("default", _NOT_GIVEN) + if default_value is not _NOT_GIVEN: # accept False as default value + state_config[key] = default_value + + return state_config # full config + + def network_get(self, binding_name: str, relation_id: Optional[int] = None): + # validation: + extra_bindings = self._charm_spec.meta.get("extra-bindings", ()) + all_endpoints = self._charm_spec.get_all_relations() + non_sub_relations = { + name for name, meta in all_endpoints if meta.get("scope") != "container" + } + + # - is binding_name a valid binding name? + if binding_name in extra_bindings: + logger.warning("extra-bindings is a deprecated feature") # fyi + + # - verify that if the binding is an extra binding, we're not ignoring a relation_id + if relation_id is not None: + # this should not happen + logger.error( + "cannot pass relation_id to network_get if the binding name is " + "that of an extra-binding. Extra-bindings are not mapped to relation IDs.", + ) + elif binding_name == "juju-info": + # implicit relation that always exists + pass + # - verify that the binding is a relation endpoint name, but not a subordinate one + elif binding_name not in non_sub_relations: + logger.error( + f"cannot get network binding for {binding_name}: is not a valid relation " + f"endpoint name nor an extra-binding.", + ) + raise RelationNotFoundError() + + # We look in State.networks for an override. If not given, we return a default network. + try: + network = self._state.get_network(binding_name) + except KeyError: + network = Network("default") # The name is not used in the output. + return network._hook_tool_output_fmt() + + # setter methods: these can mutate the state. + def application_version_set(self, version: str): + if workload_version := self._state.workload_version: + # do not record if empty = unset + self._context.workload_version_history.append(workload_version) + + self._state._update_workload_version(version) + + def status_set( + self, + status: _SettableStatusName, + message: str = "", + *, + is_app: bool = False, + ): + valid_names = get_args(_SettableStatusName) + if status not in valid_names: + raise ModelError( + f'ERROR invalid status "{status}", expected one of [{", ".join(valid_names)}]', + ) + self._context._record_status(self._state, is_app) + status_obj = _EntityStatus.from_status_name(status, message) + self._state._update_status(status_obj, is_app) + + def juju_log(self, level: str, message: str): + self._context.juju_log.append(JujuLogLine(level, message)) + + def relation_set(self, relation_id: int, key: str, value: str, is_app: bool): + self._check_app_data_access(is_app) + relation = self._get_relation_by_id(relation_id) + if is_app: + if not self._state.leader: + # will in practice not be reached because RelationData will check leadership + # and raise RelationDataAccessError upstream on this path + raise RuntimeError("needs leadership to set app data") + tgt = relation.local_app_data + else: + tgt = relation.local_unit_data + tgt[key] = value + + def secret_add( + self, + content: Dict[str, str], + *, + label: Optional[str] = None, + description: Optional[str] = None, + expire: Optional[datetime.datetime] = None, + rotate: Optional[SecretRotate] = None, + owner: Optional[Literal["unit", "app"]] = None, + ) -> str: + from .state import Secret + + secret = Secret( + content, + label=label, + description=description, + expire=expire, + rotate=rotate, + owner=owner, + ) + secrets = set(self._state.secrets) + secrets.add(secret) + self._state._update_secrets(frozenset(secrets)) + return secret.id + + def _check_can_manage_secret( + self, + secret: "Secret", + ): + if secret.owner is None: + raise SecretNotFoundError( + "this secret is not owned by this unit/app or granted to it. " + "Did you forget passing it to State.secrets?", + ) + if secret.owner == "app" and not self.is_leader(): + understandable_error = SecretNotFoundError( + f"App-owned secret {secret.id!r} can only be " + f"managed by the leader.", + ) + # charm-facing side: respect ops error + raise ModelError("ERROR permission denied") from understandable_error + + def secret_get( + self, + *, + id: Optional[str] = None, + label: Optional[str] = None, + refresh: bool = False, + peek: bool = False, + ) -> Dict[str, str]: + secret = self._get_secret(id, label) + # If both the id and label are provided, then update the label. + if id is not None and label is not None: + secret._set_label(label) + juju_version = self._context.juju_version + if not (juju_version == "3.1.7" or juju_version >= "3.3.1"): + # In this medieval Juju chapter, + # secret owners always used to track the latest revision. + # ref: https://bugs.launchpad.net/juju/+bug/2037120 + if secret.owner is not None: + refresh = True + + if peek or refresh: + if refresh: + secret._track_latest_revision() + assert secret.latest_content is not None + return secret.latest_content + + return secret.tracked_content + + def secret_info_get( + self, + *, + id: Optional[str] = None, + label: Optional[str] = None, + ) -> SecretInfo: + secret = self._get_secret(id, label) + # If both the id and label are provided, then update the label. + if id is not None and label is not None: + secret._set_label(label) + + # only "manage"=write access level can read secret info + self._check_can_manage_secret(secret) + + return SecretInfo( + id=secret.id, + label=secret.label, + revision=secret._latest_revision, + expires=secret.expire, + rotation=secret.rotate, + rotates=None, # not implemented yet. + ) + + def secret_set( + self, + id: str, + *, + content: Optional[Dict[str, str]] = None, + label: Optional[str] = None, + description: Optional[str] = None, + expire: Optional[datetime.datetime] = None, + rotate: Optional[SecretRotate] = None, + ): + secret = self._get_secret(id, label) + self._check_can_manage_secret(secret) + + if content == secret.latest_content: + # In Juju 3.6 and higher, this is a no-op, but it's good to warn + # charmers if they are doing this, because it's not generally good + # practice. + # https://bugs.launchpad.net/juju/+bug/2069238 + logger.warning( + f"secret {id} contents set to the existing value: new revision not needed", + ) + + secret._update_metadata( + content=content, + label=label, + description=description, + expire=expire, + rotate=rotate, + ) + + def secret_grant(self, id: str, relation_id: int, *, unit: Optional[str] = None): + secret = self._get_secret(id) + self._check_can_manage_secret(secret) + + grantee = unit or self.relation_remote_app_name( + relation_id, + _raise_on_error=True, + ) + + if not secret.remote_grants.get(relation_id): + secret.remote_grants[relation_id] = set() + + secret.remote_grants[relation_id].add(cast(str, grantee)) + + def secret_revoke(self, id: str, relation_id: int, *, unit: Optional[str] = None): + secret = self._get_secret(id) + self._check_can_manage_secret(secret) + + grantee = unit or self.relation_remote_app_name( + relation_id, + _raise_on_error=True, + ) + secret.remote_grants[relation_id].remove(cast(str, grantee)) + if not secret.remote_grants[relation_id]: + del secret.remote_grants[relation_id] + + def secret_remove(self, id: str, *, revision: Optional[int] = None): + secret = self._get_secret(id) + self._check_can_manage_secret(secret) + + # Removing all revisions means that the secret is removed, so is no + # longer in the state. + if revision is None: + secrets = set(self._state.secrets) + secrets.remove(secret) + self._state._update_secrets(frozenset(secrets)) + return + + # Juju does not prevent removing the tracked or latest revision, but it + # is always a mistake to do this. Rather than having the state model a + # secret where the tracked/latest revision cannot be retrieved but the + # secret still exists, we raise instead, so that charms know that there + # is a problem with their code. + if revision in (secret._tracked_revision, secret._latest_revision): + raise ValueError( + "Charms should not remove the latest revision of a secret. " + "Add a new revision with `set_content()` instead, and the previous " + "revision will be cleaned up by the secret owner when no longer in use.", + ) + + # For all other revisions, the content is not visible to the charm + # (this is as designed: the secret is being removed, so it should no + # longer be in use). That means that the state does not need to be + # modified - however, unit tests should be able to verify that the remove call was + # executed, so we track that in a history list in the context. + self._context.removed_secret_revisions.append(revision) + + def relation_remote_app_name( + self, + relation_id: int, + _raise_on_error: bool = False, + ) -> Optional[str]: + # ops catches RelationNotFoundErrors and returns None: + try: + relation = self._get_relation_by_id(relation_id) + except RelationNotFoundError: + if _raise_on_error: + raise + return None + + if isinstance(relation, PeerRelation): + return self.app_name + if isinstance(relation, (Relation, SubordinateRelation)): + return relation.remote_app_name + raise TypeError("relation_remote_app_name: unknown relation type") + + def action_set(self, results: Dict[str, Any]): + if not self._event.action: + raise ActionMissingFromContextError( + "not in the context of an action event: cannot action-set", + ) + # let ops validate the results dict + _format_action_result_dict(results) + # but then we will store it in its unformatted, + # original form for testing ease + if self._context.action_results: + self._context.action_results.update(results) + else: + self._context.action_results = results + + def action_fail(self, message: str = ""): + if not self._event.action: + raise ActionMissingFromContextError( + "not in the context of an action event: cannot action-fail", + ) + self._context._action_failure_message = message + + def action_log(self, message: str): + if not self._event.action: + raise ActionMissingFromContextError( + "not in the context of an action event: cannot action-log", + ) + self._context.action_logs.append(message) + + def action_get(self): + action = self._event.action + if not action: + raise ActionMissingFromContextError( + "not in the context of an action event: cannot action-get", + ) + return action.params + + def storage_add(self, name: str, count: int = 1): + if not isinstance(count, int) or isinstance(count, bool): + raise TypeError( + f"storage count must be integer, got: {count} ({type(count)})", + ) + + if "/" in name: + # this error is raised by Harness but not by ops at runtime + raise ModelError('storage name cannot contain "/"') + + self._context.requested_storages[name] = count + + def storage_list(self, name: str) -> List[int]: + return [ + storage.index for storage in self._state.storages if storage.name == name + ] + + def _storage_event_details(self) -> Tuple[int, str]: + storage = self._event.storage + if not storage: + # only occurs if this method is called when outside the scope of a storage event + raise RuntimeError('unable to find storage key in ""') + fs_path = storage.get_filesystem(self._context) + return storage.index, str(fs_path) + + def storage_get(self, storage_name_id: str, attribute: str) -> str: + if not len(attribute) > 0: # assume it's an empty string. + raise RuntimeError( + 'calling storage_get with `attribute=""` will return a dict ' + "and not a string. This usage is not supported.", + ) + + if attribute != "location": + # this should not happen: in ops it's hardcoded to be "location" + raise NotImplementedError( + f"storage-get not implemented for attribute={attribute}", + ) + + name, index = storage_name_id.split("/") + index = int(index) + storages: List[Storage] = [ + s for s in self._state.storages if s.name == name and s.index == index + ] + + # should not really happen: sanity checks. In practice, ops will guard against these paths. + if not storages: + raise RuntimeError(f"Storage with name={name} and index={index} not found.") + if len(storages) > 1: + raise RuntimeError( + f"Multiple Storage instances with name={name} and index={index} found. " + f"Inconsistent state.", + ) + + storage = storages[0] + fs_path = storage.get_filesystem(self._context) + return str(fs_path) + + def planned_units(self) -> int: + return self._state.planned_units + + # legacy ops API that we don't intend to mock: + def pod_spec_set( + self, + spec: Mapping[str, Any], # noqa: U100 + k8s_resources: Optional[Mapping[str, Any]] = None, # noqa: U100 + ) -> NoReturn: + raise NotImplementedError( + "pod-spec-set is not implemented in Scenario (and probably never will be: " + "it's deprecated API)", + ) + + def add_metrics( + self, + metrics: Mapping[str, Union[int, float]], # noqa: U100 + labels: Optional[Mapping[str, str]] = None, # noqa: U100 + ) -> NoReturn: + raise NotImplementedError( + "add-metrics is not implemented in Scenario (and probably never will be: " + "it's deprecated API)", + ) + + def resource_get(self, resource_name: str) -> str: + # We assume that there are few enough resources that a linear search + # will perform well enough. + for resource in self._state.resources: + if resource.name == resource_name: + return str(resource.path) + # ops will not let us get there if the resource name is unknown from metadata. + # but if the user forgot to add it in State, then we remind you of that. + raise RuntimeError( + f"Inconsistent state: " + f"resource {resource_name} not found in State. please pass it.", + ) + + def credential_get(self) -> CloudSpec_Ops: + if not self._context.app_trusted: + raise ModelError( + "ERROR charm is not trusted, initialise Context with `app_trusted=True`", + ) + if not self._state.model.cloud_spec: + raise ModelError( + "ERROR cloud spec is empty, initialise it with " + "`State(model=Model(..., cloud_spec=ops.CloudSpec(...)))`", + ) + return self._state.model.cloud_spec._to_ops() + + +class _MockPebbleClient(_TestingPebbleClient): + def __init__( + self, + socket_path: str, + container_root: Path, + mounts: Dict[str, Mount], + *, + state: "State", + event: "_Event", + charm_spec: "_CharmSpec[CharmType]", + context: "Context", + container_name: str, + ): + self._state = state + self.socket_path = socket_path + self._event = event + self._charm_spec = charm_spec + self._context = context + self._container_name = container_name + + # wipe just in case + if container_root.exists(): + # Path.rmdir will fail if root is nonempty + shutil.rmtree(container_root) + + # initialize simulated filesystem + container_root.mkdir(parents=True) + for _, mount in mounts.items(): + path = Path(mount.location).parts + mounting_dir = container_root.joinpath(*path[1:]) + mounting_dir.parent.mkdir(parents=True, exist_ok=True) + mounting_dir.symlink_to(mount.source) + + self._root = container_root + + self._notices: Dict[Tuple[str, str], pebble.Notice] = {} + self._last_notice_id = 0 + self._changes: Dict[str, pebble.Change] = {} + + # load any existing notices and check information from the state + self._notices: Dict[Tuple[str, str], pebble.Notice] = {} + self._check_infos: Dict[str, pebble.CheckInfo] = {} + for container in state.containers: + for notice in container.notices: + if hasattr(notice.type, "value"): + notice_type = cast(pebble.NoticeType, notice.type).value + else: + notice_type = str(notice.type) + self._notices[notice_type, notice.key] = notice._to_ops() + for check in container.check_infos: + self._check_infos[check.name] = check._to_ops() + + def get_plan(self) -> pebble.Plan: + return self._container.plan + + @property + def _container(self) -> "ContainerSpec": + container_name = self.socket_path.split("/")[-2] + try: + return next( + filter(lambda x: x.name == container_name, self._state.containers), + ) + except StopIteration: + raise RuntimeError( + f"container with name={container_name!r} not found. " + f"Did you forget a Container, or is the socket path " + f"{self.socket_path!r} wrong?", + ) + + @property + def _layers(self) -> Dict[str, pebble.Layer]: + return self._container.layers + + @property + def _service_status(self) -> Dict[str, pebble.ServiceStatus]: + return self._container.service_statuses + + # Based on a method of the same name from Harness. + def _find_exec_handler(self, command: List[str]) -> Optional["Exec"]: + handlers = {exec.command_prefix: exec for exec in self._container.execs} + # Start with the full command and, each loop iteration, drop the last + # element, until it matches one of the command prefixes in the execs. + # This includes matching against the empty list, which will match any + # command, if there is not a more specific match. + for prefix_len in reversed(range(len(command) + 1)): + command_prefix = tuple(command[:prefix_len]) + if command_prefix in handlers: + return handlers[command_prefix] + # None of the command prefixes in the execs matched the command, no + # matter how much of it was used, so we have failed to find a handler. + return None + + def exec( + self, + command: List[str], + *, + environment: Optional[Dict[str, str]] = None, + working_dir: Optional[str] = None, + timeout: Optional[float] = None, + user_id: Optional[int] = None, + user: Optional[str] = None, + group_id: Optional[int] = None, + group: Optional[str] = None, + stdin: Optional[Union[str, bytes, TextIO]] = None, + stdout: Optional[TextIO] = None, + stderr: Optional[TextIO] = None, + encoding: Optional[str] = "utf-8", + combine_stderr: bool = False, + **kwargs: Any, + ): + handler = self._find_exec_handler(command) + if not handler: + raise ExecError( + command, + 127, + "", + f"mock for cmd {command} not found. Please patch out whatever " + f"leads to the call, or pass to the Container {self._container.name} " + f"a scenario.Exec mock for the command your charm is attempting " + f"to run, such as " + f"'Container(..., execs={{scenario.Exec({list(command)}, ...)}})'", + ) + + if stdin is None: + proc_stdin = self._transform_exec_handler_output("", encoding) + else: + proc_stdin = None + stdin = stdin.read() if hasattr(stdin, "read") else stdin # type: ignore + if stdout is None: + proc_stdout = self._transform_exec_handler_output(handler.stdout, encoding) + else: + proc_stdout = None + stdout.write(handler.stdout) + if stderr is None: + proc_stderr = self._transform_exec_handler_output(handler.stderr, encoding) + else: + proc_stderr = None + stderr.write(handler.stderr) + + args = ExecArgs( + command=command, + environment=environment or {}, + working_dir=working_dir, + timeout=timeout, + user_id=user_id, + user=user, + group_id=group_id, + group=group, + stdin=stdin, # type:ignore # If None, will be replaced by proc_stdin.read() later. + encoding=encoding, + combine_stderr=combine_stderr, + ) + try: + self._context.exec_history[self._container_name].append(args) + except KeyError: + self._context.exec_history[self._container_name] = [args] + + change_id = handler._run() + return cast( + pebble.ExecProcess[Any], + _MockExecProcess( + change_id=change_id, + args=args, + return_code=handler.return_code, + stdin=proc_stdin, + stdout=proc_stdout, + stderr=proc_stderr, + ), + ) + + def _check_connection(self): + if not self._container.can_connect: + msg = ( + f"Cannot connect to Pebble; did you forget to set " + f"can_connect=True for container {self._container.name}?" + ) + raise pebble.ConnectionError(msg) diff --git a/testing/build/lib/scenario/ops_main_mock.py b/testing/build/lib/scenario/ops_main_mock.py new file mode 100644 index 000000000..5e4846eba --- /dev/null +++ b/testing/build/lib/scenario/ops_main_mock.py @@ -0,0 +1,270 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +import inspect +import os +import pathlib +import sys +from typing import TYPE_CHECKING, Any, Generic, Optional, Sequence, Type, cast + +import ops.charm +import ops.framework +import ops.jujucontext +import ops.model +import ops.storage +from ops import CharmBase + +# use logger from ops._main so that juju_log will be triggered +from ops._main import CHARM_STATE_FILE, _Dispatcher, _get_event_args +from ops._main import logger as ops_logger +from ops.charm import CharmMeta +from ops.log import setup_root_logging + +from .errors import BadOwnerPath, NoObserverError +from .state import CharmType + +if TYPE_CHECKING: # pragma: no cover + from .context import Context + from .state import State, _CharmSpec, _Event + +# pyright: reportPrivateUsage=false + + +def _get_owner(root: Any, path: Sequence[str]) -> ops.ObjectEvents: + """Walk path on root to an ObjectEvents instance.""" + obj = root + for step in path: + try: + obj = getattr(obj, step) + except AttributeError: + raise BadOwnerPath( + f"event_owner_path {path!r} invalid: {step!r} leads to nowhere.", + ) + if not isinstance(obj, ops.ObjectEvents): + raise BadOwnerPath( + f"event_owner_path {path!r} invalid: does not lead to " + f"an ObjectEvents instance.", + ) + return obj + + +def _emit_charm_event( + charm: "CharmBase", + event_name: str, + juju_context: ops.jujucontext._JujuContext, + event: Optional["_Event"] = None, +): + """Emits a charm event based on a Juju event name. + + Args: + charm: A charm instance to emit an event from. + event_name: A Juju event name to emit on a charm. + event: Event to emit. + juju_context: Juju context to use for the event. + """ + owner = _get_owner(charm, event.owner_path) if event else charm.on + + try: + event_to_emit = getattr(owner, event_name) + except AttributeError: + ops_logger.debug("Event %s not defined for %s.", event_name, charm) + raise NoObserverError( + f"Cannot fire {event_name!r} on {owner}: " + f"invalid event (not on charm.on).", + ) + + args, kwargs = _get_event_args(charm, event_to_emit, juju_context) + ops_logger.debug("Emitting Juju event %s.", event_name) + event_to_emit.emit(*args, **kwargs) + + +def setup_framework( + charm_dir: pathlib.Path, + state: "State", + event: "_Event", + context: "Context[CharmType]", + charm_spec: "_CharmSpec[CharmType]", + juju_context: Optional[ops.jujucontext._JujuContext] = None, +): + from .mocking import _MockModelBackend + + if juju_context is None: + juju_context = ops.jujucontext._JujuContext.from_dict(os.environ) + model_backend = _MockModelBackend( + state=state, + event=event, + context=context, + charm_spec=charm_spec, + juju_context=juju_context, + ) + setup_root_logging(model_backend, debug=juju_context.debug) + # ops sets sys.excepthook to go to Juju's debug-log, but that's not useful + # in a testing context, so reset it. + sys.excepthook = sys.__excepthook__ + ops_logger.debug( + "Operator Framework %s up and running.", + ops.__version__, + ) + + metadata = (charm_dir / "metadata.yaml").read_text() + actions_meta = charm_dir / "actions.yaml" + if actions_meta.exists(): + actions_metadata = actions_meta.read_text() + else: + actions_metadata = None + + meta = CharmMeta.from_yaml(metadata, actions_metadata) + + # ops >= 2.10 + if inspect.signature(ops.model.Model).parameters.get("broken_relation_id"): + # If we are in a RelationBroken event, we want to know which relation is + # broken within the model, not only in the event's `.relation` attribute. + broken_relation_id = ( + event.relation.id # type: ignore + if event.name.endswith("_relation_broken") + else None + ) + + model = ops.model.Model( + meta, + model_backend, + broken_relation_id=broken_relation_id, + ) + else: + ops_logger.warning( + "It looks like this charm is using an older `ops` version. " + "You may experience weirdness. Please update ops.", + ) + model = ops.model.Model(meta, model_backend) + + charm_state_path = charm_dir / CHARM_STATE_FILE + + # TODO: add use_juju_for_storage support + store = ops.storage.SQLiteStorage(charm_state_path) + framework = ops.Framework(store, charm_dir, meta, model) + framework.set_breakpointhook() + return framework + + +def setup_charm( + charm_class: Type[ops.CharmBase], framework: ops.Framework, dispatcher: _Dispatcher +): + sig = inspect.signature(charm_class) + sig.bind(framework) # signature check + + charm = charm_class(framework) + dispatcher.ensure_event_links(charm) + return charm + + +def setup( + state: "State", + event: "_Event", + context: "Context[CharmType]", + charm_spec: "_CharmSpec[CharmType]", + juju_context: Optional[ops.jujucontext._JujuContext] = None, +): + """Setup dispatcher, framework and charm objects.""" + charm_class = charm_spec.charm_type + if juju_context is None: + juju_context = ops.jujucontext._JujuContext.from_dict(os.environ) + charm_dir = juju_context.charm_dir + + dispatcher = _Dispatcher(charm_dir, juju_context) + dispatcher.run_any_legacy_hook() + + framework = setup_framework( + charm_dir, state, event, context, charm_spec, juju_context + ) + charm = setup_charm(charm_class, framework, dispatcher) + return dispatcher, framework, charm + + +class Ops(Generic[CharmType]): + """Class to manage stepping through ops setup, event emission and framework commit.""" + + def __init__( + self, + state: "State", + event: "_Event", + context: "Context[CharmType]", + charm_spec: "_CharmSpec[CharmType]", + juju_context: Optional[ops.jujucontext._JujuContext] = None, + ): + self.state = state + self.event = event + self.context = context + self.charm_spec = charm_spec + if juju_context is None: + juju_context = ops.jujucontext._JujuContext.from_dict(os.environ) + self.juju_context = juju_context + + # set by setup() + self.dispatcher: Optional[_Dispatcher] = None + self.framework: Optional[ops.Framework] = None + self.charm: Optional["CharmType"] = None + + self._has_setup = False + self._has_emitted = False + self._has_committed = False + + def setup(self): + """Setup framework, charm and dispatcher.""" + self._has_setup = True + self.dispatcher, self.framework, self.charm = setup( + self.state, + self.event, + self.context, + self.charm_spec, + self.juju_context, + ) + + def emit(self): + """Emit the event on the charm.""" + if not self._has_setup: + raise RuntimeError("should .setup() before you .emit()") + self._has_emitted = True + + dispatcher = cast(_Dispatcher, self.dispatcher) + charm = cast(CharmBase, self.charm) + framework = cast(ops.Framework, self.framework) + + try: + if not dispatcher.is_restricted_context(): + framework.reemit() + + _emit_charm_event( + charm, dispatcher.event_name, self.juju_context, self.event + ) + + except Exception: + framework.close() + raise + + def commit(self): + """Commit the framework and teardown.""" + if not self._has_emitted: + raise RuntimeError("should .emit() before you .commit()") + + framework = cast(ops.Framework, self.framework) + charm = cast(CharmBase, self.charm) + + # emit collect-status events + ops.charm._evaluate_status(charm) + + self._has_committed = True + + try: + framework.commit() + finally: + framework.close() + + def finalize(self): + """Step through all non-manually-called procedures and run them.""" + if not self._has_setup: + self.setup() + if not self._has_emitted: + self.emit() + if not self._has_committed: + self.commit() diff --git a/testing/build/lib/scenario/py.typed b/testing/build/lib/scenario/py.typed new file mode 100644 index 000000000..e69de29bb diff --git a/testing/build/lib/scenario/runtime.py b/testing/build/lib/scenario/runtime.py new file mode 100644 index 000000000..3ad2fd0a2 --- /dev/null +++ b/testing/build/lib/scenario/runtime.py @@ -0,0 +1,577 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""Test framework runtime.""" + +import copy +import dataclasses +import marshal +import re +import tempfile +import typing +from contextlib import contextmanager +from pathlib import Path +from typing import ( + TYPE_CHECKING, + Any, + Dict, + FrozenSet, + List, + Optional, + Set, + Type, + TypeVar, + Union, +) + +import yaml +from ops import ( + CollectStatusEvent, + pebble, + CommitEvent, + EventBase, + Framework, + Handle, + NoTypeError, + PreCommitEvent, +) +from ops.jujucontext import _JujuContext +from ops.storage import NoSnapshotError, SQLiteStorage +from ops.framework import _event_regex +from ops._private.harness import ActionFailed + +from .errors import NoObserverError, UncaughtCharmError +from .logger import logger as scenario_logger +from .state import ( + DeferredEvent, + PeerRelation, + Relation, + StoredState, + SubordinateRelation, +) + +if TYPE_CHECKING: # pragma: no cover + from .context import Context + from .state import CharmType, State, _CharmSpec, _Event + +logger = scenario_logger.getChild("runtime") +STORED_STATE_REGEX = re.compile( + r"((?P.*)\/)?(?P<_data_type_name>\D+)\[(?P.*)\]", +) +EVENT_REGEX = re.compile(_event_regex) + +RUNTIME_MODULE = Path(__file__).parent + + +class UnitStateDB: + """Represents the unit-state.db.""" + + def __init__(self, db_path: Union[Path, str]): + self._db_path = db_path + self._state_file = Path(self._db_path) + + def _open_db(self) -> SQLiteStorage: + """Open the db.""" + return SQLiteStorage(self._state_file) + + def get_stored_states(self) -> FrozenSet["StoredState"]: + """Load any StoredState data structures from the db.""" + + db = self._open_db() + + stored_states: Set[StoredState] = set() + for handle_path in db.list_snapshots(): + if not EVENT_REGEX.match(handle_path) and ( + match := STORED_STATE_REGEX.match(handle_path) + ): + stored_state_snapshot = db.load_snapshot(handle_path) + kwargs = match.groupdict() + sst = StoredState(content=stored_state_snapshot, **kwargs) + stored_states.add(sst) + + db.close() + return frozenset(stored_states) + + def get_deferred_events(self) -> List["DeferredEvent"]: + """Load any DeferredEvent data structures from the db.""" + + db = self._open_db() + + deferred: List[DeferredEvent] = [] + for handle_path in db.list_snapshots(): + if EVENT_REGEX.match(handle_path): + notices = db.notices(handle_path) + for handle, owner, observer in notices: + try: + snapshot_data = db.load_snapshot(handle) + except NoSnapshotError: + snapshot_data: Dict[str, Any] = {} + + event = DeferredEvent( + handle_path=handle, + owner=owner, + observer=observer, + snapshot_data=snapshot_data, + ) + deferred.append(event) + + db.close() + return deferred + + def apply_state(self, state: "State"): + """Add DeferredEvent and StoredState from this State instance to the storage.""" + db = self._open_db() + for event in state.deferred: + db.save_notice(event.handle_path, event.owner, event.observer) + try: + marshal.dumps(event.snapshot_data) + except ValueError as e: + raise ValueError( + f"unable to save the data for {event}, it must contain only simple types.", + ) from e + db.save_snapshot(event.handle_path, event.snapshot_data) + + for stored_state in state.stored_states: + db.save_snapshot(stored_state._handle_path, stored_state.content) + + db.close() + + +class _OpsMainContext: # type: ignore + """Context manager representing ops.main execution context. + + When entered, ops.main sets up everything up until the charm. + When .emit() is called, ops.main proceeds with emitting the event. + When exited, if .emit has not been called manually, it is called automatically. + """ + + def __init__(self): + self._has_emitted = False + + def __enter__(self): + pass + + def emit(self): + """Emit the event. + + Within the test framework, this only requires recording that it was emitted. + """ + self._has_emitted = True + + def __exit__(self, exc_type: Any, exc_val: Any, exc_tb: Any): # noqa: U100 + if not self._has_emitted: + self.emit() + + +class Runtime: + """Charm runtime wrapper. + + This object bridges a local environment and a charm artifact. + """ + + def __init__( + self, + charm_spec: "_CharmSpec[CharmType]", + charm_root: Optional[Union[str, Path]] = None, + juju_version: str = "3.0.0", + app_name: Optional[str] = None, + unit_id: Optional[int] = 0, + ): + self._charm_spec = charm_spec + self._juju_version = juju_version + self._charm_root = charm_root + + app_name = app_name or self._charm_spec.meta.get("name") + if not app_name: + raise ValueError('invalid metadata: mandatory "name" field is missing.') + + self._app_name = app_name + self._unit_id = unit_id + + def _get_event_env(self, state: "State", event: "_Event", charm_root: Path): + """Build the simulated environment the operator framework expects.""" + env = { + "JUJU_VERSION": self._juju_version, + "JUJU_UNIT_NAME": f"{self._app_name}/{self._unit_id}", + "_": "./dispatch", + "JUJU_DISPATCH_PATH": f"hooks/{event.name}", + "JUJU_MODEL_NAME": state.model.name, + "JUJU_MODEL_UUID": state.model.uuid, + "JUJU_CHARM_DIR": str(charm_root.absolute()), + } + + if event._is_action_event and (action := event.action): + env.update( + { + "JUJU_ACTION_NAME": action.name.replace("_", "-"), + "JUJU_ACTION_UUID": action.id, + }, + ) + + if event._is_relation_event and (relation := event.relation): + if isinstance(relation, PeerRelation): + remote_app_name = self._app_name + elif isinstance(relation, (Relation, SubordinateRelation)): + remote_app_name = relation.remote_app_name + else: + raise ValueError(f"Unknown relation type: {relation}") + env.update( + { + "JUJU_RELATION": relation.endpoint, + "JUJU_RELATION_ID": str(relation.id), + "JUJU_REMOTE_APP": remote_app_name, + }, + ) + + remote_unit_id = event.relation_remote_unit_id + + # don't check truthiness because remote_unit_id could be 0 + if remote_unit_id is None and not event.name.endswith( + ("_relation_created", "relation_broken"), + ): + remote_unit_ids = relation._remote_unit_ids + + if len(remote_unit_ids) == 1: + remote_unit_id = remote_unit_ids[0] + logger.info( + "there's only one remote unit, so we set JUJU_REMOTE_UNIT to it, " + "but you probably should be parametrizing the event with `remote_unit_id` " + "to be explicit.", + ) + elif len(remote_unit_ids) > 1: + remote_unit_id = remote_unit_ids[0] + logger.warning( + "remote unit ID unset, and multiple remote unit IDs are present; " + "We will pick the first one and hope for the best. You should be passing " + "`remote_unit_id` to the Event constructor.", + ) + else: + logger.warning( + "remote unit ID unset; no remote unit data present. " + "Is this a realistic scenario?", + ) + + if remote_unit_id is not None: + remote_unit = f"{remote_app_name}/{remote_unit_id}" + env["JUJU_REMOTE_UNIT"] = remote_unit + if event.name.endswith("_relation_departed"): + if event.relation_departed_unit_id: + env["JUJU_DEPARTING_UNIT"] = ( + f"{remote_app_name}/{event.relation_departed_unit_id}" + ) + else: + env["JUJU_DEPARTING_UNIT"] = remote_unit + + if container := event.container: + env.update({"JUJU_WORKLOAD_NAME": container.name}) + + if notice := event.notice: + if hasattr(notice.type, "value"): + notice_type = typing.cast(pebble.NoticeType, notice.type).value + else: + notice_type = str(notice.type) + env.update( + { + "JUJU_NOTICE_ID": notice.id, + "JUJU_NOTICE_TYPE": notice_type, + "JUJU_NOTICE_KEY": notice.key, + }, + ) + + if check_info := event.check_info: + env["JUJU_PEBBLE_CHECK_NAME"] = check_info.name + + if storage := event.storage: + env.update({"JUJU_STORAGE_ID": f"{storage.name}/{storage.index}"}) + + if secret := event.secret: + env.update( + { + "JUJU_SECRET_ID": secret.id, + "JUJU_SECRET_LABEL": secret.label or "", + }, + ) + # Don't check truthiness because revision could be 0. + if event.secret_revision is not None: + env["JUJU_SECRET_REVISION"] = str(event.secret_revision) + + return env + + @staticmethod + def _wrap(charm_type: Type["CharmType"]) -> Type["CharmType"]: + # dark sorcery to work around framework using class attrs to hold on to event sources + # this should only be needed if we call play multiple times on the same runtime. + class WrappedEvents(charm_type.on.__class__): + """The charm's event sources, but wrapped.""" + + pass + + WrappedEvents.__name__ = charm_type.on.__class__.__name__ + + class WrappedCharm(charm_type): + """The test charm's type, but with events wrapped.""" + + on = WrappedEvents() + + WrappedCharm.__name__ = charm_type.__name__ + return typing.cast(Type["CharmType"], WrappedCharm) + + @contextmanager + def _virtual_charm_root(self): + # If we are using runtime on a real charm, we can make some assumptions about the + # directory structure we are going to find. + # If we're, say, dynamically defining charm types and doing tests on them, we'll have to + # generate the metadata files ourselves. To be sure, we ALWAYS use a tempdir. Ground truth + # is what the user passed via the CharmSpec + spec = self._charm_spec + + if charm_virtual_root := self._charm_root: + charm_virtual_root_is_custom = True + virtual_charm_root = Path(charm_virtual_root) + else: + charm_virtual_root = tempfile.TemporaryDirectory() + virtual_charm_root = Path(charm_virtual_root.name) + charm_virtual_root_is_custom = False + + metadata_yaml = virtual_charm_root / "metadata.yaml" + config_yaml = virtual_charm_root / "config.yaml" + actions_yaml = virtual_charm_root / "actions.yaml" + + metadata_files_present: Dict[Path, Optional[str]] = { + file: file.read_text() if file.exists() else None + for file in (metadata_yaml, config_yaml, actions_yaml) + } + + any_metadata_files_present_in_charm_virtual_root = any( + v is not None for v in metadata_files_present.values() + ) + + if spec.is_autoloaded and charm_virtual_root_is_custom: + # since the spec is autoloaded, in theory the metadata contents won't differ, so we can + # overwrite away even if the custom vroot is the real charm root (the local repo). + # Still, log it for clarity. + if any_metadata_files_present_in_charm_virtual_root: + logger.debug( + f"metadata files found in custom charm_root {charm_virtual_root}. " + f"The spec was autoloaded so the contents should be identical. " + f"Proceeding...", + ) + + elif ( + not spec.is_autoloaded and any_metadata_files_present_in_charm_virtual_root + ): + logger.warning( + f"Some metadata files found in custom user-provided charm_root " + f"{charm_virtual_root} while you have passed meta, config or actions to " + f"Context.run(). " + "Single source of truth are the arguments passed to Context.run(). " + "charm_root metadata files will be overwritten for the " + "duration of this test, and restored afterwards. " + "To avoid this, clean any metadata files from the charm_root before calling run.", + ) + + metadata_yaml.write_text(yaml.safe_dump(spec.meta)) + config_yaml.write_text(yaml.safe_dump(spec.config or {})) + actions_yaml.write_text(yaml.safe_dump(spec.actions or {})) + + yield virtual_charm_root + + if charm_virtual_root_is_custom: + for file, previous_content in metadata_files_present.items(): + if previous_content is None: # None == file did not exist before + file.unlink() + else: + file.write_text(previous_content) + + else: + # charm_virtual_root is a tempdir + typing.cast(tempfile.TemporaryDirectory, charm_virtual_root).cleanup() # type: ignore + + @staticmethod + def _get_state_db(temporary_charm_root: Path): + charm_state_path = temporary_charm_root / ".unit-state.db" + return UnitStateDB(charm_state_path) + + def _initialize_storage(self, state: "State", temporary_charm_root: Path): + """Before we start processing this event, store the relevant parts of State.""" + store = self._get_state_db(temporary_charm_root) + store.apply_state(state) + + def _close_storage(self, state: "State", temporary_charm_root: Path): + """Now that we're done processing this event, read the charm state and expose it.""" + store = self._get_state_db(temporary_charm_root) + deferred = store.get_deferred_events() + stored_state = store.get_stored_states() + return dataclasses.replace(state, deferred=deferred, stored_states=stored_state) + + @contextmanager + def _exec_ctx(self, ctx: "Context"): + """python 3.8 compatibility shim""" + with self._virtual_charm_root() as temporary_charm_root: + with _capture_events( + include_deferred=ctx.capture_deferred_events, + include_framework=ctx.capture_framework_events, + ) as captured: + yield (temporary_charm_root, captured) + + @contextmanager + def exec( + self, + state: "State", + event: "_Event", + context: "Context", + ): + """Runs an event with this state as initial state on a charm. + + Returns the 'output state', that is, the state as mutated by the charm during the + event handling. + + This will set the environment up and call ops.main(). + After that it's up to ops. + """ + from ._consistency_checker import check_consistency # avoid cycles + + check_consistency(state, event, self._charm_spec, self._juju_version) + + charm_type = self._charm_spec.charm_type + logger.info(f"Preparing to fire {event.name} on {charm_type.__name__}") + + # we make a copy to avoid mutating the input state + output_state = copy.deepcopy(state) + + logger.info(" - generating virtual charm root") + with self._exec_ctx(context) as (temporary_charm_root, captured): + logger.info(" - initializing storage") + self._initialize_storage(state, temporary_charm_root) + + logger.info(" - preparing env") + env = self._get_event_env( + state=state, + event=event, + charm_root=temporary_charm_root, + ) + juju_context = _JujuContext.from_dict(env) + + logger.info(" - Entering ops.main (mocked).") + from .ops_main_mock import Ops # noqa: F811 + + try: + ops = Ops( + state=output_state, + event=event, + context=context, + charm_spec=dataclasses.replace( + self._charm_spec, + charm_type=self._wrap(charm_type), + ), + juju_context=juju_context, + ) + ops.setup() + + yield ops + + # if the caller did not manually emit or commit: do that. + ops.finalize() + + except (NoObserverError, ActionFailed): + raise # propagate along + except Exception as e: + raise UncaughtCharmError( + f"Uncaught exception ({type(e)}) in operator/charm code: {e!r}", + ) from e + + finally: + logger.info(" - Exited ops.main.") + + logger.info(" - closing storage") + output_state = self._close_storage(output_state, temporary_charm_root) + + context.emitted_events.extend(captured) + logger.info("event dispatched. done.") + context._set_output_state(output_state) + + +_T = TypeVar("_T", bound=EventBase) + + +@contextmanager +def _capture_events( + *types: Type[EventBase], + include_framework: bool = False, + include_deferred: bool = True, +): + """Capture all events of type `*types` (using instance checks). + + Arguments exposed so that you can define your own fixtures if you want to. + + Example:: + >>> from ops import StartEvent + >>> from scenario import Event, State + >>> from charm import MyCustomEvent, MyCharm # noqa + >>> + >>> def test_my_event(): + >>> with capture_events(StartEvent, MyCustomEvent) as captured: + >>> trigger(State(), ("start", MyCharm, meta=MyCharm.META) + >>> + >>> assert len(captured) == 2 + >>> e1, e2 = captured + >>> assert isinstance(e2, MyCustomEvent) + >>> assert e2.custom_attr == 'foo' + """ + allowed_types = types or (EventBase,) + + captured: List[EventBase] = [] + _real_emit = Framework._emit + _real_reemit = Framework.reemit + + def _wrapped_emit(self: Framework, evt: EventBase): + if not include_framework and isinstance( + evt, + (PreCommitEvent, CommitEvent, CollectStatusEvent), + ): + return _real_emit(self, evt) + + if isinstance(evt, allowed_types): + # dump/undump the event to ensure any custom attributes are (re)set by restore() + evt.restore(evt.snapshot()) + captured.append(evt) + + return _real_emit(self, evt) + + def _wrapped_reemit(self: Framework): + # Framework calls reemit() before emitting the main juju event. We intercept that call + # and capture all events in storage. + + if not include_deferred: + return _real_reemit(self) + + # load all notices from storage as events. + for event_path, _, _ in self._storage.notices(): + event_handle = Handle.from_path(event_path) + try: + event = self.load_snapshot(event_handle) + except NoTypeError: + continue + event = typing.cast(EventBase, event) + event.deferred = False + self._forget(event) # prevent tracking conflicts + + if not include_framework and isinstance( + event, + (PreCommitEvent, CommitEvent), + ): + continue + + if isinstance(event, allowed_types): + captured.append(event) + + return _real_reemit(self) + + Framework._emit = _wrapped_emit # type: ignore + Framework.reemit = _wrapped_reemit + + yield captured + + Framework._emit = _real_emit + Framework.reemit = _real_reemit diff --git a/testing/build/lib/scenario/state.py b/testing/build/lib/scenario/state.py new file mode 100644 index 000000000..d7e24bffc --- /dev/null +++ b/testing/build/lib/scenario/state.py @@ -0,0 +1,2030 @@ +#!/usr/bin/env python3 +# Copyright 2023 Canonical Ltd. +# See LICENSE file for licensing details. + +"""The core State object, and the components inside it.""" + +from __future__ import annotations + +import dataclasses +import datetime +import inspect +import pathlib +import random +import re +import string +from enum import Enum +from itertools import chain +from typing import ( + TYPE_CHECKING, + Any, + Callable, + ClassVar, + Dict, + Final, + Generic, + Iterable, + List, + Literal, + Mapping, + NoReturn, + Sequence, + TypeVar, + Union, + cast, +) +from uuid import uuid4 + +import yaml + +import ops +from ops import pebble, CharmBase, CharmEvents, SecretRotate, StatusBase +from ops import CloudCredential as CloudCredential_Ops +from ops import CloudSpec as CloudSpec_Ops + +from .errors import MetadataNotFoundError, StateValidationError +from .logger import logger as scenario_logger + +if TYPE_CHECKING: # pragma: no cover + from . import Context + +AnyJson = Union[str, bool, Dict[str, "AnyJson"], int, float, List["AnyJson"]] +RawSecretRevisionContents = RawDataBagContents = Dict[str, str] +UnitID = int + +CharmType = TypeVar("CharmType", bound=CharmBase) + +logger = scenario_logger.getChild("state") + +ATTACH_ALL_STORAGES = "ATTACH_ALL_STORAGES" +CREATE_ALL_RELATIONS = "CREATE_ALL_RELATIONS" +BREAK_ALL_RELATIONS = "BREAK_ALL_RELATIONS" +DETACH_ALL_STORAGES = "DETACH_ALL_STORAGES" + +_ACTION_EVENT_SUFFIX = "_action" +# all builtin events except secret events. They're special because they carry secret metadata. +_BUILTIN_EVENTS = { + "start", + "stop", + "install", + "install", + "start", + "stop", + "remove", + "update_status", + "config_changed", + "upgrade_charm", + "pre_series_upgrade", + "post_series_upgrade", + "leader_elected", + "leader_settings_changed", + "collect_metrics", +} +_FRAMEWORK_EVENTS = { + "pre_commit", + "commit", + "collect_app_status", + "collect_unit_status", +} +_PEBBLE_READY_EVENT_SUFFIX = "_pebble_ready" +_PEBBLE_CUSTOM_NOTICE_EVENT_SUFFIX = "_pebble_custom_notice" +_PEBBLE_CHECK_FAILED_EVENT_SUFFIX = "_pebble_check_failed" +_PEBBLE_CHECK_RECOVERED_EVENT_SUFFIX = "_pebble_check_recovered" +_RELATION_EVENTS_SUFFIX = { + "_relation_changed", + "_relation_broken", + "_relation_joined", + "_relation_departed", + "_relation_created", +} +_STORAGE_EVENTS_SUFFIX = { + "_storage_detaching", + "_storage_attached", +} + +_SECRET_EVENTS = { + "secret_changed", + "secret_remove", + "secret_rotate", + "secret_expired", +} + + +# This can be replaced with the KW_ONLY dataclasses functionality in Python 3.10+. +def _max_posargs(n: int): + class _MaxPositionalArgs: + """Raises TypeError when instantiating objects if arguments are not passed as keywords. + + Looks for a `_max_positional_args` class attribute, which should be an int + indicating the maximum number of positional arguments that can be passed to + `__init__` (excluding `self`). + """ + + _max_positional_args = n + + def __new__(cls, *args: Any, **kwargs: Any): + # inspect.signature guarantees the order of parameters is as + # declared, which aligns with dataclasses. Simpler ways of + # getting the arguments (like __annotations__) do not have that + # guarantee, although in practice it is the case. + parameters = inspect.signature(cls.__init__).parameters + required_args = [ + name + for name in tuple(parameters) + if parameters[name].default is inspect.Parameter.empty + and name not in kwargs + and name != "self" + ] + n_posargs = len(args) + max_n_posargs = cls._max_positional_args + kw_only = { + name + for name in tuple(parameters)[max_n_posargs:] + if not name.startswith("_") + } + if n_posargs > max_n_posargs: + raise TypeError( + f"{cls.__name__} takes {max_n_posargs} positional " + f"argument{'' if max_n_posargs == 1 else 's'} but " + f"{n_posargs} {'was' if n_posargs == 1 else 'were'} " + f"given. The following arguments are keyword-only: " + f"{', '.join(kw_only)}", + ) from None + # Also check if there are just not enough arguments at all, because + # the default TypeError message will incorrectly describe some of + # the arguments as positional. + if n_posargs < len(required_args): + required_pos = [ + f"'{arg}'" + for arg in required_args[n_posargs:] + if arg not in kw_only + ] + required_kw = { + f"'{arg}'" for arg in required_args[n_posargs:] if arg in kw_only + } + if required_pos and required_kw: + details = f"positional: {', '.join(required_pos)} and keyword: {', '.join(required_kw)} arguments" + elif required_pos: + details = f"positional argument{'' if len(required_pos) == 1 else 's'}: {', '.join(required_pos)}" + else: + details = f"keyword argument{'' if len(required_kw) == 1 else 's'}: {', '.join(required_kw)}" + raise TypeError(f"{cls.__name__} missing required {details}") from None + return super().__new__(cls) + + def __reduce__(self): + # The default __reduce__ doesn't understand that some arguments have + # to be passed as keywords, so using the copy module fails. + attrs = cast(Dict[str, Any], super().__reduce__()[2]) + return (lambda: self.__class__(**attrs), ()) + + return _MaxPositionalArgs + + +@dataclasses.dataclass(frozen=True) +class JujuLogLine(_max_posargs(2)): + """An entry in the Juju debug-log.""" + + level: str + """The level of the message, for example ``INFO`` or ``ERROR``.""" + message: str + """The log message.""" + + +@dataclasses.dataclass(frozen=True) +class CloudCredential(_max_posargs(0)): + __doc__ = ops.CloudCredential.__doc__ + + auth_type: str + """Authentication type.""" + + attributes: dict[str, str] = dataclasses.field(default_factory=dict) + """A dictionary containing cloud credentials. + For example, for AWS, it contains `access-key` and `secret-key`; + for Azure, `application-id`, `application-password` and `subscription-id` + can be found here. + """ + + redacted: list[str] = dataclasses.field(default_factory=list) + """A list of redacted generic cloud API secrets.""" + + def _to_ops(self) -> CloudCredential_Ops: + return CloudCredential_Ops( + auth_type=self.auth_type, + attributes=self.attributes, + redacted=self.redacted, + ) + + +@dataclasses.dataclass(frozen=True) +class CloudSpec(_max_posargs(1)): + __doc__ = ops.CloudSpec.__doc__ + + type: str + """Type of the cloud.""" + + name: str = "localhost" + """Juju cloud name.""" + + region: str | None = None + """Region of the cloud.""" + + endpoint: str | None = None + """Endpoint of the cloud.""" + + identity_endpoint: str | None = None + """Identity endpoint of the cloud.""" + + storage_endpoint: str | None = None + """Storage endpoint of the cloud.""" + + credential: CloudCredential | None = None + """Cloud credentials with key-value attributes.""" + + ca_certificates: list[str] = dataclasses.field(default_factory=list) + """A list of CA certificates.""" + + skip_tls_verify: bool = False + """Whether to skip TLS verification.""" + + is_controller_cloud: bool = False + """If this is the cloud used by the controller.""" + + def _to_ops(self) -> CloudSpec_Ops: + return CloudSpec_Ops( + type=self.type, + name=self.name, + region=self.region, + endpoint=self.endpoint, + identity_endpoint=self.identity_endpoint, + storage_endpoint=self.storage_endpoint, + credential=self.credential._to_ops() if self.credential else None, + ca_certificates=self.ca_certificates, + skip_tls_verify=self.skip_tls_verify, + is_controller_cloud=self.is_controller_cloud, + ) + + +def _generate_secret_id(): + # This doesn't account for collisions, but the odds are so low that it + # should not be possible in any realistic test run. + secret_id = "".join( + random.choice(string.ascii_lowercase + string.digits) for _ in range(20) + ) + return f"secret:{secret_id}" + + +@dataclasses.dataclass(frozen=True) +class Secret(_max_posargs(1)): + """A Juju secret. + + This class is used for both user and charm secrets. + """ + + tracked_content: RawSecretRevisionContents + """The content of the secret that the charm is currently tracking. + + This is the content the charm will receive with a + :meth:`ops.Secret.get_content` call.""" + latest_content: RawSecretRevisionContents | None = None + """The content of the latest revision of the secret. + + This is the content the charm will receive with a + :meth:`ops.Secret.peek_content` call.""" + + id: str = dataclasses.field(default_factory=_generate_secret_id) + """The Juju ID of the secret. + + This is automatically assigned and should not usually need to be explicitly set. + """ + + owner: Literal["unit", "app", None] = None + """Indicates if the secret is owned by *this* unit, *this* application, or + another application/unit. + + If None, the implication is that read access to the secret has been granted + to this unit. + """ + + remote_grants: dict[int, set[str]] = dataclasses.field(default_factory=dict) + """Mapping from relation IDs to remote units and applications to which this + secret has been granted.""" + + label: str | None = None + """A human-readable label the charm can use to retrieve the secret. + + If this is set, it implies that the charm has previously set the label. + """ + description: str | None = None + """A human-readable description of the secret.""" + expire: datetime.datetime | None = None + """The time at which the secret will expire.""" + rotate: SecretRotate | None = None + """The rotation policy for the secret.""" + + # what revision is currently tracked by this charm. Only meaningful if owner=False + _tracked_revision: int = 1 + + # what revision is the latest for this secret. + _latest_revision: int = 1 + + def __hash__(self) -> int: + return hash(self.id) + + def __post_init__(self): + if self.latest_content is None: + # bypass frozen dataclass + object.__setattr__(self, "latest_content", self.tracked_content) + + def _set_label(self, label: str): + # bypass frozen dataclass + object.__setattr__(self, "label", label) + + def _track_latest_revision(self): + """Set the current revision to the tracked revision.""" + # bypass frozen dataclass + object.__setattr__(self, "_tracked_revision", self._latest_revision) + object.__setattr__(self, "tracked_content", self.latest_content) + + def _update_metadata( + self, + content: RawSecretRevisionContents | None = None, + label: str | None = None, + description: str | None = None, + expire: datetime.datetime | None = None, + rotate: SecretRotate | None = None, + ): + """Update the metadata.""" + # bypass frozen dataclass + object.__setattr__(self, "_latest_revision", self._latest_revision + 1) + if content: + object.__setattr__(self, "latest_content", content) + if label: + object.__setattr__(self, "label", label) + if description: + object.__setattr__(self, "description", description) + if expire: + if isinstance(expire, datetime.timedelta): + expire = datetime.datetime.now() + expire + object.__setattr__(self, "expire", expire) + if rotate: + object.__setattr__(self, "rotate", rotate) + + +def _normalise_name(s: str): + """Event names, in Scenario, uniformly use underscores instead of dashes.""" + return s.replace("-", "_") + + +@dataclasses.dataclass(frozen=True) +class Address(_max_posargs(1)): + """An address in a Juju network space.""" + + value: str + """The IP address in the space.""" + hostname: str = "" + """A host name that maps to the address in :attr:`value`.""" + cidr: str = "" + """The CIDR of the address in :attr:`value`.""" + + @property + def address(self): + """A deprecated alias for :attr:`value`.""" + return self.value + + @address.setter + def address(self, value: str): + object.__setattr__(self, "value", value) + + +@dataclasses.dataclass(frozen=True) +class BindAddress(_max_posargs(1)): + """An address bound to a network interface in a Juju space.""" + + addresses: list[Address] + """The addresses in the space.""" + interface_name: str = "" + """The name of the network interface.""" + mac_address: str | None = None + """The MAC address of the interface.""" + + def _hook_tool_output_fmt(self): + """Dumps itself to dict in the same format the hook tool would.""" + dct = { + "interface-name": self.interface_name, + "addresses": [dataclasses.asdict(addr) for addr in self.addresses], + } + if self.mac_address: + dct["mac-address"] = self.mac_address + return dct + + +@dataclasses.dataclass(frozen=True) +class Network(_max_posargs(2)): + """A Juju network space.""" + + binding_name: str + """The name of the network space.""" + bind_addresses: list[BindAddress] = dataclasses.field( + default_factory=lambda: [BindAddress([Address("192.0.2.0")])], + ) + """Addresses that the charm's application should bind to.""" + ingress_addresses: list[str] = dataclasses.field( + default_factory=lambda: ["192.0.2.0"], + ) + """Addresses other applications should use to connect to the unit.""" + egress_subnets: list[str] = dataclasses.field( + default_factory=lambda: ["192.0.2.0/24"], + ) + """Subnets that other units will see the charm connecting from.""" + + def __hash__(self) -> int: + return hash(self.binding_name) + + def _hook_tool_output_fmt(self): + # dumps itself to dict in the same format the hook tool would + return { + "bind-addresses": [ + ba._hook_tool_output_fmt() for ba in self.bind_addresses + ], + "egress-subnets": self.egress_subnets, + "ingress-addresses": self.ingress_addresses, + } + + +_next_relation_id_counter = 1 + + +def _next_relation_id(*, update: bool = True): + """Get the ID the next relation to be created will get. + + Pass update=False if you're only inspecting it. + Pass update=True if you also want to bump it. + """ + global _next_relation_id_counter + cur = _next_relation_id_counter + if update: + _next_relation_id_counter += 1 + return cur + + +@dataclasses.dataclass(frozen=True) +class RelationBase(_max_posargs(2)): + """Base class for the various types of integration (relation).""" + + endpoint: str + """Relation endpoint name. Must match some endpoint name defined in the metadata.""" + + interface: str | None = None + """Interface name. Must match the interface name attached to this endpoint in the metadata. + If left empty, it will be automatically derived from the metadata.""" + + id: int = dataclasses.field(default_factory=_next_relation_id) + """Juju relation ID. Every new Relation instance gets a unique one, + if there's trouble, override.""" + + local_app_data: RawDataBagContents = dataclasses.field(default_factory=dict) + """This application's databag for this relation.""" + + local_unit_data: RawDataBagContents = dataclasses.field( + default_factory=lambda: _DEFAULT_JUJU_DATABAG.copy(), + ) + """This unit's databag for this relation.""" + + @property + def relation_id(self) -> NoReturn: + """Use `.id` instead of `.relation_id`. + + :private: + """ + raise AttributeError("use .id instead of .relation_id") + + @property + def _databags(self): + """Yield all databags in this relation.""" + yield self.local_app_data + yield self.local_unit_data + + @property + def _remote_unit_ids(self) -> tuple[UnitID, ...]: + """Ids of the units on the other end of this relation.""" + raise NotImplementedError() + + def _get_databag_for_remote( + self, + unit_id: int, # noqa: U100 + ) -> RawDataBagContents: + """Return the databag for some remote unit ID.""" + raise NotImplementedError() + + def __post_init__(self): + if type(self) is RelationBase: + raise RuntimeError( + "RelationBase cannot be instantiated directly; " + "please use Relation, PeerRelation, or SubordinateRelation", + ) + + for databag in self._databags: + self._validate_databag(databag) + + def __hash__(self) -> int: + return hash(self.id) + + def _validate_databag(self, databag: dict[str, str]): + if not isinstance(databag, dict): + raise StateValidationError( + f"all databags should be dicts, not {type(databag)}", + ) + for v in databag.values(): + if not isinstance(v, str): + raise StateValidationError( + f"all databags should be Dict[str,str]; " + f"found a value of type {type(v)}", + ) + + +_DEFAULT_IP = "192.0.2.0" +_DEFAULT_JUJU_DATABAG = { + "egress-subnets": _DEFAULT_IP, + "ingress-address": _DEFAULT_IP, + "private-address": _DEFAULT_IP, +} + + +@dataclasses.dataclass(frozen=True) +class Relation(RelationBase): + """An integration between the charm and another application.""" + + remote_app_name: str = "remote" + """The name of the remote application, as in the charm's metadata.""" + + # local limit + limit: int = 1 + """The maximum number of integrations on this endpoint.""" + + remote_app_data: RawDataBagContents = dataclasses.field(default_factory=dict) + """The current content of the application databag.""" + remote_units_data: dict[UnitID, RawDataBagContents] = dataclasses.field( + default_factory=lambda: {0: _DEFAULT_JUJU_DATABAG.copy()}, # dedup + ) + """The current content of the databag for each unit in the relation.""" + + def __hash__(self) -> int: + return hash(self.id) + + @property + def _remote_app_name(self) -> str: + """Who is on the other end of this relation?""" + return self.remote_app_name + + @property + def _remote_unit_ids(self) -> tuple[UnitID, ...]: + """Ids of the units on the other end of this relation.""" + return tuple(self.remote_units_data) + + def _get_databag_for_remote(self, unit_id: UnitID) -> RawDataBagContents: + """Return the databag for some remote unit ID.""" + return self.remote_units_data[unit_id] + + @property + def _databags(self): # type: ignore + """Yield all databags in this relation.""" + yield self.local_app_data + yield self.local_unit_data + yield self.remote_app_data + yield from self.remote_units_data.values() + + +@dataclasses.dataclass(frozen=True) +class SubordinateRelation(RelationBase): + """A relation to share data between a subordinate and a principal charm.""" + + remote_app_data: RawDataBagContents = dataclasses.field(default_factory=dict) + """The current content of the remote application databag.""" + remote_unit_data: RawDataBagContents = dataclasses.field( + default_factory=lambda: _DEFAULT_JUJU_DATABAG.copy(), + ) + """The current content of the remote unit databag.""" + + remote_app_name: str = "remote" + """The name of the remote application that *this unit* is attached to.""" + remote_unit_id: int = 0 + """The ID of the remote unit that *this unit* is attached to.""" + + def __hash__(self) -> int: + return hash(self.id) + + @property + def _remote_unit_ids(self) -> tuple[int]: + """Ids of the units on the other end of this relation.""" + return (self.remote_unit_id,) + + def _get_databag_for_remote(self, unit_id: int) -> RawDataBagContents: + """Return the databag for some remote unit ID.""" + if unit_id is not self.remote_unit_id: + raise ValueError( + f"invalid unit id ({unit_id}): subordinate relation only has one " + f"remote and that has id {self.remote_unit_id}", + ) + return self.remote_unit_data + + @property + def _databags(self): + """Yield all databags in this relation.""" + yield self.local_app_data + yield self.local_unit_data + yield self.remote_app_data + yield self.remote_unit_data + + @property + def remote_unit_name(self) -> str: + """The full name of the remote unit, in the form ``remote/0``.""" + return f"{self.remote_app_name}/{self.remote_unit_id}" + + +@dataclasses.dataclass(frozen=True) +class PeerRelation(RelationBase): + """A relation to share data between units of the charm.""" + + peers_data: dict[UnitID, RawDataBagContents] = dataclasses.field( + default_factory=lambda: {0: _DEFAULT_JUJU_DATABAG.copy()}, + ) + """Current contents of the peer databags.""" + # Consistency checks will validate that *this unit*'s ID is not in here. + + def __hash__(self) -> int: + return hash(self.id) + + @property + def _databags(self): # type: ignore + """Yield all databags in this relation.""" + yield self.local_app_data + yield self.local_unit_data + yield from self.peers_data.values() + + @property + def _remote_unit_ids(self) -> tuple[UnitID, ...]: + """Ids of the units on the other end of this relation.""" + return tuple(self.peers_data) + + def _get_databag_for_remote(self, unit_id: UnitID) -> RawDataBagContents: + """Return the databag for some remote unit ID.""" + return self.peers_data[unit_id] + + +def _random_model_name(): + space = string.ascii_letters + string.digits + return "".join(random.choice(space) for _ in range(20)) + + +@dataclasses.dataclass(frozen=True) +class Model(_max_posargs(1)): + """The Juju model in which the charm is deployed.""" + + name: str = dataclasses.field(default_factory=_random_model_name) + """The name of the model.""" + uuid: str = dataclasses.field(default_factory=lambda: str(uuid4())) + """A unique identifier for the model, typically generated by Juju.""" + + # whatever juju models --format=json | jq '.models[].type' gives back. + type: Literal["kubernetes", "lxd"] = "kubernetes" + """The type of Juju model.""" + + cloud_spec: CloudSpec | None = None + """Cloud specification information (metadata) including credentials.""" + + +_CHANGE_IDS = 0 + + +def _generate_new_change_id(): + global _CHANGE_IDS + _CHANGE_IDS += 1 # type: ignore + logger.info( + f"change ID unset; automatically assigning {_CHANGE_IDS}. " + f"If there are problems, pass one manually.", + ) + return _CHANGE_IDS + + +@dataclasses.dataclass(frozen=True) +class Exec(_max_posargs(1)): + """Mock data for simulated :meth:`ops.Container.exec` calls.""" + + command_prefix: Sequence[str] + return_code: int = 0 + """The return code of the process. + + Use 0 to mock the process ending successfully, and other values for failure. + """ + stdout: str = "" + """Any content written to stdout by the process. + + Provide content that the real process would write to stdout, which can be + read by the charm. + """ + stderr: str = "" + """Any content written to stderr by the process. + + Provide content that the real process would write to stderr, which can be + read by the charm. + """ + + # change ID: used internally to keep track of mocked processes + _change_id: int = dataclasses.field(default_factory=_generate_new_change_id) + + def __post_init__(self): + # The command prefix can be any sequence type, and a list is tidier to + # write when there's only one string. However, this object needs to be + # hashable, so can't contain a list. We 'freeze' the sequence to a tuple + # to support that. + object.__setattr__(self, "command_prefix", tuple(self.command_prefix)) + + def _run(self) -> int: + return self._change_id + + +@dataclasses.dataclass(frozen=True) +class Mount(_max_posargs(0)): + """Maps local files to a :class:`Container` filesystem.""" + + location: str | pathlib.PurePosixPath + """The location inside of the container.""" + source: str | pathlib.Path + """The content to provide when the charm does :meth:`ops.Container.pull`.""" + + +def _now_utc(): + return datetime.datetime.now(tz=datetime.timezone.utc) + + +_next_notice_id_counter = 1 + + +def _next_notice_id(*, update: bool = True): + """Get the ID the next Pebble notice to be created will get. + + Pass update=False if you're only inspecting it. + Pass update=True if you also want to bump it. + """ + global _next_notice_id_counter + cur = _next_notice_id_counter + if update: + _next_notice_id_counter += 1 + return str(cur) + + +@dataclasses.dataclass(frozen=True) +class Notice(_max_posargs(1)): + """A Pebble notice.""" + + key: str + """The notice key, a string that differentiates notices of this type. + + This is in the format ``domain/path``; for example: + ``canonical.com/postgresql/backup`` or ``example.com/mycharm/notice``. + """ + + id: str = dataclasses.field(default_factory=_next_notice_id) + """Unique ID for this notice.""" + + user_id: int | None = None + """UID of the user who may view this notice (None means notice is public).""" + + type: pebble.NoticeType | str = pebble.NoticeType.CUSTOM + """Type of the notice.""" + + first_occurred: datetime.datetime = dataclasses.field(default_factory=_now_utc) + """The first time one of these notices (type and key combination) occurs.""" + + last_occurred: datetime.datetime = dataclasses.field(default_factory=_now_utc) + """The last time one of these notices occurred.""" + + last_repeated: datetime.datetime = dataclasses.field(default_factory=_now_utc) + """The time this notice was last repeated. + + See Pebble's `Notices documentation `_ + for an explanation of what "repeated" means. + """ + + occurrences: int = 1 + """The number of times one of these notices has occurred.""" + + last_data: dict[str, str] = dataclasses.field(default_factory=dict) + """Additional data captured from the last occurrence of one of these notices.""" + + repeat_after: datetime.timedelta | None = None + """Minimum time after one of these was last repeated before Pebble will repeat it again.""" + + expire_after: datetime.timedelta | None = None + """How long since one of these last occurred until Pebble will drop the notice.""" + + def _to_ops(self) -> pebble.Notice: + return pebble.Notice( + id=self.id, + user_id=self.user_id, + type=self.type, + key=self.key, + first_occurred=self.first_occurred, + last_occurred=self.last_occurred, + last_repeated=self.last_repeated, + occurrences=self.occurrences, + last_data=self.last_data, + repeat_after=self.repeat_after, + expire_after=self.expire_after, + ) + + +@dataclasses.dataclass(frozen=True) +class CheckInfo(_max_posargs(1)): + """A health check for a Pebble workload container.""" + + name: str + """Name of the check.""" + + level: pebble.CheckLevel | None = None + """Level of the check.""" + + status: pebble.CheckStatus = pebble.CheckStatus.UP + """Status of the check. + + :attr:`ops.pebble.CheckStatus.UP` means the check is healthy (the number of + failures is fewer than the threshold), :attr:`ops.pebble.CheckStatus.DOWN` + means the check is unhealthy (the number of failures has reached the + threshold). + """ + + failures: int = 0 + """Number of failures since the check last succeeded.""" + + threshold: int = 3 + """Failure threshold. + + This is how many consecutive failures for the check to be considered 'down'. + """ + + def _to_ops(self) -> pebble.CheckInfo: + return pebble.CheckInfo( + name=self.name, + level=self.level, + status=self.status, + failures=self.failures, + threshold=self.threshold, + ) + + +@dataclasses.dataclass(frozen=True) +class Container(_max_posargs(1)): + """A Kubernetes container where a charm's workload runs.""" + + name: str + """Name of the container, as found in the charm metadata.""" + + can_connect: bool = False + """When False, all Pebble operations will fail.""" + + # This is the base plan. On top of it, one can add layers. + # We need to model pebble in this way because it's impossible to retrieve the layers from + # pebble or derive them from the resulting plan (which one CAN get from pebble). + # So if we are instantiating Container by fetching info from a 'live' charm, the 'layers' + # will be unknown. all that we can know is the resulting plan (the 'computed plan'). + _base_plan: dict[str, Any] = dataclasses.field(default_factory=dict) + # We expect most of the user-facing testing to be covered by this 'layers' attribute, + # as it is all that will be known when unit-testing. + layers: dict[str, pebble.Layer] = dataclasses.field(default_factory=dict) + """All :class:`ops.pebble.Layer` definitions that have already been added to the container.""" + + service_statuses: dict[str, pebble.ServiceStatus] = dataclasses.field( + default_factory=dict, + ) + """The current status of each Pebble service running in the container.""" + + # this is how you specify the contents of the filesystem: suppose you want to express that your + # container has: + # - /home/foo/bar.py + # - /bin/bash + # - /bin/baz + # + # this becomes: + # mounts = { + # 'foo': Mount(location='/home/foo/', source=Path('/path/to/local/dir/containing/bar/py/')) + # 'bin': Mount(location='/bin/', source=Path('/path/to/local/dir/containing/bash/and/baz/')) + # } + # when the charm runs `pebble.pull`, it will return .open() from one of those paths. + # when the charm pushes, it will either overwrite one of those paths (careful!) or it will + # create a tempfile and insert its path in the mock filesystem tree + mounts: dict[str, Mount] = dataclasses.field(default_factory=dict) + """Provides access to the contents of the simulated container filesystem. + + For example, suppose you want to express that your container has: + + * ``/home/foo/bar.py`` + * ``/bin/bash`` + * ``/bin/baz`` + + this becomes:: + + mounts = { + 'foo': Mount('/home/foo', pathlib.Path('/path/to/local/dir/containing/bar/py/')), + 'bin': Mount('/bin/', pathlib.Path('/path/to/local/dir/containing/bash/and/baz/')), + } + """ + + execs: Iterable[Exec] = frozenset() + """Simulate executing commands in the container. + + Specify each command the charm might run in the container and an :class:`Exec` + containing its return code and any stdout/stderr. + + For example:: + + container = Container( + name='foo', + execs={ + Exec(['whoami'], return_code=0, stdout='ubuntu'), + Exec( + ['dig', '+short', 'canonical.com'], + return_code=0, + stdout='185.125.190.20\\n185.125.190.21', + ), + } + ) + """ + + notices: list[Notice] = dataclasses.field(default_factory=list) + """Any Pebble notices that already exist in the container.""" + + check_infos: frozenset[CheckInfo] = frozenset() + """All Pebble health checks that have been added to the container.""" + + def __hash__(self) -> int: + return hash(self.name) + + def __post_init__(self): + if not isinstance(self.execs, frozenset): + # Allow passing a regular set (or other iterable) of Execs. + object.__setattr__(self, "execs", frozenset(self.execs)) + + def _render_services(self): + # copied over from ops.testing._TestingPebbleClient._render_services() + services: dict[str, pebble.Service] = {} + for key in sorted(self.layers.keys()): + layer = self.layers[key] + for name, service in layer.services.items(): + services[name] = service + return services + + @property + def plan(self) -> pebble.Plan: + """The 'computed' Pebble plan. + + This is the base plan plus the layers that have been added on top. + You should run your assertions on this plan, not so much on the layers, + as those are input data. + """ + + # copied over from ops.testing._TestingPebbleClient.get_plan(). + plan = pebble.Plan(yaml.safe_dump(self._base_plan)) + services = self._render_services() + if not services: + return plan + for name in sorted(services.keys()): + plan.services[name] = services[name] + return plan + + @property + def services(self) -> dict[str, pebble.ServiceInfo]: + """The Pebble services as rendered in the plan.""" + services = self._render_services() + infos: dict[str, pebble.ServiceInfo] = {} + names = sorted(services.keys()) + for name in names: + try: + service = services[name] + except KeyError: + # in pebble, it just returns "nothing matched" if there are 0 matches, + # but it ignores services it doesn't recognize + continue + status = self.service_statuses.get(name, pebble.ServiceStatus.INACTIVE) + if service.startup == "": + startup = pebble.ServiceStartup.DISABLED + else: + startup = pebble.ServiceStartup(service.startup) + info = pebble.ServiceInfo( + name, + startup=startup, + current=pebble.ServiceStatus(status), + ) + infos[name] = info + return infos + + def get_filesystem(self, ctx: Context) -> pathlib.Path: + """Simulated Pebble filesystem in this context. + + Returns: + A temporary filesystem containing any files or directories the + charm pushed to the container. + """ + return ctx._get_container_root(self.name) + + +_RawStatusLiteral = Literal[ + "waiting", + "blocked", + "active", + "unknown", + "error", + "maintenance", +] + + +@dataclasses.dataclass(frozen=True) +class _EntityStatus: + """This class represents StatusBase and should not be interacted with directly.""" + + # Why not use StatusBase directly? Because that can't be used with + # dataclasses.asdict to then be JSON-serializable. + + name: _RawStatusLiteral + message: str = "" + + _entity_statuses: ClassVar[dict[str, type[_EntityStatus]]] = {} + + def __eq__(self, other: Any): + if isinstance(other, (StatusBase, _EntityStatus)): + return (self.name, self.message) == (other.name, other.message) + return super().__eq__(other) + + def __repr__(self): + status_type_name = self.name.title() + "Status" + if self.name == "unknown": + return f"{status_type_name}()" + return f"{status_type_name}('{self.message}')" + + @classmethod + def from_status_name( + cls, + name: _RawStatusLiteral, + message: str = "", + ) -> _EntityStatus: + """Convert the status name, such as 'active', to the class, such as ActiveStatus.""" + # Note that this won't work for UnknownStatus. + # All subclasses have a default 'name' attribute, but the type checker can't tell that. + return cls._entity_statuses[name](message=message) # type:ignore + + @classmethod + def from_ops(cls, obj: StatusBase) -> _EntityStatus: + """Convert from the ops.StatusBase object to the matching _EntityStatus object.""" + return cls.from_status_name(obj.name, obj.message) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class UnknownStatus(_EntityStatus, ops.UnknownStatus): + __doc__ = ops.UnknownStatus.__doc__ + + name: Literal["unknown"] = "unknown" + + def __init__(self): + super().__init__(name=self.name) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class ErrorStatus(_EntityStatus, ops.ErrorStatus): + __doc__ = ops.ErrorStatus.__doc__ + + name: Literal["error"] = "error" + + def __init__(self, message: str = ""): + super().__init__(name="error", message=message) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class ActiveStatus(_EntityStatus, ops.ActiveStatus): + __doc__ = ops.ActiveStatus.__doc__ + + name: Literal["active"] = "active" + + def __init__(self, message: str = ""): + super().__init__(name="active", message=message) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class BlockedStatus(_EntityStatus, ops.BlockedStatus): + __doc__ = ops.BlockedStatus.__doc__ + + name: Literal["blocked"] = "blocked" + + def __init__(self, message: str = ""): + super().__init__(name="blocked", message=message) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class MaintenanceStatus(_EntityStatus, ops.MaintenanceStatus): + __doc__ = ops.MaintenanceStatus.__doc__ + + name: Literal["maintenance"] = "maintenance" + + def __init__(self, message: str = ""): + super().__init__(name="maintenance", message=message) + + +@dataclasses.dataclass(frozen=True, eq=False, repr=False) +class WaitingStatus(_EntityStatus, ops.WaitingStatus): + __doc__ = ops.WaitingStatus.__doc__ + + name: Literal["waiting"] = "waiting" + + def __init__(self, message: str = ""): + super().__init__(name="waiting", message=message) + + +_EntityStatus._entity_statuses.update( + unknown=UnknownStatus, + error=ErrorStatus, + active=ActiveStatus, + blocked=BlockedStatus, + maintenance=MaintenanceStatus, + waiting=WaitingStatus, +) + + +@dataclasses.dataclass(frozen=True) +class StoredState(_max_posargs(1)): + """Represents unit-local state that persists across events.""" + + name: str = "_stored" + """The attribute in the parent Object where the state is stored. + + For example, ``_stored`` in this class:: + + class MyCharm(ops.CharmBase): + _stored = ops.StoredState() + + """ + + owner_path: str | None = None + """The path to the owner of this StoredState instance. + + If None, the owner is the Framework. Otherwise, /-separated object names, + for example MyCharm/MyCharmLib. + """ + + # Ideally, the type here would be only marshallable types, rather than Any. + # However, it's complex to describe those types, since it's a recursive + # definition - even in TypeShed the _Marshallable type includes containers + # like list[Any], which seems to defeat the point. + content: dict[str, Any] = dataclasses.field(default_factory=dict) + """The content of the :class:`ops.StoredState` instance.""" + + _data_type_name: str = "StoredStateData" + + @property + def _handle_path(self): + return f"{self.owner_path or ''}/{self._data_type_name}[{self.name}]" + + def __hash__(self) -> int: + return hash(self._handle_path) + + +_RawPortProtocolLiteral = Literal["tcp", "udp", "icmp"] + + +@dataclasses.dataclass(frozen=True) +class Port(_max_posargs(1)): + """Represents a port on the charm host. + + Port objects should not be instantiated directly: use :class:`TCPPort`, + :class:`UDPPort`, or :class:`ICMPPort` instead. + """ + + port: int | None = None + """The port to open. Required for TCP and UDP; not allowed for ICMP.""" + + protocol: _RawPortProtocolLiteral = "tcp" + """The protocol that data transferred over the port will use.""" + + def __post_init__(self): + if type(self) is Port: + raise RuntimeError( + "Port cannot be instantiated directly; " + "please use TCPPort, UDPPort, or ICMPPort", + ) + + def __eq__(self, other: object) -> bool: + if isinstance(other, (Port, ops.Port)): + return (self.protocol, self.port) == (other.protocol, other.port) + return False + + +@dataclasses.dataclass(frozen=True) +class TCPPort(Port): + """Represents a TCP port on the charm host.""" + + port: int # type: ignore + """The port to open.""" + protocol: _RawPortProtocolLiteral = "tcp" + """The protocol that data transferred over the port will use. + + :meta private: + """ + + def __post_init__(self): + super().__post_init__() + if not (1 <= self.port <= 65535): + raise StateValidationError( + f"`port` outside bounds [1:65535], got {self.port}", + ) + + +@dataclasses.dataclass(frozen=True) +class UDPPort(Port): + """Represents a UDP port on the charm host.""" + + port: int # type: ignore + """The port to open.""" + protocol: _RawPortProtocolLiteral = "udp" + """The protocol that data transferred over the port will use. + + :meta private: + """ + + def __post_init__(self): + super().__post_init__() + if not (1 <= self.port <= 65535): + raise StateValidationError( + f"`port` outside bounds [1:65535], got {self.port}", + ) + + +@dataclasses.dataclass(frozen=True) +class ICMPPort(Port): + """Represents an ICMP port on the charm host.""" + + protocol: _RawPortProtocolLiteral = "icmp" + """The protocol that data transferred over the port will use. + + :meta private: + """ + + _max_positional_args: Final = 0 + + def __post_init__(self): + super().__post_init__() + if self.port is not None: + raise StateValidationError("`port` cannot be set for `ICMPPort`") + + +_port_cls_by_protocol = { + "tcp": TCPPort, + "udp": UDPPort, + "icmp": ICMPPort, +} + + +_next_storage_index_counter = 0 # storage indices start at 0 + + +def _next_storage_index(*, update: bool = True): + """Get the index (used to be called ID) the next Storage to be created will get. + + Pass update=False if you're only inspecting it. + Pass update=True if you also want to bump it. + """ + global _next_storage_index_counter + cur = _next_storage_index_counter + if update: + _next_storage_index_counter += 1 + return cur + + +@dataclasses.dataclass(frozen=True) +class Storage(_max_posargs(1)): + """Represents an (attached) storage made available to the charm container.""" + + name: str + """The name of the storage, as found in the charm metadata.""" + + index: int = dataclasses.field(default_factory=_next_storage_index) + """The index of this storage instance. + + For Kubernetes charms, this will always be 1. For machine charms, each new + Storage instance gets a new index.""" + + def __eq__(self, other: object) -> bool: + if isinstance(other, (Storage, ops.Storage)): + return (self.name, self.index) == (other.name, other.index) + return False + + def get_filesystem(self, ctx: Context) -> pathlib.Path: + """Simulated filesystem root in this context.""" + return ctx._get_storage_root(self.name, self.index) + + +@dataclasses.dataclass(frozen=True) +class Resource(_max_posargs(0)): + """Represents a resource made available to the charm.""" + + name: str + """The name of the resource, as found in the charm metadata.""" + path: str | pathlib.Path + """A local path that will be provided to the charm as the content of the resource.""" + + +@dataclasses.dataclass(frozen=True) +class State(_max_posargs(0)): + """Represents the Juju-owned portion of a unit's state. + + Roughly speaking, it wraps all hook-tool- and pebble-mediated data a charm can access in its + lifecycle. For example, status-get will return data from `State.unit_status`, is-leader will + return data from `State.leader`, and so on. + """ + + config: dict[str, str | int | float | bool] = dataclasses.field( + default_factory=dict, + ) + """The present configuration of this charm.""" + relations: Iterable[RelationBase] = dataclasses.field(default_factory=frozenset) + """All relations that currently exist for this charm.""" + networks: Iterable[Network] = dataclasses.field(default_factory=frozenset) + """Manual overrides for any relation and extra bindings currently provisioned for this charm. + If a metadata-defined relation endpoint is not explicitly mapped to a Network in this field, + it will be defaulted. + + .. warning:: + `extra-bindings` is a deprecated, regretful feature in Juju/ops. For completeness we + support it, but use at your own risk. If a metadata-defined extra-binding is left empty, + it will be defaulted. + """ + containers: Iterable[Container] = dataclasses.field(default_factory=frozenset) + """All containers (whether they can connect or not) that this charm is aware of.""" + storages: Iterable[Storage] = dataclasses.field(default_factory=frozenset) + """All **attached** storage instances for this charm. + + If a storage is not attached, omit it from this listing.""" + + # we don't use sets to make json serialization easier + opened_ports: Iterable[Port] = dataclasses.field(default_factory=frozenset) + """Ports opened by Juju on this charm.""" + leader: bool = False + """Whether this charm has leadership.""" + model: Model = Model() + """The model this charm lives in.""" + secrets: Iterable[Secret] = dataclasses.field(default_factory=frozenset) + """The secrets this charm has access to (as an owner, or as a grantee). + + The presence of a secret in this list entails that the charm can read it. + Whether it can manage it or not depends on the individual secret's `owner` flag.""" + resources: Iterable[Resource] = dataclasses.field(default_factory=frozenset) + """All resources that this charm can access.""" + planned_units: int = 1 + """Number of non-dying planned units that are expected to be running this application. + + Use with caution.""" + + # Represents the OF's event queue. These events will be emitted before the event being + # dispatched, and represent the events that had been deferred during the previous run. + # If the charm defers any events during "this execution", they will be appended + # to this list. + deferred: list[DeferredEvent] = dataclasses.field(default_factory=list) + """Events that have been deferred on this charm by some previous execution.""" + stored_states: Iterable[StoredState] = dataclasses.field( + default_factory=frozenset, + ) + """Contents of a charm's stored state.""" + + # the current statuses. + app_status: _EntityStatus = UnknownStatus() + """Status of the application.""" + unit_status: _EntityStatus = UnknownStatus() + """Status of the unit.""" + workload_version: str = "" + """Workload version.""" + + def __post_init__(self): + # Let people pass in the ops classes, and convert them to the appropriate Scenario classes. + for name in ["app_status", "unit_status"]: + val = getattr(self, name) + if isinstance(val, _EntityStatus): + pass + elif isinstance(val, StatusBase): + object.__setattr__(self, name, _EntityStatus.from_ops(val)) + else: + raise TypeError(f"Invalid status.{name}: {val!r}") + normalised_ports = [ + Port(protocol=port.protocol, port=port.port) + if isinstance(port, ops.Port) + else port + for port in self.opened_ports + ] + if self.opened_ports != normalised_ports: + object.__setattr__(self, "opened_ports", normalised_ports) + normalised_storage = [ + Storage(name=storage.name, index=storage.index) + if isinstance(storage, ops.Storage) + else storage + for storage in self.storages + ] + if self.storages != normalised_storage: + object.__setattr__(self, "storages", normalised_storage) + + # ops.Container, ops.Model, ops.Relation, ops.Secret should not be instantiated by charmers. + # ops.Network does not have the relation name, so cannot be converted. + # ops.Resources does not contain the source of the resource, so cannot be converted. + # ops.StoredState is not convenient to initialise with data, so not useful here. + + # It's convenient to pass a set, but we really want the attributes to be + # frozen sets to increase the immutability of State objects. + for name in [ + "relations", + "containers", + "storages", + "networks", + "opened_ports", + "secrets", + "resources", + "stored_states", + ]: + val = getattr(self, name) + # It's ok to pass any iterable (of hashable objects), but you'll get + # a frozenset as the actual attribute. + if not isinstance(val, frozenset): + object.__setattr__(self, name, frozenset(val)) + + def _update_workload_version(self, new_workload_version: str): + """Update the current app version and record the previous one.""" + # We don't keep a full history because we don't expect the app version to change more + # than once per hook. + + # bypass frozen dataclass + object.__setattr__(self, "workload_version", new_workload_version) + + def _update_status( + self, + new_status: _EntityStatus, + is_app: bool = False, + ): + """Update the current app/unit status.""" + name = "app_status" if is_app else "unit_status" + # bypass frozen dataclass + object.__setattr__(self, name, new_status) + + def _update_opened_ports(self, new_ports: frozenset[Port]): + """Update the current opened ports.""" + # bypass frozen dataclass + object.__setattr__(self, "opened_ports", new_ports) + + def _update_secrets(self, new_secrets: frozenset[Secret]): + """Update the current secrets.""" + # bypass frozen dataclass + object.__setattr__(self, "secrets", new_secrets) + + def get_container(self, container: str, /) -> Container: + """Get container from this State, based on its name.""" + for state_container in self.containers: + if state_container.name == container: + return state_container + raise KeyError(f"container: {container} not found in the State") + + def get_network(self, binding_name: str, /) -> Network: + """Get network from this State, based on its binding name.""" + for network in self.networks: + if network.binding_name == binding_name: + return network + raise KeyError(f"network: {binding_name} not found in the State") + + def get_secret( + self, + *, + id: str | None = None, + label: str | None = None, + ) -> Secret: + """Get secret from this State, based on the secret's id or label.""" + if id is None and label is None: + raise ValueError("An id or label must be provided.") + + for secret in self.secrets: + if ( + (id and label and secret.id == id and secret.label == label) + or (id and label is None and secret.id == id) + or (id is None and label and secret.label == label) + ): + return secret + raise KeyError("secret: not found in the State") + + def get_stored_state( + self, + stored_state: str, + /, + *, + owner_path: str | None = None, + ) -> StoredState: + """Get stored state from this State, based on the stored state's name and owner_path.""" + for ss in self.stored_states: + if ss.name == stored_state and ss.owner_path == owner_path: + return ss + raise ValueError(f"stored state: {stored_state} not found in the State") + + def get_storage( + self, + storage: str, + /, + *, + index: int | None = 0, + ) -> Storage: + """Get storage from this State, based on the storage's name and index.""" + for state_storage in self.storages: + if state_storage.name == storage and storage.index == index: + return state_storage + raise ValueError( + f"storage: name={storage}, index={index} not found in the State", + ) + + def get_relation(self, relation: int, /) -> RelationBase: + """Get relation from this State, based on the relation's id.""" + for state_relation in self.relations: + if state_relation.id == relation: + return state_relation + raise KeyError(f"relation: id={relation} not found in the State") + + def get_relations(self, endpoint: str) -> tuple[RelationBase, ...]: + """Get all relations on this endpoint from the current state.""" + + # we rather normalize the endpoint than worry about cursed metadata situations such as: + # requires: + # foo-bar: ... + # foo_bar: ... + + normalized_endpoint = _normalise_name(endpoint) + return tuple( + r + for r in self.relations + if _normalise_name(r.endpoint) == normalized_endpoint + ) + + +def _is_valid_charmcraft_25_metadata(meta: dict[str, Any]): + # Check whether this dict has the expected mandatory metadata fields according to the + # charmcraft >2.5 charmcraft.yaml schema + if (config_type := meta.get("type")) != "charm": + logger.debug( + f"Not a charm: charmcraft yaml config ``.type`` is {config_type!r}.", + ) + return False + if not all(field in meta for field in {"name", "summary", "description"}): + logger.debug("Not a charm: charmcraft yaml misses some required fields") + return False + return True + + +@dataclasses.dataclass(frozen=True) +class _CharmSpec(Generic[CharmType]): + """Charm spec.""" + + charm_type: type[CharmBase] + meta: dict[str, Any] + actions: dict[str, Any] | None = None + config: dict[str, Any] | None = None + + # autoloaded means: we are running a 'real' charm class, living in some + # /src/charm.py, and the metadata files are 'real' metadata files. + is_autoloaded: bool = False + + @staticmethod + def _load_metadata_legacy(charm_root: pathlib.Path): + """Load metadata from charm projects created with Charmcraft < 2.5.""" + # back in the days, we used to have separate metadata.yaml, config.yaml and actions.yaml + # files for charm metadata. + metadata_path = charm_root / "metadata.yaml" + meta: dict[str, Any] = ( + yaml.safe_load(metadata_path.open()) if metadata_path.exists() else {} + ) + + config_path = charm_root / "config.yaml" + config = yaml.safe_load(config_path.open()) if config_path.exists() else None + + actions_path = charm_root / "actions.yaml" + actions = yaml.safe_load(actions_path.open()) if actions_path.exists() else None + return meta, config, actions + + @staticmethod + def _load_metadata(charm_root: pathlib.Path): + """Load metadata from charm projects created with Charmcraft >= 2.5.""" + metadata_path = charm_root / "charmcraft.yaml" + meta: dict[str, Any] = ( + yaml.safe_load(metadata_path.open()) if metadata_path.exists() else {} + ) + if not _is_valid_charmcraft_25_metadata(meta): + meta = {} + config = meta.pop("config", None) + actions = meta.pop("actions", None) + return meta, config, actions + + @staticmethod + def autoload(charm_type: type[CharmBase]) -> _CharmSpec[CharmType]: + """Construct a ``_CharmSpec`` object by looking up the metadata from the charm's repo root. + + Will attempt to load the metadata off the ``charmcraft.yaml`` file + """ + charm_source_path = pathlib.Path(inspect.getfile(charm_type)) + charm_root = charm_source_path.parent.parent + + # attempt to load metadata from unified charmcraft.yaml + meta, config, actions = _CharmSpec._load_metadata(charm_root) + + if not meta: + # try to load using legacy metadata.yaml/actions.yaml/config.yaml files + meta, config, actions = _CharmSpec._load_metadata_legacy(charm_root) + + if not meta: + # still no metadata? bug out + raise MetadataNotFoundError( + f"invalid charm root {charm_root!r}; " + f"expected to contain at least a `charmcraft.yaml` file " + f"(or a `metadata.yaml` file if it's an old charm).", + ) + + return _CharmSpec( + charm_type=charm_type, + meta=meta, + actions=actions, + config=config, + is_autoloaded=True, + ) + + def get_all_relations(self) -> list[tuple[str, dict[str, str]]]: + """A list of all relation endpoints defined in the metadata.""" + return list( + chain( + self.meta.get("requires", {}).items(), + self.meta.get("provides", {}).items(), + self.meta.get("peers", {}).items(), + ), + ) + + +@dataclasses.dataclass(frozen=True) +class DeferredEvent: + """An event that has been deferred to run prior to the next Juju event. + + Tests should not instantiate this class directly: use the `deferred` method + of the event instead. For example: + + ctx = Context(MyCharm) + deferred_start = ctx.on.start().deferred(handler=MyCharm._on_start) + state = State(deferred=[deferred_start]) + """ + + handle_path: str + owner: str + observer: str + + # needs to be marshal.dumps-able. + snapshot_data: dict[Any, Any] = dataclasses.field(default_factory=dict) + + # It would be nicer if people could do something like: + # `isinstance(state.deferred[0], ops.StartEvent)` + # than comparing with the string names, but there's only one `_Event` + # class in Scenario, and it also needs to be created from the context, + # which is not available here. For the ops classes, it's complex to create + # them because they need a Handle. + @property + def name(self): + """A comparable name for the event.""" + return self.handle_path.split("/")[-1].split("[")[0] + + +class _EventType(str, Enum): + FRAMEWORK = "framework" + BUILTIN = "builtin" + RELATION = "relation" + ACTION = "action" + SECRET = "secret" + STORAGE = "storage" + WORKLOAD = "workload" + CUSTOM = "custom" + + +class _EventPath(str): + if TYPE_CHECKING: # pragma: no cover + name: str + owner_path: list[str] + suffix: str + prefix: str + is_custom: bool + type: _EventType + + def __new__(cls, string: str): + string = _normalise_name(string) + instance = super().__new__(cls, string) + + instance.name = name = string.split(".")[-1] + instance.owner_path = string.split(".")[:-1] or ["on"] + + instance.suffix, instance.type = suffix, _ = _EventPath._get_suffix_and_type( + name, + ) + if suffix: + instance.prefix, _ = string.rsplit(suffix) + else: + instance.prefix = string + + instance.is_custom = suffix == "" + return instance + + @staticmethod + def _get_suffix_and_type(s: str) -> tuple[str, _EventType]: + for suffix in _RELATION_EVENTS_SUFFIX: + if s.endswith(suffix): + return suffix, _EventType.RELATION + + if s.endswith(_ACTION_EVENT_SUFFIX): + return _ACTION_EVENT_SUFFIX, _EventType.ACTION + + if s in _SECRET_EVENTS: + return s, _EventType.SECRET + + if s in _FRAMEWORK_EVENTS: + return s, _EventType.FRAMEWORK + + # Whether the event name indicates that this is a storage event. + for suffix in _STORAGE_EVENTS_SUFFIX: + if s.endswith(suffix): + return suffix, _EventType.STORAGE + + # Whether the event name indicates that this is a workload event. + if s.endswith(_PEBBLE_READY_EVENT_SUFFIX): + return _PEBBLE_READY_EVENT_SUFFIX, _EventType.WORKLOAD + if s.endswith(_PEBBLE_CUSTOM_NOTICE_EVENT_SUFFIX): + return _PEBBLE_CUSTOM_NOTICE_EVENT_SUFFIX, _EventType.WORKLOAD + if s.endswith(_PEBBLE_CHECK_FAILED_EVENT_SUFFIX): + return _PEBBLE_CHECK_FAILED_EVENT_SUFFIX, _EventType.WORKLOAD + if s.endswith(_PEBBLE_CHECK_RECOVERED_EVENT_SUFFIX): + return _PEBBLE_CHECK_RECOVERED_EVENT_SUFFIX, _EventType.WORKLOAD + + if s in _BUILTIN_EVENTS: + return "", _EventType.BUILTIN + + return "", _EventType.CUSTOM + + +@dataclasses.dataclass(frozen=True) +class _Event: # type: ignore + """A Juju, ops, or custom event that can be run against a charm. + + Typically, for simple events, the string name (e.g. ``install``) can be used, + and for more complex events, an ``event`` property will be available on the + related object (e.g. ``relation.joined_event``). + """ + + path: str + args: tuple[Any, ...] = () + kwargs: dict[str, Any] = dataclasses.field(default_factory=dict) + + storage: Storage | None = None + """If this is a storage event, the storage it refers to.""" + relation: RelationBase | None = None + """If this is a relation event, the relation it refers to.""" + relation_remote_unit_id: int | None = None + relation_departed_unit_id: int | None = None + + secret: Secret | None = None + """If this is a secret event, the secret it refers to.""" + + # if this is a secret-removed or secret-expired event, the secret revision it refers to + secret_revision: int | None = None + + container: Container | None = None + """If this is a workload (container) event, the container it refers to.""" + + notice: Notice | None = None + """If this is a Pebble notice event, the notice it refers to.""" + + check_info: CheckInfo | None = None + """If this is a Pebble check event, the check info it provides.""" + + action: _Action | None = None + """If this is an action event, the :class:`Action` it refers to.""" + + _owner_path: list[str] = dataclasses.field(default_factory=list) + + def __post_init__(self): + path = _EventPath(self.path) + # bypass frozen dataclass + object.__setattr__(self, "path", path) + + @property + def _path(self) -> _EventPath: + # we converted it in __post_init__, but the type checker doesn't know about that + return cast(_EventPath, self.path) + + @property + def name(self) -> str: + """Full event name. + + Consists of a 'prefix' and a 'suffix'. The suffix denotes the type of the event, the + prefix the name of the entity the event is about. + + "foo-relation-changed": + - "foo"=prefix (name of a relation), + - "-relation-changed"=suffix (relation event) + """ + return self._path.name + + @property + def owner_path(self) -> list[str]: + """Path to the ObjectEvents instance owning this event. + + If this event is defined on the toplevel charm class, it should be ['on']. + """ + return self._path.owner_path + + @property + def _is_relation_event(self) -> bool: + """Whether the event name indicates that this is a relation event.""" + return self._path.type is _EventType.RELATION + + @property + def _is_action_event(self) -> bool: + """Whether the event name indicates that this is a relation event.""" + return self._path.type is _EventType.ACTION + + @property + def _is_secret_event(self) -> bool: + """Whether the event name indicates that this is a secret event.""" + return self._path.type is _EventType.SECRET + + @property + def _is_storage_event(self) -> bool: + """Whether the event name indicates that this is a storage event.""" + return self._path.type is _EventType.STORAGE + + @property + def _is_workload_event(self) -> bool: + """Whether the event name indicates that this is a workload event.""" + return self._path.type is _EventType.WORKLOAD + + # this method is private because _CharmSpec is not quite user-facing; also, + # the user should know. + def _is_builtin_event(self, charm_spec: _CharmSpec[CharmType]) -> bool: + """Determine whether the event is a custom-defined one or a builtin one.""" + event_name = self.name + + # simple case: this is an event type owned by our charm base.on + if hasattr(charm_spec.charm_type.on, event_name): + return hasattr(CharmEvents, event_name) + + # this could be an event defined on some other Object, e.g. a charm lib. + # We don't support (yet) directly emitting those, but they COULD have names that conflict + # with events owned by the base charm. E.g. if the charm has a `foo` relation, the charm + # will get a charm.on.foo_relation_created. Your charm lib is free to define its own + # `foo_relation_created` custom event, because its handle will be + # `charm.lib.on.foo_relation_created` and therefore be unique and the Framework is happy. + # However, our Event data structure ATM has no knowledge of which Object/Handle it is + # owned by. So the only thing we can do right now is: check whether the event name, + # assuming it is owned by the charm, LOOKS LIKE that of a builtin event or not. + return self._path.type is not _EventType.CUSTOM + + def deferred(self, handler: Callable[..., Any], event_id: int = 1) -> DeferredEvent: + """Construct a DeferredEvent from this Event.""" + handler_repr = repr(handler) + handler_re = re.compile(r"") + match = handler_re.match(handler_repr) + if not match: + raise ValueError( + f"cannot construct DeferredEvent from {handler}; please create one manually.", + ) + owner_name, handler_name = match.groups()[0].split(".")[-2:] + handle_path = f"{owner_name}/on/{self.name}[{event_id}]" + + # Many events have no snapshot data: install, start, stop, remove, config-changed, + # upgrade-charm, pre-series-upgrade, post-series-upgrade, leader-elected, + # leader-settings-changed, collect-metrics + snapshot_data: dict[str, Any] = {} + + # fixme: at this stage we can't determine if the event is a builtin one or not; if it is + # not, then the coming checks are meaningless: the custom event could be named like a + # relation event but not *be* one. + if self._is_workload_event: + # Enforced by the consistency checker, but for type checkers: + assert self.container is not None + snapshot_data["container_name"] = self.container.name + if self.notice: + if hasattr(self.notice.type, "value"): + notice_type = cast(pebble.NoticeType, self.notice.type).value + else: + notice_type = str(self.notice.type) + snapshot_data.update( + { + "notice_id": self.notice.id, + "notice_key": self.notice.key, + "notice_type": notice_type, + }, + ) + elif self.check_info: + snapshot_data["check_name"] = self.check_info.name + + elif self._is_relation_event: + # Enforced by the consistency checker, but for type checkers: + assert self.relation is not None + relation = self.relation + if isinstance(relation, PeerRelation): + # FIXME: relation.unit for peers should point to , but we + # don't have access to the local app name in this context. + remote_app = "local" + elif isinstance(relation, (Relation, SubordinateRelation)): + remote_app = relation.remote_app_name + else: + raise RuntimeError(f"unexpected relation type: {relation!r}") + + snapshot_data.update( + { + "relation_name": relation.endpoint, + "relation_id": relation.id, + "app_name": remote_app, + }, + ) + if not self.name.endswith(("_created", "_broken")): + snapshot_data["unit_name"] = ( + f"{remote_app}/{self.relation_remote_unit_id}" + ) + if self.name.endswith("_departed"): + snapshot_data["departing_unit"] = ( + f"{remote_app}/{self.relation_departed_unit_id}" + ) + + elif self._is_storage_event: + # Enforced by the consistency checker, but for type checkers: + assert self.storage is not None + snapshot_data.update( + { + "storage_name": self.storage.name, + "storage_index": self.storage.index, + # "storage_location": str(self.storage.get_filesystem(self._context)), + }, + ) + + elif self._is_secret_event: + # Enforced by the consistency checker, but for type checkers: + assert self.secret is not None + snapshot_data.update( + {"secret_id": self.secret.id, "secret_label": self.secret.label}, + ) + if self.name.endswith(("_remove", "_expired")): + snapshot_data["secret_revision"] = self.secret_revision + + elif self._is_action_event: + # Enforced by the consistency checker, but for type checkers: + assert self.action is not None + snapshot_data["id"] = self.action.id + + return DeferredEvent( + handle_path, + owner_name, + handler_name, + snapshot_data=snapshot_data, + ) + + +_next_action_id_counter = 1 + + +def _next_action_id(*, update: bool = True): + """Get the ID the next action to be created will get. + + Pass update=False if you're only inspecting it. + Pass update=True if you also want to bump it. + """ + global _next_action_id_counter + cur = _next_action_id_counter + if update: + _next_action_id_counter += 1 + # Juju currently uses numbers for the ID, but in the past used UUIDs, so + # we need these to be strings. + return str(cur) + + +@dataclasses.dataclass(frozen=True) +class _Action(_max_posargs(1)): + """A ``juju run`` command. + + Used to simulate ``juju run``, passing in any parameters. For example:: + + def test_backup_action(): + ctx = Context(MyCharm) + state = ctx.run( + ctx.on.action('do_backup', params={'filename': 'foo'}), + State(), + ) + assert ctx.action_results == ... + """ + + name: str + """Juju action name, as found in the charm metadata.""" + + params: Mapping[str, AnyJson] = dataclasses.field(default_factory=dict) + """Parameter values passed to the action.""" + + id: str = dataclasses.field(default_factory=_next_action_id) + """Juju action ID. + + Every action invocation is automatically assigned a new one. Override in + the rare cases where a specific ID is required.""" diff --git a/testing/dist/ops_scenario-7.0.5-py3-none-any.whl b/testing/dist/ops_scenario-7.0.5-py3-none-any.whl new file mode 100644 index 000000000..d4bf9fd42 Binary files /dev/null and b/testing/dist/ops_scenario-7.0.5-py3-none-any.whl differ diff --git a/testing/src/ops_scenario.egg-info/PKG-INFO b/testing/src/ops_scenario.egg-info/PKG-INFO new file mode 100644 index 000000000..288ae8623 --- /dev/null +++ b/testing/src/ops_scenario.egg-info/PKG-INFO @@ -0,0 +1,1208 @@ +Metadata-Version: 2.1 +Name: ops-scenario +Version: 7.0.6.dev0 +Summary: Python library providing a state-transition testing API for Operator Framework charms. +Author-email: Pietro Pasotti +Maintainer-email: "The Charm Tech team at Canonical Ltd." +License: Apache-2.0 +Project-URL: Homepage, https://github.com/canonical/operator +Project-URL: Bug Tracker, https://github.com/canonical/operator/issues +Keywords: juju,test +Classifier: Development Status :: 5 - Production/Stable +Classifier: License :: OSI Approved :: Apache Software License +Classifier: Framework :: Pytest +Classifier: Intended Audience :: Developers +Classifier: Topic :: Software Development :: Quality Assurance +Classifier: Topic :: Software Development :: Testing +Classifier: Topic :: Utilities +Requires-Python: >=3.8 +Description-Content-Type: text/markdown +Requires-Dist: ops~=2.17 +Requires-Dist: PyYAML>=6.0.1 + +# Scenario + +[![Build](https://github.com/canonical/ops-scenario/actions/workflows/build_wheels.yaml/badge.svg)](https://github.com/canonical/ops-scenario/actions/workflows/build_wheels.yaml) +[![QC](https://github.com/canonical/ops-scenario/actions/workflows/quality_checks.yaml/badge.svg?event=pull_request)](https://github.com/canonical/ops-scenario/actions/workflows/quality_checks.yaml?event=pull_request) +[![Discourse Status](https://img.shields.io/discourse/status?server=https%3A%2F%2Fdiscourse.charmhub.io&style=flat&label=CharmHub%20Discourse)](https://discourse.charmhub.io) +[![foo](https://img.shields.io/badge/everything-charming-blueviolet)](https://github.com/PietroPasotti/jhack) +[![Awesome](https://cdn.rawgit.com/sindresorhus/awesome/d7305f38d29fed78fa85652e3a63e154dd8e8829/media/badge.svg)](https://discourse.charmhub.io/t/rethinking-charm-testing-with-ops-scenario/8649) +[![Python >= 3.8](https://img.shields.io/badge/python-3.8-blue.svg)](https://www.python.org/downloads/release/python-380/) + +Scenario is a state-transition testing SDK for Operator Framework charms. + +Where the Harness enables you to procedurally mock pieces of the state the charm needs to function, Scenario tests allow +you to declaratively define the state all at once, and use it as a sort of context against which you can fire a single +event on the charm and execute its logic. + +This puts scenario tests somewhere in between unit and integration tests: some say 'functional', some say 'contract', I prefer 'state-transition'. + +Scenario tests nudge you into thinking of a charm as an input->output function. The input is the +union of an `Event` (why am I, charm, being executed), a `State` (am I leader? what is my relation data? what is my +config?...) and the charm's execution `Context` (what relations can I have? what containers can I have?...). The output is another `State`: the state after the charm has had a chance to interact with the +mocked Juju model and affect the initial state back. + +![state transition model depiction](https://raw.githubusercontent.com/canonical/ops-scenario/main/resources/state-transition-model.png) + +For example: a charm currently in `unknown` status is executed with a `start` event, and based on whether it has leadership or not (according to its input state), it will decide to set `active` or `blocked` status (which will be reflected in the output state). + +Scenario-testing a charm, then, means verifying that: + +- the charm does not raise uncaught exceptions while handling the event +- the output state (or the diff with the input state) is as expected. + +## Core concepts as a metaphor + +I like metaphors, so here we go: + +- There is a theatre stage. +- You pick an actor (a Charm) to put on the stage. Not just any actor: an improv one. +- You arrange the stage with content that the actor will have to interact with. This consists of selecting: + - An initial situation (`State`) in which the actor is, e.g. is the actor the main role or an NPC (`is_leader`), or what + other actors are there around it, what is written in those pebble-shaped books on the table? + - Something that has just happened (an `Event`) and to which the actor has to react (e.g. one of the NPCs leaves the + stage (`relation-departed`), or the content of one of the books changes). +- How the actor will react to the event will have an impact on the context: e.g. the actor might knock over a table (a + container), or write something down into one of the books. + +## Core concepts not as a metaphor + +Scenario tests are about running assertions on atomic state transitions treating the charm being tested like a black +box. An initial state goes in, an event occurs (say, `'start'`) and a new state comes out. Scenario tests are about +validating the transition, that is, consistency-checking the delta between the two states, and verifying the charm +author's expectations. + +Comparing scenario tests with `Harness` tests: + +- Harness exposes an imperative API: the user is expected to call methods on the Harness driving it to the desired + state, then verify its validity by calling charm methods or inspecting the raw data. In contrast, Scenario is declarative. You fully specify an initial state, an execution context and an event, then you run the charm and inspect the results. +- Harness instantiates the charm once, then allows you to fire multiple events on the charm, which is breeding ground + for subtle bugs. Scenario tests are centered around testing single state transitions, that is, one event at a time. + This ensures that the execution environment is as clean as possible (for a unit test). +- Harness maintains a model of the Juju Model, which is a maintenance burden and adds complexity. Scenario mocks at the + level of hook tools and stores all mocking data in a monolithic data structure (the State), which makes it more + lightweight and portable. + +# Writing scenario tests + +A scenario test consists of three broad steps: + +- **Arrange**: + - declare the context + - declare the input state + - select an event to fire +- **Act**: + - run the context (i.e. obtain the output state, given the input state and the event) +- **Assert**: + - verify that the output state (or the delta with the input state) is how you expect it to be + - verify that the charm has seen a certain sequence of statuses, events, and `juju-log` calls + - optionally, you can use a context manager to get a hold of the charm instance and run assertions on internal APIs and the internal state of the charm and operator framework. + +The most basic scenario is one in which all is defaulted and barely any data is +available. The charm has no config, no relations, no leadership, and its status is `unknown`. + +With that, we can write the simplest possible scenario test: + +```python +def test_scenario_base(): + ctx = scenario.Context(MyCharm, meta={"name": "foo"}) + out = ctx.run(ctx.on.start(), scenario.State()) + assert out.unit_status == scenario.UnknownStatus() +``` + +Note that you should always compare the app and unit status using `==`, not `is`. You can compare +them to either the `scenario` objects, or the `ops` ones. + +Now let's start making it more complicated. Our charm sets a special state if it has leadership on 'start': + +```python +import pytest + + +class MyCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + framework.observe(self.on.start, self._on_start) + + def _on_start(self, _): + if self.unit.is_leader(): + self.unit.status = ops.ActiveStatus('I rule') + else: + self.unit.status = ops.ActiveStatus('I am ruled') + + +@pytest.mark.parametrize('leader', (True, False)) +def test_status_leader(leader): + ctx = scenario.Context(MyCharm, meta={"name": "foo"}) + out = ctx.run(ctx.on.start(), scenario.State(leader=leader)) + assert out.unit_status == scenario.ActiveStatus('I rule' if leader else 'I am ruled') +``` + +By defining the right state we can programmatically define what answers will the charm get to all the questions it can +ask the Juju model: am I leader? What are my relations? What is the remote unit I'm talking to? etc... + +## Statuses + +One of the simplest types of black-box testing available to charmers is to execute the charm and verify that the charm +sets the expected unit/application status. We have seen a simple example above including leadership. But what if the +charm transitions through a sequence of statuses? + +```python +# charm code: +def _on_event(self, _event): + self.unit.status = ops.MaintenanceStatus('determining who the ruler is...') + try: + if self._call_that_takes_a_few_seconds_and_only_passes_on_leadership(): + self.unit.status = ops.ActiveStatus('I rule') + else: + self.unit.status = ops.WaitingStatus('checking this is right...') + self._check_that_takes_some_more_time() + self.unit.status = ops.ActiveStatus('I am ruled') + except: + self.unit.status = ops.BlockedStatus('something went wrong') +``` + +More broadly, often we want to test 'side effects' of executing a charm, such as what events have been emitted, what +statuses it went through, etc... Before we get there, we have to explain what the `Context` represents, and its +relationship with the `State`. + +## Context and State + +Consider the following tests. Suppose we want to verify that while handling a given top-level Juju event: + +- a specific chain of (custom) events was emitted on the charm +- the charm `juju-log`ged these specific strings +- the charm went through this sequence of app/unit statuses (e.g. `maintenance`, then `waiting`, then `active`) + +These types of test have a place in Scenario, but that is not State: the contents of the Juju log or the status history +are side effects of executing a charm, but are not persisted in a charm-accessible "state" in any meaningful way. +In other words: those data streams are, from the charm's perspective, write-only. + +As such, they do not belong in `scenario.State` but in `scenario.Context`: the object representing the charm's execution +context. + +## Status history + +You can verify that the charm has followed the expected path by checking the unit/app status history like so: + +```python +def test_statuses(): + ctx = scenario.Context(MyCharm, meta={"name": "foo"}) + out = ctx.run(ctx.on.start(), scenario.State(leader=False)) + assert ctx.unit_status_history == [ + scenario.UnknownStatus(), + scenario.MaintenanceStatus('determining who the ruler is...'), + scenario.WaitingStatus('checking this is right...'), + ] + assert out.unit_status == scenario.ActiveStatus("I am ruled") + + # similarly you can check the app status history: + assert ctx.app_status_history == [ + scenario.UnknownStatus(), + ... + ] +``` + +Note that the *current* status is **not** in the unit status history. + +Also note that, unless you initialize the State with a preexisting status, the first status in the history will always +be `unknown`. That is because, so far as Scenario is concerned, each event is "the first event this charm has ever +seen". + +If you want to simulate a situation in which the charm already has seen some event, and is in a status other than +Unknown (the default status every charm is born with), you will have to pass the 'initial status' to State. + +```python +class MyCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + framework.observe(self.on.start, self._on_start) + + def _on_start(self, event): + self.model.unit.status = ops.ActiveStatus("foo") + +# ... +ctx = scenario.Context(MyCharm, meta={"name": "foo"}) +ctx.run(ctx.on.start(), scenario.State(unit_status=scenario.ActiveStatus('foo'))) +assert ctx.unit_status_history == [ + scenario.ActiveStatus('foo'), # now the first status is active: 'foo'! + # ... +] +``` + +## Workload version history + +Using a similar api to `*_status_history`, you can assert that the charm has set one or more workload versions during a +hook execution: + +```python +# ... +ctx = scenario.Context(HistoryCharm, meta={"name": "foo"}) +ctx.run(ctx.on.start(), scenario.State()) +assert ctx.workload_version_history == ['1', '1.2', '1.5'] +# ... +``` + +Note that the *current* version is not in the version history, as with the status history. + +## Emitted events + +If your charm deals with deferred events, custom events, and charm libs that in turn emit their own custom events, it +can be hard to examine the resulting control flow. In these situations it can be useful to verify that, as a result of a +given Juju event triggering (say, 'start'), a specific chain of events is emitted on the charm. The +resulting state, black-box as it is, gives little insight into how exactly it was obtained. + +```python +def test_foo(): + ctx = scenario.Context(...) + ctx.run(ctx.on.start(), ...) + + assert len(ctx.emitted_events) == 1 + assert isinstance(ctx.emitted_events[0], ops.StartEvent) +``` + +You can configure what events will be captured by passing the following arguments to `Context`: +- `capture_deferred_events`: If you want to include re-emitted deferred events. +- `capture_framework_events`: If you want to include framework events (`pre-commit`, `commit`, and `collect-status`). + +For example: +```python +def test_emitted_full(): + ctx = scenario.Context( + MyCharm, + capture_deferred_events=True, + capture_framework_events=True, + ) + ctx.run(ctx.on.start(), scenario.State(deferred=[ctx.on.update_status().deferred(MyCharm._foo)])) + + assert len(ctx.emitted_events) == 5 + assert [e.handle.kind for e in ctx.emitted_events] == [ + "update_status", + "start", + "collect_unit_status", + "pre_commit", + "commit", + ] +``` + +## Relations + +You can write scenario tests to verify the shape of relation data: + +```python +# This charm copies over remote app data to local unit data +class MyCharm(ops.CharmBase): + ... + + def _on_event(self, event): + rel = event.relation + assert rel.app.name == 'remote' + assert rel.data[self.unit]['abc'] == 'foo' + rel.data[self.unit]['abc'] = rel.data[event.app]['cde'] + + +def test_relation_data(): + rel = scenario.Relation( + endpoint="foo", + interface="bar", + remote_app_name="remote", + local_unit_data={"abc": "foo"}, + remote_app_data={"cde": "baz!"}, + ) + state_in = scenario.State(relations={rel}) + ctx = scenario.Context(MyCharm, meta={"name": "foo"}) + + state_out = ctx.run(ctx.on.start(), state_in) + + assert state_out.get_relation(rel.id).local_unit_data == {"abc": "baz!"} + # You can do this to check that there are no other differences: + assert state_out.relations == { + scenario.Relation( + endpoint="foo", + interface="bar", + remote_app_name="remote", + local_unit_data={"abc": "baz!"}, + remote_app_data={"cde": "baz!"}, + ), + } + +# which is very idiomatic and superbly explicit. +``` + +The only mandatory argument to `Relation` (and other relation types, see below) is `endpoint`. The `interface` will be +derived from the charm's metadata. When fully defaulted, a relation is 'empty'. There are no remote units, the +remote application is called `'remote'` and only has a single unit `remote/0`, and nobody has written any data to the +databags yet. + +That is typically the state of a relation when the first unit joins it. + +When you use `Relation`, you are specifying a regular (conventional) relation. But that is not the only type of +relation. There are also peer relations and subordinate relations. While in the background the data model is the same, +the data access rules and the consistency constraints on them are very different. For example, it does not make sense +for a peer relation to have a different 'remote app' than its 'local app', because it's the same application. + +### PeerRelation + +To declare a peer relation, you should use `scenario.PeerRelation`. The core difference with regular relations is +that peer relations do not have a "remote app" (it's this app, in fact). So unlike `Relation`, a `PeerRelation` does not +have `remote_app_name` or `remote_app_data` arguments. Also, it talks in terms of `peers`: + +- `Relation.remote_units_data` maps to `PeerRelation.peers_data` + +```python +relation = scenario.PeerRelation( + endpoint="peers", + peers_data={1: {}, 2: {}, 42: {'foo': 'bar'}}, +) +``` + +be mindful when using `PeerRelation` not to include **"this unit"**'s ID in `peers_data` or `peers_ids`, as that would +be flagged by the Consistency Checker: + +```python +state_in = scenario.State(relations={ + scenario.PeerRelation( + endpoint="peers", + peers_data={1: {}, 2: {}, 42: {'foo': 'bar'}}, + )}) + +meta = { + "name": "invalid", + "peers": { + "peers": { + "interface": "foo", + } + } +} +ctx = scenario.Context(ops.CharmBase, meta=meta, unit_id=1) +ctx.run(ctx.on.start(), state_in) # invalid: this unit's id cannot be the ID of a peer. +``` + +### SubordinateRelation + +To declare a subordinate relation, you should use `scenario.SubordinateRelation`. The core difference with regular +relations is that subordinate relations always have exactly one remote unit (there is always exactly one remote unit +that this unit can see). +Because of that, `SubordinateRelation`, compared to `Relation`, always talks in terms of `remote`: + +- `Relation.remote_units_data` becomes `SubordinateRelation.remote_unit_data` taking a single `Dict[str:str]`. The remote unit ID can be provided as a separate argument. +- `Relation.remote_unit_ids` becomes `SubordinateRelation.remote_unit_id` (a single ID instead of a list of IDs) +- `Relation.remote_units_data` becomes `SubordinateRelation.remote_unit_data` (a single databag instead of a mapping from unit IDs to databags) + +```python +relation = scenario.SubordinateRelation( + endpoint="peers", + remote_unit_data={"foo": "bar"}, + remote_app_name="zookeeper", + remote_unit_id=42 +) +relation.remote_unit_name # "zookeeper/42" +``` + +### Triggering Relation Events + +If you want to trigger relation events, use `ctx.on.relation_changed` (and so +on for the other relation events) and pass the relation object: + +```python +ctx = scenario.Context(MyCharm, meta=MyCharm.META) + +relation = scenario.Relation(endpoint="foo", interface="bar") +changed_event = ctx.on.relation_changed(relation=relation) +joined_event = ctx.on.relation_joined(relation=relation) +# ... +``` + +The reason for this construction is that the event is associated with some relation-specific metadata, that Scenario +needs to set up the process that will run `ops.main` with the right environment variables. + +### Additional event parameters + +All relation events have some additional metadata that does not belong in the Relation object, such as, for a +relation-joined event, the name of the (remote) unit that is joining the relation. That is what determines what +`ops.model.Unit` you get when you get `RelationJoinedEvent().unit` in an event handler. + +In order to supply this parameter, as well as the relation object, pass as `remote_unit` the id of the +remote unit that the event is about. The reason that this parameter is not supplied to `Relation` but to relation +events, is that the relation already ties 'this app' to some 'remote app' (cfr. the `Relation.remote_app_name` attr), +but not to a specific unit. What remote unit this event is about is not a `State` concern but an `Event` one. + +```python +ctx = scenario.Context(MyCharm, meta=MyCharm.META) + +relation = scenario.Relation(endpoint="foo", interface="bar") +remote_unit_2_is_joining_event = ctx.on.relation_joined(relation, remote_unit=2) +``` + +## Networks + +Simplifying a bit the Juju "spaces" model, each integration endpoint a charm defines in its metadata is associated with a network. Regardless of whether there is a living relation over that endpoint, that is. + +If your charm has a relation `"foo"` (defined in its metadata), then the charm will be able at runtime to do `self.model.get_binding("foo").network`. +The network you'll get by doing so is heavily defaulted (see `state.Network`) and good for most use-cases because the charm should typically not be concerned about what IP it gets. + +On top of the relation-provided network bindings, a charm can also define some `extra-bindings` in its metadata and access them at runtime. Note that this is a deprecated feature that should not be relied upon. For completeness, we support it in Scenario. + +If you want to, you can override any of these relation or extra-binding associated networks with a custom one by passing it to `State.networks`. + +```python +state = scenario.State(networks={ + scenario.Network("foo", [scenario.BindAddress([scenario.Address('192.0.2.1')])]) +}) +``` + +Where `foo` can either be the name of an `extra-bindings`-defined binding, or a relation endpoint. + +## Containers + +When testing a Kubernetes charm, you can mock container interactions. When using the null state (`State()`), there will +be no containers. So if the charm were to `self.unit.containers`, it would get back an empty dict. + +To give the charm access to some containers, you need to pass them to the input state, like so: +`State(containers={...})` + +An example of a state including some containers: + +```python +state = scenario.State(containers={ + scenario.Container(name="foo", can_connect=True), + scenario.Container(name="bar", can_connect=False) +}) +``` + +In this case, `self.unit.get_container('foo').can_connect()` would return `True`, while for 'bar' it would give `False`. + +### Container filesystem setup + +You can configure a container to have some files in it: + +```python +import pathlib + +local_file = pathlib.Path('/path/to/local/real/file.txt') + +container = scenario.Container( + name="foo", + can_connect=True, + mounts={'local': scenario.Mount(location='/local/share/config.yaml', source=local_file)} + ) +state = scenario.State(containers={container}) +``` + +In this case, if the charm were to: + +```python +def _on_start(self, _): + foo = self.unit.get_container('foo') + content = foo.pull('/local/share/config.yaml').read() +``` + +then `content` would be the contents of our locally-supplied `file.txt`. You can use `tempfile` for nicely wrapping +data and passing it to the charm via the container. + +`container.push` works similarly, so you can write a test like: + +```python +import tempfile + + +class MyCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + framework.observe(self.on.foo_pebble_ready, self._on_pebble_ready) + + def _on_pebble_ready(self, _): + foo = self.unit.get_container('foo') + foo.push('/local/share/config.yaml', "TEST", make_dirs=True) + + +def test_pebble_push(): + with tempfile.NamedTemporaryFile() as local_file: + container = scenario.Container( + name='foo', + can_connect=True, + mounts={'local': Mount(location='/local/share/config.yaml', source=local_file.name)} + ) + state_in = State(containers={container}) + ctx = Context( + MyCharm, + meta={"name": "foo", "containers": {"foo": {}}} + ) + ctx.run( + ctx.on.pebble_ready(container), + state_in, + ) + assert local_file.read().decode() == "TEST" +``` + +`container.pebble_ready_event` is syntactic sugar for: `Event("foo-pebble-ready", container=container)`. The reason we +need to associate the container with the event is that the Framework uses an envvar to determine which container the +pebble-ready event is about (it does not use the event name). Scenario needs that information, similarly, for injecting +that envvar into the charm's runtime. + +### Container filesystem post-mortem + +If the charm writes files to a container (to a location you didn't Mount as a temporary folder you have access to), you will be able to inspect them using the `get_filesystem` api. + +```python +class MyCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + framework.observe(self.on.foo_pebble_ready, self._on_pebble_ready) + + def _on_pebble_ready(self, _): + foo = self.unit.get_container('foo') + foo.push('/local/share/config.yaml', "TEST", make_dirs=True) + + +def test_pebble_push(): + container = scenario.Container(name='foo', can_connect=True) + state_in = scenario.State(containers={container}) + ctx = scenario.Context( + MyCharm, + meta={"name": "foo", "containers": {"foo": {}}} + ) + + ctx.run(ctx.on.start(), state_in) + + # This is the root of the simulated container filesystem. Any mounts will be symlinks in it. + container_root_fs = container.get_filesystem(ctx) + cfg_file = container_root_fs / 'local' / 'share' / 'config.yaml' + assert cfg_file.read_text() == "TEST" +``` + +### `Container.exec` mocks + +`container.exec` is a tad more complicated, but if you get to this low a level of simulation, you probably will have far +worse issues to deal with. You need to specify, for each possible command the charm might run on the container, what the +result of that would be: its return code, what will be written to stdout/stderr. + +```python +LS_LL = """ +.rw-rw-r-- 228 ubuntu ubuntu 18 jan 12:05 -- charmcraft.yaml +.rw-rw-r-- 497 ubuntu ubuntu 18 jan 12:05 -- config.yaml +.rw-rw-r-- 900 ubuntu ubuntu 18 jan 12:05 -- CONTRIBUTING.md +drwxrwxr-x - ubuntu ubuntu 18 jan 12:06 -- lib +""" + + +class MyCharm(ops.CharmBase): + def _on_start(self, _): + foo = self.unit.get_container('foo') + proc = foo.exec(['ls', '-ll']) + proc.stdin.write("...") + stdout, _ = proc.wait_output() + assert stdout == LS_LL + + +def test_pebble_exec(): + container = scenario.Container( + name='foo', + execs={ + scenario.Exec( + command_prefix=['ls'], + return_code=0, + stdout=LS_LL, + ), + } + ) + state_in = scenario.State(containers={container}) + ctx = scenario.Context( + MyCharm, + meta={"name": "foo", "containers": {"foo": {}}}, + ) + state_out = ctx.run( + ctx.on.pebble_ready(container), + state_in, + ) + assert ctx.exec_history[container.name][0].command == ['ls', '-ll'] + assert ctx.exec_history[container.name][0].stdin == "..." +``` + +Scenario will attempt to find the right `Exec` object by matching the provided +command prefix against the command used in the ops `container.exec()` call. For +example if the command is `['ls', '-ll']` then the searching will be: + + 1. an `Exec` with exactly the same as command prefix, `('ls', '-ll')` + 2. an `Exec` with the command prefix `('ls', )` + 3. an `Exec` with the command prefix `()` + +If none of these are found Scenario will raise an `ExecError`. + +### Pebble Notices + +Pebble can generate notices, which Juju will detect, and wake up the charm to +let it know that something has happened in the container. The most common +use-case is Pebble custom notices, which is a mechanism for the workload +application to trigger a charm event. +- +When the charm is notified, there might be a queue of existing notices, or just +the one that has triggered the event: + +```python +class MyCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + framework.observe(self.on["my-container"].pebble_custom_notice, self._on_notice) + + def _on_notice(self, event): + event.notice.key # == "example.com/c" + for notice in self.unit.get_container("my-container").get_notices(): + ... + +ctx = scenario.Context(MyCharm, meta={"name": "foo", "containers": {"my-container": {}}}) +notices = [ + scenario.Notice(key="example.com/a", occurrences=10), + scenario.Notice(key="example.com/b", last_data={"bar": "baz"}), + scenario.Notice(key="example.com/c"), +] +container = scenario.Container("my-container", notices=notices) +state = scenario.State(containers={container}) +ctx.run(ctx.on.pebble_custom_notice(container=container, notice=notices[-1]), state) +``` + +### Pebble Checks + +A Pebble plan can contain checks, and when those checks exceed the configured +failure threshold, or start succeeding again after, Juju will emit a +pebble-check-failed or pebble-check-recovered event. In order to simulate these +events, you need to add a `CheckInfo` to the container. Note that the status of the +check doesn't have to match the event being generated: by the time that Juju +sends a pebble-check-failed event the check might have started passing again. + +```python +ctx = scenario.Context(MyCharm, meta={"name": "foo", "containers": {"my_container": {}}}) +check_info = scenario.CheckInfo("http-check", failures=7, status=ops.pebble.CheckStatus.DOWN) +container = scenario.Container("my_container", check_infos={check_info}) +state = scenario.State(containers={container}) +ctx.run(ctx.on.pebble_check_failed(info=check_info, container=container), state=state) +``` + +## Storage + +If your charm defines `storage` in its metadata, you can use `scenario.Storage` to instruct Scenario to make (mocked) filesystem storage available to the charm at runtime. + +Using the same `get_filesystem` API as `Container`, you can access the temporary directory used by Scenario to mock the filesystem root before and after the scenario runs. + +```python +# Some charm with a 'foo' filesystem-type storage defined in its metadata: +ctx = scenario.Context(MyCharm, meta=MyCharm.META) +storage = scenario.Storage("foo") + +# Setup storage with some content: +(storage.get_filesystem(ctx) / "myfile.txt").write_text("helloworld") + +with ctx(ctx.on.update_status(), scenario.State(storages={storage})) as manager: + foo = manager.charm.model.storages["foo"][0] + loc = foo.location + path = loc / "myfile.txt" + assert path.exists() + assert path.read_text() == "helloworld" + + myfile = loc / "path.py" + myfile.write_text("helloworlds") + +# post-mortem: inspect fs contents. +assert ( + storage.get_filesystem(ctx) / "path.py" +).read_text() == "helloworlds" +``` + +Note that State only wants to know about **attached** storages. A storage which is not attached to the charm can simply be omitted from State and the charm will be none the wiser. + +### Storage-add + +If a charm requests adding more storage instances while handling some event, you can inspect that from the `Context.requested_storage` API. + +```python notest +# In MyCharm._on_foo: +# The charm requests two new "foo" storage instances to be provisioned: +self.model.storages.request("foo", 2) +``` + +From test code, you can inspect that: + +```python notest +ctx = scenario.Context(MyCharm, meta=MyCharm.META) +ctx.run(ctx.on.some_event_that_will_cause_on_foo_to_be_called(), scenario.State()) + +# the charm has requested two 'foo' storages to be provisioned: +assert ctx.requested_storages['foo'] == 2 +``` + +Requesting storages has no other consequence in Scenario. In real life, this request will trigger Juju to provision the storage and execute the charm again with `foo-storage-attached`. +So a natural follow-up Scenario test suite for this case would be: + +```python +ctx = scenario.Context(MyCharm, meta=MyCharm.META) +foo_0 = scenario.Storage('foo') +# The charm is notified that one of the storages it has requested is ready: +ctx.run(ctx.on.storage_attached(foo_0), scenario.State(storages={foo_0})) + +foo_1 = scenario.Storage('foo') +# The charm is notified that the other storage is also ready: +ctx.run(ctx.on.storage_attached(foo_1), scenario.State(storages={foo_0, foo_1})) +``` + +## Ports + +Since `ops 2.6.0`, charms can invoke the `open-port`, `close-port`, and `opened-ports` hook tools to manage the ports opened on the host VM/container. Using the `State.opened_ports` API, you can: + +- simulate a charm run with a port opened by some previous execution +ctx = scenario.Context(MyCharm, meta=MyCharm.META) +ctx.run(ctx.on.start(), scenario.State(opened_ports={scenario.TCPPort(42)})) +``` +- assert that a charm has called `open-port` or `close-port`: +```python +ctx = scenario.Context(PortCharm, meta=MyCharm.META) +state1 = ctx.run(ctx.on.start(), scenario.State()) +assert state1.opened_ports == [scenario.TCPPort(42)] + +state2 = ctx.run(ctx.on.stop(), state1) +assert state2.opened_ports == {} +``` + +## Secrets + +Scenario has secrets. Here's how you use them. + +```python +state = scenario.State( + secrets={ + scenario.Secret( + tracked_content={'key': 'public'}, + latest_content={'key': 'public', 'cert': 'private'}, + ) + } +) +``` + +The only mandatory arguments to Secret is the `tracked_content` dict: a `str:str` +mapping representing the content of the revision. If there is a newer revision +of the content than the one the unit that's handling the event is tracking, then +`latest_content` should also be provided - if it's not, then Scenario assumes +that `latest_content` is the `tracked_content`. If there are other revisions of +the content, simply don't include them: the unit has no way of knowing about +these. + +There are three cases: +- the secret is owned by this app but not this unit, in which case this charm can only manage it if we are the leader +- the secret is owned by this unit, in which case this charm can always manage it (leader or not) +- (default) the secret is not owned by this app nor unit, which means we can't manage it but only view it (this includes user secrets) + +Thus by default, the secret is not owned by **this charm**, but, implicitly, by some unknown 'other charm' (or a user), and that other has granted us view rights. + +The presence of the secret in `State.secrets` entails that we have access to it, either as owners or as grantees. Therefore, if we're not owners, we must be grantees. Absence of a Secret from the known secrets list means we are not entitled to obtaining it in any way. The charm, indeed, shouldn't even know it exists. + +[note] +If this charm does not own the secret, but also it was not granted view rights by the (remote) owner, you model this in Scenario by _not adding it to State.secrets_! The presence of a `Secret` in `State.secrets` means, in other words, that the charm has view rights (otherwise, why would we put it there?). If the charm owns the secret, or is leader, it will _also_ have manage rights on top of view ones. +[/note] + +To specify a secret owned by this unit (or app): + +```python +rel = scenario.Relation("web") +state = scenario.State( + secrets={ + scenario.Secret( + {'key': 'private'}, + owner='unit', # or 'app' + # The secret owner has granted access to the "remote" app over some relation: + remote_grants={rel.id: {"remote"}} + ) + } +) +``` + +To specify a secret owned by some other application, or a user secret, and give this unit (or app) access to it: + +```python +state = scenario.State( + secrets={ + scenario.Secret( + {'key': 'public'}, + # owner=None, which is the default + ) + } +) +``` + +When handling the `secret-expired` and `secret-remove` events, the charm must remove the specified revision of the secret. For `secret-remove`, the revision will no longer be in the `State`, because it's no longer in use (which is why the `secret-remove` event was triggered). To ensure that the charm is removing the secret, check the context for the history of secret removal: + +```python +class SecretCharm(ops.CharmBase): + def __init__(self, framework): + super().__init__(framework) + self.framework.observe(self.on.secret_remove, self._on_secret_remove) + + def _on_secret_remove(self, event): + event.secret.remove_revision(event.revision) + + +ctx = scenario.Context(SecretCharm, meta={"name": "foo"}) +secret = scenario.Secret({"password": "xxxxxxxx"}, owner="app") +old_revision = 42 +state = ctx.run( + ctx.on.secret_remove(secret, revision=old_revision), + scenario.State(leader=True, secrets={secret}) +) +assert ctx.removed_secret_revisions == [old_revision] +``` + +## StoredState + +Scenario can simulate StoredState. You can define it on the input side as: + +```python +class MyCharmType(ops.CharmBase): + my_stored_state = ops.StoredState() + + def __init__(self, framework): + super().__init__(framework) + assert self.my_stored_state.foo == 'bar' # this will pass! + + +state = scenario.State(stored_states={ + scenario.StoredState( + owner_path="MyCharmType", + name="my_stored_state", + content={ + 'foo': 'bar', + 'baz': {42: 42}, + }), + }, +) +``` + +And the charm's runtime will see `self.my_stored_state.foo` and `.baz` as expected. Also, you can run assertions on it on +the output side the same as any other bit of state. + +## Resources + +If your charm requires access to resources, you can make them available to it through `State.resources`. +From the perspective of a 'real' deployed charm, if your charm _has_ resources defined in its metadata, they _must_ be made available to the charm. That is a Juju-enforced constraint: you can't deploy a charm without attaching all resources it needs to it. +However, when testing, this constraint is unnecessarily strict (and it would also mean the great majority of all existing tests would break) since a charm will only notice that a resource is not available when it explicitly asks for it, which not many charms do. + +So, the only consistency-level check we enforce in Scenario when it comes to resource is that if a resource is provided in State, it needs to have been declared in the metadata. + +```python +import pathlib + +ctx = scenario.Context(MyCharm, meta={'name': 'juliette', "resources": {"foo": {"type": "oci-image"}}}) +resource = scenario.Resource(name='foo', path='/path/to/resource.tar') +with ctx(ctx.on.start(), scenario.State(resources={resource})) as manager: + # If the charm, at runtime, were to call self.model.resources.fetch("foo"), it would get '/path/to/resource.tar' back. + path = manager.charm.model.resources.fetch('foo') + assert path == pathlib.Path('/path/to/resource.tar') +``` + +## Model + +Charms don't usually need to be aware of the model in which they are deployed, +but if you need to set the model name or UUID, you can provide a `scenario.Model` +to the state: + +```python +ctx = scenario.Context(MyCharm, meta={"name": "foo"}) +state_in = scenario.State(model=scenario.Model(name="my-model")) +out = ctx.run(ctx.on.start(), state_in) +assert out.model.name == "my-model" +assert out.model.uuid == state_in.model.uuid +``` + +### CloudSpec + +You can set CloudSpec information in the model (only `type` and `name` are required). + +Example: + +```python +import scenario + +cloud_spec=scenario.CloudSpec( + type="lxd", + endpoint="https://127.0.0.1:8443", + credential=scenario.CloudCredential( + auth_type="clientcertificate", + attributes={ + "client-cert": "foo", + "client-key": "bar", + "server-cert": "baz", + }, + ), +) +state = scenario.State( + model=scenario.Model(name="my-vm-model", type="lxd", cloud_spec=cloud_spec), +) +``` + +Then you can access it by `Model.get_cloud_spec()`: + +```python +# charm.py +class MyVMCharm(ops.CharmBase): + def __init__(self, framework: ops.Framework): + super().__init__(framework) + framework.observe(self.on.start, self._on_start) + + def _on_start(self, event: ops.StartEvent): + self.cloud_spec = self.model.get_cloud_spec() +``` + +# Actions + +An action is a special sort of event, even though `ops` handles them almost identically. +In most cases, you'll want to inspect the 'results' of an action, or whether it has failed or +logged something while executing. Many actions don't have a direct effect on the output state. + +How to test actions with scenario: + +## Actions without parameters + +```python +def test_backup_action(): + ctx = scenario.Context(MyCharm) + + # If you didn't declare do_backup in the charm's metadata, + # the `ConsistencyChecker` will slap you on the wrist and refuse to proceed. + state = ctx.run(ctx.on.action("do_backup"), scenario.State()) + + # You can assert on action results and logs using the context: + assert ctx.action_logs == ['baz', 'qux'] + assert ctx.action_results == {'foo': 'bar'} +``` + +## Failing Actions + +If the charm code calls `event.fail()` to indicate that the action has failed, +an `ActionFailed` exception will be raised. This avoids having to include +success checks in every test where the action is successful. + +```python +def test_backup_action_failed(): + ctx = scenario.Context(MyCharm) + + with pytest.raises(ActionFailed) as exc_info: + ctx.run(ctx.on.action("do_backup"), scenario.State()) + assert exc_info.value.message == "sorry, couldn't do the backup" + # The state is also available if that's required: + assert exc_info.value.state.get_container(...) + + # You can still assert action results and logs that occurred as well as the failure: + assert ctx.action_logs == ['baz', 'qux'] + assert ctx.action_results == {'foo': 'bar'} +``` + +## Parametrized Actions + +If the action takes parameters, you can pass those in the call. + +```python +def test_backup_action(): + ctx = scenario.Context(MyCharm) + + # If the parameters (or their type) don't match what is declared in the metadata, + # the `ConsistencyChecker` will slap you on the other wrist. + state = ctx.run( + ctx.on.action("do_backup", params={'a': 'b'}), + scenario.State() + ) + + # ... +``` + +# Deferred events + +Scenario allows you to accurately simulate the Operator Framework's event queue. The event queue is responsible for +keeping track of the deferred events. On the input side, you can verify that if the charm triggers with this and that +event in its queue (they would be there because they had been deferred in the previous run), then the output state is +valid. You generate the deferred data structure using the event's `deferred()` method: + +```python +class MyCharm(ops.CharmBase): + ... + + def _on_update_status(self, event): + event.defer() + + def _on_start(self, event): + event.defer() + + +def test_start_on_deferred_update_status(): + """Test charm execution if a 'start' is dispatched when in the previous run an update-status had been deferred.""" + ctx = scenario.Context(MyCharm) + state_in = scenario.State( + deferred=[ + ctx.on.update_status().deferred(handler=MyCharm._on_update_status) + ] + ) + state_out = ctx.run(ctx.on.start(), state_in) + assert len(state_out.deferred) == 1 + assert state_out.deferred[0].name == 'start' +``` + +On the output side, you can verify that an event that you expect to have been deferred during this trigger, has indeed +been deferred. + +```python +class MyCharm(ops.CharmBase): + ... + + def _on_start(self, event): + event.defer() + + +def test_defer(MyCharm): + out = scenario.Context(MyCharm).run(ctx.on.start(), scenario.State()) + assert len(out.deferred) == 1 + assert out.deferred[0].name == 'start' +``` + +# Live charm introspection + +Scenario is a black-box, state-transition testing framework. It makes it trivial to assert that a status went from A to +B, but not to assert that, in the context of this charm execution, with this state, a certain charm-internal method was called and returned a +given piece of data, or would return this and that _if_ it had been called. + +The Scenario `Context` object can be used as a context manager for this use case specifically: + +```python notest +from charms.bar.lib_name.v1.charm_lib import CharmLib + + +class MyCharm(ops.CharmBase): + META = {"name": "mycharm"} + _stored = ops.StoredState() + + def __init__(self, framework): + super().__init__(framework) + self._stored.set_default(a="a") + self.my_charm_lib = CharmLib() + framework.observe(self.on.start, self._on_start) + + def _on_start(self, event): + self._stored.a = "b" + + +def test_live_charm_introspection(mycharm): + ctx = scenario.Context(mycharm, meta=mycharm.META) + with ctx(ctx.on.start(), scenario.State()) as manager: + # This is your charm instance, after ops has set it up: + charm: MyCharm = manager.charm + + # We can check attributes on nested Objects or the charm itself: + assert charm.my_charm_lib.foo == "foo" + # such as stored state: + assert charm._stored.a == "a" + + # This will tell ops.main to proceed with normal execution and emit the "start" event on the charm: + state_out = manager.run() + + # After that is done, we are handed back control, and we can again do some introspection: + assert charm.my_charm_lib.foo == "bar" + # and check that the charm's internal state is as we expect: + assert charm._stored.a == "b" + + # state_out is, as in regular scenario tests, a State object you can assert on: + assert state_out.unit_status == ... +``` + +Note that you can't call `manager.run()` multiple times: the object is a context that ensures that `ops.main` 'pauses' right +before emitting the event to hand you some introspection hooks, but for the rest this is a regular Scenario test: you +can't emit multiple events in a single charm execution. + +# The virtual charm root + +Before executing the charm, Scenario copies the charm's `/src`, any libs, the metadata, config, and actions `yaml`s to a temporary directory. The +charm will see that temporary directory as its 'root'. This allows us to keep things simple when dealing with metadata that can be +either inferred from the charm type being passed to `Context` or be passed to it as an argument, thereby overriding +the inferred one. This also allows you to test charms defined on the fly, as in: + +```python +class MyCharmType(ops.CharmBase): + pass + + +ctx = scenario.Context(charm_type=MyCharmType, meta={'name': 'my-charm-name'}) +ctx.run(ctx.on.start(), scenario.State()) +``` + +A consequence of this fact is that you have no direct control over the temporary directory that we are creating to put the metadata +you are passing to `.run()` (because `ops` expects it to be a file...). That is, unless you pass your own: + +```python +import tempfile + + +class MyCharmType(ops.CharmBase): + pass + + +td = tempfile.TemporaryDirectory() +ctx = scenario.Context( + charm_type=MyCharmType, + meta={'name': 'my-charm-name'}, + charm_root=td.name +) +state = ctx.run(ctx.on.start(), scenario.State()) +``` + +Do this, and you will be able to set up said directory as you like before the charm is run, as well as verify its +contents after the charm has run. Do keep in mind that any metadata files you create in it will be overwritten by Scenario, and therefore +ignored, if you pass any metadata keys to `Context`. Omit `meta` in the call +above, and Scenario will instead attempt to read metadata from the +temporary directory. + +# Immutability + +All of the data structures in `state`, e.g. `State, Relation, Container`, etc... are implemented as frozen dataclasses. + +This means that all components of the state that goes into a `context.run()` call are not mutated by the call, and the +state that you obtain in return is a different instance, and all parts of it have been (deep)copied. +This ensures that you can do delta-based comparison of states without worrying about them being mutated by Scenario. + +If you want to modify any of these data structures, you will need to either reinstantiate it from scratch, or use +the dataclasses `replace` api. + +```python +import dataclasses + +relation = scenario.Relation('foo', remote_app_data={"1": "2"}) +# make a copy of relation, but with remote_app_data set to {"3": "4"} +relation2 = dataclasses.replace(relation, remote_app_data={"3": "4"}) +``` + +# Consistency checks + +A Scenario, that is, the combination of an event, a state, and a charm, is consistent if it's plausible in JujuLand. For +example, Juju can't emit a `foo-relation-changed` event on your charm unless your charm has declared a `foo` relation +endpoint in its metadata. If that happens, that's a Juju bug. Scenario however assumes that Juju is bug-free, +therefore, so far as we're concerned, that can't happen, and therefore we help you verify that the scenarios you create +are consistent and raise an exception if that isn't so. + +That happens automatically behind the scenes whenever you trigger an event; +`scenario._consistency_checker.check_consistency` is called and verifies that the scenario makes sense. + +## Caveats: + +- False positives: not all checks are implemented yet; more will come. +- False negatives: it is possible that a scenario you know to be consistent is seen as inconsistent. That is probably a + bug in the consistency checker itself; please report it. +- Inherent limitations: if you have a custom event whose name conflicts with a builtin one, the consistency constraints + of the builtin one will apply. For example: if you decide to name your custom event `bar-pebble-ready`, but you are + working on a machine charm or don't have either way a `bar` container in your `metadata.yaml`, Scenario will flag that + as inconsistent. + +## Bypassing the checker + +If you have a clear false negative, are explicitly testing 'edge', inconsistent situations, or for whatever reason the +checker is in your way, you can set the `SCENARIO_SKIP_CONSISTENCY_CHECKS` envvar and skip it altogether. Hopefully you +don't need that. + +# Jhack integrations + +Up until `v5.6.0`, Scenario shipped with a cli tool called `snapshot`, used to interact with a live charm's state. +The functionality [has been moved over to `jhack`](https://github.com/PietroPasotti/jhack/pull/111), +to allow us to keep working on it independently, and to streamline +the profile of Scenario itself as it becomes more broadly adopted and ready for widespread usage. diff --git a/testing/src/ops_scenario.egg-info/SOURCES.txt b/testing/src/ops_scenario.egg-info/SOURCES.txt new file mode 100644 index 000000000..7f1426b79 --- /dev/null +++ b/testing/src/ops_scenario.egg-info/SOURCES.txt @@ -0,0 +1,52 @@ +CHANGES.md +CONTRIBUTING.md +README.md +UPGRADING.md +pyproject.toml +tox.ini +src/ops_scenario.egg-info/PKG-INFO +src/ops_scenario.egg-info/SOURCES.txt +src/ops_scenario.egg-info/dependency_links.txt +src/ops_scenario.egg-info/requires.txt +src/ops_scenario.egg-info/top_level.txt +src/scenario/__init__.py +src/scenario/_consistency_checker.py +src/scenario/_ops_main_mock.py +src/scenario/_runtime.py +src/scenario/context.py +src/scenario/errors.py +src/scenario/logger.py +src/scenario/mocking.py +src/scenario/py.typed +src/scenario/state.py +tests/__init__.py +tests/helpers.py +tests/readme-conftest.py +tests/test_charm_spec_autoload.py +tests/test_consistency_checker.py +tests/test_context.py +tests/test_context_on.py +tests/test_emitted_events_util.py +tests/test_plugin.py +tests/test_runtime.py +tests/test_e2e/__init__.py +tests/test_e2e/test_actions.py +tests/test_e2e/test_cloud_spec.py +tests/test_e2e/test_config.py +tests/test_e2e/test_deferred.py +tests/test_e2e/test_event.py +tests/test_e2e/test_juju_log.py +tests/test_e2e/test_manager.py +tests/test_e2e/test_network.py +tests/test_e2e/test_pebble.py +tests/test_e2e/test_play_assertions.py +tests/test_e2e/test_ports.py +tests/test_e2e/test_relations.py +tests/test_e2e/test_resource.py +tests/test_e2e/test_rubbish_events.py +tests/test_e2e/test_secrets.py +tests/test_e2e/test_state.py +tests/test_e2e/test_status.py +tests/test_e2e/test_storage.py +tests/test_e2e/test_stored_state.py +tests/test_e2e/test_vroot.py \ No newline at end of file diff --git a/testing/src/ops_scenario.egg-info/dependency_links.txt b/testing/src/ops_scenario.egg-info/dependency_links.txt new file mode 100644 index 000000000..8b1378917 --- /dev/null +++ b/testing/src/ops_scenario.egg-info/dependency_links.txt @@ -0,0 +1 @@ + diff --git a/testing/src/ops_scenario.egg-info/requires.txt b/testing/src/ops_scenario.egg-info/requires.txt new file mode 100644 index 000000000..99af883cb --- /dev/null +++ b/testing/src/ops_scenario.egg-info/requires.txt @@ -0,0 +1,2 @@ +ops~=2.17 +PyYAML>=6.0.1 diff --git a/testing/src/ops_scenario.egg-info/top_level.txt b/testing/src/ops_scenario.egg-info/top_level.txt new file mode 100644 index 000000000..de43fb96d --- /dev/null +++ b/testing/src/ops_scenario.egg-info/top_level.txt @@ -0,0 +1 @@ +scenario