The primary purpose of runvenv is to solve the problem of launching executable python scripts within a named environment using the shebang system of Unix-like systems. For example:
testscript.py:
#!/usr/bin/env -S runvenv myscriptenv
import sys
print(sys.executable)
With the above script marked as an executable file, a Unix-like system will execute "/usr/bin/env -S runvenv myscriptenv testscript.py
". Then runvenv will look in its search path for the environment named myscriptenv
, find the python executable within that environment, and execute testscript.py
with it. Preferencing the usage of /usr/bin/env
to call runvenv allows the same shebang line to work identically across platforms, and both with and without root access for installing runvenv, requiring only a common environment name. For single system tools, the direct path location of runvenv can be placed on the shebang line.
Because running python directly picks up how python was invoked and not where an environment was created, this can cause cross-linked dependencies to quietly arise in environments with python -m pip
usage that have symlinks in the path. To help with managing a clean invocation of pip that corresponds to the single defitional path that an environment was created with, this tool also provides a runpip myenvironment
which can be used to reliably invoke pip.
When invoked, runvenv will look for venv or conda python environments with the given name in the following order, silently skipping over options which do not exist:
- In the realpath directory in which the script is located.
- In
~/.venv/
- In
/usr/local/share/venv/
- Environments found in
~/.conda/environments.txt
- Environments found in
envs_dirs:
directory entries in the~/.condarc
yaml file. - In
~/.conda/envs/
The first match found of a myscriptenv/bin/python
will be used to execute the script.
Create a git repo with a script.py
that requires an environment, and provide support for creating a hidden ./.venv/
directory, listed in .gitignore
, which serves as its enviroment. Create a symlink in ~/bin/script.py
to the script.py
within the repo, and set script.py
to have the shebang line:
#!/usr/bin/env -S runvenv .venv
Then the symlink within the path in ~/bin/script.py
will be executed within the environment setup in the repo directory, executing repo/.venv/bin/python
.
Add an environment to ~/.venv
with "python -m venv ~/.venv/mysharedenv
" and then set scripts to use this with:
#!/usr/bin/env -S runvenv mysharedenv
This example is for system-level scripts requiring environments, or for multi-user systems (e.g. for data analysis) benefitting from users having access to a common environment. Create environments under /usr/local/share/venv/
such as /usr/local/share/venv/mysystemenv
which are readable by all, and set scripts to use:
#!/usr/bin/env -S runvenv mysystemenv
Create a conda environment, enabling support for different python versions, such as with "conda create -n py313 python=3.13
" and then use this without activating any environment by setting scripts to use:
#!/usr/bin/env -S runvenv py313
runpip py313 install ipython
runpip py313 list
The runpip
utility provided will run the corresponding bin/pip
found within the environment, and pass along everything on the line directly to pip.
runipyth py313
The runipyth
convenience utility will launch ipython within any provided environment.
ipython313:
#!/usr/bin/env -S runvenv py313
import re
import sys
from IPython import start_ipython
if __name__ == '__main__':
sys.argv[0] = re.sub(r'(-script\.pyw|\.exe)?$', '', sys.argv[0])
sys.exit(start_ipython())
Run make
. There are no non-standard dependencies.
Copy the produced runvenv executable into /usr/bin/
or /usr/local/bin/
for a system level install, or somewhere in the path like ~/bin/
for a user level install.
The main part of runvenv is licensed under GPLv3, with the included RCLib files being separately licensed under the permissive Boost license.