파이썬 - Version 문자열 다루기(semver 패키지)
의외로 파이썬 환경에서 버전 문자열을 타입(C#처럼 Version 타입)을 다루는 방법이 쉽지 않습니다. ^^;
원래는 distutils에 있었는데요,
C:\temp\Python37> python
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 22:22:05) [MSC v.1916 64 bit (AMD64)] on win32
Type "help", "copyright", "credits" or "license" for more information.
>>> from distutils.version import LooseVersion
>>> v = LooseVersion('3.4.0')
>>> print(v)
3.4.0
>>>
파이썬 3.12부터 distutils가 기본 패키지에서 제거되었습니다. ^^;
C:\python312\org> python
Python 3.12.3 (tags/v3.12.3:f6650f9, Apr 9 2024, 14:05:25) [MSC v.1938 64 bit (AMD64)] on win32
>>> from distutils.version import LooseVersion
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ModuleNotFoundError: No module named 'distutils'
>>>
따라서 distutils를 3.12부터 사용하려면 별도로
setuptools 패키지를 설치해야 한다는 데요, 그나마도 LooserVersion 등의 타입은 없어졌고 오직 Version 타입만 제공합니다.
>>> from setuptools.dist import Version
>>> v = Version('5.6')
>>> print(v)
5.6
차라리 이럴 거라면 애당초 외부 패키지를 사용하든가 직접 만드는 것이 나을 듯합니다. 만약 외부 패키지를 사용하겠다면, 그나마 좋은 대안이 semver 패키지로 보이는데요,
// Docs » Using semver
// ; https://python-semver.readthedocs.io/en/2.9.0/usage.html
python3 -m pip install semver
그런데 이것도 문서만 보고 무작정 사용해서는 안 됩니다. 가령, semver 버전에 따라 Version/VersionInfo이 모두 있거나,
>>> import semver
>>> semver.Version.parse('3.0.1')
Version(major=3, minor=0, patch=1, prerelease=None, build=None)
>>> semver.VersionInfo.parse('3.0.1')
Version(major=3, minor=0, patch=1, prerelease=None, build=None)
>>> print(semver.__version__)
3.0.4
없을 수도 있기 때문입니다.
>>> print(semver.__version__)
2.9.0
>>> semver.Version.parse('3.0.1')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: module 'semver' has no attribute 'Version'
따라서 현실적으로
semver.parse_version_info를 사용하는 것이 좋겠습니다. ^^
그런데 간혹 예외가 발생할 수 있습니다. 예를 들어 제 경우에 파이썬 패키지들의 버전을 semver을 이용해 체크하는데요, 일부 패키지들은 버전 문자열을 희한하게 반환해서,
// pip install psycopg2-binary==2.8
>>> import psycopg2
>>> print(psycopg2.__version__)
2.8 (dt dec pq3 ext lo64)
저런 형태인 경우에는 semver을 직접 사용할 수 없습니다. ^^;
>>> semver.parse_version_info(psycopg2.__version__)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/home/testusr/.local/lib/python3.8/site-packages/semver/_deprecated.py", line 81, in wrapper
return func(*args, **kwargs) # type: ignore
File "/home/testusr/.local/lib/python3.8/site-packages/semver/_deprecated.py", line 170, in parse_version_info
return Version.parse(version)
File "/home/testusr/.local/lib/python3.8/site-packages/semver/version.py", line 644, in parse
raise ValueError(f"{version} is not valid SemVer string")
ValueError: 2.8 (dt dec pq3 ext lo64) is not valid SemVer string
그래서 공백 문자로 나눈 후 버전 문자열만 가져와 넘겼더니 이번에는 "2.8"과 같은 "[major].[minor]"만 있는 문자열은 처리하지 못해 예외가 발생합니다.
>>> semver.parse_version_info('2.8')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/home/testusr/.local/lib/python3.8/site-packages/semver/_deprecated.py", line 81, in wrapper
return func(*args, **kwargs) # type: ignore
File "/home/testusr/.local/lib/python3.8/site-packages/semver/_deprecated.py", line 170, in parse_version_info
return Version.parse(version)
File "/home/testusr/.local/lib/python3.8/site-packages/semver/version.py", line 644, in parse
raise ValueError(f"{version} is not valid SemVer string")
ValueError: 2.8 is not valid SemVer string
이해할 수 없군요, ^^; "3.4.5-pre.2+build.4" 이런 식의 쓸데없는 문자열 처리도 잘 하면서, 왜 굳이 "3.0"과 같은 간단한 문자열을 처리하지 못하게 막았을까요? 암튼 100% 마음에 드는 방법이 없습니다. ^^
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]