mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-21 15:01:03 +00:00
130 lines
3.4 KiB
TypeScript
130 lines
3.4 KiB
TypeScript
import type { OpenClawConfig } from "openclaw/plugin-sdk/zalo";
|
|
import { resolveZaloAccount } from "./accounts.js";
|
|
import type { ZaloFetch } from "./api.js";
|
|
import { sendMessage, sendPhoto } from "./api.js";
|
|
import { resolveZaloProxyFetch } from "./proxy.js";
|
|
import { resolveZaloToken } from "./token.js";
|
|
|
|
export type ZaloSendOptions = {
|
|
token?: string;
|
|
accountId?: string;
|
|
cfg?: OpenClawConfig;
|
|
mediaUrl?: string;
|
|
caption?: string;
|
|
verbose?: boolean;
|
|
proxy?: string;
|
|
};
|
|
|
|
export type ZaloSendResult = {
|
|
ok: boolean;
|
|
messageId?: string;
|
|
error?: string;
|
|
};
|
|
|
|
function resolveSendContext(options: ZaloSendOptions): {
|
|
token: string;
|
|
fetcher?: ZaloFetch;
|
|
} {
|
|
if (options.cfg) {
|
|
const account = resolveZaloAccount({
|
|
cfg: options.cfg,
|
|
accountId: options.accountId,
|
|
});
|
|
const token = options.token || account.token;
|
|
const proxy = options.proxy ?? account.config.proxy;
|
|
return { token, fetcher: resolveZaloProxyFetch(proxy) };
|
|
}
|
|
|
|
const token = options.token ?? resolveZaloToken(undefined, options.accountId).token;
|
|
const proxy = options.proxy;
|
|
return { token, fetcher: resolveZaloProxyFetch(proxy) };
|
|
}
|
|
|
|
function resolveValidatedSendContext(
|
|
chatId: string,
|
|
options: ZaloSendOptions,
|
|
): { ok: true; chatId: string; token: string; fetcher?: ZaloFetch } | { ok: false; error: string } {
|
|
const { token, fetcher } = resolveSendContext(options);
|
|
if (!token) {
|
|
return { ok: false, error: "No Zalo bot token configured" };
|
|
}
|
|
const trimmedChatId = chatId?.trim();
|
|
if (!trimmedChatId) {
|
|
return { ok: false, error: "No chat_id provided" };
|
|
}
|
|
return { ok: true, chatId: trimmedChatId, token, fetcher };
|
|
}
|
|
|
|
export async function sendMessageZalo(
|
|
chatId: string,
|
|
text: string,
|
|
options: ZaloSendOptions = {},
|
|
): Promise<ZaloSendResult> {
|
|
const context = resolveValidatedSendContext(chatId, options);
|
|
if (!context.ok) {
|
|
return { ok: false, error: context.error };
|
|
}
|
|
|
|
if (options.mediaUrl) {
|
|
return sendPhotoZalo(context.chatId, options.mediaUrl, {
|
|
...options,
|
|
token: context.token,
|
|
caption: text || options.caption,
|
|
});
|
|
}
|
|
|
|
try {
|
|
const response = await sendMessage(
|
|
context.token,
|
|
{
|
|
chat_id: context.chatId,
|
|
text: text.slice(0, 2000),
|
|
},
|
|
context.fetcher,
|
|
);
|
|
|
|
if (response.ok && response.result) {
|
|
return { ok: true, messageId: response.result.message_id };
|
|
}
|
|
|
|
return { ok: false, error: "Failed to send message" };
|
|
} catch (err) {
|
|
return { ok: false, error: err instanceof Error ? err.message : String(err) };
|
|
}
|
|
}
|
|
|
|
export async function sendPhotoZalo(
|
|
chatId: string,
|
|
photoUrl: string,
|
|
options: ZaloSendOptions = {},
|
|
): Promise<ZaloSendResult> {
|
|
const context = resolveValidatedSendContext(chatId, options);
|
|
if (!context.ok) {
|
|
return { ok: false, error: context.error };
|
|
}
|
|
|
|
if (!photoUrl?.trim()) {
|
|
return { ok: false, error: "No photo URL provided" };
|
|
}
|
|
|
|
try {
|
|
const response = await sendPhoto(
|
|
context.token,
|
|
{
|
|
chat_id: context.chatId,
|
|
photo: photoUrl.trim(),
|
|
caption: options.caption?.slice(0, 2000),
|
|
},
|
|
context.fetcher,
|
|
);
|
|
|
|
if (response.ok && response.result) {
|
|
return { ok: true, messageId: response.result.message_id };
|
|
}
|
|
|
|
return { ok: false, error: "Failed to send photo" };
|
|
} catch (err) {
|
|
return { ok: false, error: err instanceof Error ? err.message : String(err) };
|
|
}
|
|
}
|