agent-messenger 2.10.1 → 2.10.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude-plugin/plugin.json +1 -1
- package/dist/package.json +1 -1
- package/dist/src/platforms/teams/token-extractor.d.ts.map +1 -1
- package/dist/src/platforms/teams/token-extractor.js +9 -2
- package/dist/src/platforms/teams/token-extractor.js.map +1 -1
- package/dist/src/shared/chromium/decryptor.d.ts +6 -0
- package/dist/src/shared/chromium/decryptor.d.ts.map +1 -1
- package/dist/src/shared/chromium/decryptor.js +26 -6
- package/dist/src/shared/chromium/decryptor.js.map +1 -1
- package/e2e/channeltalk.e2e.test.ts +13 -13
- package/e2e/channeltalkbot.e2e.test.ts +13 -13
- package/e2e/discord.e2e.test.ts +24 -24
- package/e2e/discordbot.e2e.test.ts +16 -16
- package/e2e/instagram.e2e.test.ts +10 -10
- package/e2e/kakaotalk.e2e.test.ts +7 -7
- package/e2e/line.e2e.test.ts +8 -8
- package/e2e/slack.e2e.test.ts +34 -34
- package/e2e/slackbot.e2e.test.ts +14 -14
- package/e2e/teams.e2e.test.ts +23 -23
- package/e2e/telegram.e2e.test.ts +8 -8
- package/e2e/webex.e2e.test.ts +14 -14
- package/e2e/whatsapp.e2e.test.ts +8 -8
- package/e2e/whatsappbot.e2e.test.ts +6 -6
- package/package.json +1 -1
- package/skills/agent-channeltalk/SKILL.md +1 -1
- package/skills/agent-channeltalkbot/SKILL.md +1 -1
- package/skills/agent-discord/SKILL.md +1 -1
- package/skills/agent-discordbot/SKILL.md +1 -1
- package/skills/agent-instagram/SKILL.md +1 -1
- package/skills/agent-kakaotalk/SKILL.md +1 -1
- package/skills/agent-line/SKILL.md +1 -1
- package/skills/agent-slack/SKILL.md +1 -1
- package/skills/agent-slackbot/SKILL.md +1 -1
- package/skills/agent-teams/SKILL.md +1 -1
- package/skills/agent-telegram/SKILL.md +1 -1
- package/skills/agent-webex/SKILL.md +1 -1
- package/skills/agent-wechatbot/SKILL.md +1 -1
- package/skills/agent-whatsapp/SKILL.md +1 -1
- package/skills/agent-whatsappbot/SKILL.md +1 -1
- package/src/platforms/channeltalk/client.test.ts +26 -26
- package/src/platforms/channeltalk/commands/auth.test.ts +16 -16
- package/src/platforms/channeltalk/commands/bot.test.ts +2 -2
- package/src/platforms/channeltalk/commands/chat.test.ts +3 -3
- package/src/platforms/channeltalk/commands/group.test.ts +4 -4
- package/src/platforms/channeltalk/commands/manager.test.ts +2 -2
- package/src/platforms/channeltalk/commands/message.test.ts +17 -17
- package/src/platforms/channeltalk/commands/snapshot.test.ts +7 -7
- package/src/platforms/channeltalk/commands/whoami.test.ts +3 -3
- package/src/platforms/channeltalk/credential-manager.test.ts +18 -18
- package/src/platforms/channeltalk/ensure-auth.test.ts +5 -5
- package/src/platforms/channeltalk/index.test.ts +23 -23
- package/src/platforms/channeltalk/token-extractor.test.ts +21 -21
- package/src/platforms/channeltalk/types.test.ts +12 -12
- package/src/platforms/channeltalkbot/client.test.ts +14 -14
- package/src/platforms/channeltalkbot/commands/auth.test.ts +16 -16
- package/src/platforms/channeltalkbot/commands/bot.test.ts +6 -6
- package/src/platforms/channeltalkbot/commands/chat.test.ts +9 -9
- package/src/platforms/channeltalkbot/commands/group.test.ts +6 -6
- package/src/platforms/channeltalkbot/commands/manager.test.ts +3 -3
- package/src/platforms/channeltalkbot/commands/message.test.ts +10 -10
- package/src/platforms/channeltalkbot/commands/snapshot.test.ts +7 -7
- package/src/platforms/channeltalkbot/commands/whoami.test.ts +4 -4
- package/src/platforms/channeltalkbot/credential-manager.test.ts +27 -27
- package/src/platforms/channeltalkbot/index.test.ts +15 -15
- package/src/platforms/discord/client.test.ts +28 -28
- package/src/platforms/discord/commands/auth.test.ts +7 -7
- package/src/platforms/discord/commands/channel.test.ts +7 -7
- package/src/platforms/discord/commands/dm.test.ts +4 -4
- package/src/platforms/discord/commands/file.test.ts +4 -4
- package/src/platforms/discord/commands/friend.test.ts +6 -6
- package/src/platforms/discord/commands/member.test.ts +5 -5
- package/src/platforms/discord/commands/mention.test.ts +5 -5
- package/src/platforms/discord/commands/message.test.ts +9 -9
- package/src/platforms/discord/commands/note.test.ts +6 -6
- package/src/platforms/discord/commands/profile.test.ts +4 -4
- package/src/platforms/discord/commands/reaction.test.ts +5 -5
- package/src/platforms/discord/commands/server.test.ts +7 -7
- package/src/platforms/discord/commands/snapshot.test.ts +6 -6
- package/src/platforms/discord/commands/thread.test.ts +6 -6
- package/src/platforms/discord/commands/user.test.ts +5 -5
- package/src/platforms/discord/commands/whoami.test.ts +6 -6
- package/src/platforms/discord/credential-manager.test.ts +16 -16
- package/src/platforms/discord/ensure-auth.test.ts +8 -8
- package/src/platforms/discord/index.test.ts +17 -17
- package/src/platforms/discord/listener.test.ts +33 -33
- package/src/platforms/discord/token-extractor.test.ts +53 -53
- package/src/platforms/discord/types.test.ts +26 -26
- package/src/platforms/discordbot/client.test.ts +31 -31
- package/src/platforms/discordbot/commands/auth.test.ts +18 -18
- package/src/platforms/discordbot/commands/channel.test.ts +11 -11
- package/src/platforms/discordbot/commands/file.test.ts +7 -7
- package/src/platforms/discordbot/commands/message.test.ts +25 -25
- package/src/platforms/discordbot/commands/reaction.test.ts +6 -6
- package/src/platforms/discordbot/commands/server.test.ts +12 -12
- package/src/platforms/discordbot/commands/snapshot.test.ts +13 -13
- package/src/platforms/discordbot/commands/thread.test.ts +10 -10
- package/src/platforms/discordbot/commands/user.test.ts +9 -9
- package/src/platforms/discordbot/commands/whoami.test.ts +4 -4
- package/src/platforms/discordbot/credential-manager.test.ts +28 -28
- package/src/platforms/instagram/client.test.ts +18 -18
- package/src/platforms/instagram/commands/auth.test.ts +11 -11
- package/src/platforms/instagram/commands/chat.test.ts +6 -6
- package/src/platforms/instagram/commands/message.test.ts +11 -11
- package/src/platforms/instagram/commands/shared.test.ts +12 -12
- package/src/platforms/instagram/commands/whoami.test.ts +3 -3
- package/src/platforms/instagram/credential-manager.test.ts +21 -21
- package/src/platforms/instagram/ensure-auth.test.ts +4 -4
- package/src/platforms/instagram/index.test.ts +9 -9
- package/src/platforms/instagram/listener.test.ts +8 -8
- package/src/platforms/instagram/token-extractor.test.ts +35 -35
- package/src/platforms/kakaotalk/client.test.ts +33 -33
- package/src/platforms/kakaotalk/commands/auth.test.ts +11 -11
- package/src/platforms/kakaotalk/commands/chat.test.ts +6 -6
- package/src/platforms/kakaotalk/commands/message.test.ts +7 -7
- package/src/platforms/kakaotalk/commands/whoami.test.ts +5 -5
- package/src/platforms/kakaotalk/credential-manager.test.ts +15 -15
- package/src/platforms/kakaotalk/index.test.ts +15 -15
- package/src/platforms/kakaotalk/listener.test.ts +17 -17
- package/src/platforms/line/client.test.ts +17 -17
- package/src/platforms/line/commands/auth.test.ts +8 -8
- package/src/platforms/line/commands/chat.test.ts +7 -7
- package/src/platforms/line/commands/friend.test.ts +6 -6
- package/src/platforms/line/commands/message.test.ts +7 -7
- package/src/platforms/line/commands/whoami.test.ts +6 -6
- package/src/platforms/line/credential-manager.test.ts +17 -17
- package/src/platforms/line/index.test.ts +10 -10
- package/src/platforms/line/listener.test.ts +15 -15
- package/src/platforms/line/types.test.ts +14 -14
- package/src/platforms/slack/cli.test.ts +8 -8
- package/src/platforms/slack/client.test.ts +151 -151
- package/src/platforms/slack/commands/activity.test.ts +13 -13
- package/src/platforms/slack/commands/auth.test.ts +34 -34
- package/src/platforms/slack/commands/bookmark.test.ts +9 -9
- package/src/platforms/slack/commands/channel.test.ts +17 -17
- package/src/platforms/slack/commands/drafts.test.ts +7 -7
- package/src/platforms/slack/commands/emoji.test.ts +3 -3
- package/src/platforms/slack/commands/file.test.ts +12 -12
- package/src/platforms/slack/commands/message.test.ts +19 -19
- package/src/platforms/slack/commands/pin.test.ts +7 -7
- package/src/platforms/slack/commands/reaction.test.ts +10 -10
- package/src/platforms/slack/commands/reminder.test.ts +9 -9
- package/src/platforms/slack/commands/saved.test.ts +7 -7
- package/src/platforms/slack/commands/sections.test.ts +5 -5
- package/src/platforms/slack/commands/snapshot.test.ts +13 -13
- package/src/platforms/slack/commands/unread.test.ts +6 -6
- package/src/platforms/slack/commands/user.test.ts +10 -10
- package/src/platforms/slack/commands/usergroup.test.ts +15 -15
- package/src/platforms/slack/commands/whoami.test.ts +6 -6
- package/src/platforms/slack/commands/workspace.test.ts +26 -26
- package/src/platforms/slack/credential-manager.test.ts +14 -14
- package/src/platforms/slack/ensure-auth.test.ts +21 -21
- package/src/platforms/slack/index.test.ts +12 -12
- package/src/platforms/slack/listener.test.ts +17 -17
- package/src/platforms/slack/token-extractor-node.test.ts +2 -2
- package/src/platforms/slack/token-extractor.test.ts +37 -37
- package/src/platforms/slack/types.test.ts +21 -21
- package/src/platforms/slackbot/client.test.ts +22 -22
- package/src/platforms/slackbot/commands/auth.test.ts +14 -14
- package/src/platforms/slackbot/commands/channel.test.ts +7 -7
- package/src/platforms/slackbot/commands/message.test.ts +13 -13
- package/src/platforms/slackbot/commands/reaction.test.ts +6 -6
- package/src/platforms/slackbot/commands/user.test.ts +7 -7
- package/src/platforms/slackbot/commands/whoami.test.ts +4 -4
- package/src/platforms/slackbot/credential-manager.test.ts +22 -22
- package/src/platforms/slackbot/types.test.ts +7 -7
- package/src/platforms/teams/client.test.ts +30 -30
- package/src/platforms/teams/commands/auth.test.ts +8 -8
- package/src/platforms/teams/commands/channel.test.ts +7 -7
- package/src/platforms/teams/commands/file.test.ts +4 -4
- package/src/platforms/teams/commands/message.test.ts +5 -5
- package/src/platforms/teams/commands/reaction.test.ts +4 -4
- package/src/platforms/teams/commands/snapshot.test.ts +7 -7
- package/src/platforms/teams/commands/team.test.ts +8 -8
- package/src/platforms/teams/commands/user.test.ts +4 -4
- package/src/platforms/teams/commands/whoami.test.ts +6 -6
- package/src/platforms/teams/credential-manager.test.ts +17 -17
- package/src/platforms/teams/ensure-auth.test.ts +13 -13
- package/src/platforms/teams/index.test.ts +15 -15
- package/src/platforms/teams/token-extractor.test.ts +109 -49
- package/src/platforms/teams/token-extractor.ts +7 -2
- package/src/platforms/teams/types.test.ts +26 -26
- package/src/platforms/telegram/app-config.test.ts +4 -4
- package/src/platforms/telegram/chat-utils.test.ts +12 -12
- package/src/platforms/telegram/client.test.ts +4 -4
- package/src/platforms/telegram/commands/auth.test.ts +16 -16
- package/src/platforms/telegram/commands/chat.test.ts +9 -9
- package/src/platforms/telegram/commands/message.test.ts +6 -6
- package/src/platforms/telegram/commands/shared.test.ts +3 -3
- package/src/platforms/telegram/commands/whoami.test.ts +3 -3
- package/src/platforms/telegram/credential-manager.test.ts +10 -10
- package/src/platforms/telegram/types.test.ts +6 -6
- package/src/platforms/webex/app-config.test.ts +8 -8
- package/src/platforms/webex/cli.test.ts +5 -5
- package/src/platforms/webex/client.test.ts +65 -65
- package/src/platforms/webex/commands/auth.test.ts +18 -18
- package/src/platforms/webex/commands/member.test.ts +5 -5
- package/src/platforms/webex/commands/message.test.ts +12 -12
- package/src/platforms/webex/commands/snapshot.test.ts +5 -5
- package/src/platforms/webex/commands/space.test.ts +10 -10
- package/src/platforms/webex/commands/whoami.test.ts +6 -6
- package/src/platforms/webex/credential-manager.test.ts +22 -22
- package/src/platforms/webex/encryption.test.ts +4 -4
- package/src/platforms/webex/ensure-auth.test.ts +5 -5
- package/src/platforms/webex/index.test.ts +5 -5
- package/src/platforms/webex/markdown-to-html.test.ts +33 -33
- package/src/platforms/webex/token-extractor.test.ts +23 -23
- package/src/platforms/webex/types.test.ts +27 -27
- package/src/platforms/wechatbot/client.test.ts +27 -27
- package/src/platforms/wechatbot/commands/auth.test.ts +15 -15
- package/src/platforms/wechatbot/commands/message.test.ts +8 -8
- package/src/platforms/wechatbot/commands/template.test.ts +9 -9
- package/src/platforms/wechatbot/commands/user.test.ts +7 -7
- package/src/platforms/wechatbot/commands/whoami.test.ts +5 -5
- package/src/platforms/wechatbot/credential-manager.test.ts +18 -18
- package/src/platforms/wechatbot/index.test.ts +10 -10
- package/src/platforms/wechatbot/types.test.ts +25 -25
- package/src/platforms/whatsapp/commands/auth.test.ts +13 -13
- package/src/platforms/whatsapp/commands/chat.test.ts +8 -8
- package/src/platforms/whatsapp/commands/message.test.ts +10 -10
- package/src/platforms/whatsapp/commands/whoami.test.ts +3 -3
- package/src/platforms/whatsapp/credential-manager.test.ts +23 -23
- package/src/platforms/whatsapp/ensure-auth.test.ts +4 -4
- package/src/platforms/whatsapp/index.test.ts +8 -8
- package/src/platforms/whatsapp/types.test.ts +42 -42
- package/src/platforms/whatsappbot/client.test.ts +27 -27
- package/src/platforms/whatsappbot/commands/auth.test.ts +14 -14
- package/src/platforms/whatsappbot/commands/message.test.ts +16 -16
- package/src/platforms/whatsappbot/commands/template.test.ts +9 -9
- package/src/platforms/whatsappbot/commands/whoami.test.ts +5 -5
- package/src/platforms/whatsappbot/credential-manager.test.ts +18 -18
- package/src/platforms/whatsappbot/index.test.ts +7 -7
- package/src/platforms/whatsappbot/types.test.ts +18 -18
- package/src/shared/chromium/browsers.test.ts +22 -22
- package/src/shared/chromium/cookie-reader.test.ts +13 -13
- package/src/shared/chromium/decryptor.test.ts +97 -32
- package/src/shared/chromium/decryptor.ts +27 -6
- package/src/shared/utils/concurrency.test.ts +6 -6
- package/src/shared/utils/derived-key-cache.test.ts +11 -11
- package/src/tui/utils.test.ts +31 -31
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { afterEach, describe, expect, spyOn,
|
|
1
|
+
import { afterEach, describe, expect, spyOn, it } from 'bun:test'
|
|
2
2
|
import { createCipheriv, randomBytes } from 'node:crypto'
|
|
3
3
|
import { existsSync, mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'node:fs'
|
|
4
4
|
import { tmpdir } from 'node:os'
|
|
@@ -18,19 +18,19 @@ describe('ChannelTokenExtractor', () => {
|
|
|
18
18
|
})
|
|
19
19
|
|
|
20
20
|
describe('getAppDataDir', () => {
|
|
21
|
-
|
|
21
|
+
it('returns null for unsupported platform', () => {
|
|
22
22
|
const extractor = new ChannelTokenExtractor('freebsd' as NodeJS.Platform)
|
|
23
23
|
expect(extractor.getAppDataDir()).toBeNull()
|
|
24
24
|
})
|
|
25
25
|
})
|
|
26
26
|
|
|
27
27
|
describe('getCookiesPath', () => {
|
|
28
|
-
|
|
28
|
+
it('returns null for unsupported platform', () => {
|
|
29
29
|
const extractor = new ChannelTokenExtractor('freebsd' as NodeJS.Platform)
|
|
30
30
|
expect(extractor.getCookiesPath()).toBeNull()
|
|
31
31
|
})
|
|
32
32
|
|
|
33
|
-
|
|
33
|
+
it('returns win32 path under AppData/Roaming/Channel Talk/Network', () => {
|
|
34
34
|
// given
|
|
35
35
|
const tempDir = mkdtempSync(join(tmpdir(), 'channel-win-'))
|
|
36
36
|
tempDirs.push(tempDir)
|
|
@@ -58,7 +58,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
58
58
|
})
|
|
59
59
|
|
|
60
60
|
describe('getBrowserCookiesPaths', () => {
|
|
61
|
-
|
|
61
|
+
it('returns browser cookie paths on macOS including Default profile', () => {
|
|
62
62
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
63
63
|
const paths = extractor.getBrowserCookiesPaths()
|
|
64
64
|
|
|
@@ -67,7 +67,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
67
67
|
expect(paths).toContain(join(chromeBase, 'Default', 'Network', 'Cookies'))
|
|
68
68
|
})
|
|
69
69
|
|
|
70
|
-
|
|
70
|
+
it('returns browser cookie paths on Linux', () => {
|
|
71
71
|
const extractor = new ChannelTokenExtractor('linux')
|
|
72
72
|
const paths = extractor.getBrowserCookiesPaths()
|
|
73
73
|
|
|
@@ -75,7 +75,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
75
75
|
expect(paths).toContain(join(chromeBase, 'Default', 'Cookies'))
|
|
76
76
|
})
|
|
77
77
|
|
|
78
|
-
|
|
78
|
+
it('returns browser cookie paths on Windows', () => {
|
|
79
79
|
const extractor = new ChannelTokenExtractor('win32')
|
|
80
80
|
const paths = extractor.getBrowserCookiesPaths()
|
|
81
81
|
|
|
@@ -84,14 +84,14 @@ describe('ChannelTokenExtractor', () => {
|
|
|
84
84
|
expect(paths).toContain(join(chromeBase, 'Default', 'Cookies'))
|
|
85
85
|
})
|
|
86
86
|
|
|
87
|
-
|
|
87
|
+
it('returns empty array for unsupported platform', () => {
|
|
88
88
|
const extractor = new ChannelTokenExtractor('freebsd' as NodeJS.Platform)
|
|
89
89
|
expect(extractor.getBrowserCookiesPaths()).toEqual([])
|
|
90
90
|
})
|
|
91
91
|
})
|
|
92
92
|
|
|
93
93
|
describe('extract', () => {
|
|
94
|
-
|
|
94
|
+
it('returns empty array when desktop cookies path does not exist', async () => {
|
|
95
95
|
class MissingPathExtractor extends ChannelTokenExtractor {
|
|
96
96
|
override getCookiesPath(): string | null {
|
|
97
97
|
return null
|
|
@@ -103,7 +103,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
103
103
|
expect(await extractor.extract()).toEqual([])
|
|
104
104
|
})
|
|
105
105
|
|
|
106
|
-
|
|
106
|
+
it('tries desktop app before browser profiles and collects both', async () => {
|
|
107
107
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
108
108
|
|
|
109
109
|
const desktopSpy = spyOn(extractor as any, 'extractFromDesktopApp').mockResolvedValue({
|
|
@@ -122,7 +122,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
122
122
|
browserSpy.mockRestore()
|
|
123
123
|
})
|
|
124
124
|
|
|
125
|
-
|
|
125
|
+
it('includes browser profiles even when desktop extraction returns null', async () => {
|
|
126
126
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
127
127
|
|
|
128
128
|
const desktopSpy = spyOn(extractor as any, 'extractFromDesktopApp').mockResolvedValue(null)
|
|
@@ -143,7 +143,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
143
143
|
browserSpy.mockRestore()
|
|
144
144
|
})
|
|
145
145
|
|
|
146
|
-
|
|
146
|
+
it('extracts plaintext cookies from a real sqlite database', async () => {
|
|
147
147
|
const tempDir = mkdtempSync(join(tmpdir(), 'channel-cookie-db-'))
|
|
148
148
|
tempDirs.push(tempDir)
|
|
149
149
|
const dbPath = join(tempDir, 'Cookies')
|
|
@@ -173,7 +173,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
173
173
|
])
|
|
174
174
|
})
|
|
175
175
|
|
|
176
|
-
|
|
176
|
+
it('returns token with undefined sessionCookie when only x-account is present', async () => {
|
|
177
177
|
const tempDir = mkdtempSync(join(tmpdir(), 'channel-cookie-db-'))
|
|
178
178
|
tempDirs.push(tempDir)
|
|
179
179
|
const dbPath = join(tempDir, 'Cookies')
|
|
@@ -199,7 +199,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
199
199
|
expect(result[0]?.sessionCookie).toBeUndefined()
|
|
200
200
|
})
|
|
201
201
|
|
|
202
|
-
|
|
202
|
+
it('returns empty array when x-account is missing', async () => {
|
|
203
203
|
const tempDir = mkdtempSync(join(tmpdir(), 'channel-cookie-db-'))
|
|
204
204
|
tempDirs.push(tempDir)
|
|
205
205
|
const dbPath = join(tempDir, 'Cookies')
|
|
@@ -222,7 +222,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
222
222
|
expect(await extractor.extract()).toEqual([])
|
|
223
223
|
})
|
|
224
224
|
|
|
225
|
-
|
|
225
|
+
it('decrypts AES-256-GCM encrypted cookies using master key', async () => {
|
|
226
226
|
// given — known master key and AES-256-GCM encrypted cookie
|
|
227
227
|
const masterKey = randomBytes(32)
|
|
228
228
|
const accountValue = 'encrypted-account-jwt'
|
|
@@ -277,7 +277,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
277
277
|
expect(result[0]?.sessionCookie).toBe('encrypted-session-jwt')
|
|
278
278
|
})
|
|
279
279
|
|
|
280
|
-
|
|
280
|
+
it('deduplicates entries with the same accountCookie from desktop and browser', async () => {
|
|
281
281
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
282
282
|
|
|
283
283
|
const desktopSpy = spyOn(extractor as any, 'extractFromDesktopApp').mockResolvedValue({
|
|
@@ -299,7 +299,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
299
299
|
browserSpy.mockRestore()
|
|
300
300
|
})
|
|
301
301
|
|
|
302
|
-
|
|
302
|
+
it('returns multiple distinct accounts from desktop and browser sources', async () => {
|
|
303
303
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
304
304
|
|
|
305
305
|
const desktopSpy = spyOn(extractor as any, 'extractFromDesktopApp').mockResolvedValue({
|
|
@@ -322,7 +322,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
322
322
|
browserSpy.mockRestore()
|
|
323
323
|
})
|
|
324
324
|
|
|
325
|
-
|
|
325
|
+
it('returns empty array when DPAPI decryption fails', async () => {
|
|
326
326
|
// given
|
|
327
327
|
const masterKey = randomBytes(32)
|
|
328
328
|
const encryptAccount = encryptAESGCM('account-jwt', masterKey)
|
|
@@ -369,14 +369,14 @@ describe('ChannelTokenExtractor', () => {
|
|
|
369
369
|
})
|
|
370
370
|
|
|
371
371
|
describe('decryptDPAPI', () => {
|
|
372
|
-
|
|
372
|
+
it('returns null on non-win32 platform', () => {
|
|
373
373
|
const extractor = new ChannelTokenExtractor('darwin')
|
|
374
374
|
expect(extractor.decryptDPAPI(Buffer.from('test'))).toBeNull()
|
|
375
375
|
})
|
|
376
376
|
})
|
|
377
377
|
|
|
378
378
|
describe('decryptBrowserCookie', () => {
|
|
379
|
-
|
|
379
|
+
it('decrypts v10-prefixed browser cookie using macOS keychain password (AES-128-CBC)', () => {
|
|
380
380
|
// given — AES-128-CBC encrypted cookie with macOS keychain-derived key
|
|
381
381
|
const { createCipheriv, pbkdf2Sync } = require('node:crypto')
|
|
382
382
|
const password = 'test-keychain-password'
|
|
@@ -400,7 +400,7 @@ describe('ChannelTokenExtractor', () => {
|
|
|
400
400
|
execSecuritySpy.mockRestore()
|
|
401
401
|
})
|
|
402
402
|
|
|
403
|
-
|
|
403
|
+
it('decrypts v10-prefixed browser cookie using Linux peanuts key (AES-128-CBC)', () => {
|
|
404
404
|
// given — AES-128-CBC encrypted cookie with Linux Chromium peanuts key
|
|
405
405
|
const { createCipheriv, pbkdf2Sync } = require('node:crypto')
|
|
406
406
|
const key = pbkdf2Sync('peanuts', 'saltysalt', 1, 16, 'sha1')
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { describe, expect,
|
|
1
|
+
import { describe, expect, it } from 'bun:test'
|
|
2
2
|
|
|
3
3
|
import {
|
|
4
4
|
ChannelAccountSchema,
|
|
@@ -10,7 +10,7 @@ import {
|
|
|
10
10
|
} from './types'
|
|
11
11
|
|
|
12
12
|
describe('channel types', () => {
|
|
13
|
-
|
|
13
|
+
it('accepts a valid workspace entry', () => {
|
|
14
14
|
const result = ChannelWorkspaceEntrySchema.safeParse({
|
|
15
15
|
workspace_id: '232986',
|
|
16
16
|
workspace_name: 'Support',
|
|
@@ -23,7 +23,7 @@ describe('channel types', () => {
|
|
|
23
23
|
expect(result.success).toBe(true)
|
|
24
24
|
})
|
|
25
25
|
|
|
26
|
-
|
|
26
|
+
it('accepts a workspace entry without session_cookie', () => {
|
|
27
27
|
const result = ChannelWorkspaceEntrySchema.safeParse({
|
|
28
28
|
workspace_id: '232986',
|
|
29
29
|
workspace_name: 'Support',
|
|
@@ -33,7 +33,7 @@ describe('channel types', () => {
|
|
|
33
33
|
expect(result.success).toBe(true)
|
|
34
34
|
})
|
|
35
35
|
|
|
36
|
-
|
|
36
|
+
it('rejects a workspace entry missing account_cookie', () => {
|
|
37
37
|
const result = ChannelWorkspaceEntrySchema.safeParse({
|
|
38
38
|
workspace_id: '232986',
|
|
39
39
|
workspace_name: 'Support',
|
|
@@ -42,7 +42,7 @@ describe('channel types', () => {
|
|
|
42
42
|
expect(result.success).toBe(false)
|
|
43
43
|
})
|
|
44
44
|
|
|
45
|
-
|
|
45
|
+
it('accepts a valid config', () => {
|
|
46
46
|
const result = ChannelConfigSchema.safeParse({
|
|
47
47
|
current: { workspace_id: '232986' },
|
|
48
48
|
workspaces: {
|
|
@@ -58,7 +58,7 @@ describe('channel types', () => {
|
|
|
58
58
|
expect(result.success).toBe(true)
|
|
59
59
|
})
|
|
60
60
|
|
|
61
|
-
|
|
61
|
+
it('accepts a valid account', () => {
|
|
62
62
|
const result = ChannelAccountSchema.safeParse({
|
|
63
63
|
id: '493041',
|
|
64
64
|
name: 'Devxoul',
|
|
@@ -72,7 +72,7 @@ describe('channel types', () => {
|
|
|
72
72
|
expect(result.success).toBe(true)
|
|
73
73
|
})
|
|
74
74
|
|
|
75
|
-
|
|
75
|
+
it('rejects an invalid account', () => {
|
|
76
76
|
const result = ChannelAccountSchema.safeParse({
|
|
77
77
|
id: '493041',
|
|
78
78
|
name: 'Devxoul',
|
|
@@ -86,7 +86,7 @@ describe('channel types', () => {
|
|
|
86
86
|
expect(result.success).toBe(false)
|
|
87
87
|
})
|
|
88
88
|
|
|
89
|
-
|
|
89
|
+
it('accepts a valid message with blocks', () => {
|
|
90
90
|
const result = ChannelMessageSchema.safeParse({
|
|
91
91
|
id: 'msg_1',
|
|
92
92
|
channelId: '232986',
|
|
@@ -99,7 +99,7 @@ describe('channel types', () => {
|
|
|
99
99
|
expect(result.success).toBe(true)
|
|
100
100
|
})
|
|
101
101
|
|
|
102
|
-
|
|
102
|
+
it('rejects an invalid message block payload', () => {
|
|
103
103
|
const result = ChannelMessageSchema.safeParse({
|
|
104
104
|
id: 'msg_1',
|
|
105
105
|
blocks: [{ type: 'text', value: 123 }],
|
|
@@ -108,7 +108,7 @@ describe('channel types', () => {
|
|
|
108
108
|
expect(result.success).toBe(false)
|
|
109
109
|
})
|
|
110
110
|
|
|
111
|
-
|
|
111
|
+
it('accepts a valid group', () => {
|
|
112
112
|
const result = ChannelGroupSchema.safeParse({
|
|
113
113
|
id: 'group_1',
|
|
114
114
|
channelId: '232986',
|
|
@@ -120,7 +120,7 @@ describe('channel types', () => {
|
|
|
120
120
|
expect(result.success).toBe(true)
|
|
121
121
|
})
|
|
122
122
|
|
|
123
|
-
|
|
123
|
+
it('rejects an invalid group', () => {
|
|
124
124
|
const result = ChannelGroupSchema.safeParse({
|
|
125
125
|
id: 'group_1',
|
|
126
126
|
channelId: '232986',
|
|
@@ -131,7 +131,7 @@ describe('channel types', () => {
|
|
|
131
131
|
expect(result.success).toBe(false)
|
|
132
132
|
})
|
|
133
133
|
|
|
134
|
-
|
|
134
|
+
it('stores the code on ChannelError', () => {
|
|
135
135
|
const error = new ChannelError('boom', 'CHANNEL_FAILED')
|
|
136
136
|
|
|
137
137
|
expect(error).toBeInstanceOf(Error)
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { afterEach, beforeEach, describe, expect,
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, it } from 'bun:test'
|
|
2
2
|
|
|
3
3
|
import { ChannelBotClient } from './client'
|
|
4
4
|
import { ChannelBotError } from './types'
|
|
@@ -46,7 +46,7 @@ describe('ChannelBotClient', () => {
|
|
|
46
46
|
)
|
|
47
47
|
}
|
|
48
48
|
|
|
49
|
-
|
|
49
|
+
it('successful GET request returns unwrapped JSON', async () => {
|
|
50
50
|
mockResponse({ channel: { id: 'ch-1', name: 'My Channel' } })
|
|
51
51
|
|
|
52
52
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -56,7 +56,7 @@ describe('ChannelBotClient', () => {
|
|
|
56
56
|
expect(fetchCalls[0].url).toBe('https://api.channel.io/open/v5/channel')
|
|
57
57
|
})
|
|
58
58
|
|
|
59
|
-
|
|
59
|
+
it('auth headers are set on every request', async () => {
|
|
60
60
|
mockResponse({ channel: { id: 'ch-1', name: 'My Channel' } })
|
|
61
61
|
mockResponse({ user: { id: 'u-1', channelId: 'ch-1' } })
|
|
62
62
|
|
|
@@ -73,7 +73,7 @@ describe('ChannelBotClient', () => {
|
|
|
73
73
|
}
|
|
74
74
|
})
|
|
75
75
|
|
|
76
|
-
|
|
76
|
+
it('429 response triggers retry with Retry-After wait', async () => {
|
|
77
77
|
mockResponse({ message: 'Rate limited' }, 429, { 'Retry-After': '0.05' })
|
|
78
78
|
mockResponse({ channel: { id: 'ch-1', name: 'My Channel' } })
|
|
79
79
|
|
|
@@ -87,7 +87,7 @@ describe('ChannelBotClient', () => {
|
|
|
87
87
|
expect(elapsed).toBeGreaterThanOrEqual(40)
|
|
88
88
|
})
|
|
89
89
|
|
|
90
|
-
|
|
90
|
+
it('500 response triggers retry with exponential backoff', async () => {
|
|
91
91
|
mockResponse({ message: 'Server error' }, 500)
|
|
92
92
|
mockResponse({ message: 'Server error' }, 500)
|
|
93
93
|
mockResponse({ channel: { id: 'ch-1', name: 'My Channel' } })
|
|
@@ -102,7 +102,7 @@ describe('ChannelBotClient', () => {
|
|
|
102
102
|
expect(elapsed).toBeGreaterThanOrEqual(280)
|
|
103
103
|
})
|
|
104
104
|
|
|
105
|
-
|
|
105
|
+
it('4xx non-429 throws immediately without retry', async () => {
|
|
106
106
|
mockResponse({ message: 'Forbidden' }, 403)
|
|
107
107
|
|
|
108
108
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -110,7 +110,7 @@ describe('ChannelBotClient', () => {
|
|
|
110
110
|
expect(fetchCalls).toHaveLength(1)
|
|
111
111
|
})
|
|
112
112
|
|
|
113
|
-
|
|
113
|
+
it('network error retries then throws ChannelBotError with code network_error', async () => {
|
|
114
114
|
;(globalThis as Record<string, unknown>).fetch = async (
|
|
115
115
|
url: string | URL | Request,
|
|
116
116
|
options?: RequestInit,
|
|
@@ -131,7 +131,7 @@ describe('ChannelBotClient', () => {
|
|
|
131
131
|
}
|
|
132
132
|
})
|
|
133
133
|
|
|
134
|
-
|
|
134
|
+
it('204 response returns undefined', async () => {
|
|
135
135
|
mockResponse(null, 204)
|
|
136
136
|
|
|
137
137
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -140,11 +140,11 @@ describe('ChannelBotClient', () => {
|
|
|
140
140
|
expect(result).toBeUndefined()
|
|
141
141
|
})
|
|
142
142
|
|
|
143
|
-
|
|
143
|
+
it('wrapTextInBlocks returns a single text block with value', () => {
|
|
144
144
|
expect(ChannelBotClient.wrapTextInBlocks('Hello world')).toEqual([{ type: 'text', value: 'Hello world' }])
|
|
145
145
|
})
|
|
146
146
|
|
|
147
|
-
|
|
147
|
+
it('sendUserChatMessage includes botName in query string', async () => {
|
|
148
148
|
mockResponse({ message: { id: 'm-1' } })
|
|
149
149
|
|
|
150
150
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -153,7 +153,7 @@ describe('ChannelBotClient', () => {
|
|
|
153
153
|
expect(fetchCalls[0].url).toBe('https://api.channel.io/open/v5/user-chats/chat-1/messages?botName=SupportBot')
|
|
154
154
|
})
|
|
155
155
|
|
|
156
|
-
|
|
156
|
+
it('sendGroupMessage includes botName in query string', async () => {
|
|
157
157
|
mockResponse({ message: { id: 'm-1' } })
|
|
158
158
|
|
|
159
159
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -162,7 +162,7 @@ describe('ChannelBotClient', () => {
|
|
|
162
162
|
expect(fetchCalls[0].url).toBe('https://api.channel.io/open/v5/groups/grp-1/messages?botName=OpsBot')
|
|
163
163
|
})
|
|
164
164
|
|
|
165
|
-
|
|
165
|
+
it('resolveGroup("@team-name") calls groups by name endpoint', async () => {
|
|
166
166
|
mockResponse({ group: { id: 'grp-1', channelId: 'ch-1', name: 'team-name' } })
|
|
167
167
|
|
|
168
168
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -171,7 +171,7 @@ describe('ChannelBotClient', () => {
|
|
|
171
171
|
expect(fetchCalls[0].url).toBe('https://api.channel.io/open/v5/groups/@team-name')
|
|
172
172
|
})
|
|
173
173
|
|
|
174
|
-
|
|
174
|
+
it('resolveGroup("grp123") calls groups by id endpoint', async () => {
|
|
175
175
|
mockResponse({ group: { id: 'grp123', channelId: 'ch-1', name: 'team-name' } })
|
|
176
176
|
|
|
177
177
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -180,7 +180,7 @@ describe('ChannelBotClient', () => {
|
|
|
180
180
|
expect(fetchCalls[0].url).toBe('https://api.channel.io/open/v5/groups/grp123')
|
|
181
181
|
})
|
|
182
182
|
|
|
183
|
-
|
|
183
|
+
it('pagination params are included in query string', async () => {
|
|
184
184
|
mockResponse({ messages: [] })
|
|
185
185
|
|
|
186
186
|
const client = await new ChannelBotClient().login({ accessKey: 'key-1', accessSecret: 'secret-1' })
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { afterEach, beforeEach, describe, expect, mock,
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, mock, it } from 'bun:test'
|
|
2
2
|
import { existsSync, rmSync } from 'node:fs'
|
|
3
3
|
import { mkdir } from 'node:fs/promises'
|
|
4
4
|
import { tmpdir } from 'node:os'
|
|
@@ -39,7 +39,7 @@ describe('auth commands', () => {
|
|
|
39
39
|
})
|
|
40
40
|
|
|
41
41
|
describe('setAction', () => {
|
|
42
|
-
|
|
42
|
+
it('validates and stores credentials with workspace info from API', async () => {
|
|
43
43
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
44
44
|
|
|
45
45
|
const result = await setAction('key123', 'secret123', { _credManager: manager })
|
|
@@ -53,7 +53,7 @@ describe('auth commands', () => {
|
|
|
53
53
|
expect(creds?.workspace_id).toBe('ch123')
|
|
54
54
|
})
|
|
55
55
|
|
|
56
|
-
|
|
56
|
+
it('uses --workspace option as workspace name', async () => {
|
|
57
57
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
58
58
|
|
|
59
59
|
const result = await setAction('key123', 'secret123', {
|
|
@@ -68,7 +68,7 @@ describe('auth commands', () => {
|
|
|
68
68
|
expect(creds?.workspace_name).toBe('My Custom Name')
|
|
69
69
|
})
|
|
70
70
|
|
|
71
|
-
|
|
71
|
+
it('handles client errors gracefully', async () => {
|
|
72
72
|
mockGetChannel.mockImplementationOnce(() => Promise.reject(new Error('Invalid credentials')))
|
|
73
73
|
|
|
74
74
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
@@ -81,7 +81,7 @@ describe('auth commands', () => {
|
|
|
81
81
|
})
|
|
82
82
|
|
|
83
83
|
describe('statusAction', () => {
|
|
84
|
-
|
|
84
|
+
it('returns no credentials when none set', async () => {
|
|
85
85
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
86
86
|
|
|
87
87
|
const result = await statusAction({ _credManager: manager })
|
|
@@ -90,7 +90,7 @@ describe('auth commands', () => {
|
|
|
90
90
|
expect(result.error).toBeDefined()
|
|
91
91
|
})
|
|
92
92
|
|
|
93
|
-
|
|
93
|
+
it('returns valid status for current workspace', async () => {
|
|
94
94
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
95
95
|
await manager.setCredentials({
|
|
96
96
|
workspace_id: 'ch123',
|
|
@@ -105,7 +105,7 @@ describe('auth commands', () => {
|
|
|
105
105
|
expect(result.workspace_id).toBe('ch123')
|
|
106
106
|
})
|
|
107
107
|
|
|
108
|
-
|
|
108
|
+
it('returns invalid when API call fails', async () => {
|
|
109
109
|
mockGetChannel.mockImplementationOnce(() => Promise.reject(new Error('Unauthorized')))
|
|
110
110
|
|
|
111
111
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
@@ -123,7 +123,7 @@ describe('auth commands', () => {
|
|
|
123
123
|
})
|
|
124
124
|
|
|
125
125
|
describe('clearAction', () => {
|
|
126
|
-
|
|
126
|
+
it('removes all stored credentials', async () => {
|
|
127
127
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
128
128
|
await manager.setCredentials({
|
|
129
129
|
workspace_id: 'ch123',
|
|
@@ -140,7 +140,7 @@ describe('auth commands', () => {
|
|
|
140
140
|
})
|
|
141
141
|
|
|
142
142
|
describe('listAction', () => {
|
|
143
|
-
|
|
143
|
+
it('returns all stored workspaces', async () => {
|
|
144
144
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
145
145
|
await manager.setCredentials({
|
|
146
146
|
workspace_id: 'ch1',
|
|
@@ -162,7 +162,7 @@ describe('auth commands', () => {
|
|
|
162
162
|
expect(result.workspaces?.find((w) => w.workspace_id === 'ch1')?.is_current).toBe(false)
|
|
163
163
|
})
|
|
164
164
|
|
|
165
|
-
|
|
165
|
+
it('returns empty list when no workspaces stored', async () => {
|
|
166
166
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
167
167
|
|
|
168
168
|
const result = await listAction({ _credManager: manager })
|
|
@@ -172,7 +172,7 @@ describe('auth commands', () => {
|
|
|
172
172
|
})
|
|
173
173
|
|
|
174
174
|
describe('useAction', () => {
|
|
175
|
-
|
|
175
|
+
it('switches current workspace', async () => {
|
|
176
176
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
177
177
|
await manager.setCredentials({
|
|
178
178
|
workspace_id: 'ch1',
|
|
@@ -193,7 +193,7 @@ describe('auth commands', () => {
|
|
|
193
193
|
expect(result.workspace_id).toBe('ch1')
|
|
194
194
|
})
|
|
195
195
|
|
|
196
|
-
|
|
196
|
+
it('returns error for unknown workspace', async () => {
|
|
197
197
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
198
198
|
|
|
199
199
|
const result = await useAction('nonexistent', { _credManager: manager })
|
|
@@ -203,7 +203,7 @@ describe('auth commands', () => {
|
|
|
203
203
|
})
|
|
204
204
|
|
|
205
205
|
describe('removeAction', () => {
|
|
206
|
-
|
|
206
|
+
it('removes a stored workspace', async () => {
|
|
207
207
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
208
208
|
await manager.setCredentials({
|
|
209
209
|
workspace_id: 'ch1',
|
|
@@ -218,7 +218,7 @@ describe('auth commands', () => {
|
|
|
218
218
|
expect(await manager.getCredentials('ch1')).toBeNull()
|
|
219
219
|
})
|
|
220
220
|
|
|
221
|
-
|
|
221
|
+
it('returns error for unknown workspace', async () => {
|
|
222
222
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
223
223
|
|
|
224
224
|
const result = await removeAction('nonexistent', { _credManager: manager })
|
|
@@ -228,7 +228,7 @@ describe('auth commands', () => {
|
|
|
228
228
|
})
|
|
229
229
|
|
|
230
230
|
describe('botAction', () => {
|
|
231
|
-
|
|
231
|
+
it('sets default bot name', async () => {
|
|
232
232
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
233
233
|
|
|
234
234
|
const result = await botAction('my-bot', { _credManager: manager })
|
|
@@ -238,7 +238,7 @@ describe('auth commands', () => {
|
|
|
238
238
|
expect(await manager.getDefaultBot()).toBe('my-bot')
|
|
239
239
|
})
|
|
240
240
|
|
|
241
|
-
|
|
241
|
+
it('updates existing default bot', async () => {
|
|
242
242
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
243
243
|
await manager.setDefaultBot('old-bot')
|
|
244
244
|
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { afterEach, beforeEach, describe, expect, mock,
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, mock, it } from 'bun:test'
|
|
2
2
|
import { existsSync, rmSync } from 'node:fs'
|
|
3
3
|
import { mkdir } from 'node:fs/promises'
|
|
4
4
|
import { tmpdir } from 'node:os'
|
|
@@ -67,7 +67,7 @@ describe('bot commands', () => {
|
|
|
67
67
|
})
|
|
68
68
|
|
|
69
69
|
describe('listAction', () => {
|
|
70
|
-
|
|
70
|
+
it('returns all bots', async () => {
|
|
71
71
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
72
72
|
const result = await listAction({ _credManager: manager })
|
|
73
73
|
|
|
@@ -78,7 +78,7 @@ describe('bot commands', () => {
|
|
|
78
78
|
})
|
|
79
79
|
|
|
80
80
|
describe('createAction', () => {
|
|
81
|
-
|
|
81
|
+
it('creates bot with name', async () => {
|
|
82
82
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
83
83
|
const result = await createAction('New Bot', { _credManager: manager })
|
|
84
84
|
|
|
@@ -88,7 +88,7 @@ describe('bot commands', () => {
|
|
|
88
88
|
expect(capturedCreateArgs[0]).toBe('New Bot')
|
|
89
89
|
})
|
|
90
90
|
|
|
91
|
-
|
|
91
|
+
it('creates bot with optional color and avatar', async () => {
|
|
92
92
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
93
93
|
await createAction('New Bot', {
|
|
94
94
|
color: '#00FF00',
|
|
@@ -101,7 +101,7 @@ describe('bot commands', () => {
|
|
|
101
101
|
})
|
|
102
102
|
|
|
103
103
|
describe('deleteAction', () => {
|
|
104
|
-
|
|
104
|
+
it('deletes bot with --force flag', async () => {
|
|
105
105
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
106
106
|
const result = await deleteAction('bot1', { force: true, _credManager: manager })
|
|
107
107
|
|
|
@@ -111,7 +111,7 @@ describe('bot commands', () => {
|
|
|
111
111
|
expect(capturedDeleteArg).toBe('bot1')
|
|
112
112
|
})
|
|
113
113
|
|
|
114
|
-
|
|
114
|
+
it('returns error without --force flag', async () => {
|
|
115
115
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
116
116
|
const result = await deleteAction('bot1', { _credManager: manager })
|
|
117
117
|
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { afterEach, beforeEach, describe, expect, mock,
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, mock, it } from 'bun:test'
|
|
2
2
|
import { existsSync, rmSync } from 'node:fs'
|
|
3
3
|
import { mkdir } from 'node:fs/promises'
|
|
4
4
|
import { tmpdir } from 'node:os'
|
|
@@ -92,7 +92,7 @@ describe('chat commands', () => {
|
|
|
92
92
|
})
|
|
93
93
|
|
|
94
94
|
describe('listAction', () => {
|
|
95
|
-
|
|
95
|
+
it('lists opened chats by default', async () => {
|
|
96
96
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
97
97
|
const result = await listAction({ _credManager: manager })
|
|
98
98
|
|
|
@@ -102,14 +102,14 @@ describe('chat commands', () => {
|
|
|
102
102
|
expect(capturedListArgs[0]).toMatchObject({ state: 'opened' })
|
|
103
103
|
})
|
|
104
104
|
|
|
105
|
-
|
|
105
|
+
it('passes state filter to API', async () => {
|
|
106
106
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
107
107
|
await listAction({ state: 'closed', _credManager: manager })
|
|
108
108
|
|
|
109
109
|
expect(capturedListArgs[0]).toMatchObject({ state: 'closed' })
|
|
110
110
|
})
|
|
111
111
|
|
|
112
|
-
|
|
112
|
+
it('passes pagination params', async () => {
|
|
113
113
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
114
114
|
await listAction({ limit: '10', sort: 'asc', since: 'cursor123', _credManager: manager })
|
|
115
115
|
|
|
@@ -118,7 +118,7 @@ describe('chat commands', () => {
|
|
|
118
118
|
})
|
|
119
119
|
|
|
120
120
|
describe('getAction', () => {
|
|
121
|
-
|
|
121
|
+
it('returns specific chat', async () => {
|
|
122
122
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
123
123
|
const result = await getAction('chat1', { _credManager: manager })
|
|
124
124
|
|
|
@@ -129,7 +129,7 @@ describe('chat commands', () => {
|
|
|
129
129
|
})
|
|
130
130
|
|
|
131
131
|
describe('closeAction', () => {
|
|
132
|
-
|
|
132
|
+
it('closes chat with bot name', async () => {
|
|
133
133
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
134
134
|
const result = await closeAction('chat1', { bot: 'my-bot', _credManager: manager })
|
|
135
135
|
|
|
@@ -138,7 +138,7 @@ describe('chat commands', () => {
|
|
|
138
138
|
expect(capturedCloseArgs[1]).toBe('my-bot')
|
|
139
139
|
})
|
|
140
140
|
|
|
141
|
-
|
|
141
|
+
it('returns error when no bot name provided', async () => {
|
|
142
142
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
143
143
|
const result = await closeAction('chat1', { _credManager: manager })
|
|
144
144
|
|
|
@@ -148,7 +148,7 @@ describe('chat commands', () => {
|
|
|
148
148
|
})
|
|
149
149
|
|
|
150
150
|
describe('deleteAction', () => {
|
|
151
|
-
|
|
151
|
+
it('deletes chat with --force flag', async () => {
|
|
152
152
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
153
153
|
const result = await deleteAction('chat1', { force: true, _credManager: manager })
|
|
154
154
|
|
|
@@ -157,7 +157,7 @@ describe('chat commands', () => {
|
|
|
157
157
|
expect(result.deleted).toBe('chat1')
|
|
158
158
|
})
|
|
159
159
|
|
|
160
|
-
|
|
160
|
+
it('returns error without --force flag', async () => {
|
|
161
161
|
const manager = new ChannelBotCredentialManager(tempDir)
|
|
162
162
|
const result = await deleteAction('chat1', { _credManager: manager })
|
|
163
163
|
|