Skip to content
Snippets Groups Projects
charm.py 6.48 KiB
Newer Older
garciadav's avatar
garciadav committed
#!/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
garciadav's avatar
garciadav committed

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,
garciadav's avatar
garciadav committed
    BlockedStatus,
    ModelError,
garciadav's avatar
garciadav committed
)

from glob import glob
from pathlib import Path
from string import Template

logger = logging.getLogger(__name__)


class NGUICharm(CharmBase):
    state = StoredState()

    def __init__(self, framework, key):
        super().__init__(framework, key)
        self.state.set_default(spec=None)
garciadav's avatar
garciadav committed
        self.state.set_default(nbi_host=None)
        self.state.set_default(nbi_port=None)
garciadav's avatar
garciadav committed

        # 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)
garciadav's avatar
garciadav committed
        self.framework.observe(
            self.on.nbi_relation_changed, self.on_nbi_relation_changed
        )

        # SSL Certificate path
        self.ssl_folder = "/certs"
        self.ssl_crt_name = "ssl_certificate.crt"
        self.ssl_key_name = "ssl_certificate.key"
garciadav's avatar
garciadav committed

    def _apply_spec(self):
        # Only apply the spec if this unit is a leader.
garciadav's avatar
garciadav committed
        unit = self.model.unit
        if not unit.is_leader():
Mark Beierl's avatar
Mark Beierl committed
            unit.status = ActiveStatus("ready")
garciadav's avatar
garciadav committed
            return
        if not self.state.nbi_host or not self.state.nbi_port:
            unit.status = WaitingStatus("Waiting for NBI")
garciadav's avatar
garciadav committed
            return
garciadav's avatar
garciadav committed
        unit.status = MaintenanceStatus("Applying new pod spec")

garciadav's avatar
garciadav committed
        new_spec = self.make_pod_spec()
        if new_spec == self.state.spec:
Mark Beierl's avatar
Mark Beierl committed
            unit.status = ActiveStatus("ready")
garciadav's avatar
garciadav committed
            return
        self.framework.model.pod.set_spec(new_spec)
        self.state.spec = new_spec
Mark Beierl's avatar
Mark Beierl committed
        unit.status = ActiveStatus("ready")
garciadav's avatar
garciadav committed

    def make_pod_spec(self):
        config = self.framework.model.config

        config_spec = {
            "http_port": config["port"],
            "https_port": config["https_port"],
garciadav's avatar
garciadav committed
            "server_name": config["server_name"],
            "client_max_body_size": config["client_max_body_size"],
garciadav's avatar
garciadav committed
            "nbi_host": self.state.nbi_host or config["nbi_host"],
            "nbi_port": self.state.nbi_port or config["nbi_port"],
            "ssl_crt": "",
            "ssl_crt_key": "",
garciadav's avatar
garciadav committed
        }

        ssl_certificate = None
        ssl_certificate_key = None
        ssl_enabled = False

        if "ssl_certificate" in config and "ssl_certificate_key" in config:
            # Get bytes of cert and key
            cert_b = base64.b64decode(config["ssl_certificate"])
            key_b = base64.b64decode(config["ssl_certificate_key"])
            # Decode key and cert
            ssl_certificate = cert_b.decode("utf-8")
            ssl_certificate_key = key_b.decode("utf-8")
            # Get paths
            cert_path = "{}/{}".format(self.ssl_folder, self.ssl_crt_name)
            key_path = "{}/{}".format(self.ssl_folder, self.ssl_key_name)

            config_spec["port"] = "{} ssl".format(config["https_port"])
            config_spec["ssl_crt"] = "ssl_certificate {};".format(cert_path)
            config_spec["ssl_crt_key"] = "ssl_certificate_key {};".format(key_path)
            ssl_enabled = True
        else:
            config_spec["ssl_crt"] = ""
            config_spec["ssl_crt_key"] = ""
garciadav's avatar
garciadav committed
        files = [
            {
                "name": "configuration",
                "mountPath": "/etc/nginx/sites-available/",
                "files": {
                    Path(filename)
                    .name: Template(Path(filename).read_text())
                    .substitute(config_spec)
                    for filename in glob("files/*")
                },
garciadav's avatar
garciadav committed
            }
garciadav's avatar
garciadav committed
        ]
        port = config["https_port"] if ssl_enabled else config["port"]
        ports = [
Mark Beierl's avatar
Mark Beierl committed
            {"name": "port", "containerPort": port, "protocol": "TCP", },
        ]

        kubernetes = {
            "readinessProbe": {
                "tcpSocket": {"port": port},
                "timeoutSeconds": 5,
                "periodSeconds": 5,
                "initialDelaySeconds": 10,
            },
            "livenessProbe": {
                "tcpSocket": {"port": port},
                "timeoutSeconds": 5,
                "initialDelaySeconds": 45,
            },
        }
garciadav's avatar
garciadav committed

        if ssl_certificate and ssl_certificate_key:
            files.append(
                {
                    "name": "ssl",
                    "mountPath": self.ssl_folder,
                    "files": {
                        self.ssl_crt_name: ssl_certificate,
                        self.ssl_key_name: ssl_certificate_key,
garciadav's avatar
garciadav committed
        logger.debug(files)
        spec = {
            "version": 2,
            "containers": [
                {
                    "name": self.framework.model.app.name,
                    "image": "{}".format(config["image"]),
                    "ports": ports,
                    "kubernetes": kubernetes,
                    "files": files,
                }
            ],
        }

        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)

garciadav's avatar
garciadav committed
    def on_nbi_relation_changed(self, event):
        unit = self.model.unit
        if not unit.is_leader():
            return
        self.state.nbi_host = event.relation.data[event.unit].get("host")
        self.state.nbi_port = event.relation.data[event.unit].get("port")
        self._apply_spec()

garciadav's avatar
garciadav committed

if __name__ == "__main__":
    main(NGUICharm)