from ipaddress import ip_network
+import json
import logging
from typing import NoReturn, Optional
from urllib.parse import urlparse
+from charms.kafka_k8s.v0.kafka import KafkaEvents, KafkaRequires
from ops.main import main
from opslib.osm.charm import CharmedOsmBase, RelationsMissing
from opslib.osm.interfaces.http import HttpServer
-from opslib.osm.interfaces.kafka import KafkaClient
from opslib.osm.interfaces.keystone import KeystoneClient
from opslib.osm.interfaces.mongo import MongoClient
from opslib.osm.interfaces.prometheus import PrometheusClient
image_pull_policy: str
debug_mode: bool
security_context: bool
+ tcpsocket_liveness_probe: Optional[str]
+ tcpsocket_readiness_probe: Optional[str]
@validator("auth_backend")
def validate_auth_backend(cls, v):
raise ValueError("value must be always, ifnotpresent or never")
return values[v]
+ @staticmethod
+ def _validate_tcpsocket_probe(probe_str) -> dict:
+ valid_attributes = (
+ "initial_delay_seconds",
+ "timeout_seconds",
+ "period_seconds",
+ "success_threshold",
+ "failure_threshold",
+ )
+ if probe_str:
+ probe_dict = json.loads(probe_str)
+ if all(attribute in valid_attributes for attribute in probe_dict):
+ return probe_dict
+ raise ValueError(
+ "One or more attributes are not accepted by the tcpsocket probe configuration"
+ )
+ return {}
+
+ @validator("tcpsocket_readiness_probe")
+ def validate_tcpsocket_readiness_probe(cls, v):
+ try:
+ return ConfigModel._validate_tcpsocket_probe(v)
+ except Exception as e:
+ raise ValueError(f"tcpsocket_readiness_probe configuration error: {e}")
+
+ @validator("tcpsocket_liveness_probe")
+ def validate_tcpsocket_liveness_probe(cls, v):
+ try:
+ return ConfigModel._validate_tcpsocket_probe(v)
+ except Exception as e:
+ raise ValueError(f"tcpsocket_liveness_probe configuration error: {e}")
+
class NbiCharm(CharmedOsmBase):
+ on = KafkaEvents()
+
def __init__(self, *args) -> NoReturn:
super().__init__(
*args,
},
)
- self.kafka_client = KafkaClient(self, "kafka")
- self.framework.observe(self.on["kafka"].relation_changed, self.configure_pod)
- self.framework.observe(self.on["kafka"].relation_broken, self.configure_pod)
+ self.kafka = KafkaRequires(self)
+ self.framework.observe(self.on.kafka_available, self.configure_pod)
+ self.framework.observe(self.on.kafka_broken, self.configure_pod)
self.mongodb_client = MongoClient(self, "mongodb")
self.framework.observe(self.on["mongodb"].relation_changed, self.configure_pod)
def _check_missing_dependencies(self, config: ConfigModel):
missing_relations = []
- if (
- self.kafka_client.is_missing_data_in_unit()
- and self.kafka_client.is_missing_data_in_app()
- ):
+ if not self.kafka.host or not self.kafka.port:
missing_relations.append("kafka")
if not config.mongodb_uri and self.mongodb_client.is_missing_data_in_unit():
missing_relations.append("mongodb")
def build_pod_spec(self, image_info):
# Validate config
config = ConfigModel(**dict(self.config))
-
if config.mongodb_uri and not self.mongodb_client.is_missing_data_in_unit():
raise Exception("Mongodb data cannot be provided via config and relation")
"command": [
"sh",
"-c",
- f"until (nc -zvw1 {self.kafka_client.host} {self.kafka_client.port} ); do sleep 3; done; exit 0",
+ f"until (nc -zvw1 {self.kafka.host} {self.kafka.port} ); do sleep 3; done; exit 0",
],
}
)
container_builder.add_port(name=self.app.name, port=PORT)
container_builder.add_tcpsocket_readiness_probe(
PORT,
- initial_delay_seconds=5,
- timeout_seconds=5,
+ initial_delay_seconds=config.tcpsocket_readiness_probe.get(
+ "initial_delay_seconds", 5
+ ),
+ timeout_seconds=config.tcpsocket_readiness_probe.get("timeout_seconds", 5),
+ period_seconds=config.tcpsocket_readiness_probe.get("period_seconds", 10),
+ success_threshold=config.tcpsocket_readiness_probe.get(
+ "success_threshold", 1
+ ),
+ failure_threshold=config.tcpsocket_readiness_probe.get(
+ "failure_threshold", 3
+ ),
)
container_builder.add_tcpsocket_liveness_probe(
PORT,
- initial_delay_seconds=45,
- timeout_seconds=10,
+ initial_delay_seconds=config.tcpsocket_liveness_probe.get(
+ "initial_delay_seconds", 45
+ ),
+ timeout_seconds=config.tcpsocket_liveness_probe.get("timeout_seconds", 10),
+ period_seconds=config.tcpsocket_liveness_probe.get("period_seconds", 10),
+ success_threshold=config.tcpsocket_liveness_probe.get(
+ "success_threshold", 1
+ ),
+ failure_threshold=config.tcpsocket_liveness_probe.get(
+ "failure_threshold", 3
+ ),
)
container_builder.add_envs(
{
"OSMNBI_SERVER_ENABLE_TEST": config.enable_test,
"OSMNBI_STATIC_DIR": "/app/osm_nbi/html_public",
# Kafka configuration
- "OSMNBI_MESSAGE_HOST": self.kafka_client.host,
+ "OSMNBI_MESSAGE_HOST": self.kafka.host,
"OSMNBI_MESSAGE_DRIVER": "kafka",
- "OSMNBI_MESSAGE_PORT": self.kafka_client.port,
+ "OSMNBI_MESSAGE_PORT": self.kafka.port,
# Database configuration
"OSMNBI_DATABASE_DRIVER": "mongo",
# Storage configuration