一个 satori-python
客户端的构建从创建 App
对象开始:
from satori.client import App
app = App()
一个配置对应了一个 Satori 连接:
class WebsocketsInfo(Config):
host: str = "localhost"
port: int = 5140
token: Optional[str] = None
或
class WebhookInfo(Config):
path: str = "v1/events"
host: str = "127.0.0.1"
port: int = 8080
token: Optional[str] = None
server_host: str = "localhost"
server_port: int = 5140
你可以在创建 App
对象时传入一个或多个 WebsocketsInfo
或 WebhookInfo
对象:
from satori.client import App, WebsocketsInfo, WebhookInfo
app = App(
WebsocketsInfo(...),
WebhookInfo(...),
)
或使用 App.apply
方法:
from satori.client import App, WebsocketsInfo, WebhookInfo
app = App()
app.apply(WebsocketsInfo(...))
app.apply(WebhookInfo(...))
同时你可以自己定义新的 Config
,只需要实现下面几类方法即可:
class Config:
@property
def identity(self) -> str:
raise NotImplementedError
@property
def api_base(self) -> URL:
raise NotImplementedError
然后在 App 注册对应的 Network:
from satori.client import App
App.register_config(YourConfig, YourNetwork)
satori-python
使用 @app.register
装饰器来增加一个通用事件处理函数:
from satori.client import App, Account
from satori.model import Event
app = App()
@app.register
async def listen(account: Account, event: Event):
print(account, event)
@app.register
需要一个参数为 Account
与 Event
的异步函数.
Account
对象代表了接受事件的 Satori 平台账号, 你可以使用它来调用 API.Event
对象代表了任意类型的 Satori 事件, 你可以使用它来获取事件的数据.
除此之外,你可以使用 @app.register_on
装饰器来增加一个确定事件类型的处理函数:
from satori import EventType
from satori.client import App, Account
from satori.event import MessageEvent
app = App()
@app.register_on(EventType.MESSAGE_CREATED)
async def listen(account: Account, event: MessageEvent):
print(account, event)
使用 App.run
方法来同步运行 App
对象:
from satori.client import App
app = App()
app.run()
或使用 App.run_async
方法来异步运行 App
对象:
from satori.client import App
app = App()
async def main():
await app.run_async()
...
App.run
可以传入自定义的 asyncio.AbstractEventLoop
对象。
如前所述,Account
对象代表了一个 Satori 平台账号,你可以通过其 protocol
属性来调用 API:
from satori.client import App, Account
from satori.model import Event
app = App()
@app.register
async def listen(account: Account, event: Event):
if event.user.id == "xxxxxx":
await account.protocol.send_message(
event.channel.id,
"Hello, world!",
)
Account.protocol
拥有现在 satori
支持的所有 API 方法。
Account
允许自主创建并请求 api:
from satori import Login
from satori.client import Account, ApiInfo
async def main():
account = Account(Login(...), ApiInfo(token="xxxx"))
await account.send_message("xxxxxxxx", "Hello, World!")
Account
可以临时切换 api:
from satori.client import App, Account
from satori.client.protocol import ApiProtocol
from satori.model import Event
app = App()
@app.register
async def listen(account: Account, event: Event):
await account.custom(host="123.456.789.012", port=5140).send(event, "Hello, World!")
class MyProtocol(ApiProtocol):
async def my_api(self, event, *args): ...
@app.register
async def listen(account: Account, event: Event):
my_account = account.custom(protocol_cls=MyProtocol)
await my_account.protocol.my_api(event, "Hello, World!")
一个 satori-python
服务端的构建从创建 Server
对象开始:
from satori.server import Server
server = Server()
server 的配置直接在构造时传入:
from satori.server import Server
server = Server(
host="0.0.0.0",
port=8080,
)
同时可以传入 webhook 目标:
from satori.server import Server, WebhookEndpoint
server = Server(
webhooks=[WebhookEndpoint("http://xxxxx:8080/v1/events")]
)
你可以使用 Server.route
方法来自定义路由:
from satori import MessageObject
from satori.const import Api
from satori.server import Server, Request, route
server = Server()
@server.route(Api.MESSAGE_CREATE)
async def on_message_create(request: Request[route.MESSAGE_CREATE]):
return [MessageObject(id="123456789", content="Hello, world!")]
route 填入的若不属于 Api
中的枚举值,会被视为是内部接口的路由。
route 装饰的函数的返回值既可以是 satori 中的模型,也可以是原始数据。
同时,你也可以通过 server.apply
传入一个满足 Router
协议的对象,这里推荐继承 RouterMixin
类来实现路由:
from satori import MessageObject
from satori.const import Api
from satori.server import Server, Request, RouterMixin, route
server = Server()
class MyRouter(RouterMixin):
def __init__(self):
self.routes = {}
@self.route(Api.MESSAGE_CREATE)
async def on_message_create(request: Request[route.MESSAGE_CREATE]):
return [MessageObject(id="123456789", content="Hello, world!")]
server.apply(MyRouter())
事件由 Provider
提供:
class Provider(Protocol):
def publisher(self) -> AsyncIterator[Event]:
...
async def get_logins(self) -> list[Login]:
...
你可以通过 server.apply
传入一个满足 Provider
协议的对象:
import asyncio
from datetime import datetime
from satori import Channel, ChannelType, Event, Login, User
from satori.server import Server
server = Server()
class MyProvider:
async def get_logins(self):
return [Login(...)]
async def publisher(self):
while True:
await asyncio.sleep(2)
yield Event("example", datetime.now(), Login(...))
server.apply(MyProvider())
适配器是一个特殊的类,它同时实现了 Provider
和 Router
协议。
from satori.server import Server, Adapter
server = Server()
server.apply(Adapter(...))
一个适配器需要实现以下方法:
get_platform
: 返回适配器所适配的平台名称.publisher
: 用于推送平台事件.ensure
: 验证客户端请求的platform
和self-id
.get_logins
: 获取平台上的登录信息.launch
: 调度逻辑.
使用 Server.run
方法来运行 Server
对象:
from satori.server import Server
server = Server()
server.run()
或使用 Server.run_async
方法来异步运行 Server
对象:
from satori.server import Server
server = Server()
async def main():
await server.run_async()
...
satori-python
使用 Element
类来表示 Satori 消息元素.
from satori import Text, At, Sharp, Link
a = Text("1234")
role = At.role_("admin")
chl = Sharp("abcd")
link = Link("www.baidu.com")
link1 = Link("github.com/RF-Tar-Railt/satori-python")(
"satori-python"
)
Image
,Audio
,Video
,File
: 资源类型,对应 资源元素.
资源类型元素可以用特殊的 .of
方法来创建:
from satori import Image
image = Image.of(url="https://example.com/image.png")
在 .of
方法中,你可以传入以下参数:
url
: 资源的 URL.path
: 资源的本地路径.raw
: 资源的二进制数据. 会要求同时传入mime
参数.
from satori import Image
from io import BytesIO
from PIL import Image as PILImage
img = PILImage.open("image.png")
data = BytesIO()
img.save(data, format="PNG")
image = Image.of(raw=data, mime="image/png")
Bold
,Italic
,Underline
,Strikethrough
, ...: 修饰类型,对应 修饰元素.
from satori import Bold, Italic, Underline, Paragraph
text = Bold(
"hello",
Italic("world,"),
Underline()(
"Satori!"
),
Paragraph("This is a paragraph.")
)
对于 Message
,你可以通过 content
参数来传入子元素:
from satori import Message, Author
message = Message(forward=True)(
Message(id="123456789"),
Message(id="987654321"),
Message(
content=[
Author(id="123456789"),
"Hello, "
]
),
Message()(
Author(id="123456789"),
"World!"
)
)
!!!Satori 下的Message 不是“消息序列”的概念!!!
Quote
的用法与 Message
一致。
Custom
: 用来构造 Satori 标准外的消息元素。Raw
: 用来构造 Satori 标准外的消息元素,直接传入文本。
satori-python
提供了一个方法 select
, 用来递归地从消息中遍历提取特定类型的元素:
from satori import Quote, Author, Text, select
msg = [Quote(id="12345678")(Author(id="987654321"), Text("Hello, World!")), Text("Hello, World!")]
authors = select(msg, Author)
参考:资源链接(实验性)
对于客户端,你可以使用 Account.upload
方法来上传资源:
from pathlib import Path
from satori.client import App, Account, Event
from satori.model import Upload
app = App()
@app.register
async def _(account: Account, event: Event):
# 直接构造 Upload 对象并传入,返回`资源链接`的列表
resp: list[str] = await account.upload(
Upload(file=b'...'),
Upload(file=Path("path/to/file")),
)
# 或者构造 Upload 对象并使用关键字传入,返回`资源链接`的字典,键为传入的关键字
resp: dict[str, str] = await account.upload(
foo=Upload(file=b'...'),
bar=Upload(file=Path("path/to/file")),
)
对于服务端,你可以通过注册 upload.create
路由来处理上传请求:
from satori.const import Api
from satori.server import Server, Request, FormData, parse_content_disposition
server = Server()
@server.route(Api.UPLOAD_CREATE)
async def on_upload_create(request: Request[FormData]):
# 上传的文件在 `request.params` 中
res = {}
for _, data in request.params.items():
if isinstance(data, str):
continue
ext = data.headers["content-type"]
disp = parse_content_disposition(data.headers["content-disposition"])
res[disp["name"]] = ... # 处理后的资源链接
return res
对于客户端,推荐使用 Account.protocol.download
方法来下载资源:
from satori.client import App, Account, Event
from satori import Image, Upload
app = App()
@app.register
async def _(account: Account, event: Event):
# 假设你获取到了一个 Image 对象, 你想要下载这个资源
img: Image = ...
# 那么你可以传入 `Image.src` 来下载资源
data: bytes = await account.protocol.download(img.src)
# 或者你想下载你通过 `Account.upload` 上传的资源
url = (await account.upload(Upload(file=b'...')))[0]
data: bytes = await account.protocol.download(url)
# 或者你直接传入一个合法的 url
data: bytes = await account.protocol.download("https://example.com/image.png")
若链接符合以下条件之一,则返回链接的代理形式 ({host}/{path}/{version}/proxy/{url}):
- 链接以 "upload://" 开头
- 链接开头出现在 account.self_info.proxy_urls 中的某一项
对于服务端:
- 如果 url 不是合法的 URL,会直接返回 400;
- 如果 url 不以任何一个 Adapter.proxy_urls 中的前缀开头,会直接返回 403;
- 如果 url 是一个内部链接,会由该内部链接的实现决定如何提供此资源 (可能的方式包括返回数据、重定向以及资源无法访问的报错);
- 如果 url 是一个外部链接 (即不以 upload:// 开头的链接),会在 SDK 侧下载该资源并返回 (通常使用流式传输)
你可以通过实现 handle_internal
方法和 handle_proxied
方法来处理内部链接和代理链接的下载请求:
from typing import Optional
from starlette.responses import Response
from satori.server import Server, Provider, Request
class MyProvider(Provider):
# 此处声明的 `proxy_urls` 会同步到 Login.proxy_urls 中
@staticmethod
def proxy_urls() -> list[str]:
return ["https://example.com"]
async def handle_internal(self, request: Request, path: str) -> Response:
# 处理下载请求
...
# prefix 为 Adapter.proxy_urls 中的某一项
# Adapter 类下 download_proxied 已有默认实现,你可以选择自己重写实现
# 若处理下载请求失败,可不做更改抛出异常或返回 None
async def handle_proxied(self, prefix: str, url: str) -> Optional[Response]:
# 处理下载请求
...
# 当 download 返回值的大小超过 stream_threshold 时,会启用流式传输。默认为 16MB
# 你可以通过传入 stream_chunk_size 来设置流式传输的块大小, 默认为 64KB
server = Server(stream_threshold=4 * 1024 * 1024)
server.apply(MyProvider())
server.run()