Практический разбор PyPI для Python-инженеров

Kate

Administrator
Команда форума
Меня зовут Юрий Бондаренко, и последние 8 лет я занимаюсь коммерческой разработкой. На данный момент я сотрудничаю с EPAM Ukraine в роли Senior Software Engineer, а в свободное от основной работы время участвую еще в паре проектов. В этой статье хотел бы рассказать о том, что такое PyPI и как он может пригодиться разработчику. И, конечно, я разберу, как создать собственный пакет. На выходе получится небольшая инструкция для всех интересующихся.

Уже во время написания статьи я наткнулся на новость о том, что издательство Manning готовит к выходу свою новую книгу Publishing Python Packages (ISBN 9781617299919). Мне это показалось хорошим маркером того, что сейчас в комьюнити есть большой интерес к этой теме.

Согласно свежему Stack Overflow Global Developers Survey 2021, Python вошел в топ-3 самых популярных технологий. Одна из причин, на мой взгляд, — возможность легко делиться своим идеями и наработками. Для этого инженеру достаточно оформить код в виде пакета и опубликовать его на pypi.org. Есть еще несколько опций, о которых мы поговорим чуть позже.

Эту статью условно можно разделить на две части: подготовка и непосредственная сборка и публикация.

Что же такое этот ваш PyPI​

Каждый, кто пробовал программировать на Python, хотя бы раз в жизни использовал команду:

pip install <package_name>

Но не каждый задумывался, что же за ней скрывается. Pip — это система управления пакетами, которая на данный момент по умолчанию включена в дистрибутив языка. Она используется для установки и управления программными пакетами, написанными на Python. Откуда же она берет сами пакеты? По умолчанию pip ищет пакеты на pypi.org, но у нас есть возможность устанавливать их и из приватных PyPI серверов и Gitрепозиториев. The Python Package Index (PyPI) — это хранилище программного обеспечения для языка Python. Он помогает нам найти и установить ПО, разработанное и распространенное сообществом Python. На момент написания статьи в нем хранится приблизительно 320К проектов. Хотя еще в 2010 году их было всего 10К.

Для чего это все​

Допускаю, что уже на этом этапе большинство задается вопросом: «А зачем мне все это?». Ответ можно разделить на две большие ветки.

Мир open source​

Большинство из нас каждый день пользуется open source проектами и даже не задумывается об этом: взять те же Pydantic, FastAPI, Django и еще много крутых инструментов. Множество Python пакетов распространяются по лицензиям, которые позволяют нам пользоваться ими бесплатно. В большинстве случаев их создание начиналось небольшой группой энтузиастов или вообще одним человеком. Если у вас есть крутая идея, ее можно упаковать и поделится в комьюнити. После прочтения этой статьи у нас будет четкая инструкция для этого.

Мир коммерческой разработки​

Сегодня мы все чаще отказываемся от монолитных решений и часто сталкиваемся с необходимостью каким-то образом делится кодовой базой между нашими проектами, сервисами, Lambda Functions (этот список можно продолжать бесконечно). Как же быть в этой ситуации? Можно упаковывать этот код в пакет и устанавливать из GIT или приватного PyPI server. Другой кейс — необходимость опубликовать клиентскую библиотеку для работы с нашим сервисом, что позволяет сделать его более привлекательным для использования. Как пример, boto3 для работы с сервисами AWS, клиентская библиотека для Stripe и так далее.

Как видите, сценариев применения много, все ограничивается только нашей фантазией.

Каким же должен быть идеальный пакет​

Давайте подумаем над этим вопросом с точки зрения конечного пользователя и разработчика. Идеальному пакету нужны:

  • крутая идея;
  • актуальная документация;
  • кроссплатформенность;
  • покрытие тестами;
  • статический анализ кода;
  • поддержка актуальных версий языка;
  • использование CI.
На мой взгляд, это минимальный набор качеств, которыми должен обладать любой проект. Давайте теперь детально поговорим о каждом из этих пунктов и попробуем собрать собственный пакет.

Идея для пакета​

Идея пакета в данном случае не слишком важна, так как цель нашего материала другая. Однако я долго размышлял над этим пунктом, хотелось чего-то небанального. И вот на днях мы смотрели с дочкой передачу про азбуку Морзе. Я подумал, что это будет вполне удачной идеей для пакета в рамках этой статьи.

Для упрощения нашего пакета будем поддерживать только символы латиницей. И да, этот пакет не несет какого-то смысла для комьюнити, но для нашей цели вполне подойдет.

from enum import Enum

morse_dictionary = (
("a", ".-"),
("b", "-..."),
("c", "-.-."),
("d", "-.."),
#......
)

lat_to_morse = {lat: morse for lat, morse in morse_dictionary}
morse_to_lat = {morse: lat for lat, morse in morse_dictionary}

class Errors(Enum):
IGNORE = "ignore"
REPLACE = "replace"
STRICT = "strict"

def _translate_char(char: str, dictionary, errors: Errors = Errors.STRICT) -> str:
char = char.lower()

try:
return dictionary[char]
except KeyError:
if errors == Errors.STRICT:
raise ValueError

if errors == Errors.IGNORE:
return ""

if errors == Errors.REPLACE:
return "?"


def translate_char_to_morse(char: str, errors: Errors = Errors.STRICT):
return _translate_char(char=char, dictionary=lat_to_morse, errors=errors)


def translate_char_to_lat(char: str, errors: Errors = Errors.STRICT):
return _translate_char(char=char, dictionary=morse_to_lat, errors=errors)


def translate_to_morse(text: str, errors: Errors = Errors.STRICT):
return "".join(
translate_char_to_morse(char=char, errors=errors) if char != " " else "" for char in text
)


def translate_to_lat(text: str, errors: Errors = Errors.STRICT):
return " ".join(translate_char_to_lat(char=char, errors=errors) for char in text.split(" "))

Теперь у нас есть код, которым мы бы хотели поделиться. Для этого упакуем его в пакет и опубликуем в PyPI. Но с чего же начать?

Документация​

Большинство хотя бы раз в жизни сталкивались с необходимостью использования внешнего пакета. И если у него есть хорошая документация с примерами использования, это в разы ускоряет интеграцию. Что же обязана содержать в себе документация?

  • Инструкцию по установке.
  • Описание методов и классов.
  • Примеры работы.
  • Описание исключений.
Все это повысит привлекательность пакета и снизит порог вхождения.

Конечно, можно воспользоваться сервисом readthedocs.org и генерацией документации с помощью Sphinx, но для нашего проекта мы обойдемся только README.md, его будет более чем достаточно.

include ./README.md

Контроль зависимостей​

Для этого можно использовать как requirements.txt, так и более продвинутые инструменты. Я буду использовать Poetry, так как он более привычный для меня. Чтобы разобраться в работе инструмента, можно пройти по пути из официальной документации.

Статический анализ кода​

Для статического анализа кода существует большое количество библиотек и инструментов и они во многом дополняют друг друга. Практически все анализаторы можно тонко настроить под свои нужды. Для этого обычно используется один из файлов: tox.ini, setup.cfg или pyproject.toml. Мне привычнее использовать второй вариант, с ним и будут примеры. И да, настроив один раз этот файл, его можно использовать и на других проектах.

iSort​

Это утилита/библиотека для сортировки импортируемых файлов в алфавитном порядке с автоматическим разделением на разделы по типу. Предоставляет утилиту командной строки и так же легко встраивается в большинство популярных редакторов.

[isort]
# See https://github.com/timothycrosley/isort#multi-line-output-modes
multi_line_output = 3
include_trailing_comma = true
sections = FUTURE,STDLIB,THIRDPARTY,FIRSTPARTY,LOCALFOLDER
default_section = FIRSTPARTY
# Should be: 100 - 1
line_length = 99
wrap_length = 99

Для запуска можно использовать:

poetry run isort .

flake8​

Инструмент, позволяющий просканировать код проекта и обнаружить в нем стилистические ошибки и нарушения различных конвенций кода на Python. Flake8 умеет работать не только с PEP 8, но и с другими правилами, к тому же он поддерживает сторонние плагины.

[flake8]
max-line-length = 100

# flake8-quotes plugin settings
inline-quotes = "

#flake8-import-order plugin settings
application-import-names = apps
import-order-style = google

# flake8-docstrings plugin settings
docstring-convention = google

max-complexity = 12
Для запуска анализа выполняем:

poetry run flake8 .

black​

Black — бескомпромиссное средство форматирования кода Python. Используя его, мы соглашаемся передать контроль над мелочами ручного форматирования. В свою очередь, Black дает нам скорость, детерминизм и свободу от ворчаний по поводу форматирования. А мы экономим время и умственную энергию для более важных дел. При этом он легко встраивается во все современные редакторы.

Для запуска можно использовать:

poetry run black .

Для нашего пакета этого списка будет более чем достаточно. При желании его можно расширить. К примеру, для контроля типизации можно использовать mypy, а для контроля сложности можно использовать xenon.

Тестирование​

Автоматизация тестирования очень важна, особенно если мы хотим поддерживать разные версии языка и разные платформы. Для тестирования выбрал свой любимый pytest. Про него могу рассказывать бесконечно, возможно, когда-нибудь сделаю отдельную статью. Но сейчас давайте вернемся к нашему пакету.

Структура проекта​

Тесты будем хранить отдельно от нашего кода, в случае дистрибуции это лучшее решение.

root/
morse_code_translator/
__init__.py
dictionary.py
translator.py
tests/
test_translator.py
conftest.py

Настройка​

В настройке pytest есть много возможностей, но нам будет достаточно минимального количества.

[tool:pytest]
python_files = tests.py test_*.py *_tests.py

# Directories that are not visited by pytest collector:
norecursedirs = *.egg .eggs dist build docs .tox .git __pycache__

# You will need to measure your tests speed with `-n auto` and without it,
# so you can see whether it gives you any performance gain, or just gives
# you an overhead. See `docs/template/development-process.rst`.
addopts = --cache-clear
--strict-markers
--tb=short
--doctest-modules
--cov=morse_code_translator
--cov-report=term-missing:skip-covered
--cov-report=xml
--cov-fail-under=100

И да, конечно же, мы будем измерять покрытие тестами.

[coverage:run]
branch = True
omit = *tests*

Пишем тесты​

import random

import pytest

from morse_code_translator.enums import Errors
from morse_code_translator.dictionary import morse_dictionary, lat_to_morse, morse_to_lat
from morse_code_translator.translator import translate_to_morse, translate_to_lat


def test_dictionary_len():
assert len(morse_dictionary) == len(lat_to_morse) == len(morse_to_lat)


def test_dictionary_map():
random_element = random.choice(morse_dictionary)
assert lat_to_morse[random_element[0]] == random_element[1]
assert morse_to_lat[random_element[1]] == random_element[0]


@pytest.mark.parametrize(
"input, output, errors",
(
(
"Morse Code Translator!!",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-.",
Errors.IGNORE,
),
(
"MORSE CODE TRANSLATOR!!",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-.",
Errors.IGNORE,
),
(
"Morse Code Translator!!",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-. ? ?",
Errors.REPLACE,
),
(
"MORSE CODE TRANSLATOR!!",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-. ? ?",
Errors.REPLACE,
),
(
"Morse Code Translator",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-.",
Errors.STRICT,
),
(
"MORSE CODE TRANSLATOR",
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-.",
Errors.STRICT,
),
),
)
def test_success_translate_to_morse(input, output, errors):
assert translate_to_morse(input, errors=errors) == output


def test_fail_translate_to_morse():
with pytest.raises(ValueError):
translate_to_morse("Morse Code Translator!!", errors=Errors.STRICT)


@pytest.mark.parametrize(
"input, output, errors",
(
(
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-. ..... -----",
"morse code translator",
Errors.IGNORE,
),
(
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-. ..... -----",
"morse code translator??",
Errors.REPLACE,
),
(
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-.",
"morse code translator",
Errors.STRICT,
),
),
)
def test_success_translate_to_lat(input, output, errors):
assert translate_to_lat(input, errors=errors) == output


def test_fail_translate_to_lat():
with pytest.raises(ValueError):
translate_to_lat(
"-- --- .-. ... . -.-. --- -.. . - .-. .- -. ... .-.. .- - --- .-. ..... -----",
errors=Errors.STRICT,
)

А для запуска будем использовать:

poetry run pytest

Лицензирование​

Вообще видов лицензий много. Так как наш будущий пакет будет выпущен как open source, то идеальной для нас лицензией будет MIT. Хоть MIT и не единственная лицензия для open source проектов, но самый распространенный выбор для подобных библиотек. Определиться с лучшим вариантом для вас поможет, к примеру, этот ресурс.

Кстати, любой наш выбор будет явно лучше, чем его полное отсутствие. Выпуск пакета без лицензии, по сути, равен не выпуску пакета. Так же это убережет нас от воровства нашего кода. Кроме того, большинство компаний не рискнет использовать пакет без лицензии по причине того, что это несет за собой большие риски в виде юридических последствий. Лицензия должна хранится в файле LICENSE в корне проекта.

MIT License

Copyright (c) 2021 Yura Bondarenko

.....

Сборка и публикация пакета​

Вот мы и подошли к самому интересному, а именно к сборке и публикации.

Подготовка к сборке​

Чтобы упаковать наш код в пакет, мы будем использовать setup.py. Этот файл содержит информацию о нашем будущем пакете, которая необходима для PyPI. Например: название, описание, версия пакета и еще много всего.

Ниже setup.py который описывает будущий пакет.

import os
from setuptools import find_packages, setup

ROOT = os.path.dirname(__file__)

with open("README.md", encoding="utf-8") as f:
README = f.read()

setup(
name="morse-code-translator2",
version="1.0.0",
description="Morse Code Translator",
long_description=README,
long_description_content_type="text/markdown",
author="Yurii Bondarenko",
author_email="ybondarenko.job@gmail.com",
url="https://github.com/",
packages=find_packages(
exclude=[
"*tests*",
"poetry.lock",
"pyproject.toml",
"TOPIC.md",
"conftest.py",
".venv",
]
),
package_dir={"morse_code_translator": "morse_code_translator"},
python_requires=">=3.6.*",
license="MIT",
install_requires=[],
classifiers=[
"Programming Language :: Python",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.6",
"Programming Language :: Python :: 3.7",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
],
project_urls={
"Documentation": "",
"Source Code": "https://github.com/",
},
tests_require=[],
)

Давайте пройдемся по файлу и посмотрим, что здесь происходит. Сначала мы импортируем из модуля setuptools несколько функций. Читаем README.md. Вызываем функцию setup с большим количеством аргументов. Хотя большинство аргументов понятны и не требуют пояснения, давайте разберем по основным:

  • author — имя автора;
  • author_email — контактный email автора;
  • license — название лицензии, в нашем случае это MIT;
  • description — короткое, однострочное описание пакета;
  • version — текущая версия пакета;
  • long_description — если переводить буквально, то это длинное описание пакета. Самым распространенным и удачным решением для этого является README.md;
  • url — URL, который ведет на главную страницу пакета или, если ее нет, то на репозиторий;
  • packages — в этом поле мы используем setuptools;
  • python_requires — описывает, какие версии языка поддерживает пакет
  • install_requires — в этом пункте указываем все внешние зависимости нашего пакета, как пример [«requests», «pytz»;
  • classifiers — так как пакетов много, то их необходимо классифицировать. Для этих целей и служит это поле. Полный список доступных значений можно посмотреть по ссылке.
Ну что ж, теперь мы готовы к самому главному: рождению сборке нашего пакета.

Сборка пакета​

Не буду вдаваться в подробности различных форматов распространения пакетов. Мы будем использовать стандартный подход. Для начала убедимся, что у нас установлена последняя версия setuptools и wheel:

pip install --upgrade setuptools wheel

Теперь собираем наш пакет. Для этого выполняем:

python setup.py sdist bdist_wheel

В результате выполнения этой команды будет создано три каталога build, dist и morse_code_translator.egg-info. Все эти каталоги добавляем в .gitignore. Но самые интересные для нас — два файла из dist, а именно:

  • morse-code-translator2-1.0.0.tar.gz — архив с нашим кодом.
  • morse_code_translator2-1.0.0-py3-none-any.whl — wheel файл.
Именно эти два файла мы и будем загружать в PyPI. Теперь мы можем перейти к загрузке нашего пакета.

Публикация​

Мы практически у цели, осталось только загрузить наш пакет на сервер PyPI. Для этого регистрируемся на pypi.org.

Но было бы неплохо протестировать загрузку пакета перед публикацией на глобальном сервере. Для этого есть тестовый сервер PyPI.

Имя пользователя и пароль можно сохранить в .pypirc в корне вашего домашнего каталога.

[distutils]
index-servers =
pypi
testpypi
[pypi]
username: username
password: myPassword
[testpypi]
repository: https://test.pypi.org/legacy/
username: username
password: myPassword

Секция testpypi не обязательна.

Для загрузки будем использовать twine, для этого добавим его в зависимости:

poetry add twine --dev

Для загрузки на тестовый сервер выполняем:

twine upload --repository testpypi dist/*

А для загрузки на глобальный сервер:

twine upload --repository dist/*

Ну вот мы и у цели. Наш пакет.

Автоматизация​

Все это, конечно, очень интересно, но хотелось бы максимально автоматизировать процесс публикации. Так как наш проект хранится в GitHub то для автоматизации будем использовать GitHub CI.

GitHub CI​

GitHub, как и большинство современных Git-репозиториев, поддерживает процесс непрерывной интеграции. В GitHub это реализовано с помощью GitHub Actions. Нам надо будет добавить Action, который будет тестировать, собирать и заливать нашу новую версию пакете в PyPI.

Actions в репозиторий можно добавлять несколькими способами. Для нас более удобным будет создать его вручную, для этого создаем две директории .github и workflows, которая вложена в первую, она и будет хранилищем наших Actions. В итоге у нас получается такая структура:

root/
.github/
workflows/
publish_package.yml
morse_code_translator/
....
tests/
....
....

Перед тем как мы начнем писать наш Action, получим токен для публикации в PyPI. Для этого идем в настройки PyPI аккаунта. И, пролистав эту страницу, мы увидим секцию «API tokens».

image_28730850021632757184585.png


Нажимаем на кнопку «Add API token», вводим название нашего токена (у меня это GitHub), выбираем наш пакет и создаем токен.

image_21851976531632757184590.png


Дальше идем в настройки Git-репозитория и создаем секрет с именем PYPI_API_TOKEN, значением будет наш токен.

image_44046905141632757238853.png


Теперь у нас есть все для того, что бы создать наш GitHub Action. Создаем .yml-файл, в котором описываем наши задачи.

.github/workflows/publish_package.yml:
name: Publish package

on:
push:
tags:
- 'v*'
jobs:
static_analysis:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-latest, macos-latest, windows-latest]
python-version: [ 3.6, 3.7, 3.8, 3.9 ]
steps:
- uses: actions/checkout@v2
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v2
with:
python-version: ${{ matrix.python-version }}
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install poetry
poetry install
- name: Lint with flake8
run: |
poetry run flake8 --count --show-source --statistics --exit-zero
- name: Lint with isort
run: |
poetry run isort .
- name: Test with pytest
run: |
poetry run pytest
deploy:
runs-on: ubuntu-latest
needs: static_analysis
steps:
- uses: actions/checkout@v2
- name: Set up Python
uses: actions/setup-python@v2
with:
python-version: '3.6'
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install poetry
poetry install --no-interaction --no-ansi --no-dev
- name: Upgrade setuptools
run: python -m pip install --upgrade setuptools wheel
- name: Build package
run: python setup.py sdist bdist_wheel
- name: Publish package
uses: pypa/gh-action-pypi-publish@27b31702a0e7fc50959f5ad993c78deac1bdfc29
with:
user: __token__
password: ${{ secrets.PYPI_API_TOKEN }}

Давайте разберемся, что же здесь происходит. Первая строчка указывает название name: Publish package.

Следующая секция описывает условия запуска. В нашем случае он будет реагировать на создание тегов, которые начинаются с v:

on:
push:
tags:
- 'v*'

Последняя секция описывает задачи, в нашем случае их две.

Первая задача это <code>static_analysis</code>, в ней мы запускаем flake8, isort и pytest. Так как мы заявляем поддержку Python 3.6+ в операционных системах Linux, macOS и Windows, то нам нужно и тестировать наш пакет в этих средах. Для этого используется такое свойство, как matrix:

strategy:
matrix:
os: [ubuntu-latest, macos-latest, windows-latest]
python-version: [ 3.6, 3.7, 3.8, 3.9 ]

В нашей матрице используется две переменные. Первая — это os, в ней мы записываем список поддерживаемых операционных систем. А во второй — python-version — перечень поддерживаемых нами версий Python. И как итог CI запустит эту задачу 12 раз. Но почему так? Все просто: он перебирает все возможные комбинации наших переменных, в данном случае их 12. Таким образом, наши тесты будут запущены во всех поддерживаемых окружениях.

Вторая задача — это deploy. У нее есть две особенности. Первая — это блок:

needs: static_analysis

Этой строчкой мы говорим, что перед тем как запустить эту задачу, должна удачно отработать задача static_analysis.

Вторая — это использование секрета, который мы создали ранее:

- name: Publish package
uses: pypa/gh-action-pypi-publish@27b31702a0e7fc50959f5ad993c78deac1bdfc29
with:
user: __token__
password: ${{ secrets.PYPI_API_TOKEN }}

В результате для того, чтобы опубликовать новую версию пакета, достаточно выполнить две команды.

Первая — создаст тег, при этом даст ему название, используя версию нашего пакета, которая указана в setup.py:

git tag -a v$(python setup.py --version) -m 'description of version'

Вторая — втолкнет новый тег в репозиторий, что и запустит наш Action:

git push --tags

Вот так это выглядит в GitHub:

image_98342039521632757238821.png


И вот так выглядит наш пакет:

image_74069896931632757238848.png


Заключение​

Изначальной целью статьи было написать инструкцию, следуя которой можно опубликовать свой пакет. Но тут я задел много смежных с этой темой аспектов, которые будут полезны не только при публикации пакетов.

 
Сверху