Skip to content

Commit

Permalink
Try singleton decorator
Browse files Browse the repository at this point in the history
  • Loading branch information
disrupted committed Jul 9, 2024
1 parent c8471e4 commit 0094054
Show file tree
Hide file tree
Showing 8 changed files with 39 additions and 47 deletions.
18 changes: 7 additions & 11 deletions kpops/components/base_components/base_defaults_component.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
from pydantic.json_schema import SkipJsonSchema

from kpops.component_handlers import ComponentHandlers
from kpops.config import KpopsConfig, get_config
from kpops.config import KpopsConfig
from kpops.const.file_type import KpopsFileType
from kpops.utils import cached_classproperty
from kpops.utils.dict_ops import (
Expand Down Expand Up @@ -129,13 +129,12 @@ def substitute_in_component(cls, **component_data: Any) -> dict[str, Any]:
:param component_as_dict: Component represented as dict
:return: Updated component
"""
config = get_config()
# Leftover variables that were previously introduced in the component by the substitution
# functions, still hardcoded, because of their names.
# TODO(Ivan Yordanov): Get rid of them
substitution_hardcoded: dict[str, JsonType] = {
"error_topic_name": config.topic_name_config.default_error_topic_name,
"output_topic_name": config.topic_name_config.default_output_topic_name,
"error_topic_name": KpopsConfig.topic_name_config.default_error_topic_name,
"output_topic_name": KpopsConfig.topic_name_config.default_output_topic_name,
}
component_substitution = generate_substitution(
component_data,
Expand All @@ -144,7 +143,7 @@ def substitute_in_component(cls, **component_data: Any) -> dict[str, Any]:
separator=".",
)
substitution = generate_substitution(
config.model_dump(mode="json"),
KpopsConfig.model_dump(mode="json"),
"config",
existing_substitution=component_substitution,
separator=".",
Expand All @@ -165,7 +164,6 @@ def extend_with_defaults(cls, **kwargs: Any) -> dict[str, Any]:
:param kwargs: The init kwargs for pydantic
:returns: Enriched kwargs with inherited defaults
"""
config = get_config()
pipeline_path_str = ENV.get(PIPELINE_PATH)
if not pipeline_path_str:
return kwargs
Expand All @@ -177,7 +175,7 @@ def extend_with_defaults(cls, **kwargs: Any) -> dict[str, Any]:
kwargs[k] = asdict(v)

defaults_file_paths_ = get_defaults_file_paths(
pipeline_path, config, ENV.get("environment")
pipeline_path, ENV.get("environment")
)
defaults = cls.load_defaults(*defaults_file_paths_)
log.debug(
Expand Down Expand Up @@ -238,9 +236,7 @@ def defaults_from_yaml(path: Path, key: str) -> dict:
return value


def get_defaults_file_paths(
pipeline_path: Path, config: KpopsConfig, environment: str | None
) -> list[Path]:
def get_defaults_file_paths(pipeline_path: Path, environment: str | None) -> list[Path]:
"""Return a list of default file paths related to the given pipeline.
This function traverses the directory hierarchy upwards till the `pipeline_base_dir`,
Expand All @@ -260,7 +256,7 @@ def get_defaults_file_paths(
raise FileNotFoundError(message)

path = pipeline_path.resolve()
pipeline_base_dir = config.pipeline_base_dir.resolve()
pipeline_base_dir = KpopsConfig.pipeline_base_dir.resolve()
if pipeline_base_dir not in path.parents:
message = f"The given pipeline base path {pipeline_base_dir} is not part of the pipeline path {path}"
raise RuntimeError(message)
Expand Down
4 changes: 2 additions & 2 deletions kpops/components/base_components/cleaner.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
create_helm_release_name,
)
from kpops.components.base_components.helm_app import HelmApp
from kpops.config import get_config
from kpops.config import KpopsConfig


class Cleaner(HelmApp, ABC):
Expand All @@ -35,7 +35,7 @@ def helm_name_override(self) -> str:
@override
def helm_flags(self) -> HelmFlags:
return HelmFlags(
create_namespace=get_config().create_namespace,
create_namespace=KpopsConfig.create_namespace,
version=self.version,
wait=True,
wait_for_jobs=True,
Expand Down
12 changes: 6 additions & 6 deletions kpops/components/base_components/helm_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@
KubernetesAppValues,
)
from kpops.components.base_components.models.resource import Resource
from kpops.config import get_config
from kpops.config import KpopsConfig
from kpops.utils.colorify import magentaify
from kpops.utils.docstring import describe_attr
from kpops.utils.pydantic import exclude_by_name
Expand Down Expand Up @@ -85,7 +85,7 @@ class HelmApp(KubernetesApp):
@cached_property
def helm(self) -> Helm:
"""Helm object that contains component-specific config such as repo."""
helm = Helm(get_config().helm_config)
helm = Helm(KpopsConfig.helm_config)
if self.repo_config is not None:
helm.add_repo(
self.repo_config.repository_name,
Expand All @@ -97,11 +97,11 @@ def helm(self) -> Helm:
@cached_property
def helm_diff(self) -> HelmDiff:
"""Helm diff object of last and current release of this component."""
return HelmDiff(get_config().helm_diff_config)
return HelmDiff(KpopsConfig.helm_diff_config)

@cached_property
def dry_run_handler(self) -> DryRunHandler:
helm_diff = HelmDiff(get_config().helm_diff_config)
helm_diff = HelmDiff(KpopsConfig.helm_diff_config)
return DryRunHandler(self.helm, helm_diff, self.namespace)

@property
Expand Down Expand Up @@ -131,15 +131,15 @@ def helm_flags(self) -> HelmFlags:
return HelmFlags(
**auth_flags,
version=self.version,
create_namespace=get_config().create_namespace,
create_namespace=KpopsConfig.create_namespace,
)

@property
def template_flags(self) -> HelmTemplateFlags:
"""Return flags for Helm template command."""
return HelmTemplateFlags(
**self.helm_flags.model_dump(),
api_version=get_config().helm_config.api_version,
api_version=KpopsConfig.helm_config.api_version,
)

@override
Expand Down
4 changes: 2 additions & 2 deletions kpops/components/base_components/kafka_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
from kpops.components.base_components.models.topic import KafkaTopic, KafkaTopicStr
from kpops.components.base_components.pipeline_component import PipelineComponent
from kpops.components.common.streams_bootstrap import StreamsBootstrap
from kpops.config import get_config
from kpops.config import KpopsConfig
from kpops.utils.docstring import describe_attr
from kpops.utils.pydantic import (
CamelCaseConfigModel,
Expand Down Expand Up @@ -114,7 +114,7 @@ async def clean(self, dry_run: bool) -> None:
log.info(f"Init cleanup job for {self.helm_release_name}")
await self.deploy(dry_run)

if not get_config().retain_clean_jobs:
if not KpopsConfig.retain_clean_jobs:
log.info(f"Uninstall cleanup job for {self.helm_release_name}")
await self.destroy(dry_run)

Expand Down
6 changes: 3 additions & 3 deletions kpops/components/base_components/kafka_connector.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@
from kpops.components.base_components.models.from_section import FromTopic
from kpops.components.base_components.models.topic import KafkaTopic
from kpops.components.base_components.pipeline_component import PipelineComponent
from kpops.config import get_config
from kpops.config import KpopsConfig
from kpops.utils.colorify import magentaify
from kpops.utils.docstring import describe_attr

Expand Down Expand Up @@ -85,7 +85,7 @@ async def reset(self, dry_run: bool) -> None:
)
await self.deploy(dry_run)

if not get_config().retain_clean_jobs:
if not KpopsConfig.retain_clean_jobs:
log.info(magentaify("Connector Cleanup: uninstall Kafka Resetter."))
await self.destroy(dry_run)

Expand Down Expand Up @@ -158,7 +158,7 @@ def _resetter(self) -> KafkaConnectorResetter:
connector_type=self._connector_type.value,
config=KafkaConnectorResetterConfig(
connector=self.full_name,
brokers=get_config().kafka_brokers,
brokers=KpopsConfig.kafka_brokers,
),
**self.resetter_values.model_dump(),
),
Expand Down
32 changes: 15 additions & 17 deletions kpops/config/__init__.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,9 @@
from __future__ import annotations

import logging
from functools import lru_cache
from pathlib import Path
from typing import ClassVar
from typing import ClassVar, TypeVar

from pydantic import AnyHttpUrl, Field, PrivateAttr, TypeAdapter
from pydantic_settings import (
Expand All @@ -16,6 +17,11 @@
from kpops.utils.docstring import describe_object
from kpops.utils.pydantic import YamlConfigSettingsSource

try:
from typing import Self # pyright: ignore[reportAttributeAccessIssue]
except ImportError:
from typing_extensions import Self

ENV_PREFIX = "KPOPS_"


Expand Down Expand Up @@ -72,11 +78,15 @@ class KafkaConnectConfig(BaseSettings):
)


@lru_cache(maxsize=1)
def singleton(cls: type[KpopsConfig]) -> KpopsConfig:
return cls()


@singleton
class KpopsConfig(BaseSettings):
"""Global configuration for KPOps project."""

_config: ClassVar[KpopsConfig] = PrivateAttr()

pipeline_base_dir: Path = Field(
default=Path(),
description="Base directory to the pipelines (default is current working directory)",
Expand Down Expand Up @@ -130,14 +140,13 @@ def create(
dotenv: list[Path] | None = None,
environment: str | None = None,
verbose: bool = False,
) -> KpopsConfig:
) -> Self:
cls.setup_logging_level(verbose)
YamlConfigSettingsSource.config_dir = config
YamlConfigSettingsSource.environment = environment
cls._config = KpopsConfig(
return cls(
_env_file=dotenv # pyright: ignore[reportCallIssue]
)
return cls._config

@staticmethod
def setup_logging_level(verbose: bool):
Expand Down Expand Up @@ -166,14 +175,3 @@ def settings_customise_sources(
dotenv_settings,
file_secret_settings,
)


def get_config() -> KpopsConfig:
if KpopsConfig._config is None:
msg = f"{KpopsConfig.__name__} has not been initialized, call {KpopsConfig.__name__}.{KpopsConfig.create.__name__}"
raise RuntimeError(msg)
return KpopsConfig._config


def set_config(config: KpopsConfig) -> None:
KpopsConfig._config = config
5 changes: 2 additions & 3 deletions tests/components/test_base_defaults_component.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@
BaseDefaultsComponent,
get_defaults_file_paths,
)
from kpops.config import KpopsConfig, set_config
from kpops.config import KpopsConfig
from kpops.const.file_type import DEFAULTS_YAML, PIPELINE_YAML, KpopsFileType
from kpops.pipeline import PIPELINE_PATH
from kpops.utils.environment import ENV
Expand Down Expand Up @@ -49,8 +49,7 @@ class EnvVarTest(BaseDefaultsComponent):
@pytest.fixture(autouse=True)
def config() -> None:
ENV[PIPELINE_PATH] = str(RESOURCES_PATH / "pipeline.yaml")
config = KpopsConfig(pipeline_base_dir=PIPELINE_BASE_DIR)
set_config(config)
KpopsConfig(pipeline_base_dir=PIPELINE_BASE_DIR)


@pytest.fixture()
Expand Down
5 changes: 2 additions & 3 deletions tests/components/test_kafka_connector.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
from kpops.components.base_components.kafka_connector import (
KafkaConnector,
)
from kpops.config import KpopsConfig, TopicNameConfig, set_config
from kpops.config import KpopsConfig, TopicNameConfig
from tests.components import PIPELINE_BASE_DIR

CONNECTOR_NAME = "test-connector-with-long-name-0123456789abcdefghijklmnop"
Expand All @@ -30,7 +30,7 @@
class TestKafkaConnector:
@pytest.fixture(autouse=True)
def config(self) -> None:
config = KpopsConfig(
KpopsConfig(
topic_name_config=TopicNameConfig(
default_error_topic_name="${component.type}-error-topic",
default_output_topic_name="${component.type}-output-topic",
Expand All @@ -39,7 +39,6 @@ def config(self) -> None:
helm_diff_config=HelmDiffConfig(),
pipeline_base_dir=PIPELINE_BASE_DIR,
)
set_config(config)

@pytest.fixture()
def handlers(self) -> ComponentHandlers:
Expand Down

0 comments on commit 0094054

Please sign in to comment.