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.
Files changed (239) hide show
  1. package/.claude-plugin/plugin.json +1 -1
  2. package/dist/package.json +1 -1
  3. package/dist/src/platforms/teams/token-extractor.d.ts.map +1 -1
  4. package/dist/src/platforms/teams/token-extractor.js +9 -2
  5. package/dist/src/platforms/teams/token-extractor.js.map +1 -1
  6. package/dist/src/shared/chromium/decryptor.d.ts +6 -0
  7. package/dist/src/shared/chromium/decryptor.d.ts.map +1 -1
  8. package/dist/src/shared/chromium/decryptor.js +26 -6
  9. package/dist/src/shared/chromium/decryptor.js.map +1 -1
  10. package/e2e/channeltalk.e2e.test.ts +13 -13
  11. package/e2e/channeltalkbot.e2e.test.ts +13 -13
  12. package/e2e/discord.e2e.test.ts +24 -24
  13. package/e2e/discordbot.e2e.test.ts +16 -16
  14. package/e2e/instagram.e2e.test.ts +10 -10
  15. package/e2e/kakaotalk.e2e.test.ts +7 -7
  16. package/e2e/line.e2e.test.ts +8 -8
  17. package/e2e/slack.e2e.test.ts +34 -34
  18. package/e2e/slackbot.e2e.test.ts +14 -14
  19. package/e2e/teams.e2e.test.ts +23 -23
  20. package/e2e/telegram.e2e.test.ts +8 -8
  21. package/e2e/webex.e2e.test.ts +14 -14
  22. package/e2e/whatsapp.e2e.test.ts +8 -8
  23. package/e2e/whatsappbot.e2e.test.ts +6 -6
  24. package/package.json +1 -1
  25. package/skills/agent-channeltalk/SKILL.md +1 -1
  26. package/skills/agent-channeltalkbot/SKILL.md +1 -1
  27. package/skills/agent-discord/SKILL.md +1 -1
  28. package/skills/agent-discordbot/SKILL.md +1 -1
  29. package/skills/agent-instagram/SKILL.md +1 -1
  30. package/skills/agent-kakaotalk/SKILL.md +1 -1
  31. package/skills/agent-line/SKILL.md +1 -1
  32. package/skills/agent-slack/SKILL.md +1 -1
  33. package/skills/agent-slackbot/SKILL.md +1 -1
  34. package/skills/agent-teams/SKILL.md +1 -1
  35. package/skills/agent-telegram/SKILL.md +1 -1
  36. package/skills/agent-webex/SKILL.md +1 -1
  37. package/skills/agent-wechatbot/SKILL.md +1 -1
  38. package/skills/agent-whatsapp/SKILL.md +1 -1
  39. package/skills/agent-whatsappbot/SKILL.md +1 -1
  40. package/src/platforms/channeltalk/client.test.ts +26 -26
  41. package/src/platforms/channeltalk/commands/auth.test.ts +16 -16
  42. package/src/platforms/channeltalk/commands/bot.test.ts +2 -2
  43. package/src/platforms/channeltalk/commands/chat.test.ts +3 -3
  44. package/src/platforms/channeltalk/commands/group.test.ts +4 -4
  45. package/src/platforms/channeltalk/commands/manager.test.ts +2 -2
  46. package/src/platforms/channeltalk/commands/message.test.ts +17 -17
  47. package/src/platforms/channeltalk/commands/snapshot.test.ts +7 -7
  48. package/src/platforms/channeltalk/commands/whoami.test.ts +3 -3
  49. package/src/platforms/channeltalk/credential-manager.test.ts +18 -18
  50. package/src/platforms/channeltalk/ensure-auth.test.ts +5 -5
  51. package/src/platforms/channeltalk/index.test.ts +23 -23
  52. package/src/platforms/channeltalk/token-extractor.test.ts +21 -21
  53. package/src/platforms/channeltalk/types.test.ts +12 -12
  54. package/src/platforms/channeltalkbot/client.test.ts +14 -14
  55. package/src/platforms/channeltalkbot/commands/auth.test.ts +16 -16
  56. package/src/platforms/channeltalkbot/commands/bot.test.ts +6 -6
  57. package/src/platforms/channeltalkbot/commands/chat.test.ts +9 -9
  58. package/src/platforms/channeltalkbot/commands/group.test.ts +6 -6
  59. package/src/platforms/channeltalkbot/commands/manager.test.ts +3 -3
  60. package/src/platforms/channeltalkbot/commands/message.test.ts +10 -10
  61. package/src/platforms/channeltalkbot/commands/snapshot.test.ts +7 -7
  62. package/src/platforms/channeltalkbot/commands/whoami.test.ts +4 -4
  63. package/src/platforms/channeltalkbot/credential-manager.test.ts +27 -27
  64. package/src/platforms/channeltalkbot/index.test.ts +15 -15
  65. package/src/platforms/discord/client.test.ts +28 -28
  66. package/src/platforms/discord/commands/auth.test.ts +7 -7
  67. package/src/platforms/discord/commands/channel.test.ts +7 -7
  68. package/src/platforms/discord/commands/dm.test.ts +4 -4
  69. package/src/platforms/discord/commands/file.test.ts +4 -4
  70. package/src/platforms/discord/commands/friend.test.ts +6 -6
  71. package/src/platforms/discord/commands/member.test.ts +5 -5
  72. package/src/platforms/discord/commands/mention.test.ts +5 -5
  73. package/src/platforms/discord/commands/message.test.ts +9 -9
  74. package/src/platforms/discord/commands/note.test.ts +6 -6
  75. package/src/platforms/discord/commands/profile.test.ts +4 -4
  76. package/src/platforms/discord/commands/reaction.test.ts +5 -5
  77. package/src/platforms/discord/commands/server.test.ts +7 -7
  78. package/src/platforms/discord/commands/snapshot.test.ts +6 -6
  79. package/src/platforms/discord/commands/thread.test.ts +6 -6
  80. package/src/platforms/discord/commands/user.test.ts +5 -5
  81. package/src/platforms/discord/commands/whoami.test.ts +6 -6
  82. package/src/platforms/discord/credential-manager.test.ts +16 -16
  83. package/src/platforms/discord/ensure-auth.test.ts +8 -8
  84. package/src/platforms/discord/index.test.ts +17 -17
  85. package/src/platforms/discord/listener.test.ts +33 -33
  86. package/src/platforms/discord/token-extractor.test.ts +53 -53
  87. package/src/platforms/discord/types.test.ts +26 -26
  88. package/src/platforms/discordbot/client.test.ts +31 -31
  89. package/src/platforms/discordbot/commands/auth.test.ts +18 -18
  90. package/src/platforms/discordbot/commands/channel.test.ts +11 -11
  91. package/src/platforms/discordbot/commands/file.test.ts +7 -7
  92. package/src/platforms/discordbot/commands/message.test.ts +25 -25
  93. package/src/platforms/discordbot/commands/reaction.test.ts +6 -6
  94. package/src/platforms/discordbot/commands/server.test.ts +12 -12
  95. package/src/platforms/discordbot/commands/snapshot.test.ts +13 -13
  96. package/src/platforms/discordbot/commands/thread.test.ts +10 -10
  97. package/src/platforms/discordbot/commands/user.test.ts +9 -9
  98. package/src/platforms/discordbot/commands/whoami.test.ts +4 -4
  99. package/src/platforms/discordbot/credential-manager.test.ts +28 -28
  100. package/src/platforms/instagram/client.test.ts +18 -18
  101. package/src/platforms/instagram/commands/auth.test.ts +11 -11
  102. package/src/platforms/instagram/commands/chat.test.ts +6 -6
  103. package/src/platforms/instagram/commands/message.test.ts +11 -11
  104. package/src/platforms/instagram/commands/shared.test.ts +12 -12
  105. package/src/platforms/instagram/commands/whoami.test.ts +3 -3
  106. package/src/platforms/instagram/credential-manager.test.ts +21 -21
  107. package/src/platforms/instagram/ensure-auth.test.ts +4 -4
  108. package/src/platforms/instagram/index.test.ts +9 -9
  109. package/src/platforms/instagram/listener.test.ts +8 -8
  110. package/src/platforms/instagram/token-extractor.test.ts +35 -35
  111. package/src/platforms/kakaotalk/client.test.ts +33 -33
  112. package/src/platforms/kakaotalk/commands/auth.test.ts +11 -11
  113. package/src/platforms/kakaotalk/commands/chat.test.ts +6 -6
  114. package/src/platforms/kakaotalk/commands/message.test.ts +7 -7
  115. package/src/platforms/kakaotalk/commands/whoami.test.ts +5 -5
  116. package/src/platforms/kakaotalk/credential-manager.test.ts +15 -15
  117. package/src/platforms/kakaotalk/index.test.ts +15 -15
  118. package/src/platforms/kakaotalk/listener.test.ts +17 -17
  119. package/src/platforms/line/client.test.ts +17 -17
  120. package/src/platforms/line/commands/auth.test.ts +8 -8
  121. package/src/platforms/line/commands/chat.test.ts +7 -7
  122. package/src/platforms/line/commands/friend.test.ts +6 -6
  123. package/src/platforms/line/commands/message.test.ts +7 -7
  124. package/src/platforms/line/commands/whoami.test.ts +6 -6
  125. package/src/platforms/line/credential-manager.test.ts +17 -17
  126. package/src/platforms/line/index.test.ts +10 -10
  127. package/src/platforms/line/listener.test.ts +15 -15
  128. package/src/platforms/line/types.test.ts +14 -14
  129. package/src/platforms/slack/cli.test.ts +8 -8
  130. package/src/platforms/slack/client.test.ts +151 -151
  131. package/src/platforms/slack/commands/activity.test.ts +13 -13
  132. package/src/platforms/slack/commands/auth.test.ts +34 -34
  133. package/src/platforms/slack/commands/bookmark.test.ts +9 -9
  134. package/src/platforms/slack/commands/channel.test.ts +17 -17
  135. package/src/platforms/slack/commands/drafts.test.ts +7 -7
  136. package/src/platforms/slack/commands/emoji.test.ts +3 -3
  137. package/src/platforms/slack/commands/file.test.ts +12 -12
  138. package/src/platforms/slack/commands/message.test.ts +19 -19
  139. package/src/platforms/slack/commands/pin.test.ts +7 -7
  140. package/src/platforms/slack/commands/reaction.test.ts +10 -10
  141. package/src/platforms/slack/commands/reminder.test.ts +9 -9
  142. package/src/platforms/slack/commands/saved.test.ts +7 -7
  143. package/src/platforms/slack/commands/sections.test.ts +5 -5
  144. package/src/platforms/slack/commands/snapshot.test.ts +13 -13
  145. package/src/platforms/slack/commands/unread.test.ts +6 -6
  146. package/src/platforms/slack/commands/user.test.ts +10 -10
  147. package/src/platforms/slack/commands/usergroup.test.ts +15 -15
  148. package/src/platforms/slack/commands/whoami.test.ts +6 -6
  149. package/src/platforms/slack/commands/workspace.test.ts +26 -26
  150. package/src/platforms/slack/credential-manager.test.ts +14 -14
  151. package/src/platforms/slack/ensure-auth.test.ts +21 -21
  152. package/src/platforms/slack/index.test.ts +12 -12
  153. package/src/platforms/slack/listener.test.ts +17 -17
  154. package/src/platforms/slack/token-extractor-node.test.ts +2 -2
  155. package/src/platforms/slack/token-extractor.test.ts +37 -37
  156. package/src/platforms/slack/types.test.ts +21 -21
  157. package/src/platforms/slackbot/client.test.ts +22 -22
  158. package/src/platforms/slackbot/commands/auth.test.ts +14 -14
  159. package/src/platforms/slackbot/commands/channel.test.ts +7 -7
  160. package/src/platforms/slackbot/commands/message.test.ts +13 -13
  161. package/src/platforms/slackbot/commands/reaction.test.ts +6 -6
  162. package/src/platforms/slackbot/commands/user.test.ts +7 -7
  163. package/src/platforms/slackbot/commands/whoami.test.ts +4 -4
  164. package/src/platforms/slackbot/credential-manager.test.ts +22 -22
  165. package/src/platforms/slackbot/types.test.ts +7 -7
  166. package/src/platforms/teams/client.test.ts +30 -30
  167. package/src/platforms/teams/commands/auth.test.ts +8 -8
  168. package/src/platforms/teams/commands/channel.test.ts +7 -7
  169. package/src/platforms/teams/commands/file.test.ts +4 -4
  170. package/src/platforms/teams/commands/message.test.ts +5 -5
  171. package/src/platforms/teams/commands/reaction.test.ts +4 -4
  172. package/src/platforms/teams/commands/snapshot.test.ts +7 -7
  173. package/src/platforms/teams/commands/team.test.ts +8 -8
  174. package/src/platforms/teams/commands/user.test.ts +4 -4
  175. package/src/platforms/teams/commands/whoami.test.ts +6 -6
  176. package/src/platforms/teams/credential-manager.test.ts +17 -17
  177. package/src/platforms/teams/ensure-auth.test.ts +13 -13
  178. package/src/platforms/teams/index.test.ts +15 -15
  179. package/src/platforms/teams/token-extractor.test.ts +109 -49
  180. package/src/platforms/teams/token-extractor.ts +7 -2
  181. package/src/platforms/teams/types.test.ts +26 -26
  182. package/src/platforms/telegram/app-config.test.ts +4 -4
  183. package/src/platforms/telegram/chat-utils.test.ts +12 -12
  184. package/src/platforms/telegram/client.test.ts +4 -4
  185. package/src/platforms/telegram/commands/auth.test.ts +16 -16
  186. package/src/platforms/telegram/commands/chat.test.ts +9 -9
  187. package/src/platforms/telegram/commands/message.test.ts +6 -6
  188. package/src/platforms/telegram/commands/shared.test.ts +3 -3
  189. package/src/platforms/telegram/commands/whoami.test.ts +3 -3
  190. package/src/platforms/telegram/credential-manager.test.ts +10 -10
  191. package/src/platforms/telegram/types.test.ts +6 -6
  192. package/src/platforms/webex/app-config.test.ts +8 -8
  193. package/src/platforms/webex/cli.test.ts +5 -5
  194. package/src/platforms/webex/client.test.ts +65 -65
  195. package/src/platforms/webex/commands/auth.test.ts +18 -18
  196. package/src/platforms/webex/commands/member.test.ts +5 -5
  197. package/src/platforms/webex/commands/message.test.ts +12 -12
  198. package/src/platforms/webex/commands/snapshot.test.ts +5 -5
  199. package/src/platforms/webex/commands/space.test.ts +10 -10
  200. package/src/platforms/webex/commands/whoami.test.ts +6 -6
  201. package/src/platforms/webex/credential-manager.test.ts +22 -22
  202. package/src/platforms/webex/encryption.test.ts +4 -4
  203. package/src/platforms/webex/ensure-auth.test.ts +5 -5
  204. package/src/platforms/webex/index.test.ts +5 -5
  205. package/src/platforms/webex/markdown-to-html.test.ts +33 -33
  206. package/src/platforms/webex/token-extractor.test.ts +23 -23
  207. package/src/platforms/webex/types.test.ts +27 -27
  208. package/src/platforms/wechatbot/client.test.ts +27 -27
  209. package/src/platforms/wechatbot/commands/auth.test.ts +15 -15
  210. package/src/platforms/wechatbot/commands/message.test.ts +8 -8
  211. package/src/platforms/wechatbot/commands/template.test.ts +9 -9
  212. package/src/platforms/wechatbot/commands/user.test.ts +7 -7
  213. package/src/platforms/wechatbot/commands/whoami.test.ts +5 -5
  214. package/src/platforms/wechatbot/credential-manager.test.ts +18 -18
  215. package/src/platforms/wechatbot/index.test.ts +10 -10
  216. package/src/platforms/wechatbot/types.test.ts +25 -25
  217. package/src/platforms/whatsapp/commands/auth.test.ts +13 -13
  218. package/src/platforms/whatsapp/commands/chat.test.ts +8 -8
  219. package/src/platforms/whatsapp/commands/message.test.ts +10 -10
  220. package/src/platforms/whatsapp/commands/whoami.test.ts +3 -3
  221. package/src/platforms/whatsapp/credential-manager.test.ts +23 -23
  222. package/src/platforms/whatsapp/ensure-auth.test.ts +4 -4
  223. package/src/platforms/whatsapp/index.test.ts +8 -8
  224. package/src/platforms/whatsapp/types.test.ts +42 -42
  225. package/src/platforms/whatsappbot/client.test.ts +27 -27
  226. package/src/platforms/whatsappbot/commands/auth.test.ts +14 -14
  227. package/src/platforms/whatsappbot/commands/message.test.ts +16 -16
  228. package/src/platforms/whatsappbot/commands/template.test.ts +9 -9
  229. package/src/platforms/whatsappbot/commands/whoami.test.ts +5 -5
  230. package/src/platforms/whatsappbot/credential-manager.test.ts +18 -18
  231. package/src/platforms/whatsappbot/index.test.ts +7 -7
  232. package/src/platforms/whatsappbot/types.test.ts +18 -18
  233. package/src/shared/chromium/browsers.test.ts +22 -22
  234. package/src/shared/chromium/cookie-reader.test.ts +13 -13
  235. package/src/shared/chromium/decryptor.test.ts +97 -32
  236. package/src/shared/chromium/decryptor.ts +27 -6
  237. package/src/shared/utils/concurrency.test.ts +6 -6
  238. package/src/shared/utils/derived-key-cache.test.ts +11 -11
  239. package/src/tui/utils.test.ts +31 -31
@@ -1,4 +1,4 @@
1
- import { afterEach, describe, expect, spyOn, test } from 'bun:test'
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
- test('returns null for unsupported platform', () => {
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
- test('returns null for unsupported platform', () => {
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
- test('returns win32 path under AppData/Roaming/Channel Talk/Network', () => {
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
- test('returns browser cookie paths on macOS including Default profile', () => {
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
- test('returns browser cookie paths on Linux', () => {
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
- test('returns browser cookie paths on Windows', () => {
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
- test('returns empty array for unsupported platform', () => {
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
- test('returns empty array when desktop cookies path does not exist', async () => {
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
- test('tries desktop app before browser profiles and collects both', async () => {
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
- test('includes browser profiles even when desktop extraction returns null', async () => {
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
- test('extracts plaintext cookies from a real sqlite database', async () => {
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
- test('returns token with undefined sessionCookie when only x-account is present', async () => {
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
- test('returns empty array when x-account is missing', async () => {
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
- test('decrypts AES-256-GCM encrypted cookies using master key', async () => {
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
- test('deduplicates entries with the same accountCookie from desktop and browser', async () => {
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
- test('returns multiple distinct accounts from desktop and browser sources', async () => {
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
- test('returns empty array when DPAPI decryption fails', async () => {
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
- test('returns null on non-win32 platform', () => {
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
- test('decrypts v10-prefixed browser cookie using macOS keychain password (AES-128-CBC)', () => {
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
- test('decrypts v10-prefixed browser cookie using Linux peanuts key (AES-128-CBC)', () => {
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, test } from 'bun:test'
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
- test('accepts a valid workspace entry', () => {
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
- test('accepts a workspace entry without session_cookie', () => {
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
- test('rejects a workspace entry missing account_cookie', () => {
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
- test('accepts a valid config', () => {
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
- test('accepts a valid account', () => {
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
- test('rejects an invalid account', () => {
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
- test('accepts a valid message with blocks', () => {
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
- test('rejects an invalid message block payload', () => {
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
- test('accepts a valid group', () => {
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
- test('rejects an invalid group', () => {
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
- test('stores the code on ChannelError', () => {
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, test } from 'bun:test'
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
- test('successful GET request returns unwrapped JSON', async () => {
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
- test('auth headers are set on every request', async () => {
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
- test('429 response triggers retry with Retry-After wait', async () => {
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
- test('500 response triggers retry with exponential backoff', async () => {
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
- test('4xx non-429 throws immediately without retry', async () => {
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
- test('network error retries then throws ChannelBotError with code network_error', async () => {
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
- test('204 response returns undefined', async () => {
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
- test('wrapTextInBlocks returns a single text block with value', () => {
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
- test('sendUserChatMessage includes botName in query string', async () => {
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
- test('sendGroupMessage includes botName in query string', async () => {
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
- test('resolveGroup("@team-name") calls groups by name endpoint', async () => {
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
- test('resolveGroup("grp123") calls groups by id endpoint', async () => {
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
- test('pagination params are included in query string', async () => {
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, test } from 'bun:test'
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
- test('validates and stores credentials with workspace info from API', async () => {
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
- test('uses --workspace option as workspace name', async () => {
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
- test('handles client errors gracefully', async () => {
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
- test('returns no credentials when none set', async () => {
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
- test('returns valid status for current workspace', async () => {
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
- test('returns invalid when API call fails', async () => {
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
- test('removes all stored credentials', async () => {
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
- test('returns all stored workspaces', async () => {
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
- test('returns empty list when no workspaces stored', async () => {
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
- test('switches current workspace', async () => {
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
- test('returns error for unknown workspace', async () => {
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
- test('removes a stored workspace', async () => {
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
- test('returns error for unknown workspace', async () => {
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
- test('sets default bot name', async () => {
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
- test('updates existing default bot', async () => {
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, test } from 'bun:test'
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
- test('returns all bots', async () => {
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
- test('creates bot with name', async () => {
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
- test('creates bot with optional color and avatar', async () => {
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
- test('deletes bot with --force flag', async () => {
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
- test('returns error without --force flag', async () => {
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, test } from 'bun:test'
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
- test('lists opened chats by default', async () => {
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
- test('passes state filter to API', async () => {
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
- test('passes pagination params', async () => {
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
- test('returns specific chat', async () => {
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
- test('closes chat with bot name', async () => {
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
- test('returns error when no bot name provided', async () => {
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
- test('deletes chat with --force flag', async () => {
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
- test('returns error without --force flag', async () => {
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