Get started
Rencana refaktor presentasi saluran
Status
Diimplementasikan untuk agen bersama, CLI, kemampuan Plugin, dan permukaan pengiriman keluar:
ReplyPayload.presentationmembawa UI pesan semantik.ReplyPayload.delivery.pinmembawa permintaan penyematan pesan terkirim.- Aksi pesan bersama mengekspos
presentation,delivery, danpin, bukancomponents,blocks,buttons, ataucardnative penyedia. - Inti merender atau menurunkan otomatis presentasi melalui kemampuan keluar yang dideklarasikan Plugin.
- Perender Discord, Slack, Telegram, Mattermost, MS Teams, dan Feishu memakai kontrak generik.
- Kode bidang kontrol saluran Discord tidak lagi mengimpor kontainer UI yang didukung Carbon.
Dokumentasi kanonis sekarang berada di Presentasi Pesan. Simpan rencana ini sebagai konteks implementasi historis; perbarui panduan kanonis untuk perubahan kontrak, perender, atau perilaku fallback.
Masalah
UI saluran saat ini terbagi di beberapa permukaan yang tidak kompatibel:
- Inti memiliki hook perender lintas-konteks berbentuk Discord melalui
buildCrossContextComponents. channel.tsDiscord dapat mengimpor UI Carbon native melaluiDiscordUiContainer, yang menarik dependensi UI runtime ke bidang kontrol Plugin saluran.- Agen dan CLI mengekspos celah keluar payload native seperti
componentsDiscord,blocksSlack,buttonsTelegram atau Mattermost, dancardTeams atau Feishu. ReplyPayload.channelDatamembawa petunjuk transport dan amplop UI native.- Model
interactivegenerik sudah ada, tetapi lebih sempit daripada tata letak yang lebih kaya yang sudah digunakan oleh Discord, Slack, Teams, Feishu, LINE, Telegram, dan Mattermost.
Ini membuat inti menyadari bentuk UI native, melemahkan kelambatan runtime Plugin, dan memberi agen terlalu banyak cara khusus penyedia untuk mengekspresikan maksud pesan yang sama.
Tujuan
- Inti menentukan presentasi semantik terbaik untuk sebuah pesan dari kemampuan yang dideklarasikan.
- Ekstensi mendeklarasikan kemampuan dan merender presentasi semantik menjadi payload transport native.
- UI Kontrol Web tetap terpisah dari UI native obrolan.
- Payload saluran native tidak diekspos melalui permukaan pesan agen bersama atau CLI.
- Fitur presentasi yang tidak didukung otomatis diturunkan ke representasi teks terbaik.
- Perilaku pengiriman seperti menyematkan pesan terkirim adalah metadata pengiriman generik, bukan presentasi.
Bukan tujuan
- Tidak ada shim kompatibilitas mundur untuk
buildCrossContextComponents. - Tidak ada celah keluar native publik untuk
components,blocks,buttons, ataucard. - Tidak ada impor inti dari pustaka UI native saluran.
- Tidak ada seam SDK khusus penyedia untuk saluran bawaan.
Model target
Tambahkan field presentation milik inti ke ReplyPayload.
type MessagePresentationTone = "neutral" | "info" | "success" | "warning" | "danger";
type MessagePresentation = {
tone?: MessagePresentationTone;
title?: string;
blocks: MessagePresentationBlock[];
};
type MessagePresentationBlock =
| { type: "text"; text: string }
| { type: "context"; text: string }
| { type: "divider" }
| { type: "buttons"; buttons: MessagePresentationButton[] }
| { type: "select"; placeholder?: string; options: MessagePresentationOption[] };
type MessagePresentationButton = {
label: string;
value?: string;
url?: string;
style?: "primary" | "secondary" | "success" | "danger";
};
type MessagePresentationOption = {
label: string;
value: string;
};
interactive menjadi subset dari presentation selama migrasi:
- Blok teks
interactivedipetakan kepresentation.blocks[].type = "text". - Blok tombol
interactivedipetakan kepresentation.blocks[].type = "buttons". - Blok pilihan
interactivedipetakan kepresentation.blocks[].type = "select".
Skema agen eksternal dan CLI sekarang menggunakan presentation; interactive tetap menjadi helper parser/perender legacy internal untuk produsen balasan yang sudah ada.
Metadata pengiriman
Tambahkan field delivery milik inti untuk perilaku kirim yang bukan UI.
type ReplyPayloadDelivery = {
pin?:
| boolean
| {
enabled: boolean;
notify?: boolean;
required?: boolean;
};
};
Semantik:
delivery.pin = trueberarti sematkan pesan pertama yang berhasil dikirim.notifydefault kefalse.requireddefault kefalse; saluran yang tidak didukung atau kegagalan penyematan otomatis diturunkan dengan melanjutkan pengiriman.- Aksi pesan manual
pin,unpin, danlist-pinstetap ada untuk pesan yang sudah ada.
Pengikatan topik ACP Telegram saat ini harus dipindahkan dari channelData.telegram.pin = true ke delivery.pin = true.
Kontrak kemampuan runtime
Tambahkan hook perender presentasi dan pengiriman ke adaptor keluar runtime, bukan Plugin saluran bidang kontrol.
type ChannelPresentationCapabilities = {
supported: boolean;
buttons?: boolean;
selects?: boolean;
context?: boolean;
divider?: boolean;
tones?: MessagePresentationTone[];
};
type ChannelDeliveryCapabilities = {
pinSentMessage?: boolean;
};
type ChannelOutboundAdapter = {
presentationCapabilities?: ChannelPresentationCapabilities;
renderPresentation?: (params: {
payload: ReplyPayload;
presentation: MessagePresentation;
ctx: ChannelOutboundSendContext;
}) => ReplyPayload | null;
deliveryCapabilities?: ChannelDeliveryCapabilities;
pinDeliveredMessage?: (params: {
cfg: OpenClawConfig;
accountId?: string | null;
to: string;
threadId?: string | number | null;
messageId: string;
notify: boolean;
}) => Promise<void>;
};
Perilaku inti:
- Selesaikan saluran target dan adaptor runtime.
- Minta kemampuan presentasi.
- Turunkan blok yang tidak didukung sebelum merender.
- Panggil
renderPresentation. - Jika tidak ada perender, konversi presentasi menjadi fallback teks.
- Setelah pengiriman berhasil, panggil
pinDeliveredMessagesaatdelivery.pindiminta dan didukung.
Pemetaan saluran
Discord:
- Render
presentationke komponen v2 dan kontainer Carbon di modul khusus runtime. - Simpan helper warna aksen di modul ringan.
- Hapus impor
DiscordUiContainerdari kode bidang kontrol Plugin saluran.
Slack:
- Render
presentationke Block Kit. - Hapus input
blocksagen dan CLI.
Telegram:
- Render teks, konteks, dan pembatas sebagai teks.
- Render aksi dan pilihan sebagai papan tombol inline saat dikonfigurasi dan diizinkan untuk permukaan target.
- Gunakan fallback teks saat tombol inline dinonaktifkan.
- Pindahkan penyematan topik ACP ke
delivery.pin.
Mattermost:
- Render aksi sebagai tombol interaktif saat dikonfigurasi.
- Render blok lain sebagai fallback teks.
MS Teams:
- Render
presentationke Adaptive Cards. - Pertahankan aksi manual pin/unpin/list-pins.
- Secara opsional implementasikan
pinDeliveredMessagejika dukungan Graph andal untuk percakapan target.
Feishu:
- Render
presentationke kartu interaktif. - Pertahankan aksi manual pin/unpin/list-pins.
- Secara opsional implementasikan
pinDeliveredMessageuntuk penyematan pesan terkirim jika perilaku API andal.
LINE:
- Render
presentationke pesan Flex atau templat jika memungkinkan. - Gunakan fallback ke teks untuk blok yang tidak didukung.
- Hapus payload UI LINE dari
channelData.
Saluran polos atau terbatas:
- Konversi presentasi menjadi teks dengan pemformatan konservatif.
Langkah refaktor
- Terapkan ulang perbaikan rilis Discord yang memisahkan
ui-colors.tsdari UI yang didukung Carbon dan menghapusDiscordUiContainerdariextensions/discord/src/channel.ts. - Tambahkan
presentationdandeliverykeReplyPayload, normalisasi payload keluar, ringkasan pengiriman, dan payload hook. - Tambahkan skema
MessagePresentationdan helper parser di subpath SDK/runtime yang sempit. - Ganti kemampuan pesan
buttons,cards,components, danblocksdengan kemampuan presentasi semantik. - Tambahkan hook adaptor keluar runtime untuk perender presentasi dan penyematan pengiriman.
- Ganti konstruksi komponen lintas-konteks dengan
buildCrossContextPresentation. - Hapus
src/infra/outbound/channel-adapters.tsdan hapusbuildCrossContextComponentsdari tipe Plugin saluran. - Ubah
maybeApplyCrossContextMarkeragar melampirkanpresentation, bukan parameter native. - Perbarui jalur kirim plugin-dispatch agar hanya memakai presentasi semantik dan metadata pengiriman.
- Hapus parameter payload native agen dan CLI:
components,blocks,buttons, dancard. - Hapus helper SDK yang membuat skema alat pesan native, dan gantikan dengan helper skema presentasi.
- Hapus amplop UI/native dari
channelData; simpan hanya metadata transport hingga setiap field tersisa ditinjau. - Migrasikan perender Discord, Slack, Telegram, Mattermost, MS Teams, Feishu, dan LINE.
- Perbarui dokumentasi untuk CLI pesan, halaman saluran, SDK Plugin, dan cookbook kemampuan.
- Jalankan profiling fanout impor untuk Discord dan entrypoint saluran yang terdampak.
Langkah 1-11 dan 13-14 diimplementasikan dalam refaktor ini untuk kontrak agen bersama, CLI, kemampuan Plugin, dan adaptor keluar. Langkah 12 tetap menjadi tahap pembersihan internal yang lebih dalam untuk amplop transport channelData privat penyedia. Langkah 15 tetap menjadi validasi lanjutan jika kita menginginkan angka fanout impor terkuantifikasi di luar gate tipe/tes.
Tes
Tambahkan atau perbarui:
- Tes normalisasi presentasi.
- Tes penurunan otomatis presentasi untuk blok yang tidak didukung.
- Tes penanda lintas-konteks untuk plugin dispatch dan jalur pengiriman inti.
- Tes matriks perender saluran untuk Discord, Slack, Telegram, Mattermost, MS Teams, Feishu, LINE, dan fallback teks.
- Tes skema alat pesan yang membuktikan field native sudah hilang.
- Tes CLI yang membuktikan flag native sudah hilang.
- Regresi kelambatan impor entrypoint Discord yang mencakup Carbon.
- Tes penyematan pengiriman yang mencakup Telegram dan fallback generik.
Pertanyaan terbuka
- Haruskah
delivery.pindiimplementasikan untuk Discord, Slack, MS Teams, dan Feishu pada tahap pertama, atau hanya Telegram terlebih dahulu? - Haruskah
deliverypada akhirnya menyerap field yang sudah ada sepertireplyToId,replyToCurrent,silent, danaudioAsVoice, atau tetap berfokus pada perilaku pascakirim? - Haruskah presentasi mendukung gambar atau referensi file secara langsung, atau media harus tetap terpisah dari tata letak UI untuk saat ini?