WeChat iLink Bot SDK for Python — async, typed, production-grade
Project description
wechatbot-sdk — Python SDK
WeChat iLink Bot SDK for Python — async, typed, production-grade.
Install
pip install wechatbot-sdk
Requires Python ≥ 3.9. Dependencies: aiohttp, cryptography.
Quick Start
from wechatbot import WeChatBot
bot = WeChatBot()
@bot.on_message
async def handle(msg):
await bot.send_typing(msg.user_id)
await bot.reply(msg, f"Echo: {msg.text}")
bot.run() # login + start in one call
Or with async control:
import asyncio
from wechatbot import WeChatBot
async def main():
bot = WeChatBot()
await bot.login()
@bot.on_message
async def handle(msg):
await bot.reply(msg, f"Echo: {msg.text}")
await bot.start()
asyncio.run(main())
Configuration
bot = WeChatBot(
base_url="https://ilinkai.weixin.qq.com", # default
cred_path="~/.wechatbot/credentials.json", # default
on_qr_url=lambda url: print(f"Scan: {url}"),
on_scanned=lambda: print("Scanned!"),
on_expired=lambda: print("Expired..."),
on_error=lambda err: print(f"Error: {err}"),
)
API Reference
| Method | Description |
|---|---|
await bot.login(force=False) |
QR login (auto-skips if creds exist) |
await bot.start() |
Start long-poll loop |
bot.run() |
Sync: login + start |
bot.stop() |
Stop gracefully |
bot.on_message(handler) |
Register handler (also works as decorator) |
await bot.reply(msg, text) |
Reply (auto context_token + stop typing) |
await bot.send(user_id, text) |
Send to user (needs prior context) |
await bot.send_typing(user_id) |
Show "typing..." indicator |
await bot.stop_typing(user_id) |
Cancel typing indicator |
await bot.reply_media(msg, content) |
Reply with media (image, file, video) |
await bot.send_media(user_id, content) |
Send media to user (needs prior context) |
await bot.download(msg) |
Download media from incoming message |
await bot.download_raw(media, aeskey) |
Download and decrypt a raw CDN media reference |
await bot.upload(data, user_id, media_type) |
Upload file to CDN (does not send) |
Media Operations
Sending Media
# Reply with an image
await bot.reply_media(msg, {"image": png_bytes})
# Reply with a file
await bot.reply_media(msg, {"file": data, "file_name": "report.pdf"})
# Reply with a video
await bot.reply_media(msg, {"video": mp4_bytes, "caption": "Check this"})
# Send media proactively (needs prior context_token)
await bot.send_media(user_id, {"image": png_bytes})
Downloading Media
@bot.on_message
async def handle(msg):
# Auto-detect and download (priority: image > file > video > voice)
media = await bot.download(msg)
if media:
print(f"Type: {media.type}, Size: {len(media.data)} bytes")
if media.file_name:
print(f"Filename: {media.file_name}")
# Or download a raw CDN reference directly
if msg.images:
raw = await bot.download_raw(msg.images[0].media, msg.images[0].aes_key)
Uploading to CDN
# Upload without sending — returns UploadResult with CDN metadata
result = await bot.upload(file_bytes, user_id, media_type=3)
Message Types
@dataclass
class IncomingMessage:
user_id: str
text: str
type: Literal["text", "image", "voice", "file", "video"]
timestamp: datetime
images: list[ImageContent]
voices: list[VoiceContent]
files: list[FileContent]
videos: list[VideoContent]
quoted_message: QuotedMessage | None
raw: dict
AES-128-ECB Crypto
from wechatbot import (
generate_aes_key, encrypt_aes_ecb, decrypt_aes_ecb, decode_aes_key
)
key = generate_aes_key()
ct = encrypt_aes_ecb(b"Hello", key)
pt = decrypt_aes_ecb(ct, key)
# Decode protocol key (all 3 formats)
k = decode_aes_key("ABEiM0RVZneImaq7zN3u/w==") # base64(raw)
k = decode_aes_key("00112233445566778899aabbccddeeff") # hex
Project Structure
python/
├── wechatbot/
│ ├── __init__.py ← Public exports
│ ├── client.py ← WeChatBot (login, start, reply, send)
│ ├── protocol.py ← Raw iLink API calls
│ ├── auth.py ← QR login + credential persistence
│ ├── types.py ← All types (dataclasses)
│ ├── errors.py ← Error hierarchy
│ └── crypto.py ← AES-128-ECB encrypt/decrypt
├── examples/
│ └── echo_bot.py
├── tests/
│ ├── test_crypto.py ← 10 tests
│ └── test_client.py ← 8 tests
└── pyproject.toml
Testing
pip install -e ".[dev]"
pytest
License
MIT
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 wechatbot_sdk-0.2.1.tar.gz.
File metadata
- Download URL: wechatbot_sdk-0.2.1.tar.gz
- Upload date:
- Size: 14.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7c06d252fdc35110e0ef6fdd701469c04a1436a602b1088e4d3c30f944ec01c3
|
|
| MD5 |
1430771040c8da08ab09c0485f7f5b5f
|
|
| BLAKE2b-256 |
42fefc419ff228e64a5d02de7ad06a5b663dcef10502e10a2263ae667b7aa80b
|
Provenance
The following attestation bundles were made for wechatbot_sdk-0.2.1.tar.gz:
Publisher:
publish-pypi.yml on corespeed-io/wechatbot
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
wechatbot_sdk-0.2.1.tar.gz -
Subject digest:
7c06d252fdc35110e0ef6fdd701469c04a1436a602b1088e4d3c30f944ec01c3 - Sigstore transparency entry: 1328693345
- Sigstore integration time:
-
Permalink:
corespeed-io/wechatbot@c081868c905d24f10e30fc6deef7b23ba1b18389 -
Branch / Tag:
refs/tags/py-v0.2.1 - Owner: https://github.com/corespeed-io
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@c081868c905d24f10e30fc6deef7b23ba1b18389 -
Trigger Event:
push
-
Statement type:
File details
Details for the file wechatbot_sdk-0.2.1-py3-none-any.whl.
File metadata
- Download URL: wechatbot_sdk-0.2.1-py3-none-any.whl
- Upload date:
- Size: 16.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
957e13161c9d689806cd78c39d5c050f1365ede4cfa8653ce4a7395ce1278b19
|
|
| MD5 |
8b018651fa3988f0eb2a80a8115dd598
|
|
| BLAKE2b-256 |
9620ddb52f93bc90e38d4112ba427a8ec7867a82fdea30e593540fe78b7dfd76
|
Provenance
The following attestation bundles were made for wechatbot_sdk-0.2.1-py3-none-any.whl:
Publisher:
publish-pypi.yml on corespeed-io/wechatbot
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
wechatbot_sdk-0.2.1-py3-none-any.whl -
Subject digest:
957e13161c9d689806cd78c39d5c050f1365ede4cfa8653ce4a7395ce1278b19 - Sigstore transparency entry: 1328693401
- Sigstore integration time:
-
Permalink:
corespeed-io/wechatbot@c081868c905d24f10e30fc6deef7b23ba1b18389 -
Branch / Tag:
refs/tags/py-v0.2.1 - Owner: https://github.com/corespeed-io
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@c081868c905d24f10e30fc6deef7b23ba1b18389 -
Trigger Event:
push
-
Statement type: