bingocode 1.0.27 → 1.0.29

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 (54) hide show
  1. package/package.json +1 -2
  2. package/.github/FUNDING.yml +0 -1
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -44
  4. package/.github/ISSUE_TEMPLATE/config.yml +0 -1
  5. package/.github/ISSUE_TEMPLATE/question.md +0 -40
  6. package/.github/workflows/build-desktop-dev.yml +0 -210
  7. package/.github/workflows/deploy-docs.yml +0 -59
  8. package/.github/workflows/release-desktop.yml +0 -162
  9. package/.spine/user.yaml +0 -5
  10. package/.spine/workspace.yaml +0 -1
  11. package/adapters/common/__tests__/chat-queue.test.ts +0 -61
  12. package/adapters/common/__tests__/format.test.ts +0 -148
  13. package/adapters/common/__tests__/http-client.test.ts +0 -105
  14. package/adapters/common/__tests__/message-buffer.test.ts +0 -84
  15. package/adapters/common/__tests__/message-dedup.test.ts +0 -57
  16. package/adapters/common/__tests__/session-store.test.ts +0 -62
  17. package/adapters/common/__tests__/ws-bridge.test.ts +0 -177
  18. package/adapters/common/attachment/__tests__/attachment-limits.test.ts +0 -52
  19. package/adapters/common/attachment/__tests__/attachment-store.test.ts +0 -108
  20. package/adapters/common/attachment/__tests__/image-block-watcher.test.ts +0 -115
  21. package/adapters/feishu/__tests__/card-errors.test.ts +0 -194
  22. package/adapters/feishu/__tests__/cardkit.test.ts +0 -295
  23. package/adapters/feishu/__tests__/extract-payload.test.ts +0 -77
  24. package/adapters/feishu/__tests__/feishu.test.ts +0 -907
  25. package/adapters/feishu/__tests__/flush-controller.test.ts +0 -290
  26. package/adapters/feishu/__tests__/markdown-style.test.ts +0 -353
  27. package/adapters/feishu/__tests__/media.test.ts +0 -120
  28. package/adapters/feishu/__tests__/streaming-card.test.ts +0 -914
  29. package/adapters/telegram/__tests__/media.test.ts +0 -86
  30. package/adapters/telegram/__tests__/telegram.test.ts +0 -115
  31. package/src/server/__tests__/conversation-service.test.ts +0 -173
  32. package/src/server/__tests__/conversations.test.ts +0 -458
  33. package/src/server/__tests__/cron-scheduler.test.ts +0 -575
  34. package/src/server/__tests__/e2e/business-flow.test.ts +0 -841
  35. package/src/server/__tests__/e2e/full-flow.test.ts +0 -357
  36. package/src/server/__tests__/fixtures/mock-sdk-cli.ts +0 -123
  37. package/src/server/__tests__/haha-oauth-api.test.ts +0 -146
  38. package/src/server/__tests__/haha-oauth-service.test.ts +0 -185
  39. package/src/server/__tests__/providers-real.test.ts +0 -244
  40. package/src/server/__tests__/providers.test.ts +0 -579
  41. package/src/server/__tests__/proxy-streaming.test.ts +0 -317
  42. package/src/server/__tests__/proxy-transform.test.ts +0 -469
  43. package/src/server/__tests__/real-llm-test.ts +0 -526
  44. package/src/server/__tests__/scheduled-tasks.test.ts +0 -371
  45. package/src/server/__tests__/sessions.test.ts +0 -786
  46. package/src/server/__tests__/settings.test.ts +0 -376
  47. package/src/server/__tests__/skills.test.ts +0 -125
  48. package/src/server/__tests__/tasks.test.ts +0 -171
  49. package/src/server/__tests__/team-watcher.test.ts +0 -400
  50. package/src/server/__tests__/teams.test.ts +0 -627
  51. package/src/server/middleware/cors.test.ts +0 -27
  52. package/src/utils/__tests__/cronFrequency.test.ts +0 -153
  53. package/src/utils/__tests__/cronTasks.test.ts +0 -204
  54. package/src/utils/computerUse/permissions.test.ts +0 -44
@@ -1,86 +0,0 @@
1
- import { describe, it, expect, beforeEach, afterEach, mock } from 'bun:test'
2
- import * as fs from 'node:fs/promises'
3
- import * as path from 'node:path'
4
- import * as os from 'node:os'
5
- import { TelegramMediaService } from '../media.js'
6
- import { AttachmentStore } from '../../common/attachment/attachment-store.js'
7
-
8
- let tmpRoot: string
9
- let originalFetch: typeof fetch
10
-
11
- function makeMockBot() {
12
- const fetchMock = mock(async (url: string | URL) => {
13
- const u = typeof url === 'string' ? url : url.toString()
14
- expect(u).toContain('/file/botFAKE_TOKEN/photos/abc.jpg')
15
- return new Response(Buffer.from('PHOTODATA'), {
16
- status: 200,
17
- headers: { 'content-type': 'image/jpeg' },
18
- })
19
- })
20
- ;(globalThis as any).fetch = fetchMock
21
- return {
22
- token: 'FAKE_TOKEN',
23
- api: {
24
- getFile: mock(async (fileId: string) => ({
25
- file_id: fileId,
26
- file_unique_id: 'unique',
27
- file_path: 'photos/abc.jpg',
28
- })),
29
- sendPhoto: mock(async () => ({ message_id: 1 })),
30
- sendDocument: mock(async () => ({ message_id: 2 })),
31
- },
32
- fetchMock,
33
- }
34
- }
35
-
36
- beforeEach(async () => {
37
- originalFetch = globalThis.fetch
38
- tmpRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'tg-media-test-'))
39
- })
40
-
41
- afterEach(async () => {
42
- globalThis.fetch = originalFetch
43
- await fs.rm(tmpRoot, { recursive: true, force: true })
44
- })
45
-
46
- describe('TelegramMediaService', () => {
47
- it('downloadFile fetches the real URL and stores a LocalAttachment', async () => {
48
- const bot = makeMockBot()
49
- const store = new AttachmentStore({ root: tmpRoot, retentionMs: 60_000 })
50
- const svc = new TelegramMediaService(bot as any, store)
51
- const local = await svc.downloadFile('fid_123', 'sess-1', {
52
- fileName: 'abc.jpg',
53
- mimeType: 'image/jpeg',
54
- })
55
- expect(local.kind).toBe('image')
56
- expect(local.name).toBe('abc.jpg')
57
- expect(local.size).toBe('PHOTODATA'.length)
58
- expect(local.buffer.toString()).toBe('PHOTODATA')
59
- const onDisk = await fs.readFile(local.path)
60
- expect(onDisk.toString()).toBe('PHOTODATA')
61
- })
62
-
63
- it('sendPhoto calls bot.api.sendPhoto with InputFile-like payload', async () => {
64
- const bot = makeMockBot()
65
- const store = new AttachmentStore({ root: tmpRoot, retentionMs: 60_000 })
66
- const svc = new TelegramMediaService(bot as any, store)
67
- await svc.sendPhoto(42, Buffer.from('IMG'), 'caption text')
68
- expect(bot.api.sendPhoto).toHaveBeenCalledTimes(1)
69
- const args = (bot.api.sendPhoto as any).mock.calls[0]
70
- expect(args[0]).toBe(42)
71
- // grammY InputFile wraps the buffer; just verify it's an object.
72
- expect(args[1]).toBeDefined()
73
- expect(args[2]?.caption).toBe('caption text')
74
- })
75
-
76
- it('sendDocument calls bot.api.sendDocument', async () => {
77
- const bot = makeMockBot()
78
- const store = new AttachmentStore({ root: tmpRoot, retentionMs: 60_000 })
79
- const svc = new TelegramMediaService(bot as any, store)
80
- await svc.sendDocument(42, Buffer.from('DOC'), 'spec.pdf')
81
- expect(bot.api.sendDocument).toHaveBeenCalledTimes(1)
82
- const args = (bot.api.sendDocument as any).mock.calls[0]
83
- expect(args[0]).toBe(42)
84
- expect(args[1]).toBeDefined()
85
- })
86
- })
@@ -1,115 +0,0 @@
1
- import { describe, it, expect } from 'bun:test'
2
- import { splitMessage, formatPermissionRequest, truncateInput, escapeMarkdownV2 } from '../../common/format.js'
3
-
4
- /**
5
- * Telegram Adapter 翻译逻辑测试
6
- *
7
- * 由于 grammy Bot 需要实际 Token 才能初始化,
8
- * 这里测试的是不依赖 Bot 实例的核心翻译逻辑。
9
- */
10
-
11
- describe('Telegram message formatting', () => {
12
- describe('long message splitting', () => {
13
- it('splits messages at Telegram 4096 char limit', () => {
14
- const longText = 'a'.repeat(8000)
15
- const chunks = splitMessage(longText, 4000)
16
- expect(chunks.length).toBe(2)
17
- expect(chunks[0]!.length).toBeLessThanOrEqual(4000)
18
- expect(chunks[1]!.length).toBeLessThanOrEqual(4000)
19
- })
20
-
21
- it('keeps short messages as single chunk', () => {
22
- const chunks = splitMessage('Hello World', 4000)
23
- expect(chunks).toEqual(['Hello World'])
24
- })
25
-
26
- it('splits at paragraph boundary when possible', () => {
27
- const text = 'A'.repeat(2000) + '\n\n' + 'B'.repeat(2000)
28
- const chunks = splitMessage(text, 3000)
29
- expect(chunks.length).toBe(2)
30
- })
31
- })
32
-
33
- describe('permission request formatting', () => {
34
- it('formats Bash command request', () => {
35
- const result = formatPermissionRequest('Bash', { command: 'npm test' }, 'abcde')
36
- expect(result).toContain('🔐')
37
- expect(result).toContain('Bash')
38
- expect(result).toContain('npm test')
39
- expect(result).toContain('abcde')
40
- })
41
-
42
- it('formats Write file request', () => {
43
- const result = formatPermissionRequest(
44
- 'Write',
45
- { file_path: '/src/index.ts', content: 'console.log("hello")' },
46
- 'fghij',
47
- )
48
- expect(result).toContain('Write')
49
- expect(result).toContain('index.ts')
50
- expect(result).toContain('fghij')
51
- })
52
-
53
- it('truncates long input in permission request', () => {
54
- const longInput = { command: 'x'.repeat(500) }
55
- const result = formatPermissionRequest('Bash', longInput, 'xxxxx')
56
- expect(result.length).toBeLessThan(600)
57
- })
58
- })
59
-
60
- describe('callback_data parsing', () => {
61
- it('parses permit:requestId:yes format', () => {
62
- const data = 'permit:abcde:yes'
63
- const parts = data.split(':')
64
- expect(parts[0]).toBe('permit')
65
- expect(parts[1]).toBe('abcde')
66
- expect(parts[2]).toBe('yes')
67
- })
68
-
69
- it('parses permit:requestId:no format', () => {
70
- const data = 'permit:abcde:no'
71
- const parts = data.split(':')
72
- expect(parts[2]).toBe('no')
73
- })
74
-
75
- it('ignores non-permit callbacks', () => {
76
- const data = 'other:action'
77
- expect(data.startsWith('permit:')).toBe(false)
78
- })
79
- })
80
-
81
- describe('MarkdownV2 escaping', () => {
82
- it('escapes underscores', () => {
83
- expect(escapeMarkdownV2('hello_world')).toBe('hello\\_world')
84
- })
85
-
86
- it('escapes multiple special chars', () => {
87
- const result = escapeMarkdownV2('file.ts (line 42)')
88
- expect(result).toBe('file\\.ts \\(line 42\\)')
89
- })
90
-
91
- it('handles code blocks safely', () => {
92
- const result = escapeMarkdownV2('`code`')
93
- expect(result).toBe('\\`code\\`')
94
- })
95
- })
96
-
97
- describe('whitelist logic', () => {
98
- it('empty allowedUsers means allow all', () => {
99
- const allowedUsers: number[] = []
100
- const isAllowed = (userId: number) =>
101
- allowedUsers.length === 0 || allowedUsers.includes(userId)
102
- expect(isAllowed(12345)).toBe(true)
103
- expect(isAllowed(99999)).toBe(true)
104
- })
105
-
106
- it('non-empty allowedUsers filters correctly', () => {
107
- const allowedUsers = [111, 222]
108
- const isAllowed = (userId: number) =>
109
- allowedUsers.length === 0 || allowedUsers.includes(userId)
110
- expect(isAllowed(111)).toBe(true)
111
- expect(isAllowed(222)).toBe(true)
112
- expect(isAllowed(333)).toBe(false)
113
- })
114
- })
115
- })
@@ -1,173 +0,0 @@
1
- import { afterEach, beforeEach, describe, expect, test } from 'bun:test'
2
- import * as fs from 'node:fs/promises'
3
- import * as os from 'node:os'
4
- import * as path from 'node:path'
5
- import { ConversationService } from '../services/conversationService.js'
6
-
7
- describe('ConversationService', () => {
8
- let tmpDir: string
9
- let originalConfigDir: string | undefined
10
- let originalAuthToken: string | undefined
11
- let originalBaseUrl: string | undefined
12
- let originalModel: string | undefined
13
- let originalEntrypoint: string | undefined
14
- let originalOAuthToken: string | undefined
15
-
16
- beforeEach(async () => {
17
- tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'bingo-conversation-service-'))
18
- originalConfigDir = process.env.CLAUDE_CONFIG_DIR
19
- originalAuthToken = process.env.ANTHROPIC_AUTH_TOKEN
20
- originalBaseUrl = process.env.ANTHROPIC_BASE_URL
21
- originalModel = process.env.ANTHROPIC_MODEL
22
- originalEntrypoint = process.env.CLAUDE_CODE_ENTRYPOINT
23
- originalOAuthToken = process.env.CLAUDE_CODE_OAUTH_TOKEN
24
-
25
- process.env.CLAUDE_CONFIG_DIR = tmpDir
26
- process.env.ANTHROPIC_AUTH_TOKEN = 'test-token'
27
- process.env.ANTHROPIC_BASE_URL = 'https://example.invalid/anthropic'
28
- process.env.ANTHROPIC_MODEL = 'test-model'
29
- process.env.CLAUDE_CODE_OAUTH_TOKEN = 'inherited-parent-oauth-token'
30
- // Clear inherited CLAUDE_CODE_ENTRYPOINT so tests can assert whether
31
- // buildChildEnv injects it or not without interference from the shell env.
32
- delete process.env.CLAUDE_CODE_ENTRYPOINT
33
- })
34
-
35
- afterEach(async () => {
36
- if (originalConfigDir === undefined) delete process.env.CLAUDE_CONFIG_DIR
37
- else process.env.CLAUDE_CONFIG_DIR = originalConfigDir
38
-
39
- if (originalAuthToken === undefined) delete process.env.ANTHROPIC_AUTH_TOKEN
40
- else process.env.ANTHROPIC_AUTH_TOKEN = originalAuthToken
41
-
42
- if (originalBaseUrl === undefined) delete process.env.ANTHROPIC_BASE_URL
43
- else process.env.ANTHROPIC_BASE_URL = originalBaseUrl
44
-
45
- if (originalModel === undefined) delete process.env.ANTHROPIC_MODEL
46
- else process.env.ANTHROPIC_MODEL = originalModel
47
-
48
- if (originalEntrypoint === undefined) delete process.env.CLAUDE_CODE_ENTRYPOINT
49
- else process.env.CLAUDE_CODE_ENTRYPOINT = originalEntrypoint
50
-
51
- if (originalOAuthToken === undefined) delete process.env.CLAUDE_CODE_OAUTH_TOKEN
52
- else process.env.CLAUDE_CODE_OAUTH_TOKEN = originalOAuthToken
53
-
54
- await fs.rm(tmpDir, { recursive: true, force: true })
55
- })
56
-
57
- test('keeps inherited provider env when no desktop provider config exists', async () => {
58
- const service = new ConversationService() as any
59
- const env = (await service.buildChildEnv('D:\\workspace\\code\\myself_code\\bingo')) as Record<string, string>
60
-
61
- expect(env.ANTHROPIC_AUTH_TOKEN).toBe('test-token')
62
- expect(env.ANTHROPIC_BASE_URL).toBe('https://example.invalid/anthropic')
63
- expect(env.ANTHROPIC_MODEL).toBe('test-model')
64
- })
65
-
66
- test('strips inherited provider env when desktop provider config exists', async () => {
67
- const ccHahaDir = path.join(tmpDir, 'bingo')
68
- await fs.mkdir(ccHahaDir, { recursive: true })
69
- await fs.writeFile(
70
- path.join(ccHahaDir, 'providers.json'),
71
- JSON.stringify({ activeId: null, providers: [] }),
72
- 'utf-8',
73
- )
74
-
75
- const service = new ConversationService() as any
76
- const env = (await service.buildChildEnv('D:\\workspace\\code\\myself_code\\bingo')) as Record<string, string>
77
-
78
- expect(env.ANTHROPIC_AUTH_TOKEN).toBeUndefined()
79
- expect(env.ANTHROPIC_BASE_URL).toBeUndefined()
80
- expect(env.ANTHROPIC_MODEL).toBeUndefined()
81
- })
82
-
83
- test('buildChildEnv injects CLAUDE_CODE_OAUTH_TOKEN when official mode + haha oauth token exists', async () => {
84
- const ccHahaDir = path.join(tmpDir, 'bingo')
85
- await fs.mkdir(ccHahaDir, { recursive: true })
86
- await fs.writeFile(
87
- path.join(ccHahaDir, 'settings.json'),
88
- JSON.stringify({ env: {} }),
89
- 'utf-8',
90
- )
91
-
92
- const { hahaOAuthService } = await import('../services/hahaOAuthService.js')
93
- await hahaOAuthService.saveTokens({
94
- accessToken: 'haha-fresh-token',
95
- refreshToken: 'haha-refresh-xxx',
96
- expiresAt: Date.now() + 30 * 60_000,
97
- scopes: ['user:inference'],
98
- subscriptionType: 'max',
99
- })
100
-
101
- const service = new ConversationService() as any
102
- const env = (await service.buildChildEnv('/tmp')) as Record<string, string>
103
-
104
- expect(env.CLAUDE_CODE_ENTRYPOINT).toBe('claude-desktop')
105
- expect(env.CLAUDE_CODE_OAUTH_TOKEN).toBe('haha-fresh-token')
106
- })
107
-
108
- test('buildChildEnv does NOT inject CLAUDE_CODE_OAUTH_TOKEN when not official mode', async () => {
109
- const ccHahaDir = path.join(tmpDir, 'bingo')
110
- await fs.mkdir(ccHahaDir, { recursive: true })
111
- await fs.writeFile(
112
- path.join(ccHahaDir, 'settings.json'),
113
- JSON.stringify({ env: { ANTHROPIC_AUTH_TOKEN: 'custom-provider-token' } }),
114
- 'utf-8',
115
- )
116
-
117
- const { hahaOAuthService } = await import('../services/hahaOAuthService.js')
118
- await hahaOAuthService.saveTokens({
119
- accessToken: 'haha-token-should-not-be-used',
120
- refreshToken: null,
121
- expiresAt: null,
122
- scopes: [],
123
- subscriptionType: null,
124
- })
125
-
126
- const service = new ConversationService() as any
127
- const env = (await service.buildChildEnv('/tmp')) as Record<string, string>
128
-
129
- expect(env.CLAUDE_CODE_OAUTH_TOKEN).toBeUndefined()
130
- expect(env.CLAUDE_CODE_ENTRYPOINT).toBeUndefined()
131
- })
132
-
133
- test('buildChildEnv does not leak inherited CLAUDE_CODE_OAUTH_TOKEN when official token is unavailable', async () => {
134
- const ccHahaDir = path.join(tmpDir, 'bingo')
135
- await fs.mkdir(ccHahaDir, { recursive: true })
136
- await fs.writeFile(
137
- path.join(ccHahaDir, 'settings.json'),
138
- JSON.stringify({ env: {} }),
139
- 'utf-8',
140
- )
141
-
142
- const service = new ConversationService() as any
143
- const env = (await service.buildChildEnv('/tmp')) as Record<string, string>
144
-
145
- expect(env.CLAUDE_CODE_ENTRYPOINT).toBe('claude-desktop')
146
- expect(env.CLAUDE_CODE_OAUTH_TOKEN).toBeUndefined()
147
- })
148
-
149
- test('buildChildEnv injects desktop Computer Use host bundle id for sdk sessions', async () => {
150
- const service = new ConversationService() as any
151
- const env = (await service.buildChildEnv(
152
- '/tmp',
153
- 'ws://127.0.0.1:3456/sdk/test-session?token=test-token',
154
- )) as Record<string, string>
155
-
156
- expect(env.CC_HAHA_COMPUTER_USE_HOST_BUNDLE_ID).toBe(
157
- 'com.claude-code-haha.desktop',
158
- )
159
- expect(env.CC_HAHA_DESKTOP_SERVER_URL).toBe('http://127.0.0.1:3456')
160
- })
161
-
162
- test('uses bun entrypoint fallback on Windows dev mode', () => {
163
- const service = new ConversationService() as any
164
- const args = service.resolveCliArgs(['--print'])
165
-
166
- if (process.platform === 'win32') {
167
- expect(args[0]).toBe(process.execPath)
168
- expect(args[1]).toContain(path.join('src', 'entrypoints', 'cli.tsx'))
169
- } else {
170
- expect(args[0]).toContain(path.join('bin', 'claude-haha'))
171
- }
172
- })
173
- })