diff --git a/lib/charms/grafana_k8s/v0/grafana_source.py b/lib/charms/grafana_k8s/v0/grafana_source.py index f3492be2..774bcd7e 100644 --- a/lib/charms/grafana_k8s/v0/grafana_source.py +++ b/lib/charms/grafana_k8s/v0/grafana_source.py @@ -154,6 +154,8 @@ def __init__(self, *args): ) from ops.model import Relation +from charms.observability_libs.v0.juju_topology import JujuTopology + # The unique Charmhub library identifier, never change it LIBID = "974705adb86f40228298156e34b460dc" @@ -162,7 +164,7 @@ def __init__(self, *args): # Increment this PATCH version before using `charmcraft publish-lib` or reset # to 0 if you are raising the major API version -LIBPATCH = 22 +LIBPATCH = 23 logger = logging.getLogger(__name__) @@ -432,13 +434,22 @@ def update_source(self, source_url: Optional[str] = ""): def get_source_uids(self) -> Dict[str, Dict[str, str]]: """Get the datasource UID(s) assigned by the remote end(s) to this datasource. - Returns a mapping from remote application names to unit names to datasource uids. + Returns a mapping from remote application UIDs to unit names to datasource uids. """ uids = {} for rel in self._charm.model.relations.get(self._relation_name, []): if not rel: continue - uids[rel.app.name] = json.loads(rel.data[rel.app]["datasource_uids"]) + app_databag = rel.data[rel.app] + grafana_uid = app_databag.get("grafana_uid") + if not grafana_uid: + logger.warning( + "remote end is using an old grafana_datasource interface: " + "`grafana_uid` field not found." + ) + continue + + uids[grafana_uid] = json.loads(app_databag.get("datasource_uids", "{}")) return uids def _set_sources_from_event(self, event: RelationJoinedEvent) -> None: @@ -568,6 +579,15 @@ def _publish_source_uids(self, rel: Relation, uids: Dict[str, str]): Assumes only leader unit will call this method """ + juju_topology = JujuTopology.from_charm(self._charm) + unique_grafana_name = "juju_{}_{}_{}_{}".format( + juju_topology.model, + juju_topology.model_uuid, + juju_topology.application, + juju_topology.unit.split("/")[1], + ) + + rel.data[self._charm.app]["grafana_uid"] = unique_grafana_name rel.data[self._charm.app]["datasource_uids"] = json.dumps(uids) def _get_source_config(self, rel: Relation): @@ -782,4 +802,4 @@ def get_peer_data(self, key: str) -> Any: return {} data = self._charm.peers.data[self._charm.app].get(key, "") # type: ignore[attr-defined] - return json.loads(data) if data else {} + return json.loads(data) if data else {} \ No newline at end of file diff --git a/lib/charms/observability_libs/v0/juju_topology.py b/lib/charms/observability_libs/v0/juju_topology.py new file mode 100644 index 00000000..a79e5d43 --- /dev/null +++ b/lib/charms/observability_libs/v0/juju_topology.py @@ -0,0 +1,301 @@ +# Copyright 2022 Canonical Ltd. +# See LICENSE file for licensing details. +"""## Overview. + +This document explains how to use the `JujuTopology` class to +create and consume topology information from Juju in a consistent manner. + +The goal of the Juju topology is to uniquely identify a piece +of software running across any of your Juju-managed deployments. +This is achieved by combining the following four elements: + +- Model name +- Model UUID +- Application name +- Unit identifier + + +For a more in-depth description of the concept, as well as a +walk-through of it's use-case in observability, see +[this blog post](https://juju.is/blog/model-driven-observability-part-2-juju-topology-metrics) +on the Juju blog. + +## Library Usage + +This library may be used to create and consume `JujuTopology` objects. +The `JujuTopology` class provides three ways to create instances: + +### Using the `from_charm` method + +Enables instantiation by supplying the charm as an argument. When +creating topology objects for the current charm, this is the recommended +approach. + +```python +topology = JujuTopology.from_charm(self) +``` + +### Using the `from_dict` method + +Allows for instantion using a dictionary of relation data, like the +`scrape_metadata` from Prometheus or the labels of an alert rule. When +creating topology objects for remote charms, this is the recommended +approach. + +```python +scrape_metadata = json.loads(relation.data[relation.app].get("scrape_metadata", "{}")) +topology = JujuTopology.from_dict(scrape_metadata) +``` + +### Using the class constructor + +Enables instantiation using whatever values you want. While this +is useful in some very specific cases, this is almost certainly not +what you are looking for as setting these values manually may +result in observability metrics which do not uniquely identify a +charm in order to provide accurate usage reporting, alerting, +horizontal scaling, or other use cases. + +```python +topology = JujuTopology( + model="some-juju-model", + model_uuid="00000000-0000-0000-0000-000000000001", + application="fancy-juju-application", + unit="fancy-juju-application/0", + charm_name="fancy-juju-application-k8s", +) +``` + +""" +from collections import OrderedDict +from typing import Dict, List, Optional +from uuid import UUID + +# The unique Charmhub library identifier, never change it +LIBID = "bced1658f20f49d28b88f61f83c2d232" + +LIBAPI = 0 +LIBPATCH = 6 + + +class InvalidUUIDError(Exception): + """Invalid UUID was provided.""" + + def __init__(self, uuid: str): + self.message = "'{}' is not a valid UUID.".format(uuid) + super().__init__(self.message) + + +class JujuTopology: + """JujuTopology is used for storing, generating and formatting juju topology information. + + DEPRECATED: This class is deprecated. Use `pip install cosl` and + `from cosl.juju_topology import JujuTopology` instead. + """ + + def __init__( + self, + model: str, + model_uuid: str, + application: str, + unit: Optional[str] = None, + charm_name: Optional[str] = None, + ): + """Build a JujuTopology object. + + A `JujuTopology` object is used for storing and transforming + Juju topology information. This information is used to + annotate Prometheus scrape jobs and alert rules. Such + annotation when applied to scrape jobs helps in identifying + the source of the scrapped metrics. On the other hand when + applied to alert rules topology information ensures that + evaluation of alert expressions is restricted to the source + (charm) from which the alert rules were obtained. + + Args: + model: a string name of the Juju model + model_uuid: a globally unique string identifier for the Juju model + application: an application name as a string + unit: a unit name as a string + charm_name: name of charm as a string + """ + if not self.is_valid_uuid(model_uuid): + raise InvalidUUIDError(model_uuid) + + self._model = model + self._model_uuid = model_uuid + self._application = application + self._charm_name = charm_name + self._unit = unit + + def is_valid_uuid(self, uuid): + """Validate the supplied UUID against the Juju Model UUID pattern. + + Args: + uuid: string that needs to be checked if it is valid v4 UUID. + + Returns: + True if parameter is a valid v4 UUID, False otherwise. + """ + try: + return str(UUID(uuid, version=4)) == uuid + except (ValueError, TypeError): + return False + + @classmethod + def from_charm(cls, charm): + """Creates a JujuTopology instance by using the model data available on a charm object. + + Args: + charm: a `CharmBase` object for which the `JujuTopology` will be constructed + Returns: + a `JujuTopology` object. + """ + return cls( + model=charm.model.name, + model_uuid=charm.model.uuid, + application=charm.model.app.name, + unit=charm.model.unit.name, + charm_name=charm.meta.name, + ) + + @classmethod + def from_dict(cls, data: dict): + """Factory method for creating `JujuTopology` children from a dictionary. + + Args: + data: a dictionary with five keys providing topology information. The keys are + - "model" + - "model_uuid" + - "application" + - "unit" + - "charm_name" + `unit` and `charm_name` may be empty, but will result in more limited + labels. However, this allows us to support charms without workloads. + + Returns: + a `JujuTopology` object. + """ + return cls( + model=data["model"], + model_uuid=data["model_uuid"], + application=data["application"], + unit=data.get("unit", ""), + charm_name=data.get("charm_name", ""), + ) + + def as_dict( + self, + *, + remapped_keys: Optional[Dict[str, str]] = None, + excluded_keys: Optional[List[str]] = None, + ) -> OrderedDict: + """Format the topology information into an ordered dict. + + Keeping the dictionary ordered is important to be able to + compare dicts without having to resort to deep comparisons. + + Args: + remapped_keys: A dictionary mapping old key names to new key names, + which will be substituted when invoked. + excluded_keys: A list of key names to exclude from the returned dict. + uuid_length: The length to crop the UUID to. + """ + ret = OrderedDict( + [ + ("model", self.model), + ("model_uuid", self.model_uuid), + ("application", self.application), + ("unit", self.unit), + ("charm_name", self.charm_name), + ] + ) + if excluded_keys: + ret = OrderedDict({k: v for k, v in ret.items() if k not in excluded_keys}) + + if remapped_keys: + ret = OrderedDict( + (remapped_keys.get(k), v) if remapped_keys.get(k) else (k, v) for k, v in ret.items() # type: ignore + ) + + return ret + + @property + def identifier(self) -> str: + """Format the topology information into a terse string. + + This crops the model UUID, making it unsuitable for comparisons against + anything but other identifiers. Mainly to be used as a display name or file + name where long strings might become an issue. + + >>> JujuTopology( \ + model = "a-model", \ + model_uuid = "00000000-0000-4000-8000-000000000000", \ + application = "some-app", \ + unit = "some-app/1" \ + ).identifier + 'a-model_00000000_some-app' + """ + parts = self.as_dict( + excluded_keys=["unit", "charm_name"], + ) + + parts["model_uuid"] = self.model_uuid_short + values = parts.values() + + return "_".join([str(val) for val in values]).replace("/", "_") + + @property + def label_matcher_dict(self) -> Dict[str, str]: + """Format the topology information into a dict with keys having 'juju_' as prefix. + + Relabelled topology never includes the unit as it would then only match + the leader unit (ie. the unit that produced the dict). + """ + items = self.as_dict( + remapped_keys={"charm_name": "charm"}, + excluded_keys=["unit"], + ).items() + + return {"juju_{}".format(key): value for key, value in items if value} + + @property + def label_matchers(self) -> str: + """Format the topology information into a promql/logql label matcher string. + + Topology label matchers should never include the unit as it + would then only match the leader unit (ie. the unit that + produced the matchers). + """ + items = self.label_matcher_dict.items() + return ", ".join(['{}="{}"'.format(key, value) for key, value in items if value]) + + @property + def model(self) -> str: + """Getter for the juju model value.""" + return self._model + + @property + def model_uuid(self) -> str: + """Getter for the juju model uuid value.""" + return self._model_uuid + + @property + def model_uuid_short(self) -> str: + """Getter for the juju model value, truncated to the first eight letters.""" + return self._model_uuid[:8] + + @property + def application(self) -> str: + """Getter for the juju application value.""" + return self._application + + @property + def charm_name(self) -> Optional[str]: + """Getter for the juju charm name value.""" + return self._charm_name + + @property + def unit(self) -> Optional[str]: + """Getter for the juju unit value.""" + return self._unit diff --git a/metadata.yaml b/metadata.yaml index f77f7ef4..b94ccc0c 100644 --- a/metadata.yaml +++ b/metadata.yaml @@ -41,7 +41,11 @@ provides: interface: grafana_dashboard receive-remote-write: interface: prometheus_remote_write - + send-datasource: + interface: grafana_datasource_exchange + description: | + Integration to share with other COS components this charm's grafana datasources, and receive theirs. + requires: metrics-endpoint: interface: prometheus_scrape diff --git a/requirements.txt b/requirements.txt index 7c4fe61a..08a78f21 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,8 +1,7 @@ -cosl +cosl>=0.0.46 cryptography jsonschema -# pinned to 2.15 as 2.16 breaks our scenario tests and 2.17 breaks our unittests -ops == 2.15 +ops pyaml requests lightkube >= 0.11 diff --git a/src/charm.py b/src/charm.py index 355b6474..44ebc3cc 100755 --- a/src/charm.py +++ b/src/charm.py @@ -10,7 +10,7 @@ import socket import subprocess from pathlib import Path -from typing import Dict, Optional, Tuple, TypedDict, cast +from typing import Dict, List, Optional, Tuple, TypedDict, cast from urllib.parse import urlparse import yaml @@ -43,6 +43,7 @@ IngressPerUnitRevokedForUnitEvent, ) from cosl import JujuTopology +from cosl.interfaces.datasource_exchange import DatasourceDict, DatasourceExchange from lightkube.core.client import Client from lightkube.core.exceptions import ApiError as LightkubeApiError from lightkube.resources.core_v1 import PersistentVolumeClaim, Pod @@ -232,6 +233,11 @@ def __init__(self, *args): self.charm_tracing_endpoint, self.server_cert = charm_tracing_config( self.charm_tracing, self._ca_cert_path ) + self.datasource_exchange = DatasourceExchange( + self, + provider_endpoint="send-datasource", + requirer_endpoint=None, + ) self.framework.observe(self.on.prometheus_pebble_ready, self._on_pebble_ready) self.framework.observe(self.on.config_changed, self._configure) @@ -246,6 +252,10 @@ def __init__(self, *args): self.framework.observe(self.alertmanager_consumer.on.cluster_changed, self._configure) self.framework.observe(self.resources_patch.on.patch_failed, self._on_k8s_patch_failed) self.framework.observe(self.on.validate_configuration_action, self._on_validate_config) + self.framework.observe(self.on.send_datasource_relation_changed, self._configure) + self.framework.observe(self.on.send_datasource_relation_departed, self._configure) + self.framework.observe(self.on.grafana_source_relation_changed, self._configure) + self.framework.observe(self.on.grafana_source_relation_departed, self._configure) self.framework.observe(self.on.collect_unit_status, self._on_collect_unit_status) def _on_collect_unit_status(self, event: CollectStatusEvent): @@ -576,6 +586,7 @@ def _configure(self, _): ) self.remote_write_provider.update_endpoint() self.catalogue.update_item(item=self._catalogue_item) + self._update_datasource_exchange() try: # Need to reload if config or alerts changed. @@ -1106,6 +1117,18 @@ def _push(self, path, contents): """Push file to container, creating subdirs as necessary.""" self.container.push(path, contents, make_dirs=True, encoding="utf-8") + def _update_datasource_exchange(self) -> None: + """Update the grafana-datasource-exchange relations.""" + grafana_uids_to_units_to_uids = self.grafana_source_provider.get_source_uids() + raw_datasources: List[DatasourceDict] = [] + + for grafana_uid, ds_uids in grafana_uids_to_units_to_uids.items(): + for _unit_name, ds_uid in ds_uids.items(): + raw_datasources.append( + {"type": "prometheus", "uid": ds_uid, "grafana_uid": grafana_uid} + ) + self.datasource_exchange.publish(datasources=raw_datasources) + @property def workload_tracing_endpoint(self) -> Optional[str]: """Tempo endpoint for workload tracing.""" diff --git a/tests/interface/conftest.py b/tests/interface/conftest.py index f2bbbac4..5be81991 100644 --- a/tests/interface/conftest.py +++ b/tests/interface/conftest.py @@ -2,11 +2,13 @@ # # See LICENSE file for licensing details. # from unittest.mock import patch +import json from unittest.mock import patch import pytest +from charms.tempo_coordinator_k8s.v0.charm_tracing import charm_tracing_disabled from interface_tester import InterfaceTester -from scenario import Container, ExecOutput, State +from scenario import Container, Exec, Relation, State from charm import PrometheusCharm @@ -27,16 +29,34 @@ def prometheus_charm(): _promtool_check_config=lambda *_: ("stdout", ""), _prometheus_version="0.1.0", ): - yield PrometheusCharm + with charm_tracing_disabled(): + yield PrometheusCharm + + +prometheus_container = Container( + name="prometheus", + can_connect=True, + execs={Exec(["update-ca-certificates", "--fresh"], return_code=0, stdout="")}, +) + +grafana_source_relation = Relation( + "grafana-source", + remote_app_data={ + "datasource_uids": json.dumps({"prometheus/0": "01234"}), + "grafana_uid": "5678", + }, +) + +grafana_datasource_exchange_relation = Relation( + "send-datasource", + remote_app_data={ + "datasources": json.dumps([{"type": "prometheus", "uid": "01234", "grafana_uid": "5678"}]) + }, +) def begin_with_initial_hooks_isolated() -> State: - container = Container( - "prometheus", - can_connect=True, - exec_mock={("update-ca-certificates", "--fresh"): ExecOutput(return_code=0, stdout="")}, - ) - state = State(containers=[container], leader=True) + state = State(containers=[prometheus_container], leader=True) return state @@ -47,3 +67,16 @@ def interface_tester(interface_tester: InterfaceTester, prometheus_charm): state_template=begin_with_initial_hooks_isolated(), ) yield interface_tester + + +@pytest.fixture +def grafana_datasource_exchange_tester(interface_tester: InterfaceTester, prometheus_charm): + interface_tester.configure( + charm_type=prometheus_charm, + state_template=State( + leader=True, + containers=[prometheus_container], + relations=[grafana_source_relation, grafana_datasource_exchange_relation], + ), + ) + yield interface_tester diff --git a/tests/interface/test_grafana_datasource_exchange.py b/tests/interface/test_grafana_datasource_exchange.py new file mode 100644 index 00000000..b2301dee --- /dev/null +++ b/tests/interface/test_grafana_datasource_exchange.py @@ -0,0 +1,13 @@ +# Copyright 2024 Canonical Ltd. +# See LICENSE file for licensing details. +from interface_tester import InterfaceTester + + +def test_grafana_datasource_exchange_v0_interface( + grafana_datasource_exchange_tester: InterfaceTester, +): + grafana_datasource_exchange_tester.configure( + interface_name="grafana_datasource_exchange", + interface_version=0, + ) + grafana_datasource_exchange_tester.run() diff --git a/tests/scenario/conftest.py b/tests/scenario/conftest.py index 10927833..3c3caf9e 100644 --- a/tests/scenario/conftest.py +++ b/tests/scenario/conftest.py @@ -4,7 +4,8 @@ from unittest.mock import patch import pytest -from scenario import Context +from charms.tempo_coordinator_k8s.v0.charm_tracing import charm_tracing_disabled +from scenario import Container, Context, Exec from charm import PrometheusCharm @@ -25,9 +26,19 @@ def prometheus_charm(): _promtool_check_config=lambda *_: ("stdout", ""), _prometheus_version="0.1.0", ): - yield PrometheusCharm + with charm_tracing_disabled(): + yield PrometheusCharm @pytest.fixture(scope="function") def context(prometheus_charm): return Context(charm_type=prometheus_charm, juju_version="3.0.3") + + +@pytest.fixture(scope="function") +def prometheus_container(): + return Container( + "prometheus", + can_connect=True, + execs={Exec(["update-ca-certificates", "--fresh"], return_code=0, stdout="")}, + ) diff --git a/tests/scenario/helpers.py b/tests/scenario/helpers.py index 492adb8a..36dee796 100644 --- a/tests/scenario/helpers.py +++ b/tests/scenario/helpers.py @@ -1,37 +1,39 @@ # Copyright 2023 Canonical Ltd. # See LICENSE file for licensing details. -from scenario import Container, Context, ExecOutput, Network, PeerRelation, Relation, State +import dataclasses + +from scenario import Container, Context, Exec, PeerRelation, Relation, State def begin_with_initial_hooks_isolated(context: Context, *, leader: bool = True) -> State: container = Container( "prometheus", can_connect=False, - exec_mock={("update-ca-certificates", "--fresh"): ExecOutput(return_code=0, stdout="")}, + execs={Exec(["update-ca-certificates", "--fresh"], return_code=0, stdout="")}, ) state = State(containers=[container]) peer_rel = PeerRelation("prometheus-peers") - state = context.run("install", state) + state = context.run(context.on.install(), state) - state = state.replace(relations=[peer_rel]) - state = context.run(peer_rel.created_event, state) + state = dataclasses.replace(state, relations=[peer_rel]) + state = context.run(context.on.relation_created(peer_rel), state) if leader: - state = state.replace(leader=True) - state = context.run("leader-elected", state) + state = dataclasses.replace(state, leader=True) + state = context.run(context.on.leader_elected(), state) else: - state = state.replace(leader=False) - state = context.run("leader-settings-changed", state) + state = dataclasses.replace(state, leader=False) + state = context.run(context.on.leader_elected(), state) - state = context.run("config-changed", state) + state = context.run(context.on.config_changed(), state) - container = container.replace(can_connect=True) - state = state.replace(containers=[container]) - state = context.run(container.pebble_ready_event, state) + container = dataclasses.replace(container, can_connect=True) + state = dataclasses.replace(state, containers=[container]) + state = context.run(context.on.pebble_ready(container), state) - state = context.run("start", state) + state = context.run(context.on.start(), state) return state @@ -39,20 +41,22 @@ def begin_with_initial_hooks_isolated(context: Context, *, leader: bool = True) def add_relation_sequence(context: Context, state: State, relation: Relation): """Helper to simulate a relation-added sequence.""" # TODO consider adding to scenario.sequences - state_with_relation = state.replace( - relations=state.relations + [relation], - networks=state.networks + [Network.default(relation.endpoint)], + state_with_relation = dataclasses.replace( + state, + relations=state.relations.union([relation]), + ) + state_after_relation_created = context.run( + context.on.relation_created(relation), state_with_relation ) - state_after_relation_created = context.run(relation.created_event, state_with_relation) # relation is not mutated! relation_1 = state_after_relation_created.get_relations(relation.endpoint)[0] state_after_relation_joined = context.run( - relation_1.joined_event, state_after_relation_created + context.on.relation_joined(relation_1), state_after_relation_created ) relation_2 = state_after_relation_joined.get_relations(relation.endpoint)[0] state_after_relation_changed = context.run( - relation_2.changed_event, state_after_relation_joined + context.on.relation_changed(relation_2), state_after_relation_joined ) return state_after_relation_changed diff --git a/tests/scenario/test_alert_expression_labels.py b/tests/scenario/test_alert_expression_labels.py index 3c77759d..10de975d 100644 --- a/tests/scenario/test_alert_expression_labels.py +++ b/tests/scenario/test_alert_expression_labels.py @@ -4,7 +4,7 @@ import json import yaml -from scenario import Container, ExecOutput, Relation, State +from scenario import Container, Exec, Relation, State def test_alert_expression_labels(context): @@ -42,10 +42,10 @@ def test_alert_expression_labels(context): container = Container( name="prometheus", can_connect=True, - exec_mock={("update-ca-certificates", "--fresh"): ExecOutput(return_code=0, stdout="")}, + execs={Exec(["update-ca-certificates", "--fresh"], return_code=0, stdout="")}, ) state = State(containers=[container], relations=[remote_write_relation]) - context.run(event=remote_write_relation.changed_event, state=state) + context.run(context.on.relation_changed(remote_write_relation), state=state) rules_file = ( container.get_filesystem(context) / "etc/prometheus/rules/juju_foobar-model_d07df316_remote-app.rules" diff --git a/tests/scenario/test_brute_isolated.py b/tests/scenario/test_brute_isolated.py index a7d074f8..04942265 100644 --- a/tests/scenario/test_brute_isolated.py +++ b/tests/scenario/test_brute_isolated.py @@ -10,10 +10,10 @@ def test_startup_shutdown_sequence(context: Context): state = begin_with_initial_hooks_isolated(context) - state = context.run("update-status", state) + state = context.run(context.on.update_status(), state) for peer_rel in state.get_relations("replicas"): - state = context.run(peer_rel.departed_event, state) + state = context.run(context.on.relation_departed(peer_rel), state) - state = context.run("stop", state) - context.run("remove", state) + state = context.run(context.on.stop(), state) + context.run(context.on.remove(), state) diff --git a/tests/scenario/test_datasource_exchange.py b/tests/scenario/test_datasource_exchange.py new file mode 100644 index 00000000..cc65ac51 --- /dev/null +++ b/tests/scenario/test_datasource_exchange.py @@ -0,0 +1,78 @@ +# Copyright 2024 Canonical Ltd. +# See LICENSE file for licensing details. + +import json + +import pytest +from cosl.interfaces.datasource_exchange import ( + DatasourceExchange, + DSExchangeAppData, + GrafanaDatasource, +) +from scenario import Relation, State + +from charm import PrometheusCharm + +ds_tempo = [ + {"type": "tempo", "uid": "3", "grafana_uid": "4"}, +] + +ds_loki = [ + {"type": "loki", "uid": "8", "grafana_uid": "9"}, +] + +loki_dsx = Relation( + "send-datasource", + remote_app_data=DSExchangeAppData(datasources=json.dumps(ds_loki)).dump(), +) +tempo_dsx = Relation( + "send-datasource", + remote_app_data=DSExchangeAppData(datasources=json.dumps(ds_tempo)).dump(), +) + +ds = Relation( + "grafana-source", + remote_app_data={ + "grafana_uid": "9", + "datasource_uids": json.dumps({"prometheus/0": "1234"}), + }, +) + + +@pytest.mark.parametrize("event_type", ("changed", "created", "joined")) +@pytest.mark.parametrize("relation_to_observe", (ds, loki_dsx, tempo_dsx)) +def test_datasource_send(context, prometheus_container, relation_to_observe, event_type): + + state_in = State( + relations=[ + ds, + loki_dsx, + tempo_dsx, + ], + containers=[prometheus_container], + leader=True, + ) + + # WHEN we receive a datasource-related event + with context( + getattr(context.on, f"relation_{event_type}")(relation_to_observe), state_in + ) as mgr: + charm: PrometheusCharm = mgr.charm + # THEN we can find all received datasource uids + dsx: DatasourceExchange = charm.datasource_exchange + received = dsx.received_datasources + assert received == ( + GrafanaDatasource(type="tempo", uid="3", grafana_uid="4"), + GrafanaDatasource(type="loki", uid="8", grafana_uid="9"), + ) + state_out = mgr.run() + + # AND THEN we publish our own datasource information to mimir and tempo + published_dsx_loki = state_out.get_relation(loki_dsx.id).local_app_data + published_dsx_tempo = state_out.get_relation(tempo_dsx.id).local_app_data + assert published_dsx_tempo == published_dsx_loki + assert json.loads(published_dsx_tempo["datasources"])[0] == { + "type": "prometheus", + "uid": "1234", + "grafana_uid": "9", + } diff --git a/tests/scenario/test_server_scheme.py b/tests/scenario/test_server_scheme.py index c70db15b..317220ad 100644 --- a/tests/scenario/test_server_scheme.py +++ b/tests/scenario/test_server_scheme.py @@ -34,9 +34,9 @@ def initial_state(self, context, fqdn, leader) -> State: # Add relations rels = [ - Relation("self-metrics-endpoint", relation_id=10), # external self-monitoring - Relation("grafana-source", relation_id=11), # grafana - Relation("receive-remote-write", relation_id=12), # grafana-agent + Relation("self-metrics-endpoint", id=10), # external self-monitoring + Relation("grafana-source", id=11), # grafana + Relation("receive-remote-write", id=12), # grafana-agent ] for rel in rels: state = add_relation_sequence(context, state, rel) @@ -56,7 +56,7 @@ def test_pebble_layer_scheme_becomes_https_if_tls_relation_added( # WHEN a tls_certificates relation joins ca = Relation( "certificates", - relation_id=100, + id=100, remote_app_data={ "certificates": json.dumps( [ diff --git a/tox.ini b/tox.ini index 34701189..b1c393ee 100644 --- a/tox.ini +++ b/tox.ini @@ -96,14 +96,9 @@ allowlist_externals = [testenv:scenario] description = Scenario tests deps = - cosl - ops >= 2.5.0 pytest - ops-scenario >=5.1,<6.0 + ops[testing] -r{toxinidir}/requirements.txt - opentelemetry-exporter-otlp-proto-http==1.21.0 # PYDEPS for tracing - importlib-metadata==6.0.0 # PYDEPS for tracing - pydantic>=2 # PYDEPS for tracing commands = pytest -v --tb native {[vars]tst_path}/scenario --log-cli-level=INFO -s {posargs} @@ -112,7 +107,6 @@ commands = description = Run interface tests deps = pytest - ops-scenario>=5.3.1 pytest-interface-tester > 2.0.0 -r{toxinidir}/requirements.txt commands =