Fast, flexible and type-safe Git command execution in Python using subprocess.
- 🧠 Typed: All commands and options are statically type-checked.
- ⚡ Fast: Minimal abstractions over subprocess, runs directly on your system Git.
- 🧩 Composable: Git commands and options can be passed around as objects.
- 🔁 Overridable: Easily override environment variables and options in a chainable, readable manner.
- 📦 Lightweight: No dependencies on heavy Git libraries or C extensions.
- 🧰 Extensible: Future support for output transformers and other plugins.
- 🚨 Exception Handling: Raises any error as a Python-recognisable exception.
- 📤 Debuggable: Exceptions capture
stdout
,stderr
, and the return code of the run command. - 💤 Lazy Execution: Inherently lazily processed.
- 📄 Transparent Output: Returns a Git command's
stdout
as-is. - 🧪 Terminal Functions: Git subcommands are terminal functions.
- 🧼 Idiomatic Python: Write commands in idiomatic Python at compile-time and be confident they’ll execute smoothly at runtime.
pip install gitbolt
Running system commands in Python can be tricky for the following reasons:
- Arguments sent to
subprocess
may not be typed correctly and result in runtime errors. - Argument groups may be mutually exclusive or required conditionally — again causing runtime issues.
- Errors from subprocess are often unhelpful and difficult to debug.
Also, using subprocess effectively means you must:
- Understand and manage process setup, piping, and teardown.
- Know your CLI command intricacies in depth.
This project exists to fix all that — with ergonomics, speed, and type-safety.
Type-checking ensures runtime safety.
Make git command interfaces as ergonomic to the user as possible.
git hash-object
supports taking multiple files and outputs a hash per file. But in practice, it's most often used to write a single file to the Git object database and return its hash. To match this real-world usage, Gitbolt offers a more ergonomic method that accepts one file and returns one hash — while still giving you the flexibility to access the full range of git hash-object
capabilities when needed.
Gitbolt lets you pass subcommands around as typed objects. This enables highly focused, minimal APIs — you can write functions that accept only the subcommands they truly need. This leads to cleaner logic, better separation of concerns, and compile-time guarantees that help prevent misuse.
import gitbolt
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
version_subcmd = git.version_subcmd
add_subcmd = git.add_subcmd
def method_which_only_adds_a_file(add_subcmd: gitbolt.base.Add):
"""
This method only requires the `add` subcommand.
"""
...
method_which_only_adds_a_file(add_subcmd)
git subcommands are modeled as terminal functions that return stdout.
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
status_out = git.status_subcmd.status()
print(status_out)
Extensive use of type-hints ensures that invalid usages fail early — at compile-time. Write at compile-time and be sure that commands run error-free at runtime.
Allow users to set/unset/reset Git environment variables and main command options using typed, chainable, Pythonic methods — just before a subcommand is executed.
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
git = git.git_envs_override(GIT_TRACE=True)
from pathlib import Path
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
git = git.git_envs_override(GIT_TRACE=1, GIT_DIR=Path('/tmp/git-dir/'), GIT_EDITOR='vim')
from pathlib import Path
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
overridden_git = git.git_envs_override(GIT_SSH=Path('/tmp/SSH')).git_envs_override(
GIT_TERMINAL_PROMPT=1,
GIT_NO_REPLACE_OBJECTS=True
)
re_overridden_git = overridden_git.git_envs_override(GIT_TRACE=True)
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
from vt.utils.commons.commons.core_py import UNSET
git = SimpleGitCommand()
overridden_git = git.git_envs_override(GIT_ADVICE=True, GIT_TRACE=True)
no_advice_unset_git = overridden_git.git_envs_override(GIT_TRACE=UNSET)
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
overridden_git = git.git_envs_override(GIT_TRACE=True)
git_trace_reset_git = overridden_git.git_envs_override(GIT_TRACE=False)
Allow users to set/unset/reset git main command options in typed and pythonic manner just before subcommand run to provide maximal flexibility.
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
git = git.git_opts_override(no_replace_objects=True)
from pathlib import Path
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
git = git.git_opts_override(no_replace_objects=True, git_dir=Path(), paginate=True)
from pathlib import Path
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
overridden_git = git.git_opts_override(exec_path=Path('tmp')).git_opts_override(
noglob_pathspecs=True,
no_advice=True
).git_opts_override(
config_env={'auth': 'suhas', 'comm': 'suyog'}
)
re_overridden_git = overridden_git.git_opts_override(glob_pathspecs=True)
from pathlib import Path
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
from vt.utils.commons.commons.core_py import UNSET
git = SimpleGitCommand()
overridden_git = git.git_opts_override(exec_path=Path('tmp'), no_advice=True)
no_advice_unset_git = overridden_git.git_opts_override(no_advice=UNSET)
from gitbolt.git_subprocess.impl.simple import SimpleGitCommand
git = SimpleGitCommand()
overridden_git = git.git_opts_override(no_advice=True)
no_advice_reset_git = overridden_git.git_opts_override(no_advice=False)
Output of git commands is returned as-is. No transformations unless explicitly requested. Transformers for formatting/parsing can be added later.
- 🔄 Composable Git commands.
- 📤 Returns raw stdout.
- 🚨 Exceptions with full context.
- 💤 Lazy execution.
- 🧠 Strong typing and compile-time guarantees.
- 🧼 Idiomatic Python.
- 🧪 Terminal subcommands.
- 💣 Fail-fast on invalid usage.
- Support
pygit2
for direct, fast Git access. - Enable
porcelain
support usingpygit2
where required.