1
0
mirror of https://github.com/openSUSE/osc.git synced 2025-01-23 05:26:16 +01:00
github.com_openSUSE_osc/osc/obs_api/request_action.py

237 lines
8.9 KiB
Python

import urllib.error
from ..util.models import * # pylint: disable=wildcard-import,unused-wildcard-import
from .package import Package
from .package_sources import PackageSources
from .request_action_acceptinfo import RequestActionAcceptinfo
from .request_action_group import RequestActionGroup
from .request_action_grouped import RequestActionGrouped
from .request_action_options import RequestActionOptions
from .request_action_person import RequestActionPerson
from .request_action_source import RequestActionSource
from .request_action_target import RequestActionTarget
from .request_sourcediff import RequestSourcediff
class RequestAction(XmlModel):
XML_TAG = "action"
class TypeEnum(str, Enum):
SUBMIT = "submit"
DELETE = "delete"
CHANGE_DEVEL = "change_devel"
ADD_ROLE = "add_role"
SET_BUGOWNER = "set_bugowner"
MAINTENANCE_INCIDENT = "maintenance_incident"
MAINTENANCE_RELEASE = "maintenance_release"
RELEASE = "release"
GROUP = "group"
type: TypeEnum = Field(
xml_attribute=True,
)
source: Optional[RequestActionSource] = Field(
)
target: Optional[RequestActionTarget] = Field(
)
person: Optional[RequestActionPerson] = Field(
)
group: Optional[RequestActionGroup] = Field(
)
grouped_list: Optional[List[RequestActionGrouped]] = Field(
xml_name="grouped",
)
options: Optional[RequestActionOptions] = Field(
)
sourcediff: Optional[RequestSourcediff] = Field(
)
acceptinfo: Optional[RequestActionAcceptinfo] = Field(
)
def __init__(self, **kwargs):
super().__init__(**kwargs)
self._allow_new_attributes = True
# source and target always come from ``self._apiurl`` while devel and factory projects may live elsewhere
self._devel_apiurl = self._apiurl
self._factory_apiurl = self._apiurl
self._factory_project = "openSUSE:Factory"
self._props = {}
self._allow_new_attributes = False
def _get_package(self, package_type):
key = f"{package_type}_package"
if key not in self._props:
func = getattr(self, f"_get_{package_type}_apiurl_project_package")
apiurl, project, package = func()
if apiurl is None:
self._props[key] = None
else:
try:
self._props[key] = Package.from_api(apiurl, project, package)
except urllib.error.HTTPError as e:
if e.code != 404:
raise
self._props[key] = None
return self._props[key]
def _get_package_sources(self, package_type, *, rev=None):
key = f"{package_type}_package_sources"
if key not in self._props:
func = getattr(self, f"_get_{package_type}_apiurl_project_package")
apiurl, project, package = func()
if apiurl is None:
self._props[key] = None
else:
try:
self._props[key] = PackageSources.from_api(apiurl, project, package, rev=rev)
except urllib.error.HTTPError as e:
if e.code != 404:
raise
self._props[key] = None
return self._props[key]
def _get_source_apiurl_project_package(self):
return self._apiurl, self.source.project, self.source.package
@property
def source_package(self) -> Optional[Package]:
"""
Return a ``Package`` object that encapsulates metadata of the source package.
"""
return self._get_package("source")
@property
def source_package_sources(self) -> Optional[PackageSources]:
"""
Return a ``PackageSources`` object that contains information about the ``source.rev`` revision of the source package sources in OBS SCM.
"""
if self.source is None:
return None
return self._get_package_sources("source", rev=self.source.rev)
def _get_target_apiurl_project_package(self):
if self.target is None:
return None, None, None
target_project, target_package = self.get_actual_target_project_package()
return self._apiurl, target_project, target_package
@property
def target_package(self) -> Optional[Package]:
"""
Return a ``Package`` object that encapsulates metadata of the target package.
"""
return self._get_package("target")
@property
def target_package_sources(self) -> Optional[PackageSources]:
"""
Return a ``PackageSources`` object that contains information about the current revision of the target package sources in OBS SCM.
"""
return self._get_package_sources("target")
def _get_factory_apiurl_project_package(self):
if self.target is None:
# a new package was submitted, it doesn't exist on target; let's read the package name from the source
target_project, target_package = None, self.source.package
else:
target_project, target_package = self.get_actual_target_project_package()
if (self._apiurl, target_project) == (self._factory_apiurl, self._factory_project):
# factory package equals the target package
return None, None, None
return self._factory_apiurl, self._factory_project, target_package
@property
def factory_package(self) -> Optional[Package]:
"""
Return a ``Package`` object that encapsulates metadata of the package in the factory project.
The name of the package equals the target package name.
"""
return self._get_package("factory")
@property
def factory_package_sources(self) -> Optional[PackageSources]:
"""
Return a ``PackageSources`` object that contains information about the current revision of the factory package sources in OBS SCM.
"""
return self._get_package_sources("factory")
def _get_devel_apiurl_project_package(self):
if self.factory_package is None:
return None, None, None
devel = self.factory_package.devel
if devel is None:
return None, None, None
return (
self._devel_apiurl,
devel.project,
devel.package or self.factory_package.name,
)
@property
def devel_package(self) -> Optional[Package]:
"""
Return a ``Package`` object that encapsulates metadata of the package in the devel project.
The devel project name and package name come from ``self.factory_package.devel``.
If the devel package name is not set, target package name is used.
"""
return self._get_package("devel")
@property
def devel_package_sources(self) -> Optional[PackageSources]:
"""
Return a ``PackageSources`` object that contains information about the current revision of the devel package sources in OBS SCM.
"""
return self._get_package_sources("devel")
def get_actual_target_project_package(self) -> Tuple[str, str]:
"""
Return the target project and package names because maintenance incidents require special handling.
The target project for maintenance incidents is virtual and cannot be queried.
The actual target project is specified in target's ``releaseproject`` field.
Also the target package for maintenance incidents is not set explicitly.
It is extracted from ``releasename`` field from the source metadata.
If ``releasename`` is not defined, source package name is used.
"""
if self.type == "maintenance_incident":
# dmach's note on security:
# The ``releaseproject`` is baked into the target information in the request and that's perfectly ok.
# The ``releasename`` is part of the source package metadata and *may change* after the request is created.
# After consulting this with OBS developers, I believe this doesn't represent any security issue
# because the project is fixed and tampering with ``releasename`` might only lead to inconsistent naming,
# the package would still end up it the same project.
# target.releaseproject is always set for a maintenance_incident
assert self.target
assert self.target.releaseproject
project = self.target.releaseproject
# the target package is not specified
# we need to extract it from source package's metadata or use source package name as a fallback
assert self.source_package
if self.source_package.releasename:
package = self.source_package.releasename.split(".")[0]
else:
package = self.source_package.name
return project, package
assert self.target
assert self.target.project
assert self.target.package
return self.target.project, self.target.package