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

Support different versions per instance of a component #559

Draft
wants to merge 6 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
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
18 changes: 12 additions & 6 deletions commodore/cluster.py
Original file line number Diff line number Diff line change
Expand Up @@ -169,7 +169,9 @@ def generate_target(
"_instance": target,
}
if not bootstrap:
parameters["_base_directory"] = str(components[component].target_directory)
parameters["_base_directory"] = str(
components[component].alias_directory(target)
)
parameters["_kustomize_wrapper"] = str(__kustomize_wrapper__)
parameters["kapitan"] = {
"vars": {
Expand Down Expand Up @@ -206,24 +208,28 @@ def render_target(
classes = [f"params.{inv.bootstrap_target}"]

for c in sorted(components):
if inv.defaults_file(c).is_file():
classes.append(f"defaults.{c}")
defaults_file = inv.defaults_file(c)
if c == component and target != component:
# Special case alias defaults symlink
defaults_file = inv.defaults_file(target)

if defaults_file.is_file():
classes.append(f"defaults.{defaults_file.stem}")
else:
click.secho(f" > Default file for class {c} missing", fg="yellow")

classes.append("global.commodore")

if not bootstrap:
if not inv.component_file(component).is_file():
if not inv.component_file(target).is_file():
raise click.ClickException(
f"Target rendering failed for {target}: component class is missing"
)
classes.append(f"components.{component}")
classes.append(f"components.{target}")

return generate_target(inv, target, components, classes, component)


# pylint: disable=unsubscriptable-object
def update_target(cfg: Config, target: str, component: Optional[str] = None):
click.secho(f"Updating Kapitan target for {target}...", bold=True)
file = cfg.inventory.target_file(target)
Expand Down
2 changes: 1 addition & 1 deletion commodore/compile.py
Original file line number Diff line number Diff line change
Expand Up @@ -240,7 +240,7 @@ def setup_compile_environment(config: Config) -> tuple[dict[str, Any], Iterable[
config.register_component_deprecations(cluster_parameters)
# Raise exception if component version override without URL is present in the
# hierarchy.
verify_version_overrides(cluster_parameters)
verify_version_overrides(cluster_parameters, config.get_component_aliases())

for component in config.get_components().values():
ckey = component.parameters_key
Expand Down
59 changes: 54 additions & 5 deletions commodore/component/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ class Component:
_version: Optional[str] = None
_dir: P
_sub_path: str
_aliases: dict[str, str]
_work_dir: Optional[P]

@classmethod
def clone(cls, cfg, clone_url: str, name: str, version: str = "master"):
Expand Down Expand Up @@ -57,6 +59,8 @@ def __init__(
self.version = version
self._sub_path = sub_path
self._repo = None
self._aliases = {self.name: self.version or ""}
self._work_dir = work_dir

@property
def name(self) -> str:
Expand All @@ -67,8 +71,12 @@ def repo(self) -> GitRepo:
if not self._repo:
if self._dependency:
dep_repo = self._dependency.bare_repo
author_name = dep_repo.author.name
author_email = dep_repo.author.email
author_name = (
dep_repo.author.name if hasattr(dep_repo, "author") else None
)
author_email = (
dep_repo.author.email if hasattr(dep_repo, "author") else None
)
else:
# Fall back to author detection if we don't have a dependency
author_name = None
Expand Down Expand Up @@ -128,19 +136,36 @@ def repo_directory(self) -> P:

@property
def target_directory(self) -> P:
return self._dir / self._sub_path
return self.alias_directory(self.name)

@property
def target_dir(self) -> P:
return self.target_directory

@property
def class_file(self) -> P:
return self.target_directory / "class" / f"{self.name}.yml"
return self.alias_class_file(self.name)

@property
def defaults_file(self) -> P:
return self.target_directory / "class" / "defaults.yml"
return self.alias_defaults_file(self.name)

def alias_directory(self, alias: str) -> P:
if not self._dependency:
return self._dir / self._sub_path
apath = self._dependency.get_component(alias)
if not apath:
raise ValueError(f"unknown alias {alias} for component {self.name}")
return apath / self._sub_path

def alias_class_file(self, alias: str) -> P:
return self.alias_directory(alias) / "class" / f"{self.name}.yml"

def alias_defaults_file(self, alias: str) -> P:
return self.alias_directory(alias) / "class" / "defaults.yml"

def has_alias(self, alias: str):
return alias in self._aliases

@property
def lib_files(self) -> Iterable[P]:
Expand Down Expand Up @@ -177,6 +202,30 @@ def checkout(self):
)
self._dependency.checkout_component(self.name, self.version)

def register_alias(self, alias: str, version: str):
if not self._work_dir:
raise ValueError(
f"Can't register alias on component {self.name} "
+ "which isn't configured with a working directory"
)
if alias in self._aliases:
raise ValueError(
f"alias {alias} already registered on component {self.name}"
)
self._aliases[alias] = version
if self._dependency:
self._dependency.register_component(
alias, component_dir(self._work_dir, alias)
)

def checkout_alias(self, alias: str):
if alias not in self._aliases:
raise ValueError(
f"alias {alias} is not registered on component {self.name}"
)
if self._dependency:
self._dependency.checkout_component(alias, self._aliases[alias])

def is_checked_out(self) -> bool:
return self.target_dir is not None and self.target_dir.is_dir()

Expand Down
70 changes: 65 additions & 5 deletions commodore/dependency_mgmt/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,27 @@ def create_component_symlinks(cfg, component: Component):
)


def create_alias_symlinks(cfg, component: Component, alias: str):
if not component.has_alias(alias):
raise ValueError(
f"component {component.name} doesn't have alias {alias} registered"
)
relsymlink(
component.alias_class_file(alias),
cfg.inventory.components_dir,
dest_name=f"{alias}.yml",
)
inventory_default = cfg.inventory.defaults_file(alias)
relsymlink(
component.alias_defaults_file(alias),
inventory_default.parent,
dest_name=inventory_default.name,
)
# TODO: How do we handle lib files when symlinking aliases? Code in the component
# alias itself should be able to find the right library. We need to define what
# version of the library is visible for other components.


def create_package_symlink(cfg, pname: str, package: Package):
"""
Create package symlink in the inventory.
Expand Down Expand Up @@ -69,7 +90,7 @@ def fetch_components(cfg: Config):
component_names, component_aliases = _discover_components(cfg)
click.secho("Registering component aliases...", bold=True)
cfg.register_component_aliases(component_aliases)
cspecs = _read_components(cfg, component_names)
cspecs = _read_components(cfg, component_aliases)
click.secho("Fetching components...", bold=True)

deps: dict[str, list] = {}
Expand All @@ -93,6 +114,26 @@ def fetch_components(cfg: Config):
deps.setdefault(cdep.url, []).append(c)
fetch_parallel(fetch_component, cfg, deps.values())

components = cfg.get_components()

for alias, component in component_aliases.items():
if alias == component:
# Nothing to setup for identity alias
continue

c = components[component]
aspec = cspecs[alias]
if aspec.url != c.repo_url or aspec.path != c._sub_path:
# TODO: Figure out how we'll handle URL/subpath overrides
raise NotImplementedError(
"URL/path override for component alias not supported"
)
print(alias, aspec)
c.register_alias(alias, aspec.version)
c.checkout_alias(alias)

create_alias_symlinks(cfg, c, alias)


def fetch_component(cfg, dependencies):
"""
Expand Down Expand Up @@ -126,7 +167,7 @@ def register_components(cfg: Config):
click.secho("Discovering included components...", bold=True)
try:
components, component_aliases = _discover_components(cfg)
cspecs = _read_components(cfg, components)
cspecs = _read_components(cfg, component_aliases)
except KeyError as e:
raise click.ClickException(f"While discovering components: {e}")
click.secho("Registering components and aliases...", bold=True)
Expand All @@ -152,9 +193,9 @@ def register_components(cfg: Config):
cfg.register_component(component)
create_component_symlinks(cfg, component)

registered_components = cfg.get_components().keys()
registered_components = cfg.get_components()
pruned_aliases = {
a: c for a, c in component_aliases.items() if c in registered_components
a: c for a, c in component_aliases.items() if c in registered_components.keys()
}
pruned = sorted(set(component_aliases.keys()) - set(pruned_aliases.keys()))
if len(pruned) > 0:
Expand All @@ -163,6 +204,21 @@ def register_components(cfg: Config):
)
cfg.register_component_aliases(pruned_aliases)

for alias, cn in pruned_aliases.items():
if alias == cn:
# Nothing to setup for identity alias
continue

c = registered_components[cn]
aspec = cspecs[alias]
if aspec.url != c.repo_url or aspec.path != c.sub_path:
raise NotImplementedError("Changing alias sub path / URL NYI")
c.register_alias(alias, aspec.version)
if not component_dir(cfg.work_dir, alias).is_dir():
raise click.ClickException(f"Missing alias checkout for '{alias} as {cn}'")

create_alias_symlinks(cfg, c, alias)


def fetch_packages(cfg: Config):
"""
Expand Down Expand Up @@ -235,9 +291,13 @@ def register_packages(cfg: Config):
create_package_symlink(cfg, p, pkg)


def verify_version_overrides(cluster_parameters):
def verify_version_overrides(cluster_parameters, component_aliases: dict[str, str]):
errors = []
aliases = set(component_aliases.keys()) - set(component_aliases.values())
for cname, cspec in cluster_parameters["components"].items():
if cname in aliases:
# We don't require an url in component alias version configs
continue
if "url" not in cspec:
errors.append(f"component '{cname}'")

Expand Down
60 changes: 52 additions & 8 deletions commodore/dependency_mgmt/version_parsing.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

from collections.abc import Iterable
from dataclasses import dataclass
from typing import Optional

from enum import Enum

Expand Down Expand Up @@ -33,18 +34,31 @@ class DependencySpec:
path: str

@classmethod
def parse(cls, info: dict[str, str]) -> DependencySpec:
if "url" not in info:
def parse(
cls,
info: dict[str, str],
base_config: Optional[DependencySpec] = None,
) -> DependencySpec:
if "url" not in info and not base_config:
raise DependencyParseError("url")

if "version" not in info:
if "version" not in info and not base_config:
raise DependencyParseError("version")

path = info.get("path", "")
if path.startswith("/"):
path = path[1:]

return DependencySpec(info["url"], info["version"], path)
if base_config:
url = info.get("url", base_config.url)
version = info.get("version", base_config.version)
if path not in info:
path = base_config.path
else:
url = info["url"]
version = info["version"]

return DependencySpec(url, version, path)


def _read_versions(
Expand All @@ -53,6 +67,8 @@ def _read_versions(
dependency_names: Iterable[str],
require_key: bool = True,
ignore_class_notfound: bool = False,
aliases: dict[str, str] = {},
fallback: dict[str, DependencySpec] = {},
) -> dict[str, DependencySpec]:
deps_key = dependency_type.value
deptype_str = dependency_type.name.lower()
Expand All @@ -71,15 +87,26 @@ def _read_versions(
# just set deps to the empty dict.
deps = {}

if aliases:
all_dep_keys = set(aliases.keys())
else:
all_dep_keys = deps.keys()
for depname in dependency_names:
if depname not in deps:
if depname not in all_dep_keys:
raise click.ClickException(
f"Unknown {deptype_str} '{depname}'."
+ f" Please add it to 'parameters.{deps_key}'"
)

try:
dep = DependencySpec.parse(deps[depname])
basename_for_dep = aliases.get(depname, depname)
print(depname, basename_for_dep)
print(deps.get(depname, {}))
print(fallback.get(basename_for_dep))
dep = DependencySpec.parse(
deps.get(depname, {}),
base_config=fallback.get(basename_for_dep),
)
except DependencyParseError as e:
raise click.ClickException(
f"{deptype_cap} '{depname}' is missing field '{e.field}'"
Expand All @@ -96,9 +123,26 @@ def _read_versions(


def _read_components(
cfg: Config, component_names: Iterable[str]
cfg: Config, component_aliases: dict[str, str]
) -> dict[str, DependencySpec]:
return _read_versions(cfg, DepType.COMPONENT, component_names)
component_names = set(component_aliases.values())
alias_names = set(component_aliases.keys()) - component_names

component_versions = _read_versions(cfg, DepType.COMPONENT, component_names)
alias_versions = _read_versions(
cfg,
DepType.COMPONENT,
alias_names,
aliases=component_aliases,
fallback=component_versions,
)

for alias, aspec in alias_versions.items():
if alias in component_versions:
raise ValueError("alias name already in component_versions?")
component_versions[alias] = aspec

return component_versions


def _read_packages(
Expand Down
1 change: 0 additions & 1 deletion commodore/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -333,7 +333,6 @@ def rm_tree_contents(basedir):
os.unlink(f)


# pylint: disable=unsubscriptable-object
def relsymlink(src: P, dest_dir: P, dest_name: Optional[str] = None):
if dest_name is None:
dest_name = src.name
Expand Down
Loading
Loading