jenkins-job-builder/jenkins_jobs/root_base.py
Vsevolod Fedorov 60e8395c62 Add source location and context to error messages
Change-Id: I2e955c01b71a195bb6ff8ba2bb6f3a64cb3e1f58
2023-04-04 13:35:42 +03:00

255 lines
8.5 KiB
Python

# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from collections import namedtuple
from dataclasses import dataclass, field
from typing import List
from .constants import MAGIC_MANAGE_STRING
from .errors import Context, JenkinsJobsException
from .loc_loader import LocDict, LocString
from .position import Pos
from .formatter import enum_str_format_required_params, enum_str_format_param_defaults
from .expander import Expander, expand_parameters
from .defaults import Defaults
from .dimensions import enum_dimensions_params, is_point_included
@dataclass
class JobViewData:
"""Expanded job or view data, with added source context. Fed into xml generator"""
data: LocDict
context: List[Context] = field(default_factory=list)
@property
def name(self):
return self.data["name"]
def with_context(self, message, pos):
return JobViewData(self.data, [Context(message, pos), *self.context])
@dataclass
class RootBase:
"""Base class for YAML root elements - job, view or template"""
_defaults: dict
_expander: Expander
_keep_descriptions: bool
_id: str
name: str
pos: Pos
description: str
defaults_name: str
params: dict
contents: dict
@property
def id(self):
if self._id:
return self._id
else:
return self.name
@property
def title(self):
return str(self).capitalize()
def _format_description(self, params):
if self.description is None:
defaults = self._pick_defaults(self.defaults_name)
description = defaults.params.get("description")
else:
if type(self.description) is LocString:
description = str(self.description)
else:
description = self.description
if description is None and self._keep_descriptions:
return {}
expanded_desc = self._expander.expand(description, params)
return {"description": (expanded_desc or "") + MAGIC_MANAGE_STRING}
def _pick_defaults(self, name, merge_global=True):
try:
defaults = self._defaults[name]
except KeyError:
if name == "global":
return Defaults.empty()
raise JenkinsJobsException(
f"{self.title} wants defaults {self.defaults_name!r}"
" but it was never defined",
pos=name.pos,
)
if name == "global":
return defaults
if merge_global:
return defaults.merged_with_global(self._pick_defaults("global"))
else:
return defaults
class NonTemplateRootMixin:
def top_level_generate_items(self):
try:
defaults = self._pick_defaults(self.defaults_name, merge_global=False)
description = self._format_description(params={})
raw_data = self._as_dict()
contents = self._expander.expand(raw_data, self.params)
data = LocDict.merge(
defaults.contents,
contents,
description,
pos=self.pos,
)
context = [Context(f"In {self}", self.pos)]
yield JobViewData(data, context)
except JenkinsJobsException as x:
raise x.with_context(f"In {self}", pos=self.pos)
def generate_items(self, defaults_name, params):
# Do not produce jobs/views from under project - they are produced when
# processed directly from roots, by top_level_generate_items.
return []
class TemplateRootMixin:
def generate_items(self, defaults_name, params):
try:
defaults = self._pick_defaults(defaults_name or self.defaults_name)
item_params = LocDict.merge(
defaults.params,
self.params,
params,
{"template-name": self.name},
)
if self._id:
item_params["id"] = self._id
contents = LocDict.merge(
defaults.contents,
self._as_dict(),
)
axes = list(enum_str_format_required_params(self.name, self.name.pos))
axes_defaults = dict(enum_str_format_param_defaults(self.name))
for dim_params in enum_dimensions_params(axes, item_params, axes_defaults):
instance_params = LocDict.merge(
item_params,
dim_params,
)
expanded_params = expand_parameters(self._expander, instance_params)
if not is_point_included(
exclude_list=expanded_params.get("exclude"),
params=expanded_params,
key_pos=expanded_params.key_pos.get("exclude"),
):
continue
description = self._format_description(expanded_params)
expanded_contents = self._expander.expand(contents, expanded_params)
data = LocDict.merge(
expanded_contents,
description,
pos=self.pos,
)
context = [Context(f"In {self}", self.pos)]
yield JobViewData(data, context)
except JenkinsJobsException as x:
raise x.with_context(f"In {self}", pos=self.pos)
class GroupBase:
Spec = namedtuple("Spec", "name params pos")
def __repr__(self):
return f"<{self}>"
@classmethod
def _specs_from_list(cls, spec_list=None):
if spec_list is None:
return []
return [
cls._spec_from_dict(item, spec_list.value_pos[idx])
for idx, item in enumerate(spec_list)
]
@classmethod
def _spec_from_dict(cls, d, pos):
if isinstance(d, (str, LocString)):
return cls.Spec(d, params={}, pos=pos)
if not isinstance(d, dict):
raise JenkinsJobsException(
"Job/view spec should name or dict,"
f" but is {type(d)} ({d!r}). Missing indent?",
pos=pos,
)
if len(d) != 1:
raise JenkinsJobsException(
"Job/view dict should be single-item,"
f" but have keys {list(d.keys())}. Missing indent?",
pos=d.pos,
)
name, params = next(iter(d.items()))
if params is None:
params = {}
else:
if not isinstance(params, dict):
raise JenkinsJobsException(
f"Job/view {name!r} params type should be dict,"
f" but is {params!r}.",
pos=params.pos,
)
return cls.Spec(name, params, pos)
def _generate_items(self, root_dicts, spec_list, defaults_name, params):
try:
for spec in spec_list:
item = self._pick_spec_item(root_dicts, spec)
item_params = LocDict.merge(
params,
self.params,
self._my_params,
spec.params,
)
for job_data in item.generate_items(defaults_name, item_params):
yield (
job_data.with_context("Defined here", spec.pos).with_context(
f"In {self}", self.pos
)
)
except JenkinsJobsException as x:
raise x.with_context(f"In {self}", self.pos)
@property
def _my_params(self):
return {}
def _pick_spec_item(self, root_dict_list, spec):
for roots_dict in root_dict_list:
try:
return roots_dict[spec.name]
except KeyError:
pass
raise JenkinsJobsException(
f"Failed to find suitable job/view/template named '{spec.name}'",
pos=spec.pos,
)
@dataclass
class Group(GroupBase):
name: str
pos: Pos
specs: list # list[Spec]
params: dict
def generate_items(self, defaults_name, params):
return self._generate_items(self._root_dicts, self.specs, defaults_name, params)