core/script/version_bump.py

217 lines
6.9 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
"""Helper script to bump the current version."""
import argparse
import re
2018-06-21 21:27:01 +00:00
import subprocess
2018-03-30 07:23:02 +00:00
from packaging.version import Version
from homeassistant import const
from homeassistant.util import dt as dt_util
2018-03-30 07:23:02 +00:00
def _bump_release(release, bump_type):
"""Bump a release tuple consisting of 3 numbers."""
major, minor, patch = release
2019-07-31 19:25:30 +00:00
if bump_type == "patch":
2018-03-30 07:23:02 +00:00
patch += 1
2019-07-31 19:25:30 +00:00
elif bump_type == "minor":
2018-03-30 07:23:02 +00:00
minor += 1
patch = 0
2018-03-30 07:23:02 +00:00
return major, minor, patch
2018-03-30 07:23:02 +00:00
def bump_version(version, bump_type):
"""Return a new version given a current version and action."""
2018-03-30 07:23:02 +00:00
to_change = {}
2019-07-31 19:25:30 +00:00
if bump_type == "minor":
2018-03-30 07:23:02 +00:00
# Convert 0.67.3 to 0.68.0
# Convert 0.67.3.b5 to 0.68.0
# Convert 0.67.3.dev0 to 0.68.0
# Convert 0.67.0.b5 to 0.67.0
# Convert 0.67.0.dev0 to 0.67.0
2019-07-31 19:25:30 +00:00
to_change["dev"] = None
to_change["pre"] = None
2018-03-30 07:23:02 +00:00
if not version.is_prerelease or version.release[2] != 0:
2019-07-31 19:25:30 +00:00
to_change["release"] = _bump_release(version.release, "minor")
2019-07-31 19:25:30 +00:00
elif bump_type == "patch":
# Convert 0.67.3 to 0.67.4
# Convert 0.67.3.b5 to 0.67.3
# Convert 0.67.3.dev0 to 0.67.3
2019-07-31 19:25:30 +00:00
to_change["dev"] = None
to_change["pre"] = None
2018-03-30 07:23:02 +00:00
if not version.is_prerelease:
2019-07-31 19:25:30 +00:00
to_change["release"] = _bump_release(version.release, "patch")
2019-07-31 19:25:30 +00:00
elif bump_type == "dev":
# Convert 0.67.3 to 0.67.4.dev0
# Convert 0.67.3.b5 to 0.67.4.dev0
# Convert 0.67.3.dev0 to 0.67.3.dev1
2018-03-30 07:23:02 +00:00
if version.is_devrelease:
2019-07-31 19:25:30 +00:00
to_change["dev"] = ("dev", version.dev + 1)
else:
2019-07-31 19:25:30 +00:00
to_change["pre"] = ("dev", 0)
to_change["release"] = _bump_release(version.release, "minor")
2019-07-31 19:25:30 +00:00
elif bump_type == "beta":
2018-03-30 07:23:02 +00:00
# Convert 0.67.5 to 0.67.6b0
# Convert 0.67.0.dev0 to 0.67.0b0
# Convert 0.67.5.b4 to 0.67.5b5
if version.is_devrelease:
2019-07-31 19:25:30 +00:00
to_change["dev"] = None
to_change["pre"] = ("b", 0)
2018-03-30 07:23:02 +00:00
elif version.is_prerelease:
2019-07-31 19:25:30 +00:00
if version.pre[0] == "a":
to_change["pre"] = ("b", 0)
if version.pre[0] == "b":
to_change["pre"] = ("b", version.pre[1] + 1)
2018-03-30 07:23:02 +00:00
else:
2019-07-31 19:25:30 +00:00
to_change["pre"] = ("b", 0)
to_change["release"] = _bump_release(version.release, "patch")
else:
2019-07-31 19:25:30 +00:00
to_change["release"] = _bump_release(version.release, "patch")
to_change["pre"] = ("b", 0)
elif bump_type == "nightly":
# Convert 0.70.0d0 to 0.70.0d20190424, fails when run on non dev release
if not version.is_devrelease:
raise ValueError("Can only be run on dev release")
to_change["dev"] = ("dev", dt_util.utcnow().strftime("%Y%m%d"))
2018-03-30 07:23:02 +00:00
else:
assert False, f"Unsupported type: {bump_type}"
2019-07-31 19:25:30 +00:00
temp = Version("0")
2018-03-30 07:23:02 +00:00
temp._version = version._version._replace(**to_change)
return Version(str(temp))
2018-03-30 07:23:02 +00:00
def write_version(version):
"""Update Home Assistant constant file with new version."""
2019-07-31 19:25:30 +00:00
with open("homeassistant/const.py") as fil:
content = fil.read()
2019-07-31 19:25:30 +00:00
major, minor, patch = str(version).split(".", 2)
2018-03-30 07:23:02 +00:00
2021-05-21 17:31:04 +00:00
content = re.sub(
"MAJOR_VERSION: Final = .*\n", f"MAJOR_VERSION: Final = {major}\n", content
)
content = re.sub(
"MINOR_VERSION: Final = .*\n", f"MINOR_VERSION: Final = {minor}\n", content
)
content = re.sub(
"PATCH_VERSION: Final = .*\n", f'PATCH_VERSION: Final = "{patch}"\n', content
)
with open("homeassistant/const.py", "w") as fil:
fil.write(content)
def write_version_metadata(version: Version) -> None:
"""Update pyproject.toml file with new version."""
with open("pyproject.toml", encoding="utf8") as fp:
content = fp.read()
content = re.sub(r"(version\W+=\W).+\n", f'\\g<1>"{version}"\n', content, count=1)
with open("pyproject.toml", "w", encoding="utf8") as fp:
fp.write(content)
def write_ci_workflow(version: Version) -> None:
"""Update ci workflow with new version."""
with open(".github/workflows/ci.yaml") as fp:
content = fp.read()
short_version = ".".join(str(version).split(".", maxsplit=2)[:2])
content = re.sub(
2023-08-30 21:56:25 +00:00
r"(\n\W+HA_SHORT_VERSION: )\"\d{4}\.\d{1,2}\"\n",
f'\\g<1>"{short_version}"\n',
content,
count=1,
)
with open(".github/workflows/ci.yaml", "w") as fp:
fp.write(content)
def main():
"""Execute script."""
2019-07-31 19:25:30 +00:00
parser = argparse.ArgumentParser(description="Bump version of Home Assistant")
parser.add_argument(
2019-07-31 19:25:30 +00:00
"type",
help="The type of the bump the version to.",
choices=["beta", "dev", "patch", "minor", "nightly"],
)
2018-06-21 21:27:01 +00:00
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--commit", action="store_true", help="Create a version bump commit."
)
arguments = parser.parse_args()
2023-08-19 12:17:17 +00:00
if (
arguments.commit
and subprocess.run(["git", "diff", "--quiet"], check=False).returncode == 1
):
print("Cannot use --commit because git is dirty.")
return
2018-03-30 07:23:02 +00:00
current = Version(const.__version__)
bumped = bump_version(current, arguments.type)
2019-07-31 19:25:30 +00:00
assert bumped > current, "BUG! New version is not newer than old version"
2018-03-30 07:23:02 +00:00
write_version(bumped)
write_version_metadata(bumped)
write_ci_workflow(bumped)
2022-02-22 10:27:38 +00:00
print(bumped)
2018-06-21 21:27:01 +00:00
if not arguments.commit:
return
subprocess.run(["git", "commit", "-nam", f"Bump version to {bumped}"], check=True)
2018-06-21 21:27:01 +00:00
def test_bump_version():
"""Make sure it all works."""
import pytest
2019-07-31 19:25:30 +00:00
assert bump_version(Version("0.56.0"), "beta") == Version("0.56.1b0")
assert bump_version(Version("0.56.0b3"), "beta") == Version("0.56.0b4")
assert bump_version(Version("0.56.0.dev0"), "beta") == Version("0.56.0b0")
assert bump_version(Version("0.56.3"), "dev") == Version("0.57.0.dev0")
assert bump_version(Version("0.56.0b3"), "dev") == Version("0.57.0.dev0")
assert bump_version(Version("0.56.0.dev0"), "dev") == Version("0.56.0.dev1")
assert bump_version(Version("0.56.3"), "patch") == Version("0.56.4")
assert bump_version(Version("0.56.3.b3"), "patch") == Version("0.56.3")
assert bump_version(Version("0.56.0.dev0"), "patch") == Version("0.56.0")
assert bump_version(Version("0.56.0"), "minor") == Version("0.57.0")
assert bump_version(Version("0.56.3"), "minor") == Version("0.57.0")
assert bump_version(Version("0.56.0.b3"), "minor") == Version("0.56.0")
assert bump_version(Version("0.56.3.b3"), "minor") == Version("0.57.0")
assert bump_version(Version("0.56.0.dev0"), "minor") == Version("0.56.0")
assert bump_version(Version("0.56.2.dev0"), "minor") == Version("0.57.0")
today = dt_util.utcnow().strftime("%Y%m%d")
assert bump_version(Version("0.56.0.dev0"), "nightly") == Version(
f"0.56.0.dev{today}"
)
with pytest.raises(ValueError):
assert bump_version(Version("0.56.0"), "nightly")
2019-07-31 19:25:30 +00:00
if __name__ == "__main__":
main()