Search code examples
pythonbuild

Is there a way to compile a python application into static binary?


What I'm trying to do is ship my code to a remote server, that may have different python version installed and/or may not have packages my app requires.

Right now to achieve such portability I have to build relocatable virtualenv with interpreter and code. That approach has some issues (for example, you have to manually copy a bunch of libraries into your virtualenv, since --always-copy doesn't work as expected) and generally slow.

There's (in theory) a way to build python itself statically.

I wonder if I could pack interpreter with my code into one binary and run my application as module. Something like that: ./mypython -m myapp run or ./mypython -m gunicorn -c ./gunicorn.conf myapp.wsgi:application.


Solution

  • There are two ways you could go about to solve your problem

    1. Use a static builder, like freeze, or pyinstaller, or py2exe
    2. Compile using cython and link against a statically-compiled version of CPython

    This answer explains how you can go about doing it using the second approach, since the first method is not cross platform and version, and has been explained in other answers. Also, using programs like pyinstaller typically results in huge file sizes, while using cython will result in a file that's much smaller

    1. First, install cython.

      sudo -H pip3 install cython
      
    2. Then, you can use cython to generate a C file out of the Python .py file (in reference to https://stackoverflow.com/a/22040484/5714445)

      cython example_file.py --embed
      
    3. Use GCC to compile it after getting your statically-compiled python version (Note: The below assumes you are trying to compile it to Python3)

      gcc -Os $(python3-config --includes) example_file.c -o output_bin_file $(python3-config --ldflags --embed)
      

    You will now have a binary file output_bin_file, which is what you are looking for

    Other things to note:

    1. Change example_file.py to whatever file you are actually trying to compile.
    2. Cython is used to use C-Type Variable definitions for static memory allocation to speed up Python programs. In your case however, you will still be using traditional Python definitions.
    3. If you are using additional libraries (like opencv, for example), you might have to provide the directory to them using -L and then specify the name of the library using -l in the GCC Flags. For more information on this, please refer to GCC flags
    4. The above method might not work for anaconda python, as you will likely have to install a version of gcc that is compatible with your conda-python.
    5. If your Python version is not statically-compiled, this will result in a dynamic binary