Skip to content

Commit

Permalink
Enable typeguard during build time and tests, fix issues found
Browse files Browse the repository at this point in the history
It is also possible to use typeguard in osc-wrapper.py
by setting OSC_TYPEGUARD=1.
  • Loading branch information
dmach committed Jan 7, 2025
1 parent d2ed088 commit 3bb44b0
Show file tree
Hide file tree
Showing 11 changed files with 110 additions and 40 deletions.
3 changes: 2 additions & 1 deletion .github/workflows/tests.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,7 @@ jobs:
sudo apt-get -y --no-install-recommends install git python3-behave diffstat diffutils python3 python3-cryptography python3-pip python3-rpm python3-ruamel.yaml python3-setuptools python3-urllib3 obs-build obs-service-set-version
# obs-scm-bridge is not available as a package at the moment, install it from github
sudo pip3 config set global.break-system-packages 1
sudo pip3 install typeguard
sudo pip3 install git+https://github.com/openSUSE/obs-scm-bridge
sudo chmod a+x /usr/local/lib/*/*/obs_scm_bridge
sudo mkdir -p /usr/lib/obs/service
Expand All @@ -149,4 +150,4 @@ jobs:
- name: "Run tests"
run: |
cd behave
behave -Dosc=../osc-wrapper.py -Dgit-obs=../git-obs.py -Dpodman_max_containers=2
OSC_TYPEGUARD=1 behave -Dosc=../osc-wrapper.py -Dgit-obs=../git-obs.py -Dpodman_max_containers=2
10 changes: 10 additions & 0 deletions contrib/osc.spec
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,13 @@
%bcond_with fdupes
%endif

# use typeguard during build on distros where typeguard is available
%if (0%{?suse_version} > 1500 || 0%{?fedora} >= 37)
%bcond_without typeguard
%else
%bcond_with typeguard
%endif

# the macro exists only on openSUSE based distros
%if %{undefined python3_fix_shebang}
%define python3_fix_shebang %nil
Expand Down Expand Up @@ -76,6 +83,9 @@ BuildRequires: %{use_python_pkg}-cryptography
BuildRequires: %{use_python_pkg}-devel >= 3.6
BuildRequires: %{use_python_pkg}-rpm
BuildRequires: %{use_python_pkg}-setuptools
%if %{with typeguard}
BuildRequires: %{use_python_pkg}-typeguard
%endif
BuildRequires: %{use_python_pkg}-urllib3
BuildRequires: %{ruamel_yaml_pkg}
BuildRequires: diffstat
Expand Down
23 changes: 23 additions & 0 deletions osc-wrapper.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,29 @@
This wrapper allows osc to be called from the source directory during development.
"""


import os


USE_TYPEGUARD = os.environ.get("OSC_TYPEGUARD", "1").lower() in ("1", "true", "on")

if USE_TYPEGUARD:
try:
from typeguard import install_import_hook
except ImportError:
install_import_hook = None

if install_import_hook is None:
try:
from typeguard.importhook import install_import_hook
except ImportError:
install_import_hook = None

if install_import_hook:
# install typeguard import hook only if available
install_import_hook("osc")


import osc.babysitter

osc.babysitter.main()
31 changes: 17 additions & 14 deletions osc/commandline.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
from pathlib import Path
from tempfile import NamedTemporaryFile
from typing import List
from typing import Optional
from urllib.parse import urlsplit
from urllib.error import HTTPError

Expand Down Expand Up @@ -316,10 +317,10 @@ def pop_args(
args,
arg1_name: str = None,
arg1_is_optional: bool = False,
arg1_default: str = None,
arg1_default: Optional[str] = None,
arg2_name: str = None,
arg2_is_optional: bool = False,
arg2_default: str = None,
arg2_default: Optional[str] = None,
):
"""
Pop 2 arguments from `args`.
Expand Down Expand Up @@ -391,9 +392,9 @@ def pop_args(
def pop_project_package_from_args(
args: List[str],
project_is_optional: bool = False,
default_project: str = None,
default_project: Optional[str] = None,
package_is_optional: bool = False,
default_package: str = None,
default_package: Optional[str] = None,
):
"""
Pop project and package from given `args`.
Expand Down Expand Up @@ -464,9 +465,9 @@ def pop_project_package_from_args(
def pop_repository_arch_from_args(
args: List[str],
repository_is_optional: bool = False,
default_repository: str = None,
default_repository: Optional[str] = None,
arch_is_optional: bool = False,
default_arch: str = None,
default_arch: Optional[str] = None,
):
"""
Pop repository and arch from given `args`.
Expand Down Expand Up @@ -503,13 +504,13 @@ def pop_repository_arch_from_args(
def pop_project_package_repository_arch_from_args(
args: List[str],
project_is_optional: bool = False,
default_project: str = None,
default_project: Optional[str] = None,
package_is_optional: bool = False,
default_package: str = None,
default_package: Optional[str] = None,
repository_is_optional: bool = False,
default_repository: str = None,
default_repository: Optional[str] = None,
arch_is_optional: bool = False,
default_arch: str = None,
default_arch: Optional[str] = None,
):
"""
Pop project, package, repository and arch from given `args`.
Expand Down Expand Up @@ -589,13 +590,13 @@ def pop_project_package_repository_arch_from_args(
def pop_project_package_targetproject_targetpackage_from_args(
args: List[str],
project_is_optional: bool = False,
default_project: str = None,
default_project: Optional[str] = None,
package_is_optional: bool = False,
default_package: str = None,
default_package: Optional[str] = None,
target_project_is_optional: bool = False,
default_target_project: str = None,
default_target_project: Optional[str] = None,
target_package_is_optional: bool = False,
default_target_package: str = None,
default_target_package: Optional[str] = None,
):
"""
Pop project, package, target project and target package from given `args`.
Expand Down Expand Up @@ -4914,6 +4915,8 @@ def do_rdiff(self, subcmd, opts, *args):
rev2 = -rev - 1
else:
return
rev1 = str(rev1)
rev2 = str(rev2)
except:
print(f'Revision \'{opts.change}\' not an integer', file=sys.stderr)
return
Expand Down
39 changes: 20 additions & 19 deletions osc/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -1325,7 +1325,7 @@ def show_package_trigger_reason(apiurl: str, prj: str, pac: str, repo: str, arch
raise


def show_package_meta(apiurl: str, prj: str, pac: str, meta=False, blame=None):
def show_package_meta(apiurl: str, prj: str, pac: str, meta=False, blame=None) -> List[bytes]:
query: Dict[str, Union[str, int]] = {}
if meta:
query['meta'] = 1
Expand Down Expand Up @@ -2320,7 +2320,7 @@ def get_request_collection(
package=None,
states=None,
review_states=None,
types: List[str] = None,
types: Optional[List[str]] = None,
ids=None,
withfullhistory=False
):
Expand Down Expand Up @@ -2863,20 +2863,20 @@ def get_source_file_diff(dir, filename, rev, oldfilename=None, olddir=None, orig

def server_diff(
apiurl: str,
old_project: str,
old_package: str,
old_revision: str,
old_project: Optional[str],
old_package: Optional[str],
old_revision: Optional[str],
new_project: str,
new_package: str,
new_revision: str,
new_revision: Optional[str],
unified=False,
missingok=False,
meta=False,
expand=True,
onlyissues=False,
full=True,
xml=False,
files: list = None,
files: Optional[list] = None,
):
query: Dict[str, Union[str, int]] = {"cmd": "diff"}
if expand:
Expand Down Expand Up @@ -2929,19 +2929,19 @@ def server_diff(

def server_diff_noex(
apiurl: str,
old_project: str,
old_package: str,
old_revision: str,
old_project: Optional[str],
old_package: Optional[str],
old_revision: Optional[str],
new_project: str,
new_package: str,
new_revision: str,
new_revision: Optional[str],
unified=False,
missingok=False,
meta=False,
expand=True,
onlyissues=False,
xml=False,
files: list = None,
files: Optional[list] = None,
):
try:
return server_diff(apiurl,
Expand Down Expand Up @@ -3096,7 +3096,7 @@ def checkout_package(
pathname=None,
prj_obj=None,
expand_link=False,
prj_dir: Path=None,
prj_dir: Optional[Path] = None,
server_service_files=None,
service_files=None,
native_obs_package=False,
Expand Down Expand Up @@ -3214,9 +3214,9 @@ def checkout_package(


def replace_pkg_meta(
pkgmeta, new_name: str, new_prj: str, keep_maintainers=False, dst_userid=None, keep_develproject=False,
pkgmeta: List[bytes], new_name: str, new_prj: str, keep_maintainers=False, dst_userid=None, keep_develproject=False,
keep_lock: bool = False, keep_scmsync: bool = True,
):
) -> str:
"""
update pkgmeta with new new_name and new_prj and set calling user as the
only maintainer (unless keep_maintainers is set). Additionally remove the
Expand Down Expand Up @@ -3450,7 +3450,7 @@ def aggregate_pac(

if meta_change:
src_meta = show_package_meta(apiurl, src_project, src_package_meta)
dst_meta = replace_pkg_meta(src_meta, dst_package_meta, dst_project)
dst_meta = replace_pkg_meta(src_meta, dst_package_meta, dst_project).split("\n")
meta_change = True

if disable_publish:
Expand Down Expand Up @@ -3808,7 +3808,7 @@ def copy_pac(
return 'Done.'


def lock(apiurl: str, project: str, package: str, msg: str = None):
def lock(apiurl: str, project: str, package: str, msg: Optional[str] = None):
url_path = ["source", project]
if package:
url_path += [package]
Expand Down Expand Up @@ -4756,6 +4756,7 @@ def get_commitlog(
# revision is srcmd5
revision_list = [i for i in revision_list if i.srcmd5 == revision]
else:
assert revision is not None
revision = int(revision)
if revision_is_empty(revision_upper):
revision_list = [i for i in revision_list if i.rev == revision]
Expand Down Expand Up @@ -5158,7 +5159,7 @@ def owner(
return res


def set_link_rev(apiurl: str, project: str, package: str, revision="", expand=False, msg: str=None, vrev: str=None):
def set_link_rev(apiurl: str, project: str, package: str, revision="", expand=False, msg: Optional[str] = None, vrev: Optional[str] = None):
url = makeurl(apiurl, ["source", project, package, "_link"])
try:
f = http_GET(url)
Expand All @@ -5179,7 +5180,7 @@ def set_link_rev(apiurl: str, project: str, package: str, revision="", expand=Fa
return revision


def _set_link_rev(apiurl: str, project: str, package: str, root, revision="", expand=False, setvrev: str=None):
def _set_link_rev(apiurl: str, project: str, package: str, root, revision="", expand=False, setvrev: Optional[str] = None):
"""
Updates the rev attribute of the _link xml. If revision is set to None
the rev and vrev attributes are removed from the _link xml.
Expand Down
1 change: 0 additions & 1 deletion osc/meter.py
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,6 @@ def create_text_meter(*args, **kwargs) -> TextMeterBase:

use_pb_fallback = kwargs.pop("use_pb_fallback", False)

meter_class: TextMeterBase
if config.quiet:
meter_class = NoTextMeter
elif not have_pb_module or not config.show_download_progress or not sys.stdout.isatty() or use_pb_fallback:
Expand Down
6 changes: 4 additions & 2 deletions osc/output/output.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,9 @@
import subprocess
import sys
import tempfile
from typing import BinaryIO
from typing import Dict
from typing import Generator
from typing import List
from typing import Optional
from typing import TextIO
Expand Down Expand Up @@ -137,7 +139,7 @@ def safe_print(*args, **kwargs):
print(*args, **kwargs)


def safe_write(file: TextIO, text: Union[str, bytes], *, add_newline: bool = False):
def safe_write(file: Union[BinaryIO, TextIO], text: Union[str, bytes], *, add_newline: bool = False):
"""
Run sanitize_text() on ``text`` and write it to ``file``.
Expand Down Expand Up @@ -211,7 +213,7 @@ def run_pager(message: Union[bytes, str], tmp_suffix: str = ""):
run_external(*cmd, env=env)


def pipe_to_pager(lines: Union[List[bytes], List[str]], *, add_newlines=False):
def pipe_to_pager(lines: Union[List[bytes], List[str], Generator[bytes, None, None], Generator[str, None, None]], *, add_newlines=False):
"""
Pipe ``lines`` to the pager.
If running in a non-interactive terminal, print the data instead.
Expand Down
2 changes: 1 addition & 1 deletion osc/util/ar.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ def __str__(self):
class ArHdr:
"""Represents an ar header entry"""

def __init__(self, fn: bytes, date: bytes, uid: bytes, gid: bytes, mode: bytes, size: bytes, fmag: bytes, off: bytes):
def __init__(self, fn: bytes, date: bytes, uid: bytes, gid: bytes, mode: bytes, size: bytes, fmag: bytes, off: int):
self.file = fn.strip()
self.date = date.strip()
self.uid = uid.strip()
Expand Down
2 changes: 1 addition & 1 deletion osc/util/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -782,7 +782,7 @@ def xml_request(
apiurl: str,
path: List[str],
query: Optional[dict] = None,
headers: Optional[str] = None,
headers: Optional[dict] = None,
data: Optional[str] = None,
) -> urllib3.response.HTTPResponse:
from ..connection import http_request
Expand Down
19 changes: 18 additions & 1 deletion osc/util/safewriter.py
Original file line number Diff line number Diff line change
@@ -1,22 +1,39 @@
import io

# be careful when debugging this code:
# don't add print statements when setting sys.stdout = SafeWriter(sys.stdout)...
class SafeWriter:
class SafeWriter(io.TextIOBase):
"""
Safely write an (unicode) str. In case of an "UnicodeEncodeError" the
the str is encoded with the "encoding" encoding.
All getattr, setattr calls are passed through to the "writer" instance.
"""

def __init__(self, writer, encoding='unicode_escape'):
super().__init__()
self._writer = writer
self._encoding = encoding

# TextIOBase requires overriding the following stub methods: detach, read, readline, and write

def detach(self, *args, **kwargs):
return self._writer.detach(*args, **kwargs)

def read(self, *args, **kwargs):
return self._writer.read(args, **kwargs)

def readline(self, *args, **kwargs):
return self._writer.readline(args, **kwargs)

def write(self, s):
try:
self._writer.write(s)
except UnicodeEncodeError as e:
self._writer.write(s.encode(self._encoding))

def fileno(self, *args, **kwargs):
return self._writer.fileno(*args, **kwargs)

def __getattr__(self, name):
return getattr(self._writer, name)

Expand Down
Loading

0 comments on commit 3bb44b0

Please sign in to comment.