It is well known that Python has a problem when it comes to installation of applications since shared libraries can sometimes have significant API changes from one version to another. Added to the complication is the fact that many linux based distros have a preinstalled version of Python that comes with old versions of certain libraries that if overwritten can affect OS stability.

In this guide I will outline the best way to install Python applications & libraries from PyPI without breaking your system and causing a dependency hell.


Is the program I am about to install an application or a library?

In the context of installation we define an application as a Python program which exposes a user interface that can be interacted with. For eg. mitmproxy and dockersh both offer a CLI when called from the terminal which you can use in real time to complete tasks.

Libraries on the other hand are the building blocks of above mentioned applications. They are usually installed on their own only when developing applications or building other libraries. For e.g requests and boto3 can only be imported from within a Python program. Calling them from the terminal serves no purpose.

Sometimes a downloadable package exposes a CLI but also pulls in libraries that you can use in your programs to build new tools. Examples of such hybrid packages are Impacket. When dealing with such a package you should look at the purpose you require the package for. Do you wish to use it’s CLI or do you wish to use the modules it provides to build an application? Depending on this purpose the method of installation changes as described below.


Installing applications

The best way to install Python applications going forward is using pipx. On an Ubuntu system type the following in the terminal

sudo apt install python3-pip pipx python3-venv
python3 -m pipx ensurepath

pipx allows you to run Python applications by installing them in isolated environments along with their dependencies. If your distro’s package repository doesn’t contain pipx you can visit this link and find installation instructions for your platform.

  • Apps that don’t require root privileges

Applications that don’t require root privileges can be installed using the following command

python3 -m pipx install PACKAGE

In the above command replace PACKAGE with the name of the package you wish to install. For eg. python3 -m pipx install mitmproxy

  • Apps that require root privileges

For applications that require root privileges use the following command

sudo PIPX_HOME=/opt/pipx PIPX_BIN_DIR=/usr/local/bin python3 -m pipx install PACKAGE

In the above command replace PACKAGE with the name of the package you wish to install. For eg. sudo PIPX_HOME=/opt/pipx PIPX_BIN_DIR=/usr/local/bin python3 -m pipx install dockersh


Installing libraries

Since isolated installation of libraries are only really done during development you should only be installing them inside of python virtual environments. To get started we first need to install venv on our system.

sudo apt install python3-venv

Now we create a new folder that will hold our project and then we will configure a virtual environment inside of it.

mkdir myproject & cd myproject
python3 -m venv myproject-venv

Be sure to add the myproject-venv/ folder to .gitignore if you are using git based workflows.


To begin development you need to activate the virtual environment which can be done by

source myproject-venv/bin/activate

This will change your prompt to (myproject-venv) user@mycomputer:~/myproject$ which signifies that the virtual environment is currently active.


You can now install any library from pip and it will be installed in the current virtual environment without affecting your system or any other application present on your system.

python3 -m pip install PACKAGE

Finally when you are done with development work and wish to revert back to your regular system, simply enter deactivate at the terminal.

deactivate

  • Using sudo inside a virtual environment

A lot of times the application you are building or a component library requires root privileges to work correctly. In order to use sudo inside a virtual environment it has to be called differently than usual.

sudo -s "PATH=$PATH" python3 main.py

This will ensure that the sudo uses the current virtual environment’s Python interpreter than the system’s interpreter.


Conclusion

Note that in the above commands we invoke pip, pipx and venv as python modules like python3 -m pipx --version instead of pipx --version. Always invoke these tools in this manner as the latter method can result in unexpected behaviour at times. The practices described above should serve adequately for most users. They will only slightly change if you are working with pyenv or a recent version of Python that you installed on your own. If you have any queries feel free to reach out to me at solamarpreet@protonmail.com