Microsoft MVP성태의 닷넷 이야기
스크립트: 76. 파이썬 - Version 문자열 다루기(semver 패키지) [링크 복사], [링크+제목 복사],
조회: 355
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
 

파이썬 - 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% 마음에 드는 방법이 없습니다. ^^




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 5/18/2025]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12401정성태11/5/202019539VS.NET IDE: 153. 닷넷 응용 프로그램에서의 "My Code" 범위와 "Enable Just My Code"의 역할 [1]
12400정성태11/5/202015666오류 유형: 679. Visual Studio - "Source Not Found" 창에 "Decompile source code" 링크가 없는 경우
12399정성태11/5/202019125.NET Framework: 961. C# 9.0 - (10) 대상으로 형식화된 조건식(Target-typed conditional expressions)파일 다운로드1
12398정성태11/4/202018727오류 유형: 678. Windows Server 2008 R2 환경에서 Powershell을 psexec로 원격 실행할 때 hang이 발생하는 문제
12397정성태11/4/202018837.NET Framework: 960. C# - 조건 연산자(?:)를 사용하는 경우 달라지는 메서드 선택 사례파일 다운로드1
12396정성태11/3/202015645VS.NET IDE: 152. Visual Studio - "Tools" / "External Tools..."에 등록된 외부 명령어에 대한 단축키 설정 방법
12395정성태11/3/202018666오류 유형: 677. SSMS로 DB 접근 시 The server principal "..." is not able to access the database "..." under the current security context.
12394정성태11/3/202016290오류 유형: 676. cacls - The Recycle Bin on ... is corrupted. Do you want to empty the Recycle Bin for this drive?
12393정성태11/3/202015643오류 유형: 675. Visual Studio - 닷넷 응용 프로그램 디버깅 시 Disassembly 창에서 BP 설정할 때 "Error while processing breakpoint." 오류
12392정성태11/2/202020135.NET Framework: 959. C# 9.0 - (9) 레코드(Records) [4]파일 다운로드1
12390정성태11/1/202020154디버깅 기술: 173. windbg - System.Configuration.ConfigurationErrorsException 예외 분석 방법
12389정성태11/1/202019209.NET Framework: 958. C# 9.0 - (8) 정적 익명 함수 (static anonymous functions)파일 다운로드1
12388정성태10/29/202018053오류 유형: 674. 어느 순간부터 닷넷 응용 프로그램 실행 시 System.Configuration.ConfigurationErrorsException 예외가 발생한다면?
12387정성태10/28/202018896.NET Framework: 957. C# - static 필드의 정보가 GC Heap에 저장될까요? [3]파일 다운로드1
12386정성태10/28/202019806Linux: 34. 사용자 정보를 함께 출력하는 리눅스의 ps 명령어 사용 방법
12385정성태10/28/202017066오류 유형: 673. openssl - req: No value provided for Subject Attribute CN, skipped
12384정성태10/27/202019518오류 유형: 672. AllowPartiallyTrustedCallers 특성이 적용된 어셈블리의 struct 멤버 메서드를 재정의하면 System.Security.VerificationException 예외 발생
12383정성태10/27/202019542.NET Framework: 956. C# 9.0 - (7) 패턴 일치 개선 사항(Pattern matching enhancements) [3]파일 다운로드1
12382정성태10/26/202017333오류 유형: 671. dotnet build - The local source '...' doesn't exist
12381정성태10/26/202019371VC++: 137. C++ stl map의 사용자 정의 타입을 key로 사용하는 방법 [1]파일 다운로드1
12380정성태10/26/202015024오류 유형: 670. Visual Studio - Squash_FailureCommitsReset
12379정성태10/21/202021060.NET Framework: 955. .NET 메서드의 Signature 바이트 코드 분석 [1]파일 다운로드2
12378정성태10/15/202019866.NET Framework: 954. C# - x86/x64 환경에 따라 달라지는 P/Invoke 함수의 export 이름 [1]파일 다운로드1
12377정성태10/15/202020433디버깅 기술: 172. windbg - 파일 열기 시점에 bp를 걸어 파일명 알아내는 방법(Managed/Unmanaged)
12376정성태10/15/202016229오류 유형: 669. windbg - sos의 name2ee 명령어 실행 시 "Failed to request module list." 오류
12375정성태10/15/202017652Windows: 177. 윈도우 탐색기에서 띄우는 cmd.exe 창의 디렉터리 구분 문자가 'Yen(&#0165;)' 기호로 나오는 경우 [1]
... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...