1
0
mirror of https://github.com/openSUSE/osc.git synced 2025-01-23 13:31:48 +01:00

Add 'git_scm' module for handling packages that live in git scm rather than usual obs scm

This commit is contained in:
Daniel Mach 2023-07-26 14:20:26 +02:00
parent 87d1c489f2
commit f60db24e15
7 changed files with 217 additions and 3 deletions

View File

@ -58,7 +58,7 @@ jobs:
zypper -n lr --details
grep -qi tumbleweed /etc/os-release && zypper -n dist-upgrade || zypper -n patch || zypper -n patch
zypper -n install git-lfs
zypper -n install diffstat diffutils python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
zypper -n install diffstat diffutils git-core python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
- name: 'Install packages (Fedora/CentOS)'
if: ${{ contains(matrix.container, '/fedora:') || contains(matrix.container, '/centos:') }}
@ -66,7 +66,7 @@ jobs:
dnf -y makecache
dnf -y distro-sync
dnf -y install git-lfs
dnf -y install diffstat diffutils python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
dnf -y install diffstat diffutils git-core python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
- name: 'Install packages (Debian/Ubuntu)'
if: ${{ contains(matrix.container, '/debian:') || contains(matrix.container, '/ubuntu:') }}
@ -74,7 +74,7 @@ jobs:
apt-get -y update
apt-get -y upgrade
apt-get -y --no-install-recommends install git-lfs
apt-get -y --no-install-recommends install diffstat diffutils python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
apt-get -y --no-install-recommends install diffstat diffutils git-core python3 python3-cryptography python3-pip python3-rpm python3-setuptools python3-urllib3
- uses: actions/checkout@v3

View File

@ -57,6 +57,8 @@ BuildRequires: %{use_python_pkg}-rpm
BuildRequires: %{use_python_pkg}-setuptools
BuildRequires: %{use_python_pkg}-urllib3
BuildRequires: diffstat
# needed for git scm tests
BuildRequires: git-core
Requires: %{use_python_pkg}-cryptography
Requires: %{use_python_pkg}-rpm
@ -78,6 +80,10 @@ Recommends: diffstat
Recommends: powerpc32
Recommends: sudo
# needed for building from git
Recommends: git-core
Recommends: git-lfs
# needed for `osc add <URL>`
Recommends: obs-service-recompress
Recommends: obs-service-download_files

4
osc/git_scm/README.md Normal file
View File

@ -0,0 +1,4 @@
# Warning
This module provides EXPERIMENTAL and UNSTABLE support for git scm such as https://src.opensuse.org/.
The code may change or disappear without a prior notice!

7
osc/git_scm/__init__.py Normal file
View File

@ -0,0 +1,7 @@
import sys
from .store import GitStore
def warn_experimental():
print("WARNING: Using EXPERIMENTAL support for git scm. The functionality may change or disappear without a prior notice!", file=sys.stderr)

151
osc/git_scm/store.py Normal file
View File

@ -0,0 +1,151 @@
import json
import os
import subprocess
import urllib.parse
from pathlib import Path
from .. import conf as osc_conf
from .. import oscerr
class GitStore:
@classmethod
def is_project_dir(cls, path):
try:
store = cls(path)
except oscerr.NoWorkingCopy:
return False
return store.is_project
@classmethod
def is_package_dir(cls, path):
try:
store = cls(path)
except oscerr.NoWorkingCopy:
return False
return store.is_package
def __init__(self, path, check=True):
self.path = path
self.abspath = os.path.abspath(self.path)
# TODO: how to determine if the current git repo contains a project or a package?
self.is_project = False
self.is_package = os.path.exists(os.path.join(self.abspath, ".git"))
self._package = None
self._project = None
if check and not any([self.is_project, self.is_package]):
msg = f"Directory '{self.path}' is not a GIT working copy"
raise oscerr.NoWorkingCopy(msg)
# TODO: decide if we need explicit 'git lfs pull' or not
# self._run_git(["lfs", "pull"])
def assert_is_project(self):
if not self.is_project:
msg = f"Directory '{self.path}' is not a GIT working copy of a project"
raise oscerr.NoWorkingCopy(msg)
def assert_is_package(self):
if not self.is_package:
msg = f"Directory '{self.path}' is not a GIT working copy of a package"
raise oscerr.NoWorkingCopy(msg)
def _run_git(self, args):
return subprocess.check_output(["git"] + args, encoding="utf-8", cwd=self.abspath).strip()
@property
def apiurl(self):
# HACK: we're using the currently configured apiurl
return osc_conf.config["apiurl"]
@property
def project(self):
if self._project is None:
# get project from the branch name
branch = self._run_git(["branch", "--show-current"])
# HACK: replace hard-coded mapping with metadata from git or the build service
if branch == "factory":
self._project = "openSUSE:Factory"
else:
print(f"ERROR: Couldn't map git branch '{branch}' to a project", file=sys.stderr)
return self._project
@project.setter
def project(self, value):
self._project = value
@property
def package(self):
if self._package is None:
origin = self._run_git(["remote", "get-url", "origin"])
self._package = Path(urllib.parse.urlsplit(origin).path).stem
return self._package
@package.setter
def package(self, value):
self._package = value
def _get_option(self, name):
try:
result = self._run_git(["config", "--local", "--get", f"osc.{name}"])
except subprocess.CalledProcessError:
result = None
return result
def _check_type(self, name, value, expected_type):
if not isinstance(value, expected_type):
raise TypeError(f"The option '{name}' should be {expected_type.__name__}, not {type(value).__name__}")
def _set_option(self, name, value):
self._run_git(["config", "--local", f"osc.{name}", value])
def _unset_option(self, name):
try:
self._run_git(["config", "--local", "--unset", f"osc.{name}"])
except subprocess.CalledProcessError:
pass
def _get_dict_option(self, name):
result = self._get_option(name)
if result is None:
return None
result = json.loads(result)
self._check_type(name, result, dict)
return result
def _set_dict_option(self, name, value):
if value is None:
self._unset_option(name)
return
self._check_type(name, value, dict)
value = json.dumps(value)
self._set_option(name, value)
@property
def last_buildroot(self):
self.assert_is_package()
result = self._get_dict_option("last-buildroot")
if result is not None:
result = (result["repo"], result["arch"], result["vm_type"])
return result
@last_buildroot.setter
def last_buildroot(self, value):
self.assert_is_package()
if len(value) != 3:
raise ValueError("A tuple with exactly 3 items is expected: (repo, arch, vm_type)")
value = {
"repo": value[0],
"arch": value[1],
"vm_type": value[2],
}
self._set_dict_option("last-buildroot", value)
@property
def scmurl(self):
return self._run_git(["remote", "get-url", "origin"])

View File

@ -35,6 +35,7 @@ packages =
osc
osc._private
osc.commands
osc.git_scm
osc.output
osc.util
install_requires =

View File

@ -0,0 +1,45 @@
import os
import shutil
import subprocess
import tempfile
import unittest
from osc.git_scm.store import GitStore
class TestGitStore(unittest.TestCase):
def setUp(self):
self.tmpdir = tempfile.mkdtemp(prefix="osc_test")
os.chdir(self.tmpdir)
subprocess.check_output(["git", "init", "-b", "factory"])
subprocess.check_output(["git", "remote", "add", "origin", "https://example.com/packages/my-package.git"])
def tearDown(self):
try:
shutil.rmtree(self.tmpdir)
except OSError:
pass
def test_package(self):
store = GitStore(self.tmpdir)
self.assertEqual(store.package, "my-package")
def test_project(self):
store = GitStore(self.tmpdir)
self.assertEqual(store.project, "openSUSE:Factory")
def test_last_buildroot(self):
store = GitStore(self.tmpdir)
self.assertEqual(store.last_buildroot, None)
store.last_buildroot = ("repo", "arch", "vm_type")
store = GitStore(self.tmpdir)
self.assertEqual(store.last_buildroot, ("repo", "arch", "vm_type"))
if not shutil.which("git"):
TestGitStore = unittest.skip("The 'git' executable is not available")(TestGitStore)
if __name__ == "__main__":
unittest.main()