blob: 727ffbeba86cfd3b1ab7d7fb011cdd69931f876f [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]
sousaedu3ddbbd12021-08-24 19:57:24 +010052 image_pull_policy: Optional[str]
sousaeducab58cb2020-11-04 18:48:17 +000053
David Garcia49379ce2021-02-24 13:48:22 +010054 @validator("log_level")
55 def validate_log_level(cls, v):
56 if v not in {"INFO", "DEBUG"}:
57 raise ValueError("value must be INFO or DEBUG")
58 return v
sousaeducab58cb2020-11-04 18:48:17 +000059
sousaedu996a5602021-05-03 00:22:43 +020060 @validator("mongoddb_uri")
61 def validate_mongodb_uri(cls, v):
62 if v and not v.startswith("mongodb://"):
63 raise ValueError("mongodb_uri is not properly formed")
64 return v
65
David Garciaaccf1172021-05-10 12:59:33 +020066 @validator("mysql_uri")
67 def validate_mysql_uri(cls, v):
68 pattern = re.compile("^mysql:\/\/.*:.*@.*:\d+\/.*$") # noqa: W605
69 if v and not pattern.search(v):
70 raise ValueError("mysql_uri is not properly formed")
71 return v
72
sousaedu3ddbbd12021-08-24 19:57:24 +010073 @validator("image_pull_policy")
74 def validate_image_pull_policy(cls, v):
75 values = {
76 "always": "Always",
77 "ifnotpresent": "IfNotPresent",
78 "never": "Never",
79 }
80 v = v.lower()
81 if v not in values.keys():
82 raise ValueError("value must be always, ifnotpresent or never")
83 return values[v]
84
sousaeducab58cb2020-11-04 18:48:17 +000085
David Garcia49379ce2021-02-24 13:48:22 +010086class PolCharm(CharmedOsmBase):
sousaeducab58cb2020-11-04 18:48:17 +000087 def __init__(self, *args) -> NoReturn:
David Garcia49379ce2021-02-24 13:48:22 +010088 super().__init__(*args, oci_image="image")
sousaeducab58cb2020-11-04 18:48:17 +000089
David Garcia49379ce2021-02-24 13:48:22 +010090 self.kafka_client = KafkaClient(self, "kafka")
91 self.framework.observe(self.on["kafka"].relation_changed, self.configure_pod)
92 self.framework.observe(self.on["kafka"].relation_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +000093
David Garcia49379ce2021-02-24 13:48:22 +010094 self.mongodb_client = MongoClient(self, "mongodb")
95 self.framework.observe(self.on["mongodb"].relation_changed, self.configure_pod)
96 self.framework.observe(self.on["mongodb"].relation_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +000097
David Garciaaccf1172021-05-10 12:59:33 +020098 self.mysql_client = MysqlClient(self, "mysql")
99 self.framework.observe(self.on["mysql"].relation_changed, self.configure_pod)
100 self.framework.observe(self.on["mysql"].relation_broken, self.configure_pod)
101
David Garcia49379ce2021-02-24 13:48:22 +0100102 def _check_missing_dependencies(self, config: ConfigModel):
103 missing_relations = []
sousaeducab58cb2020-11-04 18:48:17 +0000104
David Garcia49379ce2021-02-24 13:48:22 +0100105 if self.kafka_client.is_missing_data_in_unit():
106 missing_relations.append("kafka")
sousaedu996a5602021-05-03 00:22:43 +0200107 if not config.mongodb_uri and self.mongodb_client.is_missing_data_in_unit():
David Garcia49379ce2021-02-24 13:48:22 +0100108 missing_relations.append("mongodb")
David Garciaaccf1172021-05-10 12:59:33 +0200109 if not config.mysql_uri and self.mysql_client.is_missing_data_in_unit():
110 missing_relations.append("mysql")
David Garcia49379ce2021-02-24 13:48:22 +0100111 if missing_relations:
112 raise RelationsMissing(missing_relations)
sousaeducab58cb2020-11-04 18:48:17 +0000113
David Garcia49379ce2021-02-24 13:48:22 +0100114 def build_pod_spec(self, image_info):
115 # Validate config
116 config = ConfigModel(**dict(self.config))
sousaedu996a5602021-05-03 00:22:43 +0200117
118 if config.mongodb_uri and not self.mongodb_client.is_missing_data_in_unit():
119 raise Exception("Mongodb data cannot be provided via config and relation")
David Garciaaccf1172021-05-10 12:59:33 +0200120 if config.mysql_uri and not self.mysql_client.is_missing_data_in_unit():
121 raise Exception("Mysql data cannot be provided via config and relation")
sousaedu996a5602021-05-03 00:22:43 +0200122
David Garcia49379ce2021-02-24 13:48:22 +0100123 # Check relations
124 self._check_missing_dependencies(config)
sousaedu996a5602021-05-03 00:22:43 +0200125
David Garcia49379ce2021-02-24 13:48:22 +0100126 # Create Builder for the PodSpec
127 pod_spec_builder = PodSpecV3Builder()
sousaedu996a5602021-05-03 00:22:43 +0200128
David Garcia49379ce2021-02-24 13:48:22 +0100129 # Build Container
sousaedu3ddbbd12021-08-24 19:57:24 +0100130 container_builder = ContainerV3Builder(
131 self.app.name, image_info, config.image_pull_policy
132 )
David Garcia49379ce2021-02-24 13:48:22 +0100133 container_builder.add_port(name=self.app.name, port=PORT)
134 container_builder.add_envs(
135 {
136 # General configuration
137 "ALLOW_ANONYMOUS_LOGIN": "yes",
138 "OSMPOL_GLOBAL_LOGLEVEL": config.log_level,
139 # Kafka configuration
140 "OSMPOL_MESSAGE_DRIVER": "kafka",
141 "OSMPOL_MESSAGE_HOST": self.kafka_client.host,
142 "OSMPOL_MESSAGE_PORT": self.kafka_client.port,
143 # Database configuration
144 "OSMPOL_DATABASE_DRIVER": "mongo",
sousaedu996a5602021-05-03 00:22:43 +0200145 "OSMPOL_DATABASE_URI": config.mongodb_uri
146 or self.mongodb_client.connection_string,
David Garciaaccf1172021-05-10 12:59:33 +0200147 "OSMPOL_SQL_DATABASE_URI": config.mysql_uri
148 or self.mysql_client.get_root_uri(DEFAULT_MYSQL_DATABASE),
David Garcia49379ce2021-02-24 13:48:22 +0100149 }
sousaeducab58cb2020-11-04 18:48:17 +0000150 )
David Garcia49379ce2021-02-24 13:48:22 +0100151 container = container_builder.build()
sousaedu996a5602021-05-03 00:22:43 +0200152
David Garcia49379ce2021-02-24 13:48:22 +0100153 # Add container to pod spec
154 pod_spec_builder.add_container(container)
sousaedu996a5602021-05-03 00:22:43 +0200155
David Garcia49379ce2021-02-24 13:48:22 +0100156 return pod_spec_builder.build()
sousaeducab58cb2020-11-04 18:48:17 +0000157
158
159if __name__ == "__main__":
160 main(PolCharm)