Telegram callback_data is one bot-wide namespace. Any extension that creates inline buttons for a bot shared with pi-telegram must use namespaced callback data.
<namespace>:<action>[:<payload>]
Examples:
vividfish:approve:123
vividfish:deny:123
myext:page:2
- Use a stable extension-owned namespace, preferably the package or extension name without scope punctuation.
- Keep the namespace lowercase ASCII:
a-z,0-9,_,-. - Do not use
pi-telegramowned prefixes:compact:,tgbtn:,menu:,model:,thinking:,status:,queue:,settings:,section:. Current app navigation usesmenu:;status:remains reserved for legacy/owned status callbacks but is not emitted by current UI.compact:is owned by the manual compaction confirmation dialog.section:is owned by the Extension Sections platform (0.10.0+), documented in Extension Sections.settings:is owned for the built-in Settings submenu. - Keep the full
callback_datawithin Telegram's 64-byte limit. - Put only opaque ids or small enum values in payloads; do not store secrets, full prompts, or large state.
- Treat callbacks as untrusted input. Validate namespace, action, and payload before executing side effects.
If pi-telegram receives callback data that is not owned by its built-in prefixes and no built-in handler consumes it, it forwards the click to Pi as:
[callback] <callback_data>
Layered extensions may intercept that message and handle their own namespace. If no extension handles it, the assistant may see the fallback message and should tell the user the callback was not handled and the environment may be misconfigured.
Telegram Extension Sections are a higher-level UI contract over this namespace rule. A section owns a canonical extension identity such as @llblab/pi-telegram-explorer, but its Telegram callback_data should use the pi-telegram owned section: prefix plus a compact token, because Telegram limits callback payloads to 64 bytes.
Conceptual form:
section:<token>:<action>[:<payload>]
The token maps back to the full section identity inside the section registry. Section authors should not hand-roll section: callbacks outside the section context helpers, and ordinary layered extensions should continue using their own namespace plus update handlers or the [callback] fallback.