blob: 7b92b453c56ccf90ae80eeca4be03b47f4f8e89b [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
David Garcia4a0db7c2022-02-21 11:48:11 +010030from charms.kafka_k8s.v0.kafka import KafkaEvents, KafkaRequires
sousaeducab58cb2020-11-04 18:48:17 +000031from ops.main import main
David Garcia49379ce2021-02-24 13:48:22 +010032from opslib.osm.charm import CharmedOsmBase, RelationsMissing
David Garciac753dc52021-03-17 15:28:47 +010033from 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,
David Garcia141d9352021-09-08 17:48:40 +020037 PodRestartPolicy,
David Garcia49379ce2021-02-24 13:48:22 +010038 PodSpecV3Builder,
39)
David Garciac753dc52021-03-17 15:28:47 +010040from opslib.osm.validator import ModelValidator, validator
David Garcia49379ce2021-02-24 13:48:22 +010041
sousaeducab58cb2020-11-04 18:48:17 +000042
43logger = logging.getLogger(__name__)
44
David Garcia49379ce2021-02-24 13:48:22 +010045PORT = 9999
David Garciaaccf1172021-05-10 12:59:33 +020046DEFAULT_MYSQL_DATABASE = "pol"
sousaeducab58cb2020-11-04 18:48:17 +000047
48
David Garcia49379ce2021-02-24 13:48:22 +010049class ConfigModel(ModelValidator):
50 log_level: str
sousaedu996a5602021-05-03 00:22:43 +020051 mongodb_uri: Optional[str]
David Garciaaccf1172021-05-10 12:59:33 +020052 mysql_uri: Optional[str]
sousaedu0dc25b32021-08-30 16:33:33 +010053 image_pull_policy: str
sousaedu540d9372021-09-29 01:53:30 +010054 debug_mode: bool
55 security_context: bool
sousaeducab58cb2020-11-04 18:48:17 +000056
David Garcia49379ce2021-02-24 13:48:22 +010057 @validator("log_level")
58 def validate_log_level(cls, v):
59 if v not in {"INFO", "DEBUG"}:
60 raise ValueError("value must be INFO or DEBUG")
61 return v
sousaeducab58cb2020-11-04 18:48:17 +000062
sousaedu996a5602021-05-03 00:22:43 +020063 @validator("mongoddb_uri")
64 def validate_mongodb_uri(cls, v):
65 if v and not v.startswith("mongodb://"):
66 raise ValueError("mongodb_uri is not properly formed")
67 return v
68
David Garciaaccf1172021-05-10 12:59:33 +020069 @validator("mysql_uri")
70 def validate_mysql_uri(cls, v):
71 pattern = re.compile("^mysql:\/\/.*:.*@.*:\d+\/.*$") # noqa: W605
72 if v and not pattern.search(v):
73 raise ValueError("mysql_uri is not properly formed")
74 return v
75
sousaedu3ddbbd12021-08-24 19:57:24 +010076 @validator("image_pull_policy")
77 def validate_image_pull_policy(cls, v):
78 values = {
79 "always": "Always",
80 "ifnotpresent": "IfNotPresent",
81 "never": "Never",
82 }
83 v = v.lower()
84 if v not in values.keys():
85 raise ValueError("value must be always, ifnotpresent or never")
86 return values[v]
87
sousaeducab58cb2020-11-04 18:48:17 +000088
David Garcia49379ce2021-02-24 13:48:22 +010089class PolCharm(CharmedOsmBase):
David Garcia4a0db7c2022-02-21 11:48:11 +010090
91 on = KafkaEvents()
92
sousaeducab58cb2020-11-04 18:48:17 +000093 def __init__(self, *args) -> NoReturn:
David Garciad680be42021-08-17 11:03:55 +020094 super().__init__(
95 *args,
96 oci_image="image",
David Garciad680be42021-08-17 11:03:55 +020097 vscode_workspace=VSCODE_WORKSPACE,
98 )
David Garciacafe31e2021-11-18 16:45:05 +010099 if self.config.get("debug_mode"):
100 self.enable_debug_mode(
101 pubkey=self.config.get("debug_pubkey"),
102 hostpaths={
103 "POL": {
104 "hostpath": self.config.get("debug_pol_local_path"),
105 "container-path": "/usr/lib/python3/dist-packages/osm_policy_module",
106 },
107 "osm_common": {
108 "hostpath": self.config.get("debug_common_local_path"),
109 "container-path": "/usr/lib/python3/dist-packages/osm_common",
110 },
111 },
112 )
David Garcia4a0db7c2022-02-21 11:48:11 +0100113 self.kafka = KafkaRequires(self)
114 self.framework.observe(self.on.kafka_available, self.configure_pod)
115 self.framework.observe(self.on.kafka_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +0000116
David Garcia49379ce2021-02-24 13:48:22 +0100117 self.mongodb_client = MongoClient(self, "mongodb")
118 self.framework.observe(self.on["mongodb"].relation_changed, self.configure_pod)
119 self.framework.observe(self.on["mongodb"].relation_broken, self.configure_pod)
sousaeducab58cb2020-11-04 18:48:17 +0000120
David Garciaaccf1172021-05-10 12:59:33 +0200121 self.mysql_client = MysqlClient(self, "mysql")
122 self.framework.observe(self.on["mysql"].relation_changed, self.configure_pod)
123 self.framework.observe(self.on["mysql"].relation_broken, self.configure_pod)
124
David Garcia49379ce2021-02-24 13:48:22 +0100125 def _check_missing_dependencies(self, config: ConfigModel):
126 missing_relations = []
sousaeducab58cb2020-11-04 18:48:17 +0000127
David Garcia4a0db7c2022-02-21 11:48:11 +0100128 if not self.kafka.host or not self.kafka.port:
David Garcia49379ce2021-02-24 13:48:22 +0100129 missing_relations.append("kafka")
sousaedu996a5602021-05-03 00:22:43 +0200130 if not config.mongodb_uri and self.mongodb_client.is_missing_data_in_unit():
David Garcia49379ce2021-02-24 13:48:22 +0100131 missing_relations.append("mongodb")
David Garciaaccf1172021-05-10 12:59:33 +0200132 if not config.mysql_uri and self.mysql_client.is_missing_data_in_unit():
133 missing_relations.append("mysql")
David Garcia49379ce2021-02-24 13:48:22 +0100134 if missing_relations:
135 raise RelationsMissing(missing_relations)
sousaeducab58cb2020-11-04 18:48:17 +0000136
David Garcia49379ce2021-02-24 13:48:22 +0100137 def build_pod_spec(self, image_info):
138 # Validate config
139 config = ConfigModel(**dict(self.config))
sousaedu996a5602021-05-03 00:22:43 +0200140
141 if config.mongodb_uri and not self.mongodb_client.is_missing_data_in_unit():
142 raise Exception("Mongodb data cannot be provided via config and relation")
David Garciaaccf1172021-05-10 12:59:33 +0200143 if config.mysql_uri and not self.mysql_client.is_missing_data_in_unit():
144 raise Exception("Mysql data cannot be provided via config and relation")
sousaedu996a5602021-05-03 00:22:43 +0200145
David Garcia49379ce2021-02-24 13:48:22 +0100146 # Check relations
147 self._check_missing_dependencies(config)
sousaedu996a5602021-05-03 00:22:43 +0200148
sousaedu540d9372021-09-29 01:53:30 +0100149 security_context_enabled = (
150 config.security_context if not config.debug_mode else False
151 )
152
David Garcia49379ce2021-02-24 13:48:22 +0100153 # Create Builder for the PodSpec
sousaedu540d9372021-09-29 01:53:30 +0100154 pod_spec_builder = PodSpecV3Builder(
155 enable_security_context=security_context_enabled
156 )
sousaedu996a5602021-05-03 00:22:43 +0200157
David Garcia141d9352021-09-08 17:48:40 +0200158 # Add secrets to the pod
159 mongodb_secret_name = f"{self.app.name}-mongodb-secret"
160 pod_spec_builder.add_secret(
161 mongodb_secret_name,
162 {"uri": config.mongodb_uri or self.mongodb_client.connection_string},
163 )
164 mysql_secret_name = f"{self.app.name}-mysql-secret"
165 pod_spec_builder.add_secret(
166 mysql_secret_name,
167 {
168 "uri": config.mysql_uri
169 or self.mysql_client.get_root_uri(DEFAULT_MYSQL_DATABASE)
170 },
171 )
172
David Garcia49379ce2021-02-24 13:48:22 +0100173 # Build Container
sousaedu3ddbbd12021-08-24 19:57:24 +0100174 container_builder = ContainerV3Builder(
sousaedu540d9372021-09-29 01:53:30 +0100175 self.app.name,
176 image_info,
177 config.image_pull_policy,
178 run_as_non_root=security_context_enabled,
sousaedu3ddbbd12021-08-24 19:57:24 +0100179 )
David Garcia49379ce2021-02-24 13:48:22 +0100180 container_builder.add_port(name=self.app.name, port=PORT)
181 container_builder.add_envs(
182 {
183 # General configuration
184 "ALLOW_ANONYMOUS_LOGIN": "yes",
185 "OSMPOL_GLOBAL_LOGLEVEL": config.log_level,
186 # Kafka configuration
187 "OSMPOL_MESSAGE_DRIVER": "kafka",
David Garcia4a0db7c2022-02-21 11:48:11 +0100188 "OSMPOL_MESSAGE_HOST": self.kafka.host,
189 "OSMPOL_MESSAGE_PORT": self.kafka.port,
David Garcia49379ce2021-02-24 13:48:22 +0100190 # Database configuration
191 "OSMPOL_DATABASE_DRIVER": "mongo",
David Garcia49379ce2021-02-24 13:48:22 +0100192 }
sousaeducab58cb2020-11-04 18:48:17 +0000193 )
David Garcia141d9352021-09-08 17:48:40 +0200194 container_builder.add_secret_envs(
195 mongodb_secret_name, {"OSMPOL_DATABASE_URI": "uri"}
196 )
197 container_builder.add_secret_envs(
198 mysql_secret_name, {"OSMPOL_SQL_DATABASE_URI": "uri"}
199 )
David Garcia49379ce2021-02-24 13:48:22 +0100200 container = container_builder.build()
sousaedu996a5602021-05-03 00:22:43 +0200201
David Garcia141d9352021-09-08 17:48:40 +0200202 # Add Pod restart policy
203 restart_policy = PodRestartPolicy()
204 restart_policy.add_secrets(
205 secret_names=(mongodb_secret_name, mysql_secret_name)
206 )
207 pod_spec_builder.set_restart_policy(restart_policy)
208
David Garcia49379ce2021-02-24 13:48:22 +0100209 # Add container to pod spec
210 pod_spec_builder.add_container(container)
sousaedu996a5602021-05-03 00:22:43 +0200211
David Garcia49379ce2021-02-24 13:48:22 +0100212 return pod_spec_builder.build()
sousaeducab58cb2020-11-04 18:48:17 +0000213
214
David Garciad680be42021-08-17 11:03:55 +0200215VSCODE_WORKSPACE = {
216 "folders": [
217 {"path": "/usr/lib/python3/dist-packages/osm_policy_module"},
218 {"path": "/usr/lib/python3/dist-packages/osm_common"},
219 ],
220 "settings": {},
221 "launch": {
222 "version": "0.2.0",
223 "configurations": [
224 {
225 "name": "POL",
226 "type": "python",
227 "request": "launch",
228 "module": "osm_policy_module.cmd.policy_module_agent",
229 "justMyCode": False,
230 }
231 ],
232 },
233}
234
235
sousaeducab58cb2020-11-04 18:48:17 +0000236if __name__ == "__main__":
237 main(PolCharm)