Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add image tag field to streams-bootstrap app values #499

Merged
merged 28 commits into from
Jul 8, 2024
Merged
Show file tree
Hide file tree
Changes from 22 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
47 changes: 43 additions & 4 deletions docs/docs/schema/defaults.json
Original file line number Diff line number Diff line change
Expand Up @@ -920,6 +920,12 @@
"additionalProperties": true,
"description": "Settings specific to producers.",
"properties": {
"imageTag": {
"default": "latest",
"description": "Docker image tag of the streams-bootstrap app.",
"title": "Imagetag",
"type": "string"
},
"nameOverride": {
"anyOf": [
{
Expand Down Expand Up @@ -1273,6 +1279,12 @@
"default": null,
"description": "Kubernetes event-driven autoscaling config"
},
"imageTag": {
"default": "latest",
"description": "Docker image tag of the streams-bootstrap app.",
"title": "Imagetag",
"type": "string"
},
"nameOverride": {
"anyOf": [
{
Expand Down Expand Up @@ -1328,10 +1340,10 @@
"app": {
"allOf": [
{
"$ref": "#/$defs/HelmAppValues"
"$ref": "#/$defs/StreamsBootstrapValues"
}
],
"description": "Helm app values"
"description": "Streams bootstrap app values"
},
"from": {
"anyOf": [
Expand Down Expand Up @@ -1409,12 +1421,39 @@
},
"required": [
"name",
"namespace",
"app"
"namespace"
],
"title": "StreamsBootstrap",
"type": "object"
},
"StreamsBootstrapValues": {
"additionalProperties": true,
"description": "Base value class for all streams bootstrap related components.",
"properties": {
"imageTag": {
"default": "latest",
"description": "Docker image tag of the streams-bootstrap app.",
"title": "Imagetag",
"type": "string"
},
"nameOverride": {
"anyOf": [
{
"maxLength": 63,
"type": "string"
},
{
"type": "null"
}
],
"default": null,
"description": "Helm chart name override, assigned automatically",
"title": "Nameoverride"
}
},
"title": "StreamsBootstrapValues",
"type": "object"
},
"StreamsConfig": {
"additionalProperties": true,
"description": "Streams Bootstrap streams section.",
Expand Down
12 changes: 12 additions & 0 deletions docs/docs/schema/pipeline.json
Original file line number Diff line number Diff line change
Expand Up @@ -588,6 +588,12 @@
"additionalProperties": true,
"description": "Settings specific to producers.",
"properties": {
"imageTag": {
"default": "latest",
"description": "Docker image tag of the streams-bootstrap app.",
"title": "Imagetag",
"type": "string"
},
"nameOverride": {
"anyOf": [
{
Expand Down Expand Up @@ -941,6 +947,12 @@
"default": null,
"description": "Kubernetes event-driven autoscaling config"
},
"imageTag": {
"default": "latest",
"description": "Docker image tag of the streams-bootstrap app.",
"title": "Imagetag",
"type": "string"
},
"nameOverride": {
"anyOf": [
{
Expand Down
5 changes: 5 additions & 0 deletions kpops/api/exception.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,8 @@ class ParsingException(Exception):

class ClassNotFoundError(Exception):
"""Similar to builtin `ModuleNotFoundError`; class doesn't exist inside module."""


class InvalidImageTagError(ValidationError):
def __init__(self, *args):
raminqaf marked this conversation as resolved.
Show resolved Hide resolved
super().__init__(*args)
9 changes: 9 additions & 0 deletions kpops/component_handlers/kafka_connect/model.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,15 @@
from pydantic import (
BaseModel,
ConfigDict,
Field,
SerializationInfo,
field_validator,
model_serializer,
)
from pydantic.json_schema import SkipJsonSchema
from typing_extensions import override

from kpops.component_handlers.kubernetes.utils import validate_image_tag
from kpops.components.base_components.helm_app import HelmAppValues
from kpops.components.base_components.models.topic import KafkaTopic, KafkaTopicStr
from kpops.utils.pydantic import (
Expand Down Expand Up @@ -123,3 +125,10 @@ class KafkaConnectorResetterConfig(CamelCaseConfigModel):
class KafkaConnectorResetterValues(HelmAppValues):
connector_type: Literal["source", "sink"]
config: KafkaConnectorResetterConfig
image_tag: str = Field(default="latest")
disrupted marked this conversation as resolved.
Show resolved Hide resolved

@pydantic.field_validator("image_tag", mode="before")
disrupted marked this conversation as resolved.
Show resolved Hide resolved
@classmethod
def validate_image_tag_field(cls, image_tag: Any) -> str:
validate_image_tag(image_tag)
return image_tag
28 changes: 28 additions & 0 deletions kpops/component_handlers/kubernetes/utils.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,14 @@
import hashlib
import logging
import re
from typing import Any

from kpops.api.exception import ValidationError

log = logging.getLogger("K8sUtils")

IMAGE_TAG_PATTERN = r"^[a-zA-Z0-9_][a-zA-Z0-9._-]{0,127}$"


def trim(max_len: int, name: str, suffix: str) -> str:
"""Shortens long K8s identifiers.
Expand All @@ -26,3 +32,25 @@ def trim(max_len: int, name: str, suffix: str) -> str:
)
return new_name
return name


def validate_image_tag(image_tag: Any) -> None:
"""Validate an image tag.

Image tags consist of lowercase and uppercase letters, digits, underscores (_), periods (.), and dashes (-).
It can be up to 128 characters long and must follow the regex pattern: [a-zA-Z0-9_][a-zA-Z0-9._-]{0,127}

:param image_tag: Docker image tag to be validated.
"""
if not (isinstance(image_tag, str) and is_valid_image_tag(image_tag)):
msg = (
"Image tag is not valid. "
"Image tags consist of lowercase and uppercase letters, digits, underscores (_), periods (.), and dashes (-). "
"It can be up to 128 characters long."
)
raise ValidationError(msg)


def is_valid_image_tag(image_tag: str) -> bool:
"""Check if the image tag is valid according to the specified regex pattern."""
return bool(re.match(IMAGE_TAG_PATTERN, image_tag))
48 changes: 47 additions & 1 deletion kpops/components/streams_bootstrap/__init__.py
Original file line number Diff line number Diff line change
@@ -1,26 +1,64 @@
from __future__ import annotations

import logging
from abc import ABC
from typing import TYPE_CHECKING, Any

import pydantic
from pydantic import Field

from kpops.component_handlers.helm_wrapper.model import HelmRepoConfig
from kpops.components.base_components.helm_app import HelmApp
from kpops.component_handlers.kubernetes.utils import validate_image_tag
from kpops.components.base_components.helm_app import HelmApp, HelmAppValues
from kpops.utils.docstring import describe_attr

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


STREAMS_BOOTSTRAP_HELM_REPO = HelmRepoConfig(
repository_name="bakdata-streams-bootstrap",
url="https://bakdata.github.io/streams-bootstrap/",
)
STREAMS_BOOTSTRAP_VERSION = "2.9.0"

log = logging.getLogger("StreamsBootstrap")


class StreamsBootstrapValues(HelmAppValues):
"""Base value class for all streams bootstrap related components.

:param image_tag: Docker image tag of the streams-bootstrap app.
"""

image_tag: str = Field(
default="latest", description=describe_attr("image_tag", __doc__)
)

@pydantic.field_validator("image_tag", mode="before")
@classmethod
def validate_image_tag_field(cls, image_tag: Any) -> str:
validate_image_tag(image_tag)
return image_tag


class StreamsBootstrap(HelmApp, ABC):
"""Base for components with a streams-bootstrap Helm chart.

:param app: Streams bootstrap app values
raminqaf marked this conversation as resolved.
Show resolved Hide resolved
:param repo_config: Configuration of the Helm chart repo to be used for
deploying the component, defaults to streams-bootstrap Helm repo
:param version: Helm chart version, defaults to "2.9.0"
"""

app: StreamsBootstrapValues = Field(
default_factory=StreamsBootstrapValues,
description=describe_attr("app", __doc__),
)

repo_config: HelmRepoConfig = Field(
default=STREAMS_BOOTSTRAP_HELM_REPO,
description=describe_attr("repo_config", __doc__),
Expand All @@ -29,3 +67,11 @@ class StreamsBootstrap(HelmApp, ABC):
default=STREAMS_BOOTSTRAP_VERSION,
description=describe_attr("version", __doc__),
)

@pydantic.model_validator(mode="after")
def warning_for_latest_image_tag(self) -> Self:
if self.validate_ and self.app.image_tag == "latest":
log.warning(
raminqaf marked this conversation as resolved.
Show resolved Hide resolved
f"The image tag for component '{self.name}' is set or defaulted to 'latest'. Please, consider providing a stable image tag."
)
return self
3 changes: 2 additions & 1 deletion kpops/components/streams_bootstrap/producer/model.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,14 +4,15 @@
KafkaAppValues,
KafkaStreamsConfig,
)
from kpops.components.streams_bootstrap import StreamsBootstrapValues
from kpops.utils.docstring import describe_attr


class ProducerStreamsConfig(KafkaStreamsConfig):
"""Kafka Streams settings specific to Producer."""


class ProducerAppValues(KafkaAppValues):
class ProducerAppValues(StreamsBootstrapValues, KafkaAppValues):
disrupted marked this conversation as resolved.
Show resolved Hide resolved
"""Settings specific to producers.

:param streams: Kafka Streams settings
Expand Down
3 changes: 2 additions & 1 deletion kpops/components/streams_bootstrap/streams/model.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
KafkaStreamsConfig,
)
from kpops.components.base_components.models.topic import KafkaTopic, KafkaTopicStr
from kpops.components.streams_bootstrap import StreamsBootstrapValues
from kpops.utils.docstring import describe_attr
from kpops.utils.pydantic import (
CamelCaseConfigModel,
Expand Down Expand Up @@ -237,7 +238,7 @@ def validate_mandatory_fields_are_set(
return self


class StreamsAppValues(KafkaAppValues):
class StreamsAppValues(StreamsBootstrapValues, KafkaAppValues):
"""streams-bootstrap app configurations.

The attributes correspond to keys and values that are used as values for the streams bootstrap helm chart.
Expand Down
37 changes: 37 additions & 0 deletions tests/component_handlers/kubernetes/test_utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
import pytest

from kpops.api.exception import ValidationError
from kpops.component_handlers.kubernetes.utils import (
validate_image_tag,
)

MAX_LENGTH = 128


def test_validate_image_tag_valid():
raminqaf marked this conversation as resolved.
Show resolved Hide resolved
valid_tags = [
"valid-tag",
"VALID_TAG",
"valid.tag",
"valid-tag_123",
"v" * MAX_LENGTH,
]
for tag in valid_tags:
validate_image_tag(tag)


def test_if_image_tag_is_invalid():
invalid_tags = [
"invalid tag!",
"",
" " * (MAX_LENGTH + 1),
"a" * (MAX_LENGTH + 1),
"@invalid",
None,
123,
{},
[],
]
for tag in invalid_tags:
with pytest.raises(ValidationError):
validate_image_tag(tag)
8 changes: 0 additions & 8 deletions tests/components/test_kafka_sink_connector.py
Original file line number Diff line number Diff line change
Expand Up @@ -107,14 +107,6 @@ def test_connector_config_parsing(
handlers: ComponentHandlers,
connector_config: KafkaConnectorConfig,
):
connector = KafkaSinkConnector(
name=CONNECTOR_NAME,
config=config,
handlers=handlers,
app=connector_config,
resetter_namespace=RESETTER_NAMESPACE,
)

topic_pattern = ".*"
connector = KafkaSinkConnector(
name=CONNECTOR_NAME,
Expand Down
3 changes: 3 additions & 0 deletions tests/components/test_streams_bootstrap.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ def test_default_configs(self, config: KpopsConfig, handlers: ComponentHandlers)
)
assert streams_bootstrap.version == "2.9.0"
assert streams_bootstrap.namespace == "test-namespace"
assert streams_bootstrap.app.image_tag == "latest"

@pytest.mark.asyncio()
async def test_should_deploy_streams_bootstrap_app(
Expand All @@ -63,6 +64,7 @@ async def test_should_deploy_streams_bootstrap_app(
**{
"namespace": "test-namespace",
"app": {
"imageTag": "1.0.0",
"streams": {
"outputTopic": "test",
"brokers": "fake-broker:9092",
Expand Down Expand Up @@ -94,6 +96,7 @@ async def test_should_deploy_streams_bootstrap_app(
"test-namespace",
{
"nameOverride": "${pipeline.name}-example-name",
"imageTag": "1.0.0",
"streams": {
"brokers": "fake-broker:9092",
"outputTopic": "test",
Expand Down
2 changes: 2 additions & 0 deletions tests/pipeline/resources/resetter_values/defaults.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,5 @@ helm-app:
kafka-sink-connector:
app:
"connector.class": "io.confluent.connect.jdbc.JdbcSinkConnector"
resetter_values:
imageTag: override-default-image-tag
Original file line number Diff line number Diff line change
Expand Up @@ -393,6 +393,7 @@
brokers: http://k8kafka-cp-kafka-headless.kpops.svc.cluster.local:9092
connector: atm-fraud-postgresql-connector
connectorType: sink
imageTag: latest
name: postgresql-connector
namespace: ${NAMESPACE}
prefix: atm-fraud-
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,7 @@
brokers: http://k8kafka-cp-kafka-headless.kpops.svc.cluster.local:9092
connector: word-count-redis-sink-connector
connectorType: sink
imageTag: latest
name: redis-sink-connector
namespace: ${NAMESPACE}
prefix: word-count-
Expand Down
Loading
Loading