Cross Compilation Adventures with Python
This post is part of a series.
TLDR; I want to build cross-platform CLI utility tools that can be compiled on my laptop and run seamlessly on other platforms.
Ask anyone what progamming language should a beginner start with and the answer would almost certainly include Python. Python is a very popular programming language with a very mature ecosystem. It is also very easy to learn and to use. It has all the features that you would expect from a programming language. I dabble in Python often, whenever I want to build a quick processing tool. Considering it to a build a CLI app, is but natural step forward. Python can run on multiple platforms, including Windows, macOS, and Linux, without the need for any additional dependencies or libraries. However note that Python itself cannot build executables for other platforms. It works the same way Java does, where the installed Python executable on each platform can execute the same python file, so in that sense it is cross platform, but still it cannot cross-compile to other platforms. There are ways to create an executable binary using a Python file, which you will learn in this post and more.
Sounds good! Let’s dive into building a very basic CLI tool.
<p>You will build the same example as in the last post.</p>
A good example to showcase would be to build a CLI tool that can convert from Β°C to F and vice versa. Our tool will take an input for value and the unit to be converted to, then output would be converted temprature value.
<p><strong>NOTE</strong>: I am using macOS (M2 Pro, Apple Silicon), so the instructions follow through using that only. However the steps should work on all platform with little tweaks.</p>
First we need to make sure we have access to python3. If you don’t have access then go ahead and install it.
Once installed, you should have access to python3
in your Terminal. If not restart your session or open a new Terminal window so it is loaded in the PATH. Follow through next steps
Create a file named run.py
.
touch run.py
Add the below code to the run.py
file and save the file.
import sys
def celsius_to_fahrenheit(celsius):
return (celsius * 9 / 5) + 32
def fahrenheit_to_celsius(fahrenheit):
return ((fahrenheit - 32) * 5 / 9)
def main():
if len(sys.argv) != 3:
print("Usage: ./run <value> <unit_to_convert_to>")
return
value = float(sys.argv[1])
unit = sys.argv[2].upper()
if not value:
print("Invalid temperature value.")
return
converted_temperature = None
if unit == "C":
converted_temperature = celsius_to_fahrenheit(value)
elif unit == "F":
converted_temperature = fahrenheit_to_celsius(value)
else:
print("Invalid unit. Please use C or F.")
return
if converted_temperature is not None:
print(f"Converted temperature: {converted_temperature}Β°{unit}")
if __name__ == "__main__":
main()
<p>I am not going to explain this code as it is simple and self explanatory.</p> <p>To understand and learn the language you can use <a href="https://learnxinyminutes.com/docs/python/" target="_blank" rel="noopener">Learn X in Y minutes: Python</a> π</p>
Now to compile, you could execute the python3
compiler with the run.py
source file:
python3 run.py
This will compile and run your code. This itself is a cross platform approach. i.e you usually would have python come pre-installed in the common OS you use. This means you can effectively run your python code on all the OS that have python installed, by simply running python3 run.py
.
To make it look like that you have an executable, you can use a trick to wrap your command inside a bash (*nix)/bat(windows) script and make it executable, as shown below
# Create the bash script
β― touch run
echo '#!/usr/bin/env bash' > run
echo 'python3 run.py $1 $2' >> run
# Make the script executable
β― chmod +x run
# Execute the bash script
β― ./run
Usage: ./run <value> <unit_to_convert_to>
# Pass args to your bash script
β― ./run 49 C
Converted temperature: 120.2Β°C
From the outside it looks like we have achieved what we wanted, but that is not true. This is not the same as an executable binary. It has a dependency on the python3
compiler to be available in the PATH. Also in order for this to work, the run.py
file needs to be in the same directory as the run
bash script.
So while this solution is a straight forward solution when using Python, it isn’t a true cross compiled binary.
That is not what I want. I would like to have an executable binary file that I can share around. I don’t want to have a dependency on Python either. Once compiled it should require no dependencies. Python doesn’t directly compile to an executable, instead its code needs to be bundled into an executable binary. There are 2 commonly known bundlers to do this job.
- Nuitika
<p>Nuitka is the optimizing Python compiler written in Python that creates executables that run without an need for a separate installer.</p>
- PyInstaller
<p>PyInstaller bundles a Python application and all its dependencies into a single package. The user can run the packaged app without installing a Python interpreter or any modules</p>
Compile into executable binary using Nuitika
Install nuitika Standard
python3 -m pip install -U nuitka
<p>There is also <a href="https://nuitka.net/index.html#nuitka-commercial" target="_blank" rel="noopener">Nuitka Commercial</a>. It additionally protects your code, data and outputs, so that users of the executable cannot access these. This a private repository of plugins that you pay to get access to. Additionally, you can purchase priority support.</p>
Build an executable binary
Execute the below command for your run.py
file
python3 -m nuitka run.py \
--output-filename=run --onefile \
--remove-output --quiet
<p><strong>NOTE</strong>: This will prompt you to download a C caching tool (to speed up repeated compilation of generated C code). Say yes to the question, when prompted.</p>
To build an Optimized executable binary, execute the same build command above for your run.py
file with --lto=yes
:
python3 -m nuitka run.py \
--output-filename=run --onefile \
--remove-output --quiet \
--lto=yes
You should now have a binary generated in the same directory with the same name as the py file i.e run
<p><strong>NOTE</strong>: I use <a href="https://github.com/bootandy/dust" target="_blank" rel="noopener"><code>dust</code></a> CLI tool to list files in directory with their sizes.</p>
Compile into executable binary using PyInstaller
Install PyInstaller
python3 -m pip install -U pyinstaller
Build an executable binary
Execute the below command for your run.py
file
pyinstaller run.py \
--onefile --distpath . \
--log-level ERROR --clean --noconfirm
To build an Optimized executable binary, execute the same build command above for your run.py
file with --strip
:
pyinstaller run.py \
--onefile --distpath . \
--log-level ERROR --clean --noconfirm \
--strip
You should now have a binary generated in the same directory with the same name as the py file i.e run
Time to execute our generated run
binary file:
β― ./run
Usage: ./run <value> <unit_to_convert_to>
Didn’t work π, but we have a helpful message stating how to use the CLI tool π
β― ./run 49 C
Converted temperature: 120.2Β°C
Done! That was a super quick intro to working with Python Compiler and Python Language in less than 5 mins π
However, this generated binary would work on only macOS system.
… the reason for that comes from known limitations in these bundlers, as none of them can cross compile to other OS platforms:
PyInstaller: To run your generated Python executable binary on multiple operating systems like Windows and macOS, you’ll need to create separate bundles for each platform using PyInstaller. Unfortunately, this means installing and running PyInstaller on each OS separately.
<p>There is a <a href="https://github.com/sayyid5416/pyinstaller" target="_blank" rel="noopener">Github Action</a>, that can help overcome this by building run.py for each OS by setting up the required python and pyinstaller environment.</p>
Nuitika: Same as Pyinstaller, it is required to install and run Nuitika on each OS separtely to compile an executable binary for that OS.
<p>There is a <a href="https://github.com/Nuitka/Nuitka-Action" target="_blank" rel="noopener">Github Action</a>, that can help overcome this by building run.py for each OS by setting up the required python and pyinstaller environment.</p>
Thats it. I think Python Language even though being one of the more maintainable and simpler language has a big issue when it comes to building CLI apps as it is not straightforward to cross compile the binaries. The work around make everything look clunky and more like duct-tapped to work together. For a mature langauge such as Pythin cross compilation should be a must have feature. Although I do agree Python does away with that drawback by being cross platform via the Python Compiler which can execute the same python code on any platform. For building CLI apps though, having dependencies makes it unattractive as a language to use.
To summarize:
Generate cross-platform binaries | Can cross-compile to platforms | Easy syntax, so maintainable code |
---|---|---|
β | β | β |
<p><strong>BONUS</strong>: <a href="https://peps.python.org/pep-0720/" target="_blank" rel="noopener">PEP-0720</a> is a proposal made in the Python Enhancement Proposals, that aims to talk about Cross-compiling Python packages. Give it a read, as it is interesting to understand the current landscape of cross compiling Python. Also there is a way to <a href="https://andreafortuna.org/2017/12/27/how-to-cross-compile-a-python-script-into-a-windows-executable-on-linux/" target="_blank" rel="noopener">compile an executable binary for Windows platform using Wine</a></p>
I’ll be trying this approach of evaluating more languages in the future. You can find the code for this post here.