Skip to main content

Utilites for composites layer

Project description

Classic Composites

Предоставляет утилиту для описания композитов в ленивом стиле.

Установка

pip install classic-composites

Rationale

Представим себе, что у нас есть приложение со следующей структурой:

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

Если опустить инфраструктурные особенности, вроде кода, считывающего настройки, или кода подключения к базам данных, то код api_entrypoint мог бы выглядеть примерно так:

from types import SimpleNamespace

from example import db, app, api


DB = SimpleNamespace()
DB.interface = db.DBInterface()
DB.some_repo = db.SomeRepo()

APP = SimpleNamespace()
APP.some_query = app.SomeQuery(DB.interface)
APP.some_command = app.SomeCommand(DB.some_repo)

API = SimpleNamespace()
API.some_handler = api.SomeHandler(
    APP.some_query, APP.some_command,
)
API.wsgi_app = api.App(API.some_handler)


if __name__ == '__main__':
    import waitress
    
    waitress.serve(API.wsgi)

Код worker_entrypoint мог бы выглядеть вот так:

from types import SimpleNamespace

from example import db, app, worker


DB = SimpleNamespace()
DB.some_repo = db.SomeRepo()

APP = SimpleNamespace()
APP.some_command = app.SomeCommand(DB.some_repo)
APP.other_command = app.OtherCommand(DB.some_repo)

WORKER = SimpleNamespace()
WORKER.listener = worker.Listener(
    APP.some_command, APP.other_command,
)


if __name__ == '__main__':
    WORKER.listener.run()

Такой пример кода хотя очень прост, все же имеет проблему - дублирование кода. В этом примере это явно не является проблемой, так как классов мало, но в реальных системах классов гораздо больше, и там это является проблемой. Классы db.SomeRepo и app.SomeCommand инстанцируются дважды. Можно попробовать решить проблему, сделав общий модуль, в котором будут инстанцированы все объекты, а в entrypoint-ах оставить только импорт нужного класса и запуск.

composite.py:

from types import SimpleNamespace

from example import db, app, api, worker


DB = SimpleNamespace()
DB.interface = db.DBInterface()
DB.some_repo = db.SomeRepo()

APP = SimpleNamespace()
APP.some_query = app.SomeQuery(DB.interface)
APP.some_command = app.SomeCommand(DB.some_repo)
APP.other_command = app.OtherCommand(DB.some_repo)

API = SimpleNamespace()
API.some_handler = api.SomeHandler(
    APP.some_query, APP.some_command,
)
API.wsgi_app = api.App(API.some_handler)

WORKER = SimpleNamespace()
WORKER.listener = worker.Listener(
    APP.some_command, APP.other_command,
)

api_entrypoint.py:

import waitress

from .composite import API

waitress.serve(API.wsgi_app)

worker_entrypoint.py

from .composites import WORKER

WORKER.listener.run()

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

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

Для этого завернем каждое инстанцирование объекта в композите в lambda, а при указании какого-либо имени в пространстве имен просто поставим скобки:

composite.py

from types import SimpleNamespace

from example import db, app, api, worker


DB = SimpleNamespace()
DB.interface = lambda: db.DBInterface()
DB.some_repo = lambda: db.SomeRepo()

APP = SimpleNamespace()
APP.some_query = lambda: app.SomeQuery(DB.interface())
APP.some_command = lambda: app.SomeCommand(DB.some_repo())
APP.other_command = lambda: app.OtherCommand(DB.some_repo())

API = SimpleNamespace()
API.some_handler = lambda: api.SomeHandler(
    APP.some_query(), APP.some_command(),
)
API.wsgi_app = lambda: api.App(API.some_handler())

WORKER = SimpleNamespace()
WORKER.listener = lambda: worker.Listener(
    APP.some_command(), APP.other_command(),
)

После исполнения этого файла в памяти останутся пространства имен, содержащие фабрики, но еще не содержащие сами объекты. Тогда, в entry_point остается только вызвать нужную функцию:

api_entrypoint.py:

import waitress

from .composite import API

waitress.serve(API.wsgi_app())

worker_entrypoint.py

from .composites import WORKER

WORKER.listener().run()

При вызове API.wsgi_app() произойдет вызов фабрики, оттуда, по цепочке, произойдет вызов API.some_handler(), оттуда APP.some_query() и APP.some_command(), и т.д., в итоге будут инстанцированы все необходимые объекты, но те фабрики, на которые не было ссылки, вызваны не будут.

Здесь все еще есть проблема - фабрики каждый раз возвращают новый объект. Нам необходимо свести инстанцирование к минимуму в большинстве случаев, потому бы хотели сделать себе что-то вроде объекта-кеша, который бы содержал в себе результаты вызова фабрик. Что-то вроде:

composite.py

from types import SimpleNamespace

from hypotetical_cache import Cache

from example import db, app, api, worker

cache = Cache()

DB = SimpleNamespace()
DB.interface = cache(lambda: db.DBInterface())
DB.some_repo = cache(lambda: db.SomeRepo())

APP = SimpleNamespace()
APP.some_query = cache(lambda: app.SomeQuery(DB.interface()))
APP.some_command = cache(lambda: app.SomeCommand(DB.some_repo()))
APP.other_command = cache(lambda: app.OtherCommand(DB.some_repo()))

API = SimpleNamespace()
API.some_handler = cache(lambda: api.SomeHandler(
    APP.some_query(), APP.some_command(),
))
API.wsgi_app = cache(lambda: api.App(API.some_handler()))

WORKER = SimpleNamespace()
WORKER.listener = cache(lambda: worker.Listener(
    APP.some_command(), APP.other_command(),
))

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

Класс Namespace умеет кешировать результаты фабрик по умолчанию:

composite.py

from classic.composites import Namespace

from example import db, app, api, worker

DB = Namespace()
DB.interface = db.DBInterface
DB.some_repo = lambda: db.SomeRepo()

APP = Namespace()
APP.some_query = lambda: app.SomeQuery(DB.interface())
APP.some_command = lambda: app.SomeCommand(DB.some_repo())
APP.other_command = lambda: app.OtherCommand(DB.some_repo())

API = Namespace()
API.some_handler = lambda: api.SomeHandler(
    APP.some_query(), APP.some_command(),
)
API.wsgi_app = lambda: api.App(API.some_handler())

WORKER = Namespace()
WORKER.listener = lambda: worker.Listener(
    APP.some_command(), APP.other_command(),
)

Выглядит очень похоже на вариант без кеша, только с кешем :)

Также есть способ отключить кеш для нужной фабрики:

from classic.composites import Namespace, no_cache


DB = Namespace()
DB.some_obj = no_cache(lambda: 1)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

classic_composites-0.0.2.tar.gz (6.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

classic_composites-0.0.2-py3-none-any.whl (6.3 kB view details)

Uploaded Python 3

File details

Details for the file classic_composites-0.0.2.tar.gz.

File metadata

  • Download URL: classic_composites-0.0.2.tar.gz
  • Upload date:
  • Size: 6.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for classic_composites-0.0.2.tar.gz
Algorithm Hash digest
SHA256 f72051838b797683b8223bb561f05a4c3d0ae721ba177364612d3282431cc651
MD5 ab2b09b5b7644593ca9576a16c4c87d2
BLAKE2b-256 81ec9f8b1cf16d994e173f94b5dca28c31d89577a1ed0df49aa3f840bbcce781

See more details on using hashes here.

Provenance

The following attestation bundles were made for classic_composites-0.0.2.tar.gz:

Publisher: publish.yml on variasov/classic-composites

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file classic_composites-0.0.2-py3-none-any.whl.

File metadata

File hashes

Hashes for classic_composites-0.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 178e6308241669d5bb39b59ecd56948c3250fab8de588cb42b6c9a6e148518f2
MD5 908abaa3e3ae0569fd0d094b3f4641aa
BLAKE2b-256 173c5110720fa5be3cb66b1dd259ce5856e22cd1492a31def015263d85fcbb65

See more details on using hashes here.

Provenance

The following attestation bundles were made for classic_composites-0.0.2-py3-none-any.whl:

Publisher: publish.yml on variasov/classic-composites

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page