Metadata-Version: 2.4
Name: smeller_db
Version: 0.1.5
Summary: Sync & async ORM-клиент, сервис-слой и CLI для управления Aroma-данными.
Author-email: SL <info@sensorylab.ru>
License: MIT
Project-URL: Homepage, https://github.com/sensoryfox/smeller_db
Project-URL: Documentation, https://github.com/sensoryfox/smeller_db#readme
Project-URL: Issues, https://github.com/sensoryfox/smeller_db/issues
Project-URL: Changelog, https://github.com/sensoryfox/smeller_db/blob/main/CHANGELOG.md
Keywords: SQLAlchemy,asyncpg,ORM,database,aroma,pydantic,rich,typer
Requires-Python: >=3.9
Description-Content-Type: text/markdown
Requires-Dist: SQLAlchemy<3.0,>=2.0
Requires-Dist: asyncpg>=0.29
Requires-Dist: psycopg2-binary>=2.9
Requires-Dist: pydantic>=2.5
Requires-Dist: rich>=13.0
Requires-Dist: typer[all]>=0.9
Requires-Dist: python-dotenv>=1.0
Provides-Extra: dev
Requires-Dist: pytest>=7.4; extra == "dev"
Requires-Dist: black>=23.10; extra == "dev"
Requires-Dist: isort>=5.12; extra == "dev"
Requires-Dist: mypy>=1.6; extra == "dev"
Provides-Extra: docs
Requires-Dist: mkdocs-material>=9.5; extra == "docs"

# Aroma DB Client 📚

> Удобный синхронно-асинхронный клиент (SQLAlchemy 2.0 style) с сервис-слоем и CLI для работы
> с базой данных проекта **Aroma Platform**.  
> Пакет можно поставить одной командой `pip install smeller_db`
> и сразу воспользоваться как из Python-кода, так и из терминала.

---

## Содержание

1. [Почему именно этот клиент?](#почему-этот-клиент)
2. [Установка](#установка)
3. [Настройка окружения (.env)](#настройка-окружения)
4. [Быстрый старт (синхронный)](#быстрый-старт-sync)
5. [Быстрый старт (асинхронный)](#быстрый-старт-async)
6. [CLI-утилита `aroma-db`](#cli-утилита)
7. [API-справка](#api-справка)
8. [Разработка / вклад](#разработка--вклад)
9. [Лицензия](#license)

---

## Почему этот клиент?

* **Два режима** — сразу готовы `ORMClient` и `AsyncORMClient`.
* **Pydantic v2** — типобезопасное превращение ORM-объектов в датаклассы и обратно.
* **Rich & Typer** — красивые таблицы и мощный CLI «из коробки».
* **Zero-Boilerplate** — одна строка для получения превью любой таблицы,
  одна команда CLI для инициализации схемы.
* **Пример-шаблон** (см. `tests/test.py`) — копируйте и начинайте.

---

## Установка

```bash
# Из PyPI
pip install smeller_db

# Локальная разработка
git clone https://github.com/sensoryfox/smeller_db.git
cd smeller_db
pip install -e .[dev]   # + dev-зависимости
```
Python ≥ 3.9 обязателен (используются `typing.Annotated`, match-statement и т.д.).

---

## Настройка окружения

`DatabaseConfig` автоматически подтянет значения из переменных окружения
(можно задать в `.env` файле). Минимальный набор:

```
POSTGRES_HOST=185.180.230.207
POSTGRES_PORT=55432
POSTGRES_DB=sl_aroma
POSTGRES_USER=slreader
POSTGRES_PASSWORD=1qa2ws#ED
POSTGRES_OPTIONS=sslmode=disable  
DB_ASYNC=true
```

Допускается форма URL-строки `postgresql://user:pass@host:port/dbname` — смотрите
`smeller_db/config/database.py`.

---
```py

from smeller_db.config.database import DatabaseConfig
from smeller_db.services.database_service import DatabaseService
db_config = DatabaseConfig.from_env()
db_service = DatabaseService(db_config, create_schema_on_init=False, drop_all_on_init=False)
# получаем список картриджей
db_service.get_all_cartridges()

## Быстрый старт (synchronous) <a name="быстрый-старт-sync"></a>
# Если есть права на создание
from smeller_db.services import DatabaseService
from smeller_db.schemas.aroma_track import AromaTrackCreate
from smeller_db.schemas.channel_control_config import ChannelControlConfig, Color

service = DatabaseService.from_env()  # использует .env

# 1) Создадим трек
track = service.create_aroma_track(
    AromaTrackCreate(
        name="Demo track",
        description="Just a couple of blocks 🙂"
    )
)

# 2) Добавим аромаблок
block = service.create_aromablock(
    AromaBlockCreate(
        name="Lime intro",
        description="Opening shot of fresh lime",
        data_type="image",
        content_link="s3://bucket/frames/lime.png",
        start_time=0.0,
        stop_time=3.0,
        aroma_track_id=track.id,
        channel_configurations={
            1: ChannelControlConfig(
                color=Color(r=0, g=255, b=0),
                intensity=0.7,
                interpolation="linear" # TODO MORE
            )
        }
    )
)

print(track)
print(block)

# 3) Красиво покажем состояние БД
service.print_database_overview()
```
---
```py
## Быстрый старт (asynchronous) <a name="быстрый-старт-async"></a>
import asyncio
from smeller_db.services.database_service_async import AsyncDatabaseService
from smeller_db.schemas.aroma_track import AromaTrackCreate

async def main():
    service = AsyncDatabaseService.from_env()

    await service.setup_schema(create_schema=True)  # или миграции alembic

    track = await service.create_aroma_track(
        AromaTrackCreate(name="Async track", description="Created asynchronously")
    )
    print(track)

asyncio.run(main())
```
---

## CLI-утилита <a name="cli-утилита"></a>

После установки пакета появляется (должна) команда `aroma-db`.
# Посмотреть все возможности
aroma-db --help
Часто используемые под-команды:
| Команда | Описание |
| ------------------------------------------- | --------------------------------------------------- |
| `smeller_db show-db` | Красивый вывод всех таблиц (+3 строки превью). |
| `smeller_db show-db --headers` | Только заголовки таблиц. |
| `smeller_db list-tables` | Сухой список таблиц. |
| `smeller_db init-schema` | Создать (если нужно – пересоздать) все таблицы. |
| `smeller_db init-schema --drop-first` | Удалить **всё** и заново создать. ⚠ Будьте осторожны! |
| `smeller_db ... --async` | Любая команда может быть выполнена через async. |

Пример:
# Одной строкой посмотреть первые 5 записей всех таблиц

```bash
smeller_db show-db --rows 5
```

---

## API-справка

smeller_db/
├─ async_orm_client.py        – «сырой» AsyncSession-обёртка (low-level)
├─ orm_client.py              – синхронный вариант
├─ services/
│    ├─ database_service.py   – высокоуровневый sync-сервис
│    └─ database_service_async.py – async-сервис
├─ models/                    – ORM-таблицы SQLAlchemy
└─ schemas/                   – Pydantic-схемы (DTO)

Основные публичные классы/функции:

| Класс/функция | Где находится | Назначение |
| ---------------------------------------- | -------------------------------------- | ------------ |
| `ORMClient`, `AsyncORMClient` | `smeller_db/orm_client.py`, `smeller_db/async_orm_client.py` | Base-клиенты для CRUD, контекстный менеджер |
| `DatabaseService`, `AsyncDatabaseService` | `smeller_db/services/…` | CRUD + удобные вспомогательные методы |
| `get_db_client()` | `smeller_db/db_client_factory.py` | Фабрика sync/async-клиентов |
| `smeller_db` (CLI) | `smeller_db/tools/db_cli.py` | Управление БД из терминала |

Полная автодока генерируется `pdoc`/`mkdocs` (см. раздел «docs»).

---

## Разработка / вклад

1. Склонируйте репозиторий.
2. `pip install -e .[dev]`
3. Запустите тесты: `pytest -q`.
4. Перед PR запустите `black . && isort . && mypy smeller_db`.

Будем рады pull-request'ам 🚀

---

## License

`smeller_db` распространяется по лицензии **MIT** – делайте с кодом, что хотите,
но не забудьте сохранить копию лицензии.
────────────────────────────────────────

Готово!  
• `pyproject.toml` — вставьте в корень проекта и запустите `python -m build`,  
  затем `twine upload dist/*` — пакет уже готов к публикации.  
• `README.md` — кладётся рядом; PyPI отобразит его как документацию.  
