* feat: add Telegram channel with agent swarm support Add Telegram as a messaging channel that can run alongside WhatsApp or standalone (TELEGRAM_ONLY mode). Includes bot pool support for agent swarms where each subagent appears as a different bot identity in the group. - Add grammy dependency for Telegram Bot API - Route messages through tg: JID prefix convention - Add storeMessageDirect for non-Baileys channels - Add sender field to IPC send_message for swarm identity - Support TELEGRAM_BOT_TOKEN, TELEGRAM_ONLY, TELEGRAM_BOT_POOL config Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * docs: add index.ts refactor plan Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * refactor: extract channel abstraction, IPC, and router from index.ts Break the 1088-line monolith into focused modules: - src/channels/whatsapp.ts: WhatsAppChannel class implementing Channel interface - src/ipc.ts: IPC watcher and task processing with dependency injection - src/router.ts: message formatting, outbound routing, channel lookup - src/types.ts: Channel interface, OnInboundMessage, OnChatMetadata types Also adds regression test suite (98 tests), updates all documentation and skill files to reflect the new architecture. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * ci: add test workflow for PRs Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * chore: remove accidentally committed pool-bot assets Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com> * fix(ci): remove grammy from base dependencies Grammy is installed by the /add-telegram skill, not a base dependency. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
92 lines
3.0 KiB
TypeScript
92 lines
3.0 KiB
TypeScript
import { describe, it, expect, beforeEach } from 'vitest';
|
|
|
|
import { _initTestDatabase, getAllChats, storeChatMetadata } from './db.js';
|
|
import { getAvailableGroups, _setRegisteredGroups } from './index.js';
|
|
|
|
beforeEach(() => {
|
|
_initTestDatabase();
|
|
_setRegisteredGroups({});
|
|
});
|
|
|
|
// --- JID ownership patterns ---
|
|
|
|
describe('JID ownership patterns', () => {
|
|
// These test the patterns that will become ownsJid() on the Channel interface
|
|
|
|
it('WhatsApp group JID: ends with @g.us', () => {
|
|
const jid = '12345678@g.us';
|
|
expect(jid.endsWith('@g.us')).toBe(true);
|
|
});
|
|
|
|
it('WhatsApp DM JID: ends with @s.whatsapp.net', () => {
|
|
const jid = '12345678@s.whatsapp.net';
|
|
expect(jid.endsWith('@s.whatsapp.net')).toBe(true);
|
|
});
|
|
|
|
it('unknown JID format: does not match WhatsApp patterns', () => {
|
|
const jid = 'unknown:12345';
|
|
expect(jid.endsWith('@g.us')).toBe(false);
|
|
expect(jid.endsWith('@s.whatsapp.net')).toBe(false);
|
|
});
|
|
});
|
|
|
|
// --- getAvailableGroups ---
|
|
|
|
describe('getAvailableGroups', () => {
|
|
it('returns only @g.us JIDs', () => {
|
|
storeChatMetadata('group1@g.us', '2024-01-01T00:00:01.000Z', 'Group 1');
|
|
storeChatMetadata('user@s.whatsapp.net', '2024-01-01T00:00:02.000Z', 'User DM');
|
|
storeChatMetadata('group2@g.us', '2024-01-01T00:00:03.000Z', 'Group 2');
|
|
|
|
const groups = getAvailableGroups();
|
|
expect(groups).toHaveLength(2);
|
|
expect(groups.every((g) => g.jid.endsWith('@g.us'))).toBe(true);
|
|
});
|
|
|
|
it('excludes __group_sync__ sentinel', () => {
|
|
storeChatMetadata('__group_sync__', '2024-01-01T00:00:00.000Z');
|
|
storeChatMetadata('group@g.us', '2024-01-01T00:00:01.000Z', 'Group');
|
|
|
|
const groups = getAvailableGroups();
|
|
expect(groups).toHaveLength(1);
|
|
expect(groups[0].jid).toBe('group@g.us');
|
|
});
|
|
|
|
it('marks registered groups correctly', () => {
|
|
storeChatMetadata('reg@g.us', '2024-01-01T00:00:01.000Z', 'Registered');
|
|
storeChatMetadata('unreg@g.us', '2024-01-01T00:00:02.000Z', 'Unregistered');
|
|
|
|
_setRegisteredGroups({
|
|
'reg@g.us': {
|
|
name: 'Registered',
|
|
folder: 'registered',
|
|
trigger: '@Andy',
|
|
added_at: '2024-01-01T00:00:00.000Z',
|
|
},
|
|
});
|
|
|
|
const groups = getAvailableGroups();
|
|
const reg = groups.find((g) => g.jid === 'reg@g.us');
|
|
const unreg = groups.find((g) => g.jid === 'unreg@g.us');
|
|
|
|
expect(reg?.isRegistered).toBe(true);
|
|
expect(unreg?.isRegistered).toBe(false);
|
|
});
|
|
|
|
it('returns groups ordered by most recent activity', () => {
|
|
storeChatMetadata('old@g.us', '2024-01-01T00:00:01.000Z', 'Old');
|
|
storeChatMetadata('new@g.us', '2024-01-01T00:00:05.000Z', 'New');
|
|
storeChatMetadata('mid@g.us', '2024-01-01T00:00:03.000Z', 'Mid');
|
|
|
|
const groups = getAvailableGroups();
|
|
expect(groups[0].jid).toBe('new@g.us');
|
|
expect(groups[1].jid).toBe('mid@g.us');
|
|
expect(groups[2].jid).toBe('old@g.us');
|
|
});
|
|
|
|
it('returns empty array when no chats exist', () => {
|
|
const groups = getAvailableGroups();
|
|
expect(groups).toHaveLength(0);
|
|
});
|
|
});
|