AG-UI adapter for line-bot-sdk-python v3
Project description
LINE AG-UI Adapter
A lightweight adapter that bridges LINE Messaging API and AG-UI using line-bot-sdk-python v3.
It converts LINE webhook message events into AG-UI requests, sends them to an AG-UI server, and converts the final AG-UI response back into LINE reply messages.
Features
- Convert LINE inbound messages to AG-UI input
- text
- image / audio / video / file as multimodal input parts
- Convert AG-UI responses back to LINE reply messages
- text
- image / audio / video when
source.type == "url" - document responses as a text message containing the document URL
- Add middleware hooks before and after the AG-UI request
- Buffer the final AG-UI response before replying, which fits LINE's non-streaming reply flow
Installation
Using uv:
uv add line-agui-adapter
Using pip:
pip install line-agui-adapter
If you want to run the FastAPI example as well:
uv add fastapi uvicorn python-dotenv
Getting Started
The FastAPI example is available in examples/fastapi/main.py.
The example assumes that:
- a LINE channel is already configured
- the webhook server is exposed over HTTPS and reachable by LINE
- an AG-UI server is already running
- the required environment variables are set
If you do not have an AG-UI server yet, you can use the sample implementation under tests/servers. For example, tests/servers/google_adk/main.py can be used as a simple AG-UI-compatible test server.
Example startup:
cp .env.example .env
uv run uvicorn examples.fastapi.main:app --port 8000
FastAPI Example
import os
from typing import cast
from dotenv import load_dotenv
from fastapi import FastAPI, Header, HTTPException, Request
from linebot.v3 import WebhookParser, WebhookPayload
from linebot.v3.exceptions import InvalidSignatureError
from linebot.v3.messaging import (
ApiClient,
Configuration,
MessagingApi,
MessagingApiBlob,
ReplyMessageRequest,
ShowLoadingAnimationRequest,
)
from linebot.v3.webhooks import MessageEvent
from line_agui_adapter import AguiHttpClient, LineAguiAdapter, create_content_fetcher
load_dotenv()
app = FastAPI()
parser = WebhookParser(channel_secret=os.environ["LINE_CHANNEL_SECRET"])
configuration = Configuration(access_token=os.environ["LINE_CHANNEL_ACCESS_TOKEN"])
agui_client = AguiHttpClient(
endpoint=os.environ["AGUI_ENDPOINT"],
headers=(
{"Authorization": f"Bearer {os.environ['AGUI_AUTH_TOKEN']}"}
if os.environ.get("AGUI_AUTH_TOKEN")
else {}
),
)
async def before_agui(request):
request.forwarded_props["tenant_id"] = "example-tenant"
request.forwarded_props["source"] = "line-fastapi-example"
return request
def after_agui(response):
for message in response.assistant_messages:
if isinstance(message.content, str) and message.content:
message.content = f"[AG-UI] {message.content}"
return response
@app.post("/callback")
async def callback(
request: Request, x_line_signature: str = Header(...)
) -> dict[str, bool]:
body = (await request.body()).decode("utf-8")
try:
payload = cast(
WebhookPayload, parser.parse(body, x_line_signature, as_payload=True)
)
except InvalidSignatureError as exc:
raise HTTPException(status_code=400, detail="invalid signature") from exc
with ApiClient(configuration) as api_client:
line_api = MessagingApi(api_client)
blob_api = MessagingApiBlob(api_client)
adapter = LineAguiAdapter(
agui_client=agui_client,
content_fetcher=create_content_fetcher(blob_api),
)
adapter.pipeline.add_before(before_agui)
adapter.pipeline.add_after(after_agui)
for event in payload.events or []:
if not isinstance(event, MessageEvent):
continue
if event.mode == "standby" or not event.reply_token:
continue
user_id = getattr(event.source, "user_id", None)
if user_id:
line_api.show_loading_animation(
ShowLoadingAnimationRequest(
chatId=user_id,
loadingSeconds=60,
)
)
messages = await adapter.handle_event(event)
line_api.reply_message(
ReplyMessageRequest(
replyToken=event.reply_token,
messages=messages,
notificationDisabled=False,
)
)
return {"ok": True}
Middleware Hooks
Middleware hooks let you modify the AG-UI request before sending it, or modify the AG-UI response before converting it back to LINE messages.
async def before_hook(request):
request.forwarded_props["tenant_id"] = "tenant-a"
return request
def after_hook(response):
for message in response.assistant_messages:
if isinstance(message.content, str) and message.content:
message.content = f"[AG-UI] {message.content}"
return response
adapter.pipeline.add_before(before_hook)
adapter.pipeline.add_after(after_hook)
Example Environment Variables
Typical variables used by the FastAPI example:
LINE_CHANNEL_SECRETLINE_CHANNEL_ACCESS_TOKENAGUI_ENDPOINTAGUI_AUTH_TOKEN(optional)LINE_AGUI_FASTAPI_EXAMPLE_LOG_LEVEL(optional)
Set LINE_AGUI_FASTAPI_EXAMPLE_LOG_LEVEL to a standard Python logging level such as DEBUG, INFO, WARNING, ERROR, or CRITICAL if you want explicit logging configuration in the example app.
Notes
- LINE replies are generated from the final AG-UI response rather than from streaming output.
- Binary output is only converted to LINE media messages when the AG-UI response provides a URL-based source.
- The included Google ADK test server in tests/servers/google_adk/main.py disables input blob artifact replacement so image inputs can be passed to the model as actual inline data.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file line_agui_adapter-0.1.0.tar.gz.
File metadata
- Download URL: line_agui_adapter-0.1.0.tar.gz
- Upload date:
- Size: 11.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f3a237ef25eae85fa0ebc31d2d976420e6a55ef1348c2aea30a9955d0aabb624
|
|
| MD5 |
67696ae2a9b58714a9f917bcde4e706c
|
|
| BLAKE2b-256 |
7879f08edbf0ca84454ae5bf68ec9743e814d947959b3f4d966a5745a8c8e82c
|
File details
Details for the file line_agui_adapter-0.1.0-py3-none-any.whl.
File metadata
- Download URL: line_agui_adapter-0.1.0-py3-none-any.whl
- Upload date:
- Size: 14.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6248de2b451b4b0623295b44174d1642b2300217d1c7d0aafd56176884508faa
|
|
| MD5 |
57afd5f64a6e0e30625aaa30024fdd23
|
|
| BLAKE2b-256 |
5888117c34f7d174e257f8bdf888c39f7bb237c6a1d23deccc1eb1d5d9d1c6b6
|