Async Yandex Dialogs Developer API client — programmatic skill creation, draft management, OAuth Device Flow.
A framework-agnostic Python library that drives the Yandex Dialogs Developer
API — the meta-API at dialogs.yandex.ru/developer-api/v2/ used to
programmatically create and manage Alice skills (Smart Home and custom dialog
skills). Where every other Yandex Alice library on PyPI handles the runtime
side (incoming webhook requests from end users), this one handles the
provisioning side: sign in via Yandex Passport Device Flow, create a skill,
upload a logo, set the webhook backend URL, publish a draft.
It exists because Yandex doesn't publish a developer-API SDK and the only
known Python implementation lived inside the Music Assistant
ma-provider-yandex-smarthome plugin. This library is that code, extracted
and made generic.
- Full skill auto-creation pipeline — CSRF/cookie session →
POST /apps→ upload logo →PATCHdraft → optional OAuth app creation →POSTdeploy. - Two channel types —
channel="smartHome"for Yandex Smart Home skills (always requires OAuth account-linking),channel="aliceSkill"for Alice custom dialog skills (OAuth-free or OAuth-attached). - Optional OAuth —
aliceSkillskills can be created without account-linking by omittingoauth_*params entirely.smartHomealways requires all four. - Incremental state machine —
SkillCreationArtifactssnapshots progress after every step. On transient failure, retry resumes from the last completed step instead of restarting. - Typed errors —
DialogsAuthError,DialogsValidationError,DialogsSkillNotFoundError,DialogsDuplicateSkillErrorwith structured field info where available. - Framework-agnostic — caller produces an authenticated
aiohttp.ClientSession(typically viaya-passport-auth) and wraps it in anAuthenticatorCMfactory. No opinion on Device Flow UX. - Strictly typed —
mypy --strictclean, PEP 561py.typedmarker. - Security-aware —
SecretStrredacts tokens in repr/str/format/tracebacks (re-exported fromya-passport-auth).
pip install ya-dialogs-apiimport asyncio
from contextlib import asynccontextmanager
from typing import AsyncIterator
import aiohttp
from ya_dialogs_api import (
AuthenticatorCM,
SMART_HOME_CHANNEL,
SkillCreationArtifacts,
SkillCreationState,
auto_create_skill,
)
# Wrap your authenticated session in an async context-manager factory.
@asynccontextmanager
async def my_authenticator() -> AsyncIterator[aiohttp.ClientSession]:
# Typically: ya_passport_auth.PassportClient.login_device_code(...)
async with aiohttp.ClientSession(cookies={"Session_id": "..."}) as session:
yield session
async def main() -> None:
artifacts = SkillCreationArtifacts(state=SkillCreationState.NONE)
result = await auto_create_skill(
authenticator=my_authenticator,
channel=SMART_HOME_CHANNEL, # "smartHome"
skill_name="My Smart Home",
artifacts=artifacts,
backend_uri="https://my-backend.example.com/yandex",
oauth_authorize_url="https://my-backend.example.com/oauth/authorize",
oauth_token_url="https://my-backend.example.com/oauth/token",
oauth_client_id="my-client-id",
oauth_client_secret="my-client-secret",
)
if result.state == SkillCreationState.DONE:
print(f"Skill created: skill_id={result.skill_id}")
else:
print(f"Failed at step {result.state}: {result.last_error}")
asyncio.run(main())from ya_dialogs_api import DIALOG_CHANNEL, auto_create_skill
result = await auto_create_skill(
authenticator=my_authenticator,
channel=DIALOG_CHANNEL, # "aliceSkill"
skill_name="My Dialog Skill",
artifacts=artifacts,
backend_uri="https://my-backend.example.com/alice",
description="A skill that does something useful",
# oauth_* params omitted → no account-linking
)Pass all four oauth_* params with channel="aliceSkill" to attach an OAuth
application (account-linking). Partial sets raise ValueError synchronously.
| Symbol | Description |
|---|---|
auto_create_skill(...) |
Full pipeline orchestrator. Resumable via SkillCreationArtifacts. |
auto_update_skill(...) |
Patch a skill draft and re-deploy. Works for both channels. |
DialogsSkillCreator |
Low-level client (one method per pipeline step, including delete_skill). |
Channel |
Literal["smartHome", "aliceSkill"] — Yandex wire values. |
SMART_HOME_CHANNEL / DIALOG_CHANNEL |
Typed constants for the two channels. |
SkillCreationArtifacts / SkillCreationState |
State machine; serialize with dump_artifacts / load_artifacts. |
AuthenticatorCM |
Protocol: no-arg async context-manager factory yielding aiohttp.ClientSession. |
DialogsApiError (base) |
All library exceptions derive from this. |
DialogsAuthError |
HTTP 401/403, HTML auth-wall, or 30x redirect to Passport. |
DialogsCsrfError |
CSRF token not found in developer console HTML. |
DialogsValidationError |
Domain-level field validation or Spring 4xx rejection. |
DialogsSkillNotFoundError |
Spring 404 "Skill not found"; skill_id attribute set. |
DialogsDuplicateSkillError |
Skill with this name already exists. |
SecretStr |
Re-exported from ya-passport-auth for token redaction. |
load_default_logo_bytes() |
Bundled 1024×1024 fallback skill logo. |
Stable (2.0.0). The public API is stable. The underlying Yandex dev-console endpoints are unofficial (reverse-engineered from DevTools traces) and may break without notice. The library is actively maintained against the current production endpoint behavior.
ya-passport-auth— Yandex Passport authentication library. Required runtime dependency.ma-provider-yandex-smarthomeandma-provider-yandex-alice— Music Assistant providers that consume this library.