Skip to content

Commit

Permalink
Fix config.yaml overriding environment variables (#353)
Browse files Browse the repository at this point in the history
Co-authored-by: Salomon Popp <[email protected]>
  • Loading branch information
sujuka99 and disrupted authored Sep 6, 2023
1 parent b20a6b3 commit 73a3ee9
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 7 deletions.
24 changes: 18 additions & 6 deletions kpops/cli/pipeline_config.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,18 @@
from __future__ import annotations

from pathlib import Path
from typing import TYPE_CHECKING, Any

from pydantic import BaseConfig, BaseSettings, Field
from pydantic.env_settings import SettingsSourceCallable

from kpops.component_handlers.helm_wrapper.model import HelmConfig, HelmDiffConfig
from kpops.utils.yaml_loading import load_yaml_file

if TYPE_CHECKING:
from collections.abc import Callable

from pydantic.env_settings import SettingsSourceCallable

ENV_PREFIX = "KPOPS_"


Expand All @@ -26,7 +33,7 @@ class PipelineConfig(BaseSettings):
"""Pipeline configuration unrelated to the components."""

defaults_path: Path = Field(
default=Path("."),
default=Path(),
example="defaults",
description="The path to the folder containing the defaults.yaml file and the environment defaults files. "
"Paths can either be absolute or relative to `config.yaml`",
Expand Down Expand Up @@ -104,17 +111,22 @@ def customise_sources(
init_settings: SettingsSourceCallable,
env_settings: SettingsSourceCallable,
file_secret_settings: SettingsSourceCallable,
):
) -> tuple[
SettingsSourceCallable | Callable[[PipelineConfig], dict[str, Any]], ...
]:
return (
env_settings,
init_settings,
yaml_config_settings_source,
env_settings,
file_secret_settings,
)


def yaml_config_settings_source(settings: PipelineConfig) -> dict | list:
def yaml_config_settings_source(settings: PipelineConfig) -> dict[str, Any]:
path_to_config = settings.Config.config_path
if path_to_config.exists():
return load_yaml_file(path_to_config)
if isinstance(source := load_yaml_file(path_to_config), dict):
return source
err_msg = f"{path_to_config} must be a mapping."
raise TypeError(err_msg)
return {}
29 changes: 28 additions & 1 deletion tests/pipeline/test_pipeline.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@

import pytest
import yaml
from pytest import MonkeyPatch
from snapshottest.module import SnapshotTest
from typer.testing import CliRunner

Expand Down Expand Up @@ -458,8 +459,34 @@ def test_default_config(self, snapshot: SnapshotTest):

snapshot.assert_match(enriched_pipeline, "test-pipeline")

def test_env_vars_precedence_over_config(
self,
monkeypatch: MonkeyPatch,
snapshot: SnapshotTest,
):
monkeypatch.setenv(name="KPOPS_KAFKA_BROKERS", value="env_broker")

result = runner.invoke(
app,
[
"generate",
"--pipeline-base-dir",
str(PIPELINE_BASE_DIR_PATH),
str(RESOURCE_PATH / "custom-config/pipeline.yaml"),
"--config",
str(RESOURCE_PATH / "custom-config/config.yaml"),
],
catch_exceptions=False,
)
assert result.exit_code == 0
enriched_pipeline: dict = yaml.safe_load(result.stdout)
assert (
enriched_pipeline["components"][0]["app"]["streams"]["brokers"]
== "env_broker"
)

def test_model_serialization(self, snapshot: SnapshotTest):
"""Test model serialization of component containing pathlib.Path attribute"""
"""Test model serialization of component containing pathlib.Path attribute."""
result = runner.invoke(
app,
[
Expand Down

0 comments on commit 73a3ee9

Please sign in to comment.