diff options
| author | dusoleil <howcansocksbereal@gmail.com> | 2023-02-23 05:29:21 -0500 | 
|---|---|---|
| committer | dusoleil <howcansocksbereal@gmail.com> | 2023-02-23 22:52:57 -0500 | 
| commit | 9f4648dae644f2fb9deb4c68859d6cdfd0bc0530 (patch) | |
| tree | 2adb421d7fd436e7f111dc9b72b7cd937fcc0cb0 | |
| parent | b016d2b55c16d1d7303cd93d5a9a3e2362a9fb58 (diff) | |
| download | nsploit-9f4648dae644f2fb9deb4c68859d6cdfd0bc0530.tar.gz nsploit-9f4648dae644f2fb9deb4c68859d6cdfd0bc0530.zip | |
Dynamically source version in toml from git
Instead of hard-coding the version into the pyproject.toml, we can
dynamically source it at build time.  Ideally, we want to use git
describe as a single authority source on the version.  The version is
stored in sploit.__version__ and can be consumed during sploit runtime
or during a build/package to populate the project's core metadata
version in the toml file.
hatchling provides a tool.hatch.version plugin that can read out the
variable during a build/package.  Because this variable is populated
from a git command, if the source tree isn't in a git repo, it will
fail.  In this case, sploit will report a PEP 440 compliant fake version
"0+unknown.version" to let the user know.
Because a packaged distribution doesn't exist in a git repo, we want to
bake in the version at build time into the package.  hatchling provides
a plugin to help with this, but it had some technical limitations that
didn't quite work for our use case.  Instead, I added a custom build
hook which will take the version sourced from the package (and by proxy
the git command), and overwrite the __init__.py with a hard-coded
version in the __version__ variable. This means that built/packaged
distributions of this project will have a fixed version hard-coded in
rather than dynamically sourcing from git.
The build hook operates just before the build executes.  It seems that
most build/packager front-ends (e.g. build, pip) will just run it in the
current source tree rather than making a temp copy.  This means that
when we modify the __init__.py, it is modifying our git tree.  Ideally,
we want this to be restored at the end of the build.  The build hook
interface allows us to write a hook that happens after the build, but it
won't run in the case of a crash or failed build.  Instead, I added a
custom solution to this using a member variable deconstructor.  If the
build ends in any way, the original contents of __init__.py are written
back out.
Signed-off-by: dusoleil <howcansocksbereal@gmail.com>
Reviewed-by: Malfurious <m@lfurio.us>
| -rw-r--r-- | hooks/bake_version.py | 26 | ||||
| -rw-r--r-- | pyproject.toml | 11 | ||||
| -rw-r--r-- | sploit/__init__.py | 2 | ||||
| -rw-r--r-- | sploit/util.py | 20 | 
4 files changed, 54 insertions, 5 deletions
| diff --git a/hooks/bake_version.py b/hooks/bake_version.py new file mode 100644 index 0000000..4d43228 --- /dev/null +++ b/hooks/bake_version.py @@ -0,0 +1,26 @@ +from hatchling.builders.hooks.plugin.interface import BuildHookInterface + +import os +import re + +filename = os.path.normpath(os.path.join("sploit","__init__.py")) + +#put the file back when the build ends +class RestoreVersionFile: +    def __init__(self,contents): +        self.contents = contents +    def __del__(self): +        with open(filename,"w") as f: +            f.write(self.contents) + +class BakeVersionBuildHook(BuildHookInterface): +    def initialize(self,version,build_data): +        with open(filename,"r") as f: +            self.restore = RestoreVersionFile(f.read()) +        pattern = r'(?i)^__version__ *= *(?P<version>.+?)$' +        match = re.search(pattern, self.restore.contents, flags=re.MULTILINE) +        if not match: +            raise ValueError("regex of version file failed") +        span = match.span('version') +        with open(filename,"w") as f: +            f.write(f'{self.restore.contents[:span[0]]}"v{self.metadata.version}"{self.restore.contents[span[1]:]}') diff --git a/pyproject.toml b/pyproject.toml index 385c6ce..d4aa8b3 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,5 @@  [project]  name = "sploit" -version = "0.1"  description = "sploit is a process interaction automation tool with software exploitation focused utilities."  readme = "README.txt"  requires-python = ">=3.8" @@ -10,6 +9,7 @@ authors = [      {name="dusoleil",email="howcansocksbereal@gmail.com"},      {name="Malfurious",email="m@lfurio.us"},  ] +dynamic = ["version"]  [project.urls]  "Homepage" = "https://github.com/dusoleil/sploit" @@ -20,3 +20,12 @@ sploit = "sploit.main:main"  [build-system]  requires = ["hatchling"]  build-backend = "hatchling.build" + +[tool.hatch.version] +source = "code" +path = "sploit/__init__.py" +search-paths = ["."] +expression = "__version__" + +[tool.hatch.build.hooks.custom] +path = "hooks/bake_version.py" diff --git a/sploit/__init__.py b/sploit/__init__.py index f5a82fc..5bac38f 100644 --- a/sploit/__init__.py +++ b/sploit/__init__.py @@ -8,3 +8,5 @@ from sploit import (          util,          rev,  ) + +__version__ = util.git_version() diff --git a/sploit/util.py b/sploit/util.py index c44ab78..3a2b842 100644 --- a/sploit/util.py +++ b/sploit/util.py @@ -1,14 +1,26 @@ +from os import path  from subprocess import run -def run_cmd(cmd): -    return run(cmd,capture_output=True).stdout.decode('utf-8').split('\n')[:-1] +def run_cmd(cmd,cwd=None): +    return run(cmd,cwd=cwd,capture_output=True,text=True,check=True).stdout.split('\n')[:-1]  __RUN_CACHE__ = {} -def run_cmd_cached(cmd): +def run_cmd_cached(cmd,cwd=None):      key = ''.join(cmd)      if key in __RUN_CACHE__:          return __RUN_CACHE__[key]      else: -        result = run_cmd(cmd) +        result = run_cmd(cmd,cwd)          __RUN_CACHE__[key] = result          return result + +#try to get the version through git +def git_version(): +    try: +        cwd = path.dirname(path.realpath(__file__)) +        version = run_cmd(["git","describe","--always","--first-parent","--dirty"],cwd=cwd)[0] +        #PEP440 compliance +        version = version.replace('-','+',1).replace('-','.') +        return version +    except: +        return "0+unknown.version" | 
