Newer
Older
#!/usr/bin/env python3
# Copyright 2020 Canonical Ltd.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import sys
import logging
sys.path.append("lib")
from ops.charm import CharmBase
from ops.framework import StoredState, Object
from ops.main import main
from ops.model import (
ActiveStatus,
MaintenanceStatus,
WaitingStatus,
)
from glob import glob
from pathlib import Path
from string import Template
logger = logging.getLogger(__name__)
class PLACharm(CharmBase):
state = StoredState()
def __init__(self, framework, key):
super().__init__(framework, key)
self.state.set_default(spec=None)
self.state.set_default(kafka_host=None)
self.state.set_default(kafka_port=None)
self.state.set_default(mongodb_uri=None)
# Observe Charm related events
self.framework.observe(self.on.config_changed, self.on_config_changed)
self.framework.observe(self.on.start, self.on_start)
self.framework.observe(self.on.upgrade_charm, self.on_upgrade_charm)
# Relations
self.framework.observe(
self.on.kafka_relation_changed, self.on_kafka_relation_changed
)
self.framework.observe(
self.on.mongo_relation_changed, self.on_mongo_relation_changed
)
def _apply_spec(self):
# Only apply the spec if this unit is a leader.
unit = self.model.unit
if not unit.is_leader():
unit.status = ActiveStatus("Ready")
return
if not self.state.kafka_host or not self.state.kafka_port:
unit.status = WaitingStatus("Waiting for Kafka")
return
if not self.state.mongodb_uri:
unit.status = WaitingStatus("Waiting for MongoDB")
unit.status = MaintenanceStatus("Applying new pod spec")
new_spec = self.make_pod_spec()
if new_spec == self.state.spec:
unit.status = ActiveStatus("Ready")
return
self.framework.model.pod.set_spec(new_spec)
self.state.spec = new_spec
unit.status = ActiveStatus("Ready")
def make_pod_spec(self):
config = self.framework.model.config
ports = [
{"name": "port", "containerPort": config["port"], "protocol": "TCP",},
]
config_spec = {
"OSMPLA_MESSAGE_DRIVER": "kafka",
"OSMPLA_MESSAGE_HOST": self.state.kafka_host,
"OSMPLA_MESSAGE_PORT": self.state.kafka_port,
"OSMPLA_DATABASE_URI": self.state.mongodb_uri,
"OSMPLA_DATABASE_COMMONKEY": config["database_common_key"],
}
spec = {
"version": 2,
"containers": [
{
"name": self.framework.model.app.name,
"image": config["image"],
"ports": ports,
"config": config_spec,
}
],
}
return spec
def on_config_changed(self, event):
"""Handle changes in configuration"""
self._apply_spec()
def on_start(self, event):
"""Called when the charm is being installed"""
self._apply_spec()
def on_upgrade_charm(self, event):
"""Upgrade the charm."""
unit = self.model.unit
unit.status = MaintenanceStatus("Upgrading charm")
self.on_start(event)
def on_kafka_relation_changed(self, event):
unit = self.model.unit
if not unit.is_leader():
return
self.state.kafka_host = event.relation.data[event.unit].get("host")
self.state.kafka_port = event.relation.data[event.unit].get("port")
self._apply_spec()
def on_mongo_relation_changed(self, event):
unit = self.model.unit
if not unit.is_leader():
return
self.state.mongodb_uri = event.relation.data[event.unit].get(
"connection_string"
)
self._apply_spec()