Compare commits

..

29 Commits

Author SHA1 Message Date
4906d867d0 Iteration 2023-07-22 05:38:34 -04:00
930d1b14b1 Iteration 2023-07-22 05:30:49 -04:00
02fe81d919 Iteration 2023-07-22 05:30:02 -04:00
849874cec6 Iteration 2023-07-22 05:28:03 -04:00
da1e716dde GitHub Actions Iterations 2023-07-22 05:24:36 -04:00
d2a3d5de6f GitHub Actions Iterations 2023-07-22 05:17:53 -04:00
f5204b496f GitHub Actions Iterations 2023-07-22 05:16:32 -04:00
3eb8b3da81 GitHub Actions Iterations 2023-07-22 05:15:27 -04:00
405ce258f2 GitHub Actions Iterations 2023-07-22 05:12:45 -04:00
1138c97865 GitHub Actions Iterations 2023-07-22 05:11:51 -04:00
8b214aab3d GitHub Actions Iterations 2023-07-22 05:11:03 -04:00
6a2b5da00b GitHub Actions Iterations 2023-07-22 05:09:18 -04:00
5c9bcfdac2 GitHub Actions Iterations 2023-07-22 05:04:59 -04:00
05b2278154 GitHub Actions Iterations 2023-07-22 05:03:40 -04:00
8976f24186 toml 2023-07-22 05:03:12 -04:00
2de8d7d597 GitHub Actions Iterations 2023-07-22 05:02:52 -04:00
29daef3b46 GitHub Actions Iterations 2023-07-22 04:59:16 -04:00
98fb6dada1 GitHub Actions Iterations 2023-07-22 04:57:52 -04:00
81f0cf33aa GitHub Actions Iterations 2023-07-22 04:54:53 -04:00
4d25ef7719 GitHub Actions Iterations 2023-07-22 04:52:29 -04:00
eecdc5661d GitHub Actions Iterations 2023-07-22 04:51:42 -04:00
f36e7be2d8 GitHub Actions Iterations 2023-07-22 04:51:11 -04:00
a273f83873 GitHub Actions Iterations 2023-07-22 04:50:33 -04:00
0ae43a314a GitHub Actions Iterations 2023-07-22 04:49:16 -04:00
df27ad2041 GitHub Actions Iterations 2023-07-22 04:48:41 -04:00
06671174b2 GitHub Actions Iterations 2023-07-22 04:48:03 -04:00
036b720f2e GitHub Actions Iterations 2023-07-22 04:47:13 -04:00
b6bfc2409c Add unit test workflow 2023-07-22 04:46:03 -04:00
28eaf580b4 Get config properly 2023-07-02 23:32:31 -04:00
6 changed files with 100 additions and 124 deletions

75
.github/workflows/unittests.yaml vendored Normal file
View File

@ -0,0 +1,75 @@
name: Pytest
on:
push:
branches:
- master
- feature/refactor
pull_request: {}
jobs:
test:
name: Unit Tests
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-22.04, windows-latest, macos-latest]
python-version: ["3.8", "3.9", "3.10", "3.11"]
steps:
- name: Checkout
uses: actions/checkout@v3
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v2
with:
python-version: ${{ matrix.python-version }}
cache: pip
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install pytest pytest-cov
pip install -r requirements.txt
- name: Test
run: |
pytest
coverage:
name: Coverage Summary
runs-on: ubuntu-22.04
permissions:
pull-requests: write
contents: read
if: github.event_name == 'pull_request'
strategy:
fail-fast: false
steps:
- name: Checkout
uses: actions/checkout@v3
- name: Set up Python
uses: actions/setup-python@v2
with:
python-version: "3.11"
cache: pip
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install pytest pytest-cov
pip install -r requirements.txt
- name: Run Coverage
run: |
pytest --junitxml=pytest.xml --cov-report=term-missing:skip-covered --cov=semver | tee pytest-coverage.txt
- name: Comment Result Summary
uses: MishaKav/pytest-coverage-comment@main
with:
pytest-coverage-path: ./pytest-coverage.txt
junitxml-path: ./pytest.xml

View File

@ -1,7 +1,7 @@
import re import re
import subprocess import subprocess
from typing import Union, List from typing import Union, List
from functools import cache from functools import lru_cache
import toml import toml
@ -9,6 +9,7 @@ from semver.scm import SCM
from semver.logger import logger from semver.logger import logger
from semver.version_type import VersionType from semver.version_type import VersionType
from semver.exceptions import SemverException from semver.exceptions import SemverException
from semver.utils import get_settings
class Git(SCM): class Git(SCM):
@ -28,7 +29,7 @@ class Git(SCM):
def _run_command( def _run_command(
self, *args: str, throwExceptions: bool = True self, *args: str, throwExceptions: bool = True
) -> subprocess.CompletedProcess[str]: ) -> subprocess.CompletedProcess:
return subprocess.run( return subprocess.run(
args, args,
capture_output=True, capture_output=True,
@ -58,7 +59,7 @@ class Git(SCM):
Get the latest tagged version from git tags Get the latest tagged version from git tags
:return: The latest tagged version :return: The latest tagged version
""" """
config: dict = toml.load("./.bumpversion.cfg") config: dict = get_settings()
tag_expression: str = config["bumpversion"]["tag_name"].replace( tag_expression: str = config["bumpversion"]["tag_name"].replace(
"{new_version}", "[0-9]*.[0-9]*.[0-9]*" "{new_version}", "[0-9]*.[0-9]*.[0-9]*"
@ -81,21 +82,26 @@ class Git(SCM):
f"Error getting latest tagged git version: {str(e.stderr).rstrip()}" f"Error getting latest tagged git version: {str(e.stderr).rstrip()}"
) )
if len(tagged_versions) > 0 and tagged_versions[-1] != "": if (
tagged_versions is not None
and len(tagged_versions) > 0
and tagged_versions[-1] != ""
):
version = tagged_versions[-1] version = tagged_versions[-1]
logger.debug(f"Tag Version: {version}") logger.debug(f"Tag Version: {version}")
return version return version
@cache @lru_cache(maxsize=None)
def get_branch(self) -> str: def get_branch(self) -> str:
""" """
Get the main branch Get the main branch
:return: The main branch
""" """
proc = self._run_command(self.git_bin, "rev-parse", "--abbrev-ref", "HEAD") proc = self._run_command(self.git_bin, "rev-parse", "--abbrev-ref", "HEAD")
return proc.stdout.rstrip() return proc.stdout.rstrip()
@cache @lru_cache(maxsize=None)
def get_merge_branch(self) -> Union[str, None]: def get_merge_branch(self) -> Union[str, None]:
""" """
Get the branches involved in the merge Get the branches involved in the merge

View File

@ -5,32 +5,18 @@ import toml
from semver.scm import SCM from semver.scm import SCM
from semver.logger import logger from semver.logger import logger
from semver.utils import get_settings
from semver.exceptions import SemverException
class Perforce(SCM): class Perforce(SCM):
def __init__(self) -> None: def __init__(self) -> None:
self.p4_bin = "p4"
super().__init__() super().__init__()
def _run_command(
self, *args: str, throwExceptions: bool = True
) -> subprocess.CompletedProcess[str]:
return subprocess.run(
args,
capture_output=True,
text=True,
check=throwExceptions,
)
def get_tag_version(self) -> str: def get_tag_version(self) -> str:
""" """
Get the latest tagged version from Perforce labels Get the latest tagged version from Perforce labels
:return: The latest tagged version :return: The latest tagged version
""" """
config: dict = get_settings() config: dict = toml.load("./.bumpversion.cfg")
tag_expression: str = config["bumpversion"]["tag_name"].replace( tag_expression: str = config["bumpversion"]["tag_name"].replace(
"{new_version}", "[0-9]*.[0-9]*.[0-9]*" "{new_version}", "[0-9]*.[0-9]*.[0-9]*"
@ -44,10 +30,15 @@ class Perforce(SCM):
# If a version is found in Perforce labels, use that the latest labeled version # If a version is found in Perforce labels, use that the latest labeled version
labeled_versions: Union[List[str], None] = None labeled_versions: Union[List[str], None] = None
try: try:
proc = self._run_command(self.p4_bin, "labels", "-e", tag_expression, "-m1") proc = subprocess.run(
["p4", "labels", "-e", tag_expression, "-m1"],
capture_output=True,
text=True,
check=True,
)
labeled_versions = proc.stdout.rstrip().split("\n") labeled_versions = proc.stdout.rstrip().split("\n")
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
raise SemverException( raise RuntimeError(
f"Error getting latest labeled Perforce version: {str(e.stderr).rstrip()}" f"Error getting latest labeled Perforce version: {str(e.stderr).rstrip()}"
) )
@ -56,97 +47,3 @@ class Perforce(SCM):
logger.debug("Label Version: " + str(version)) logger.debug("Label Version: " + str(version))
return version return version
def get_branch(self) -> str:
"""
Returns the name of the current Perforce stream
:return: The current Perforce stream
"""
try:
proc = self._run_command(self.p4_bin, "client", "-o")
client = toml.loads(proc.stdout)
return client["Stream"]
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error getting current Perforce stream: {str(e.stderr).rstrip()}"
)
def get_merge_branch(self) -> Union[str, None]:
"""
Returns the name of the stream being intergrated into mainline
:return: The Perforce stream being intergrated into mainline
"""
try:
proc = self._run_command(self.p4_bin, "client", "-o")
client = toml.loads(proc.stdout)
return client["StreamAtChange"]
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error getting current Perforce stream: {str(e.stderr).rstrip()}"
)
def commit_and_push(self, branch: str) -> None:
"""
Creates a changelist and submits it to Perforce
:param branch: The current Perforce stream
"""
try:
proc = self._run_command(self.p4_bin, "change", "-o")
change = toml.loads(proc.stdout)
change["Description"] = f"Version Bump"
change["Files"] = [f"//{branch}/..."]
proc = subprocess.Popen([self.p4_bin, "change", "-i"], stdin=subprocess.PIPE)
proc.communicate(input=toml.dumps(change).encode())
proc.wait()
proc = self._run_command(self.p4_bin, "submit", "-c", str(change["Change"]))
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error creating a CL and submitting to Perforce: {str(e.stderr).rstrip()}"
)
def tag_version(self, version: str) -> None:
"""
Creates a Perforce label
:param version: The version to label
"""
try:
proc = self._run_command(self.p4_bin, "label", "-o", version)
label = toml.loads(proc.stdout)
label["Description"] = f"Version {version}"
label["Options"] = "locked"
label["Revision"] = f"//{self.get_branch()}/..."
proc = subprocess.Popen([self.p4_bin, "label", "-i"], stdin=subprocess.PIPE)
proc.communicate(input=toml.dumps(label).encode())
proc.wait()
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error creating a Perforce label: {str(e.stderr).rstrip()}"
)
def get_version_hash(self, version: str) -> str:
"""
Returns the hash of the version label
:param version: The version label
:return: The hash of the version label
"""
try:
proc = self._run_command(self.p4_bin, "labels", "-e", version, "-m1")
label = toml.loads(proc.stdout)
return label["Revision"]
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error getting hash of version label: {str(e.stderr).rstrip()}"
)
def get_hash(self) -> str:
"""
Returns the hash of the current commit
:return: The hash of the current commit
"""
try:
proc = self._run_command(self.p4_bin, "changes", "-m1")
change = toml.loads(proc.stdout)
return change["Change"]
except subprocess.CalledProcessError as e:
raise SemverException(
f"Error getting hash of current commit: {str(e.stderr).rstrip()}"
)

View File

@ -67,10 +67,8 @@ class TestSemVer(unittest.TestCase):
@mock.patch("toml.load") @mock.patch("toml.load")
@mock.patch("pathlib.Path.is_file") @mock.patch("pathlib.Path.is_file")
@mock.patch("builtins.open", mock.mock_open()) @mock.patch("builtins.open", mock.mock_open())
@mock.patch("semver.logger.warning")
def test_update_file_version( def test_update_file_version(
self, self,
mock_logger: mock.Mock,
mock_path_is_file: mock.Mock, mock_path_is_file: mock.Mock,
mock_toml_load: mock.Mock, mock_toml_load: mock.Mock,
): ):
@ -83,11 +81,9 @@ class TestSemVer(unittest.TestCase):
} }
mock_path_is_file.return_value = True mock_path_is_file.return_value = True
self.semver._update_file_version("1.0.1", "1.0.0") self.semver._update_file_version("1.0.1", "1.0.0")
mock_logger.assert_not_called()
mock_path_is_file.return_value = False mock_path_is_file.return_value = False
self.semver._update_file_version("1.0.1", "1.0.0") self.semver._update_file_version("1.0.1", "1.0.0")
mock_logger.assert_called_once()
@mock.patch("semver.semver.SemVer._version_repo", mock.MagicMock()) @mock.patch("semver.semver.SemVer._version_repo", mock.MagicMock())
def test_run_ok(self): def test_run_ok(self):

View File

@ -1,6 +1,6 @@
from typing import List from typing import List
from pathlib import Path from pathlib import Path
from functools import cache from functools import lru_cache
import configparser import configparser
import toml import toml
@ -8,7 +8,7 @@ import toml
from semver.exceptions import SemverException from semver.exceptions import SemverException
@cache @lru_cache(maxsize=None)
def get_settings() -> dict: def get_settings() -> dict:
""" """
Get the settings from the config file Get the settings from the config file

View File

@ -24,7 +24,9 @@ classifiers =
[options] [options]
include_package_data = True include_package_data = True
packages = find: packages = find:
python_requires = >=3.7, <4 python_requires = >=3.8, <4
install_requires =
toml
[options.packages.find] [options.packages.find]
exclude = exclude =