How to install, download and build Python wheels

Python Packaging Index (PyPI) is a repository containing several hundred thousand packages.

Install package
# Install a PyPl indexed package optionally with version v.v
pip install <packagename>[==v.v]

# test install with dry-run
pip install --dry-run <packagename>

# Upgrade an installed package
pip install --upgrade <packagename>

# Uninstall a package
pip uninstall <packagename>

# Install a package from a repository other than PyPI, such as Github
pip install -e git+<https://github.com/myrepo.git#egg=packagename>

# Install a package from specific index url
pip install <packagename> --index-url https://some.index.url

# Install a package with extra index url
pip install <packagename> --extra-index-url https://some.extra.index.url

# Install package from local .whl file
pip install /path/to/some/package.whl

pip install --find-links /path/to/the/wheel/file/ <package-name>

pip wheel [--no-deps] -w /path/to/the/wheel/files/
Build your own .whl file
Pure python package

When it comes to Python packaging, if your package consists purely of Python code, you can do the following:

1. Make sure Wheel and the latest version of setuptools is installed on your system by running:

python -m pip install -U wheel setuptools

2. Then run:

python setup.py sdist bdist_wheel

This will create both a source distribution (sdist) and a wheel file (bdist_wheel) , along with all of its dependencies. You can now upload your built distributions to PyPI. For more information, see Sharing Your Labor of Love: PyPI Quick and Dirty.

Python package with C libraries

If your package has linked C libraries, you’ll need to create specific build environments, and then compile your package separately for each target operating system you want to support.

An example of wheel building:

# file structure:
setup.py
src/
    mypkg/
        __init__.py
        module.py
        data/
            tables.dat
            spoons.dat
            forks.dat 

Content of setup.py:

from setuptools import setup

setup(name='foo',
      version='1.0',
      description='Python Distribution Example',
      author='lin',
      packages=['mypkg'],
      package_dir={'mypkg': 'src/mypkg'},
      package_data={'mypkg': ['data/*.dat']},
      )

Another setup.py for importing prebuilt external lib dependencies through a dummy package:

# file structure:
setup.py
cvcuda_lib/
    libcvcuda.so.0.3.1
    libcvcuda.so.0
    libcvcuda.so
    libnvcv_types.so.0.3.1
    libnvcv_types.so.0
    libnvcv_types.so
    cvcuda.cpython-38-x86_64-linux-gnu.so
    nvcv.cpython-38-x86_64-linux-gnu.so
from setuptools import setup

setup(
    name='cvcuda_import',
    version='0.3.1',
    description='proxy wheel to bring in cvcuda Library',
    data_files=[('lib', ['cvcuda_lib/libcvcuda.so.0.3.1',
                         'cvcuda_lib/libcvcuda.so.0',
                         'cvcuda_lib/libcvcuda.so',
                         'cvcuda_lib/libnvcv_types.so.0.3.1',
                         'cvcuda_lib/libnvcv_types.so.0',
                         'cvcuda_lib/libnvcv_types.so']),
                ('lib/python3.8/site-packages',['cvcuda_lib/cvcuda.cpython-38-x86_64-linux-gnu.so',
                                                'cvcuda_lib/nvcv.cpython-38-x86_64-linux-gnu.so'])],

    install_requires=["numpy"],
 )
Test package installation

Check the content of the whl file:

unzip -l dist/*.whl


# Test it with a conda env
echo | conda create --name cvcuda-py38 python=3.8
conda activate cvcuda-py38

# Different ways of installing wheel:

# use file name with install:
# pip install [--install-option="--prefix=$PREFIX_PATH"] cvcuda-0.3.1-cp38-cp38-linux_x86_64.whl

# pip install dist/*.whl

# use package name with install:
pip install --find-links ~/temp/cvcuda_wheel/dist/ cvcuda_import

# wheel can also be installed using pip wheel command:
pip wheel [--no-deps] -w ~/temp/cvcuda_wheel/dist/

# Remove the testing env
conda deactivate
conda remove --name cvcuda-py38 --all
Upload your package to PyPI

1. Wheel naming is automatically generated from the wheel building

{dist}-{version}(-{build})?-{python.version}-{os_platform}.whl
# deployment with Python 2.7 on 32 bit Windows example:
# PyYAML-5.3.1-cp27-cp27m-win32.whl

2. Test local installation

pip install --find-links /path/to/the/wheel/file/ <package-name>

3. Upload to the index server

Get the publishing tool twine for this:

pip install -U twine
# Since both sdist and bdist_wheel output to dist/ by default, you can safely tell twine to upload everything under dist/ using a shell wildcard (dist/*).
twine upload dist/* [--repository-url https://$USER:$API_TOKEN@url.to.the.artifactory]
# if no url is provided, the default index server will be PyPI.

References