blob: 1ad1e266415e7ffa32c45220f4e9e3f55a734cb5 [file] [log] [blame]
sousaeducab58cb2020-11-04 18:48:17 +00001#!/usr/bin/env python3
David Garcia49379ce2021-02-24 13:48:22 +01002# Copyright 2021 Canonical Ltd.
sousaeducab58cb2020-11-04 18:48:17 +00003#
4# Licensed under the Apache License, Version 2.0 (the "License"); you may
5# not use this file except in compliance with the License. You may obtain
6# a copy of the License at
7#
8# http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13# License for the specific language governing permissions and limitations
14# under the License.
15#
16# For those usages not covered by the Apache License, Version 2.0 please
17# contact: legal@canonical.com
18#
19# To get in touch with the maintainers, please contact:
20# osm-charmers@lists.launchpad.net
21##
22
David Garcia49379ce2021-02-24 13:48:22 +010023# pylint: disable=E0213
24
25
sousaeducab58cb2020-11-04 18:48:17 +000026import logging
David Garciaaccf1172021-05-10 12:59:33 +020027import re
sousaedu996a5602021-05-03 00:22:43 +020028from typing import NoReturn, Optional
sousaeducab58cb2020-11-04 18:48:17 +000029
sousaeducab58cb2020-11-04 18:48:17 +000030from ops.main import main
David Garcia49379ce2021-02-24 13:48:22 +010031from opslib.osm.charm import CharmedOsmBase, RelationsMissing
David Garciac753dc52021-03-17 15:28:47 +010032from opslib.osm.interfaces.kafka import KafkaClient
33from opslib.osm.interfaces.mongo import MongoClient
David Garciaaccf1172021-05-10 12:59:33 +020034from opslib.osm.interfaces.mysql import MysqlClient
David Garcia49379ce2021-02-24 13:48:22 +010035from opslib.osm.pod import (
36 ContainerV3Builder,
37 PodSpecV3Builder,
38)
David Garciac753dc52021-03-17 15:28:47 +010039from opslib.osm.validator import ModelValidator, validator
David Garcia49379ce2021-02-24 13:48:22 +010040
sousaeducab58cb2020-11-04 18:48:17 +000041
42logger = logging.getLogger(__name__)
43
David Garcia49379ce2021-02-24 13:48:22 +010044PORT = 9999
David Garciaaccf1172021-05-10 12:59:33 +020045DEFAULT_MYSQL_DATABASE = "pol"
sousaeducab58cb2020-11-04 18:48:17 +000046
47
David Garcia49379ce2021-02-24 13:48:22 +010048class ConfigModel(ModelValidator):
49 log_level: str
sousaedu996a5602021-05-03 00:22:43 +020050 mongodb_uri: Optional[str]
David Garciaaccf1172021-05-10 12:59:33 +020051 mysql_uri: Optional[str]
sousaeducab58cb2020-11-04 18:48:17 +000052
David Garcia49379ce2021-02-24 13:48:22 +010053 @validator("log_level")
54 def validate_log_level(cls, v):
55 if v not in {"INFO", "DEBUG"}:
56 raise ValueError("value must be INFO or DEBUG")
57 return v
sousaeducab58cb2020-11-04 18:48:17 +000058
sousaedu996a5602021-05-03 00:22:43 +020059 @validator("mongoddb_uri")
60 def validate_mongodb_uri(cls, v):
61 if v and not v.startswith("mongodb://"):
62 raise ValueError("mongodb_uri is not properly formed")
63 return v
64
David Garciaaccf1172021-05-10 12:59:33 +020065 @validator("mysql_uri")
66 def validate_mysql_uri(cls, v):
67 pattern = re.compile("^mysql:\/\/.*:.*@.*:\d+\/.*$") # noqa: W605
68 if v and not pattern.search(v):
69 raise ValueError("mysql_uri is not properly formed")
70 return v
71
sousaeducab58cb2020-11-04 18:48:17 +000072
David Garcia49379ce2021-02-24 13:48:22 +010073class PolCharm(CharmedOsmBase):
sousaeducab58cb2020-11-04 18:48:17 +000074 def __init__(self, *args) -> NoReturn:
David Garcia49379ce2021-02-24 13:48:22 +010075 super().__init__(*args, oci_image="image")
sousaeducab58cb2020-11-04 18:48:17 +000076
David Garcia49379ce2021-02-24 13:48:22 +010077 self.kafka_client = KafkaClient(self, "kafka")
78 self.framework.observe(self.on["kafka"].relation_changed, self.configure_pod)
79 self.framework.observe(self.on["kafka"].relation_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +000080
David Garcia49379ce2021-02-24 13:48:22 +010081 self.mongodb_client = MongoClient(self, "mongodb")
82 self.framework.observe(self.on["mongodb"].relation_changed, self.configure_pod)
83 self.framework.observe(self.on["mongodb"].relation_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +000084
David Garciaaccf1172021-05-10 12:59:33 +020085 self.mysql_client = MysqlClient(self, "mysql")
86 self.framework.observe(self.on["mysql"].relation_changed, self.configure_pod)
87 self.framework.observe(self.on["mysql"].relation_broken, self.configure_pod)
88
David Garcia49379ce2021-02-24 13:48:22 +010089 def _check_missing_dependencies(self, config: ConfigModel):
90 missing_relations = []
sousaeducab58cb2020-11-04 18:48:17 +000091
David Garcia49379ce2021-02-24 13:48:22 +010092 if self.kafka_client.is_missing_data_in_unit():
93 missing_relations.append("kafka")
sousaedu996a5602021-05-03 00:22:43 +020094 if not config.mongodb_uri and self.mongodb_client.is_missing_data_in_unit():
David Garcia49379ce2021-02-24 13:48:22 +010095 missing_relations.append("mongodb")
David Garciaaccf1172021-05-10 12:59:33 +020096 if not config.mysql_uri and self.mysql_client.is_missing_data_in_unit():
97 missing_relations.append("mysql")
David Garcia49379ce2021-02-24 13:48:22 +010098 if missing_relations:
99 raise RelationsMissing(missing_relations)
sousaeducab58cb2020-11-04 18:48:17 +0000100
David Garcia49379ce2021-02-24 13:48:22 +0100101 def build_pod_spec(self, image_info):
102 # Validate config
103 config = ConfigModel(**dict(self.config))
sousaedu996a5602021-05-03 00:22:43 +0200104
105 if config.mongodb_uri and not self.mongodb_client.is_missing_data_in_unit():
106 raise Exception("Mongodb data cannot be provided via config and relation")
David Garciaaccf1172021-05-10 12:59:33 +0200107 if config.mysql_uri and not self.mysql_client.is_missing_data_in_unit():
108 raise Exception("Mysql data cannot be provided via config and relation")
sousaedu996a5602021-05-03 00:22:43 +0200109
David Garcia49379ce2021-02-24 13:48:22 +0100110 # Check relations
111 self._check_missing_dependencies(config)
sousaedu996a5602021-05-03 00:22:43 +0200112
David Garcia49379ce2021-02-24 13:48:22 +0100113 # Create Builder for the PodSpec
114 pod_spec_builder = PodSpecV3Builder()
sousaedu996a5602021-05-03 00:22:43 +0200115
David Garcia49379ce2021-02-24 13:48:22 +0100116 # Build Container
117 container_builder = ContainerV3Builder(self.app.name, image_info)
118 container_builder.add_port(name=self.app.name, port=PORT)
119 container_builder.add_envs(
120 {
121 # General configuration
122 "ALLOW_ANONYMOUS_LOGIN": "yes",
123 "OSMPOL_GLOBAL_LOGLEVEL": config.log_level,
124 # Kafka configuration
125 "OSMPOL_MESSAGE_DRIVER": "kafka",
126 "OSMPOL_MESSAGE_HOST": self.kafka_client.host,
127 "OSMPOL_MESSAGE_PORT": self.kafka_client.port,
128 # Database configuration
129 "OSMPOL_DATABASE_DRIVER": "mongo",
sousaedu996a5602021-05-03 00:22:43 +0200130 "OSMPOL_DATABASE_URI": config.mongodb_uri
131 or self.mongodb_client.connection_string,
David Garciaaccf1172021-05-10 12:59:33 +0200132 "OSMPOL_SQL_DATABASE_URI": config.mysql_uri
133 or self.mysql_client.get_root_uri(DEFAULT_MYSQL_DATABASE),
David Garcia49379ce2021-02-24 13:48:22 +0100134 }
sousaeducab58cb2020-11-04 18:48:17 +0000135 )
David Garcia49379ce2021-02-24 13:48:22 +0100136 container = container_builder.build()
sousaedu996a5602021-05-03 00:22:43 +0200137
David Garcia49379ce2021-02-24 13:48:22 +0100138 # Add container to pod spec
139 pod_spec_builder.add_container(container)
sousaedu996a5602021-05-03 00:22:43 +0200140
David Garcia49379ce2021-02-24 13:48:22 +0100141 return pod_spec_builder.build()
sousaeducab58cb2020-11-04 18:48:17 +0000142
143
144if __name__ == "__main__":
145 main(PolCharm)