Repository

Looks good to me!

User Tools

Site Tools


kb:intranet:software:python3

Python 3

Installation

Probably the more straightforward way to download is by installing natively. Modern Linux systems today tend to come with a pre-packaged version of Python in order to run some userspace pre-packaged programs (ufw is one).

To avoid conflicts with system binaries, Python versions can be managed via pyenv instead. Some installation instructions for Ubuntu 20.04 and 22.04 LTS:

# 1. Pull required binaries into build environment, needed to build python
#    See: https://github.com/pyenv/pyenv/wiki
sudo apt-get update
sudo apt-get install make build-essential libssl-dev zlib1g-dev libbz2-dev \
libreadline-dev libsqlite3-dev wget curl llvm libncursesw5-dev xz-utils tk-dev \
libxml2-dev libxmlsec1-dev libffi-dev liblzma-dev

# 2. Ensure git's autocrlf is set to false
#    See: https://github.com/pyenv/pyenv/issues/1725#issuecomment-724689291
#    Symptoms include: `/usr/bin/env: ‘bash\r’: No such file or directory` when running pyenv
git config --global core.autocrlf false

# 3. Run the automated installer
#    See: https://github.com/pyenv/pyenv-installer
curl -L https://github.com/pyenv/pyenv-installer/raw/master/bin/pyenv-installer | bash

# 4. Add scripts to shell to enable pyenv shims
#    For Bash, run the following commands:
echo 'export PYENV_ROOT="$HOME/.pyenv"' >> ~/.bashrc
echo 'command -v pyenv >/dev/null || export PATH="$PYENV_ROOT/bin:$PATH"' >> ~/.bashrc
echo 'eval "$(pyenv init -)"' >> ~/.bashrc
echo 'export PYENV_ROOT="$HOME/.pyenv"' >> ~/.profile
echo 'command -v pyenv >/dev/null || export PATH="$PYENV_ROOT/bin:$PATH"' >> ~/.profile
echo 'eval "$(pyenv init -)"' >> ~/.profile
echo 'export PYENV_ROOT="$HOME/.pyenv"' >> ~/.bash_profile
echo 'command -v pyenv >/dev/null || export PATH="$PYENV_ROOT/bin:$PATH"' >> ~/.bash_profile
echo 'eval "$(pyenv init -)"' >> ~/.bash_profile

# 5. Restart shell
exec "$SHELL"

# 6. Install Python version, e.g. 3.10.4
pyenv install 3.10.4

# 7. Set Python version as global version to use
     To unset, set it to 'system' insetad.
pyenv global 3.10.4

One great benefit of using pyenv is the ability to seamlessly transition between virtual environments when moving into a project directory, simply by creating a virtual environment and assigning a Python version to it. Multiple versions can be assigned as well.

On Windows platforms, Python should be uninstalled manually via the "Add & Remove Programs" settings page, instead of the usual pyenv uninstall .... The former seems to handle additional processes like removing entries from the registry, while the latter only deletes the program directory.

For OpenSUSE

Scripts

Small scripts to park here, until I manage to get my boilerplate library properly setup:

#!/usr/bin/env python3
# Rename files in "Screenshot from 2022-12-07 09-23-06.png" format to "20221207_092306_" 
# Justin, 2022-12-12
 
"""
Old contents:
#!/bin/sh
ls | grep -E "Screenshot .* ([0-9]{4})" | sed -r "s/([0-9]+)-([0-9]+)-([0-9]+) ([0-9]+)-([0-9]+)-([0-9]+)/\1\2\3_\4\5\6/g"
# TODO: Not complete!
"""
 
import pathlib
 
for path in pathlib.Path().glob("Screenshot from *.png"):
 
    # Extract all numbers
    target = "".join([c for c in str(path) if "0" <= c <= "9"])
    target = target[:8] + "_" + target[8:] + "_.png"
 
    # Rename
    path.rename(path.with_name(target).absolute())

setup.py?

PEP517 in a nutshell:

PS C:\Users\Justin> pip install git+https://github.com/pyuxiang/boiler.git
Collecting git+https://github.com/pyuxiang/boiler.git
  Cloning https://github.com/pyuxiang/boiler.git to c:\users\justin\appdata\local\temp\pip-req-build-rzbbiuw0
  Running command git clone --filter=blob:none --quiet https://github.com/pyuxiang/boiler.git 'C:\Users\Justin\AppData\Local\Temp\pip-req-build-rzbbiuw0'
  Resolved https://github.com/pyuxiang/boiler.git to commit 63522bae91982a15919d07160f68d4068169c336
  Preparing metadata (setup.py) ... done
Requirement already satisfied: numpy in c:\users\justin\.pyenv\pyenv-win\versions\3.11.1\lib\site-packages (from boiler==0.0.5) (1.24.1)
Installing collected packages: boiler
  DEPRECATION: boiler is being installed using the legacy 'setup.py install' method, because it does not have a 'pyproject.toml' and the 'wheel' package is not installed. pip 23.1 will enforce this behaviour change. A possible replacement is to enable the '--use-pep517' option. Discussion can be found at https://github.com/pypa/pip/issues/8559
  Running setup.py install for boiler ... done
Successfully installed boiler-0.0.5

Saving Python history file, without outputs, to copy into script:

import readline
readline.write_history_file("history.txt")

For interactive mode, try:

user:~$ python3 -i -c "from {{MODULE}} import *"

To expose all variables from single function (for quick scripting purposes), use globals().update(locals()).

A post-install script from ''freqcd.c''

See libraries that are likely to be manually imported, i.e. libraries that are not dependencies:

user:~$ pip3 list --not-required

A list of popular libraries on my side:

requirements.txt
# General stuff
arrow
joblib
matplotlib
numpy
pandas
pyserial
requests
requests-cache
scipy
tqdm
uncertainties

# Linting
black
isort

# S15 stuff
setuptools
Cython
fpfind @ git+https://github.com/s-fifteen-instruments/fpfind.git
S15lib @ git+https://github.com/s-fifteen-instruments/pyS15.git

# Personal
kochen @ git+https://github.com/pyuxiang/kochen.git

Running a single file code remotely can be done by passing the script via stdin over SSH (assuming the dependency libraries are already available on the remote side), then reading the output back. If a more complex data structure needs to be passed, one could use a message passing service or construct... or just pickle it if it's back to another Python module. See:

# Remote script, "producer.py"
if __name__ == "__main__":
    import contextlib, pickle, sys
    with contextlib.redirect_stdout(None):
        result = myfunc()
    sys.stdout.buffer.write(pickle.dumps(result))
 
# Local script, "consumer.py"
import pickle, sys
result = pickle.loads(sys.stdin.buffer.read())
user:~$ cat producer.py | ssh <HOST> python3 - <ARGS> | ./consumer.py
 
# If using pyenv, then may need to use a login shell to trigger .bashrc source
user:~$ cat producer.py | ssh <HOST> 'bash -lc "python3 - <ARGS>"' | ./consumer.py

The Python interpreter performance can be improved by setting compile-time optimization flags, as shown in this page, e.g.

env PYTHON_CONFIGURE_OPTS='--enable-optimizations --with-lto' PYTHON_CFLAGS='-march=native -mtune=native' pyenv install 3.10.6

Command-line scripts

Some libraries are designed to be installed as a Python module, with command line tools exposed. If the main method was defined, but no command line tool is available, consider mapping your own:

/usr/bin/aubio
#!/usr/bin/env python3
from aubio import cmd
if __name__ == "__main__":
    cmd.main()

In the case of Aubio, it was just that the commands are exposed only after restarting the shell...

uv

uv is fast becoming one of my favorite package + Python managers, by virtue of how blazingly fast it is. It also caches existing compilations so there's no need for manual building which is what pyenv does. Not really a good idea to expect other users to follow though (need to learn the extra uv commands), so the way to transparently add Python:

user:~$ uv python install 3.12
user:~$ uv venv
 
# Hide virtual environment name by setting to empty string
user:~$ sed -i -E 's/(export VIRTUAL_ENV_PROMPT)/\1=""/g' .venv/bin/activate
 
# Source Python installation
user:~$ echo '. "$HOME/.venv/bin/activate"' >> .profile
user:~$ uv pip install pip

Restarting the shell will result in a base Python + Pip, just like pyenv, but with users none the wiser :)

Some quirks exist though:

  • tkinter is not packaged in the prebuilt Pythons supplied by uv, resulting in matplotlib interactive plotting issues.
    • It does look like installing PyQt6 will allow matplotlib to automatically search for the qt6 backend and use it for interactive plots.
  • A more detailed list of quirks can be found here.
  • clang is used by default when python-standalone is compiled (see here), which necessitates clang and manual linking with numpy headers.
    • One solution is to force usage of gcc by overriding Python's preferred CC backend using sysconfig:
diff --git a/S15lib/g2lib/setup.py b/S15lib/g2lib/setup.py
index 0f38af1..aee29ca 100644
--- a/S15lib/g2lib/setup.py
+++ b/S15lib/g2lib/setup.py
@@ -10,4 +10,10 @@ package = Extension(
         ("NPY_NO_DEPRECATED_API", "NPY_1_7_API_VERSION")
     ],  # https://stackoverflow.com/a/64915608
 )
+
+# Override clang with gcc
+import sysconfig
+sysconfig.get_config_vars()['CC'] = 'gcc'
+sysconfig.get_config_vars()['LDSHARED'] = 'gcc -shared'
+
 setup(ext_modules=cythonize([package], language_level="3"))
kb/intranet/software/python3.txt · Last modified: 6 weeks ago ( 6 December 2024) by justin