Replace the built-in Feishu SDK with the community-maintained clawdbot-feishu plugin by @m1heng. Changes: - Remove src/feishu/ directory (19 files) - Remove src/channels/plugins/outbound/feishu.ts - Remove src/channels/plugins/normalize/feishu.ts - Remove src/config/types.feishu.ts - Remove feishu exports from plugin-sdk/index.ts - Remove FeishuConfig from types.channels.ts New features in community plugin: - Document tools (read/create/edit Feishu docs) - Wiki tools (navigate/manage knowledge base) - Drive tools (folder/file management) - Bitable tools (read/write table records) - Permission tools (collaborator management) - Emoji reactions support - Typing indicators - Rich media support (bidirectional image/file transfer) - @mention handling - Skills for feishu-doc, feishu-wiki, feishu-drive, feishu-perm Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
160 lines
4.6 KiB
TypeScript
160 lines
4.6 KiB
TypeScript
import type { ClawdbotConfig } from "openclaw/plugin-sdk";
|
|
import type { FeishuConfig } from "./types.js";
|
|
import { createFeishuClient } from "./client.js";
|
|
import { normalizeFeishuTarget } from "./targets.js";
|
|
|
|
export type FeishuDirectoryPeer = {
|
|
kind: "user";
|
|
id: string;
|
|
name?: string;
|
|
};
|
|
|
|
export type FeishuDirectoryGroup = {
|
|
kind: "group";
|
|
id: string;
|
|
name?: string;
|
|
};
|
|
|
|
export async function listFeishuDirectoryPeers(params: {
|
|
cfg: ClawdbotConfig;
|
|
query?: string;
|
|
limit?: number;
|
|
}): Promise<FeishuDirectoryPeer[]> {
|
|
const feishuCfg = params.cfg.channels?.feishu as FeishuConfig | undefined;
|
|
const q = params.query?.trim().toLowerCase() || "";
|
|
const ids = new Set<string>();
|
|
|
|
for (const entry of feishuCfg?.allowFrom ?? []) {
|
|
const trimmed = String(entry).trim();
|
|
if (trimmed && trimmed !== "*") ids.add(trimmed);
|
|
}
|
|
|
|
for (const userId of Object.keys(feishuCfg?.dms ?? {})) {
|
|
const trimmed = userId.trim();
|
|
if (trimmed) ids.add(trimmed);
|
|
}
|
|
|
|
return Array.from(ids)
|
|
.map((raw) => raw.trim())
|
|
.filter(Boolean)
|
|
.map((raw) => normalizeFeishuTarget(raw) ?? raw)
|
|
.filter((id) => (q ? id.toLowerCase().includes(q) : true))
|
|
.slice(0, params.limit && params.limit > 0 ? params.limit : undefined)
|
|
.map((id) => ({ kind: "user" as const, id }));
|
|
}
|
|
|
|
export async function listFeishuDirectoryGroups(params: {
|
|
cfg: ClawdbotConfig;
|
|
query?: string;
|
|
limit?: number;
|
|
}): Promise<FeishuDirectoryGroup[]> {
|
|
const feishuCfg = params.cfg.channels?.feishu as FeishuConfig | undefined;
|
|
const q = params.query?.trim().toLowerCase() || "";
|
|
const ids = new Set<string>();
|
|
|
|
for (const groupId of Object.keys(feishuCfg?.groups ?? {})) {
|
|
const trimmed = groupId.trim();
|
|
if (trimmed && trimmed !== "*") ids.add(trimmed);
|
|
}
|
|
|
|
for (const entry of feishuCfg?.groupAllowFrom ?? []) {
|
|
const trimmed = String(entry).trim();
|
|
if (trimmed && trimmed !== "*") ids.add(trimmed);
|
|
}
|
|
|
|
return Array.from(ids)
|
|
.map((raw) => raw.trim())
|
|
.filter(Boolean)
|
|
.filter((id) => (q ? id.toLowerCase().includes(q) : true))
|
|
.slice(0, params.limit && params.limit > 0 ? params.limit : undefined)
|
|
.map((id) => ({ kind: "group" as const, id }));
|
|
}
|
|
|
|
export async function listFeishuDirectoryPeersLive(params: {
|
|
cfg: ClawdbotConfig;
|
|
query?: string;
|
|
limit?: number;
|
|
}): Promise<FeishuDirectoryPeer[]> {
|
|
const feishuCfg = params.cfg.channels?.feishu as FeishuConfig | undefined;
|
|
if (!feishuCfg?.appId || !feishuCfg?.appSecret) {
|
|
return listFeishuDirectoryPeers(params);
|
|
}
|
|
|
|
try {
|
|
const client = createFeishuClient(feishuCfg);
|
|
const peers: FeishuDirectoryPeer[] = [];
|
|
const limit = params.limit ?? 50;
|
|
|
|
const response = await client.contact.user.list({
|
|
params: {
|
|
page_size: Math.min(limit, 50),
|
|
},
|
|
});
|
|
|
|
if (response.code === 0 && response.data?.items) {
|
|
for (const user of response.data.items) {
|
|
if (user.open_id) {
|
|
const q = params.query?.trim().toLowerCase() || "";
|
|
const name = user.name || "";
|
|
if (!q || user.open_id.toLowerCase().includes(q) || name.toLowerCase().includes(q)) {
|
|
peers.push({
|
|
kind: "user",
|
|
id: user.open_id,
|
|
name: name || undefined,
|
|
});
|
|
}
|
|
}
|
|
if (peers.length >= limit) break;
|
|
}
|
|
}
|
|
|
|
return peers;
|
|
} catch {
|
|
return listFeishuDirectoryPeers(params);
|
|
}
|
|
}
|
|
|
|
export async function listFeishuDirectoryGroupsLive(params: {
|
|
cfg: ClawdbotConfig;
|
|
query?: string;
|
|
limit?: number;
|
|
}): Promise<FeishuDirectoryGroup[]> {
|
|
const feishuCfg = params.cfg.channels?.feishu as FeishuConfig | undefined;
|
|
if (!feishuCfg?.appId || !feishuCfg?.appSecret) {
|
|
return listFeishuDirectoryGroups(params);
|
|
}
|
|
|
|
try {
|
|
const client = createFeishuClient(feishuCfg);
|
|
const groups: FeishuDirectoryGroup[] = [];
|
|
const limit = params.limit ?? 50;
|
|
|
|
const response = await client.im.chat.list({
|
|
params: {
|
|
page_size: Math.min(limit, 100),
|
|
},
|
|
});
|
|
|
|
if (response.code === 0 && response.data?.items) {
|
|
for (const chat of response.data.items) {
|
|
if (chat.chat_id) {
|
|
const q = params.query?.trim().toLowerCase() || "";
|
|
const name = chat.name || "";
|
|
if (!q || chat.chat_id.toLowerCase().includes(q) || name.toLowerCase().includes(q)) {
|
|
groups.push({
|
|
kind: "group",
|
|
id: chat.chat_id,
|
|
name: name || undefined,
|
|
});
|
|
}
|
|
}
|
|
if (groups.length >= limit) break;
|
|
}
|
|
}
|
|
|
|
return groups;
|
|
} catch {
|
|
return listFeishuDirectoryGroups(params);
|
|
}
|
|
}
|