@doist/twist-cli 2.20.0 → 2.21.1

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 (198) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/package.json +3 -5
  3. package/dist/__mocks__/chalk.d.ts +0 -3
  4. package/dist/__mocks__/chalk.d.ts.map +0 -1
  5. package/dist/__mocks__/chalk.js +0 -8
  6. package/dist/__mocks__/chalk.js.map +0 -1
  7. package/dist/commands/auth.d.ts +0 -3
  8. package/dist/commands/auth.d.ts.map +0 -1
  9. package/dist/commands/auth.js +0 -155
  10. package/dist/commands/auth.js.map +0 -1
  11. package/dist/commands/away.d.ts +0 -3
  12. package/dist/commands/away.d.ts.map +0 -1
  13. package/dist/commands/away.js +0 -120
  14. package/dist/commands/away.js.map +0 -1
  15. package/dist/commands/changelog.d.ts +0 -8
  16. package/dist/commands/changelog.d.ts.map +0 -1
  17. package/dist/commands/changelog.js +0 -113
  18. package/dist/commands/changelog.js.map +0 -1
  19. package/dist/commands/channel.d.ts +0 -3
  20. package/dist/commands/channel.d.ts.map +0 -1
  21. package/dist/commands/channel.js +0 -53
  22. package/dist/commands/channel.js.map +0 -1
  23. package/dist/commands/comment.d.ts +0 -3
  24. package/dist/commands/comment.d.ts.map +0 -1
  25. package/dist/commands/comment.js +0 -99
  26. package/dist/commands/comment.js.map +0 -1
  27. package/dist/commands/completion.d.ts +0 -3
  28. package/dist/commands/completion.d.ts.map +0 -1
  29. package/dist/commands/completion.js +0 -121
  30. package/dist/commands/completion.js.map +0 -1
  31. package/dist/commands/conversation.d.ts +0 -3
  32. package/dist/commands/conversation.d.ts.map +0 -1
  33. package/dist/commands/conversation.js +0 -394
  34. package/dist/commands/conversation.js.map +0 -1
  35. package/dist/commands/inbox.d.ts +0 -3
  36. package/dist/commands/inbox.d.ts.map +0 -1
  37. package/dist/commands/inbox.js +0 -127
  38. package/dist/commands/inbox.js.map +0 -1
  39. package/dist/commands/msg.d.ts +0 -3
  40. package/dist/commands/msg.d.ts.map +0 -1
  41. package/dist/commands/msg.js +0 -103
  42. package/dist/commands/msg.js.map +0 -1
  43. package/dist/commands/react.d.ts +0 -3
  44. package/dist/commands/react.d.ts.map +0 -1
  45. package/dist/commands/react.js +0 -102
  46. package/dist/commands/react.js.map +0 -1
  47. package/dist/commands/search.d.ts +0 -3
  48. package/dist/commands/search.d.ts.map +0 -1
  49. package/dist/commands/search.js +0 -160
  50. package/dist/commands/search.js.map +0 -1
  51. package/dist/commands/skill.d.ts +0 -3
  52. package/dist/commands/skill.d.ts.map +0 -1
  53. package/dist/commands/skill.js +0 -119
  54. package/dist/commands/skill.js.map +0 -1
  55. package/dist/commands/thread/create.d.ts +0 -7
  56. package/dist/commands/thread/create.d.ts.map +0 -1
  57. package/dist/commands/thread/create.js +0 -45
  58. package/dist/commands/thread/create.js.map +0 -1
  59. package/dist/commands/thread/helpers.d.ts +0 -10
  60. package/dist/commands/thread/helpers.d.ts.map +0 -1
  61. package/dist/commands/thread/helpers.js +0 -27
  62. package/dist/commands/thread/helpers.js.map +0 -1
  63. package/dist/commands/thread/mutate.d.ts +0 -5
  64. package/dist/commands/thread/mutate.d.ts.map +0 -1
  65. package/dist/commands/thread/mutate.js +0 -21
  66. package/dist/commands/thread/mutate.js.map +0 -1
  67. package/dist/commands/thread/reply.d.ts +0 -9
  68. package/dist/commands/thread/reply.d.ts.map +0 -1
  69. package/dist/commands/thread/reply.js +0 -68
  70. package/dist/commands/thread/reply.js.map +0 -1
  71. package/dist/commands/thread/view.d.ts +0 -9
  72. package/dist/commands/thread/view.d.ts.map +0 -1
  73. package/dist/commands/thread/view.js +0 -175
  74. package/dist/commands/thread/view.js.map +0 -1
  75. package/dist/commands/thread.d.ts +0 -3
  76. package/dist/commands/thread.d.ts.map +0 -1
  77. package/dist/commands/thread.js +0 -54
  78. package/dist/commands/thread.js.map +0 -1
  79. package/dist/commands/update.d.ts +0 -14
  80. package/dist/commands/update.d.ts.map +0 -1
  81. package/dist/commands/update.js +0 -91
  82. package/dist/commands/update.js.map +0 -1
  83. package/dist/commands/user.d.ts +0 -3
  84. package/dist/commands/user.d.ts.map +0 -1
  85. package/dist/commands/user.js +0 -67
  86. package/dist/commands/user.js.map +0 -1
  87. package/dist/commands/view.d.ts +0 -3
  88. package/dist/commands/view.d.ts.map +0 -1
  89. package/dist/commands/view.js +0 -63
  90. package/dist/commands/view.js.map +0 -1
  91. package/dist/commands/workspace.d.ts +0 -3
  92. package/dist/commands/workspace.d.ts.map +0 -1
  93. package/dist/commands/workspace.js +0 -48
  94. package/dist/commands/workspace.js.map +0 -1
  95. package/dist/index.d.ts +0 -3
  96. package/dist/index.d.ts.map +0 -1
  97. package/dist/index.js +0 -133
  98. package/dist/index.js.map +0 -1
  99. package/dist/lib/api.d.ts +0 -20
  100. package/dist/lib/api.d.ts.map +0 -1
  101. package/dist/lib/api.js +0 -220
  102. package/dist/lib/api.js.map +0 -1
  103. package/dist/lib/auth.d.ts +0 -21
  104. package/dist/lib/auth.d.ts.map +0 -1
  105. package/dist/lib/auth.js +0 -189
  106. package/dist/lib/auth.js.map +0 -1
  107. package/dist/lib/completion.d.ts +0 -31
  108. package/dist/lib/completion.d.ts.map +0 -1
  109. package/dist/lib/completion.js +0 -173
  110. package/dist/lib/completion.js.map +0 -1
  111. package/dist/lib/config.d.ts +0 -13
  112. package/dist/lib/config.d.ts.map +0 -1
  113. package/dist/lib/config.js +0 -26
  114. package/dist/lib/config.js.map +0 -1
  115. package/dist/lib/dates.d.ts +0 -3
  116. package/dist/lib/dates.d.ts.map +0 -1
  117. package/dist/lib/dates.js +0 -44
  118. package/dist/lib/dates.js.map +0 -1
  119. package/dist/lib/input.d.ts +0 -3
  120. package/dist/lib/input.d.ts.map +0 -1
  121. package/dist/lib/input.js +0 -52
  122. package/dist/lib/input.js.map +0 -1
  123. package/dist/lib/markdown.d.ts +0 -2
  124. package/dist/lib/markdown.d.ts.map +0 -1
  125. package/dist/lib/markdown.js +0 -16
  126. package/dist/lib/markdown.js.map +0 -1
  127. package/dist/lib/oauth-server.d.ts +0 -13
  128. package/dist/lib/oauth-server.d.ts.map +0 -1
  129. package/dist/lib/oauth-server.js +0 -824
  130. package/dist/lib/oauth-server.js.map +0 -1
  131. package/dist/lib/oauth.d.ts +0 -31
  132. package/dist/lib/oauth.d.ts.map +0 -1
  133. package/dist/lib/oauth.js +0 -140
  134. package/dist/lib/oauth.js.map +0 -1
  135. package/dist/lib/options.d.ts +0 -17
  136. package/dist/lib/options.d.ts.map +0 -1
  137. package/dist/lib/options.js +0 -2
  138. package/dist/lib/options.js.map +0 -1
  139. package/dist/lib/output.d.ts +0 -25
  140. package/dist/lib/output.d.ts.map +0 -1
  141. package/dist/lib/output.js +0 -128
  142. package/dist/lib/output.js.map +0 -1
  143. package/dist/lib/permissions.d.ts +0 -4
  144. package/dist/lib/permissions.d.ts.map +0 -1
  145. package/dist/lib/permissions.js +0 -35
  146. package/dist/lib/permissions.js.map +0 -1
  147. package/dist/lib/pkce.d.ts +0 -16
  148. package/dist/lib/pkce.d.ts.map +0 -1
  149. package/dist/lib/pkce.js +0 -35
  150. package/dist/lib/pkce.js.map +0 -1
  151. package/dist/lib/progress.d.ts +0 -29
  152. package/dist/lib/progress.d.ts.map +0 -1
  153. package/dist/lib/progress.js +0 -101
  154. package/dist/lib/progress.js.map +0 -1
  155. package/dist/lib/public-channels.d.ts +0 -5
  156. package/dist/lib/public-channels.d.ts.map +0 -1
  157. package/dist/lib/public-channels.js +0 -35
  158. package/dist/lib/public-channels.js.map +0 -1
  159. package/dist/lib/refs.d.ts +0 -37
  160. package/dist/lib/refs.d.ts.map +0 -1
  161. package/dist/lib/refs.js +0 -219
  162. package/dist/lib/refs.js.map +0 -1
  163. package/dist/lib/search-api.d.ts +0 -25
  164. package/dist/lib/search-api.d.ts.map +0 -1
  165. package/dist/lib/search-api.js +0 -85
  166. package/dist/lib/search-api.js.map +0 -1
  167. package/dist/lib/secure-store.d.ts +0 -11
  168. package/dist/lib/secure-store.d.ts.map +0 -1
  169. package/dist/lib/secure-store.js +0 -57
  170. package/dist/lib/secure-store.js.map +0 -1
  171. package/dist/lib/skills/content.d.ts +0 -5
  172. package/dist/lib/skills/content.d.ts.map +0 -1
  173. package/dist/lib/skills/content.js +0 -274
  174. package/dist/lib/skills/content.js.map +0 -1
  175. package/dist/lib/skills/create-installer.d.ts +0 -9
  176. package/dist/lib/skills/create-installer.d.ts.map +0 -1
  177. package/dist/lib/skills/create-installer.js +0 -60
  178. package/dist/lib/skills/create-installer.js.map +0 -1
  179. package/dist/lib/skills/index.d.ts +0 -7
  180. package/dist/lib/skills/index.d.ts.map +0 -1
  181. package/dist/lib/skills/index.js +0 -54
  182. package/dist/lib/skills/index.js.map +0 -1
  183. package/dist/lib/skills/types.d.ts +0 -30
  184. package/dist/lib/skills/types.d.ts.map +0 -1
  185. package/dist/lib/skills/types.js +0 -2
  186. package/dist/lib/skills/types.js.map +0 -1
  187. package/dist/lib/skills/update-installed.d.ts +0 -9
  188. package/dist/lib/skills/update-installed.d.ts.map +0 -1
  189. package/dist/lib/skills/update-installed.js +0 -20
  190. package/dist/lib/skills/update-installed.js.map +0 -1
  191. package/dist/lib/spinner.d.ts +0 -24
  192. package/dist/lib/spinner.d.ts.map +0 -1
  193. package/dist/lib/spinner.js +0 -126
  194. package/dist/lib/spinner.js.map +0 -1
  195. package/dist/postinstall.d.ts +0 -2
  196. package/dist/postinstall.d.ts.map +0 -1
  197. package/dist/postinstall.js +0 -3
  198. package/dist/postinstall.js.map +0 -1
@@ -1,99 +0,0 @@
1
- import { getTwistClient } from '../lib/api.js';
2
- import { formatRelativeDate } from '../lib/dates.js';
3
- import { openEditor, readStdin } from '../lib/input.js';
4
- import { renderMarkdown } from '../lib/markdown.js';
5
- import { colors, formatJson } from '../lib/output.js';
6
- import { resolveCommentId } from '../lib/refs.js';
7
- async function viewComment(ref, options) {
8
- const commentId = resolveCommentId(ref);
9
- const client = await getTwistClient();
10
- const comment = await client.comments.getComment(commentId);
11
- const userResponse = await client.workspaceUsers.getUserById({ workspaceId: comment.workspaceId, userId: comment.creator }, { batch: false });
12
- const creatorName = userResponse.name;
13
- if (options.json) {
14
- const output = { ...comment, creatorName };
15
- console.log(formatJson(output, options.full ? undefined : 'comment', options.full));
16
- return;
17
- }
18
- const author = colors.author(creatorName);
19
- const time = colors.timestamp(formatRelativeDate(comment.posted));
20
- console.log(`${author} ${time} ${colors.timestamp(`id:${comment.id}`)}`);
21
- console.log(options.raw ? comment.content : renderMarkdown(comment.content));
22
- console.log('');
23
- }
24
- async function updateComment(ref, content, options) {
25
- const commentId = resolveCommentId(ref);
26
- let newContent = await readStdin();
27
- if (!newContent && content) {
28
- newContent = content;
29
- }
30
- if (!newContent) {
31
- newContent = await openEditor();
32
- }
33
- if (!newContent || newContent.trim() === '') {
34
- console.error('No content provided.');
35
- process.exit(1);
36
- }
37
- if (options.dryRun) {
38
- console.log(`Dry run: would update comment ${commentId}`);
39
- console.log('');
40
- console.log(newContent);
41
- return;
42
- }
43
- const client = await getTwistClient();
44
- const comment = await client.comments.updateComment({
45
- id: commentId,
46
- content: newContent,
47
- });
48
- if (options.json) {
49
- console.log(formatJson(comment, 'comment', options.full));
50
- return;
51
- }
52
- console.log(`Comment updated: ${comment.url}`);
53
- }
54
- async function deleteComment(ref, options) {
55
- const commentId = resolveCommentId(ref);
56
- if (options.dryRun) {
57
- console.log(`Dry run: would delete comment ${commentId}`);
58
- return;
59
- }
60
- const client = await getTwistClient();
61
- await client.comments.deleteComment(commentId);
62
- if (options.json) {
63
- console.log(formatJson({ id: commentId, deleted: true }));
64
- return;
65
- }
66
- console.log(`Comment ${commentId} deleted.`);
67
- }
68
- export function registerCommentCommand(program) {
69
- const comment = program
70
- .command('comment')
71
- .description('Thread comment operations (view, update, delete)');
72
- comment
73
- .command('view [comment-ref]', { isDefault: true })
74
- .description('View a single thread comment')
75
- .option('--raw', 'Show raw markdown instead of rendered')
76
- .option('--json', 'Output as JSON')
77
- .option('--full', 'Include all fields in JSON output')
78
- .action((ref, options) => {
79
- if (!ref) {
80
- comment.help();
81
- return;
82
- }
83
- return viewComment(ref, options);
84
- });
85
- comment
86
- .command('update <comment-ref> [content]')
87
- .description('Update a thread comment')
88
- .option('--dry-run', 'Show what would be updated without updating')
89
- .option('--json', 'Output updated comment as JSON')
90
- .option('--full', 'Include all fields in JSON output')
91
- .action(updateComment);
92
- comment
93
- .command('delete <comment-ref>')
94
- .description('Delete a thread comment')
95
- .option('--dry-run', 'Show what would happen without executing')
96
- .option('--json', 'Output result as JSON')
97
- .action(deleteComment);
98
- }
99
- //# sourceMappingURL=comment.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"comment.js","sourceRoot":"","sources":["../../src/commands/comment.ts"],"names":[],"mappings":"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"}
@@ -1,3 +0,0 @@
1
- import { Command } from 'commander';
2
- export declare function registerCompletionCommand(program: Command): void;
3
- //# sourceMappingURL=completion.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"completion.d.ts","sourceRoot":"","sources":["../../src/commands/completion.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AA2CnC,wBAAgB,yBAAyB,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAkGhE"}
@@ -1,121 +0,0 @@
1
- import { existsSync } from 'node:fs';
2
- import { homedir } from 'node:os';
3
- import { join, resolve } from 'node:path';
4
- import { getCompletions, parseCompLine } from '../lib/completion.js';
5
- // pwsh is included because tabtab supports it — installedShells() needs to
6
- // detect and clean up pwsh completion files even though we don't advertise it.
7
- const COMPLETION_EXTENSIONS = {
8
- bash: 'bash',
9
- zsh: 'zsh',
10
- fish: 'fish',
11
- pwsh: 'ps1',
12
- };
13
- /**
14
- * Find which shells have tw completions installed by checking for the
15
- * completion script files that tabtab creates.
16
- *
17
- * FIXME: Workaround for https://github.com/pnpm/tabtab/issues/34 —
18
- * tabtab.uninstall() without a shell tries all shells and throws ENOENT
19
- * for ones that were never installed. We detect which shells are actually
20
- * installed and uninstall only those. Remove once the upstream issue is fixed.
21
- */
22
- function installedShells() {
23
- return Object.entries(COMPLETION_EXTENSIONS)
24
- .filter(([shell, ext]) => existsSync(join(homedir(), '.config', 'tabtab', shell, `tw.${ext}`)))
25
- .map(([shell]) => shell);
26
- }
27
- function resolveCompleterCommand() {
28
- const invokedScript = process.argv[1];
29
- if (!invokedScript) {
30
- return 'tw';
31
- }
32
- const resolvedScript = resolve(invokedScript);
33
- if (existsSync(resolvedScript)) {
34
- return resolvedScript;
35
- }
36
- return 'tw';
37
- }
38
- export function registerCompletionCommand(program) {
39
- const completion = program.command('completion').description('Manage shell completions');
40
- completion
41
- .command('install [shell]')
42
- .description('Install shell completions (bash, zsh, fish)')
43
- .action(async (shell) => {
44
- const tabtab = await import('@pnpm/tabtab');
45
- const completer = resolveCompleterCommand();
46
- if (shell && !tabtab.isShellSupported(shell)) {
47
- console.error(`Unsupported shell: ${shell}. Supported: ${tabtab.SUPPORTED_SHELLS.join(', ')}`);
48
- process.exitCode = 1;
49
- return;
50
- }
51
- await tabtab.install({
52
- name: 'tw',
53
- // Use the executable path used to install completions so shell
54
- // completion doesn't accidentally call an older `tw` on PATH.
55
- completer,
56
- shell: shell,
57
- });
58
- console.log('Shell completions installed successfully.');
59
- console.log('Restart your shell or source your shell config to activate.');
60
- });
61
- completion
62
- .command('uninstall')
63
- .description('Remove shell completions')
64
- .action(async () => {
65
- // FIXME: Replace with plain tabtab.uninstall({ name: 'tw' }) once
66
- // https://github.com/pnpm/tabtab/issues/34 is fixed.
67
- const shells = installedShells();
68
- if (shells.length === 0) {
69
- console.log('No shell completions installed.');
70
- return;
71
- }
72
- const tabtab = await import('@pnpm/tabtab');
73
- for (const shell of shells) {
74
- await tabtab.uninstall({
75
- name: 'tw',
76
- shell,
77
- });
78
- }
79
- console.log('Shell completions removed.');
80
- });
81
- // Hidden command invoked by the shell completion script at TAB time
82
- program
83
- .command('completion-server', { hidden: true })
84
- .description('Completion server (internal)')
85
- .allowUnknownOption()
86
- .allowExcessArguments()
87
- .action(async () => {
88
- const tabtab = await import('@pnpm/tabtab');
89
- const env = tabtab.parseEnv(process.env);
90
- if (!env.complete) {
91
- return;
92
- }
93
- const shell = tabtab.getShellFromEnv(process.env);
94
- const words = parseCompLine(env.line);
95
- let current = env.last;
96
- // The fish/zsh completion templates always append a trailing space
97
- // to COMP_LINE, making env.last empty even when the cursor is right
98
- // after '--flag='. Use env.lastPartial (cursor-position-aware) to
99
- // detect this case and restore the actual word being completed.
100
- if (current === '' && env.lastPartial.includes('=')) {
101
- if (words.at(-1) === '')
102
- words.pop();
103
- current = env.lastPartial;
104
- }
105
- const completions = getCompletions(program, words, current);
106
- // Bash treats '=' as a word break (COMP_WORDBREAKS), so readline
107
- // replaces only the part after '=' with COMPREPLY values. If we
108
- // return '--flag=value', bash produces '--flag=--flag=value'.
109
- // Strip the flag prefix so bash gets just the value part.
110
- if (shell === 'bash' && current.includes('=')) {
111
- const prefix = current.slice(0, current.indexOf('=') + 1);
112
- const values = completions
113
- .map((c) => (c.name.startsWith(prefix) ? c.name.slice(prefix.length) : c.name))
114
- .filter(Boolean);
115
- console.log(values.join('\n'));
116
- return;
117
- }
118
- tabtab.log(completions, shell);
119
- });
120
- }
121
- //# sourceMappingURL=completion.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"completion.js","sourceRoot":"","sources":["../../src/commands/completion.ts"],"names":[],"mappings":"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"}
@@ -1,3 +0,0 @@
1
- import { Command } from 'commander';
2
- export declare function registerConversationCommand(program: Command): void;
3
- //# sourceMappingURL=conversation.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"conversation.d.ts","sourceRoot":"","sources":["../../src/commands/conversation.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AA4enC,wBAAgB,2BAA2B,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA0DlE"}
@@ -1,394 +0,0 @@
1
- import chalk from 'chalk';
2
- import { getCurrentWorkspaceId, getSessionUser, getTwistClient } from '../lib/api.js';
3
- import { formatRelativeDate } from '../lib/dates.js';
4
- import { openEditor, readStdin } from '../lib/input.js';
5
- import { renderMarkdown } from '../lib/markdown.js';
6
- import { colors, filterEntityFields, formatJson, formatNdjson, isAccessible, } from '../lib/output.js';
7
- import { resolveConversationId, resolveUserRefs, resolveWorkspaceRef } from '../lib/refs.js';
8
- const CONVERSATION_PAGE_LIMIT = 100;
9
- function buildConversationTitle(conversation, userMap) {
10
- const participants = conversation.userIds
11
- .map((id) => userMap.get(id) || `user:${id}`)
12
- .join(', ');
13
- return conversation.title || `Conversation with ${participants}`;
14
- }
15
- function sortByLastActiveDescending(a, b) {
16
- return new Date(b.lastActive).getTime() - new Date(a.lastActive).getTime();
17
- }
18
- async function getAllConversations(workspaceId) {
19
- const [activeConversations, archivedConversations] = await Promise.all([
20
- getConversationPages({ workspaceId, limit: CONVERSATION_PAGE_LIMIT }),
21
- getConversationPages({ workspaceId, archived: true, limit: CONVERSATION_PAGE_LIMIT }),
22
- ]);
23
- const byId = new Map();
24
- for (const conversation of [...activeConversations, ...archivedConversations]) {
25
- byId.set(conversation.id, conversation);
26
- }
27
- return [...byId.values()].sort(sortByLastActiveDescending);
28
- }
29
- async function getConversationPages(initialArgs) {
30
- const client = await getTwistClient();
31
- const conversations = [];
32
- let beforeId = initialArgs.beforeId;
33
- while (true) {
34
- const pageArgs = {
35
- ...initialArgs,
36
- beforeId,
37
- };
38
- const page = await client.conversations.getConversations(pageArgs);
39
- if (page.length === 0) {
40
- break;
41
- }
42
- conversations.push(...page);
43
- if (page.length < initialArgs.limit) {
44
- break;
45
- }
46
- const nextBeforeId = page[page.length - 1]?.id;
47
- if (!nextBeforeId || nextBeforeId === beforeId) {
48
- break;
49
- }
50
- beforeId = nextBeforeId;
51
- }
52
- return conversations;
53
- }
54
- async function findDirectConversation(workspaceId, sessionUserId, targetUserId) {
55
- const client = await getTwistClient();
56
- let groupConversationCount = 0;
57
- for (const archived of [false, true]) {
58
- let beforeId;
59
- while (true) {
60
- const pageArgs = {
61
- workspaceId,
62
- archived: archived || undefined,
63
- limit: CONVERSATION_PAGE_LIMIT,
64
- beforeId,
65
- };
66
- const page = await client.conversations.getConversations(pageArgs);
67
- if (page.length === 0) {
68
- break;
69
- }
70
- for (const conversation of page) {
71
- if (!conversation.userIds.includes(targetUserId)) {
72
- continue;
73
- }
74
- const isSelfConversation = sessionUserId === targetUserId;
75
- const isDirect = isSelfConversation
76
- ? conversation.userIds.length === 1
77
- : conversation.userIds.length === 2 &&
78
- conversation.userIds.includes(sessionUserId);
79
- if (isDirect) {
80
- return { directConversation: conversation, groupConversationCount };
81
- }
82
- if (conversation.userIds.length > (isSelfConversation ? 1 : 2)) {
83
- groupConversationCount += 1;
84
- }
85
- }
86
- beforeId = page[page.length - 1]?.id;
87
- }
88
- }
89
- return { groupConversationCount };
90
- }
91
- async function listConversationsWithUser(conversations, workspaceId, options) {
92
- if (conversations.length === 0) {
93
- console.log('No matching conversations found.');
94
- return;
95
- }
96
- const client = await getTwistClient();
97
- const userIds = new Set();
98
- for (const conversation of conversations) {
99
- for (const userId of conversation.userIds) {
100
- userIds.add(userId);
101
- }
102
- }
103
- const userCalls = [...userIds].map((userId) => client.workspaceUsers.getUserById({ workspaceId, userId }, { batch: true }));
104
- const userResponses = await client.batch(...userCalls);
105
- const userMap = new Map(userResponses.map((response) => [response.data.id, response.data.name]));
106
- const output = conversations.map((conversation) => ({
107
- ...conversation,
108
- participantNames: conversation.userIds.map((id) => userMap.get(id)),
109
- }));
110
- if (options.json) {
111
- console.log(formatJson(output, 'conversation', options.full));
112
- return;
113
- }
114
- if (options.ndjson) {
115
- console.log(formatNdjson(output, 'conversation', options.full));
116
- return;
117
- }
118
- for (const conversation of conversations) {
119
- const participants = conversation.userIds
120
- .map((id) => userMap.get(id) || `user:${id}`)
121
- .join(', ');
122
- const title = buildConversationTitle(conversation, userMap);
123
- const archivedBadge = conversation.archived
124
- ? chalk.yellow(isAccessible() ? ' (archived)' : ' [archived]')
125
- : '';
126
- console.log(`${chalk.bold(title)}${archivedBadge}`);
127
- console.log(` ${colors.timestamp(`id:${conversation.id}`)} ${colors.author(participants)}`);
128
- if (options.snippet && conversation.snippet) {
129
- console.log(renderMarkdown(conversation.snippet));
130
- }
131
- console.log(` ${colors.timestamp(formatRelativeDate(conversation.lastActive))}`);
132
- console.log(` ${colors.url(conversation.url)}`);
133
- console.log('');
134
- }
135
- }
136
- async function showUnread(workspaceRef, options) {
137
- if (workspaceRef && options.workspace) {
138
- throw new Error('Cannot specify workspace both as argument and --workspace flag');
139
- }
140
- let workspaceId;
141
- const ref = workspaceRef || options.workspace;
142
- if (ref) {
143
- const workspace = await resolveWorkspaceRef(ref);
144
- workspaceId = workspace.id;
145
- }
146
- else {
147
- workspaceId = await getCurrentWorkspaceId();
148
- }
149
- const client = await getTwistClient();
150
- const unreadConversations = await client.conversations.getUnread(workspaceId);
151
- if (unreadConversations.length === 0) {
152
- console.log('No unread conversations.');
153
- return;
154
- }
155
- const conversationCalls = unreadConversations.map((uc) => client.conversations.getConversation(uc.conversationId, { batch: true }));
156
- const conversationResponses = await client.batch(...conversationCalls);
157
- const conversations = conversationResponses.map((r) => r.data);
158
- const userIds = new Set();
159
- for (const conv of conversations) {
160
- for (const id of conv.userIds) {
161
- userIds.add(id);
162
- }
163
- }
164
- const userCalls = [...userIds].map((id) => client.workspaceUsers.getUserById({ workspaceId, userId: id }, { batch: true }));
165
- const userResponses = await client.batch(...userCalls);
166
- const userMap = new Map(userResponses.map((r) => [r.data.id, r.data.name]));
167
- if (options.json) {
168
- const output = conversations.map((c) => ({
169
- ...c,
170
- participantNames: c.userIds.map((id) => userMap.get(id)),
171
- }));
172
- console.log(formatJson(output, 'conversation', options.full));
173
- return;
174
- }
175
- if (options.ndjson) {
176
- const output = conversations.map((c) => ({
177
- ...c,
178
- participantNames: c.userIds.map((id) => userMap.get(id)),
179
- }));
180
- console.log(formatNdjson(output, 'conversation', options.full));
181
- return;
182
- }
183
- for (const conv of conversations) {
184
- const participants = conv.userIds.map((id) => userMap.get(id) || `user:${id}`).join(', ');
185
- const title = conv.title || `Conversation with ${participants}`;
186
- const unreadInfo = unreadConversations.find((uc) => uc.conversationId === conv.id);
187
- const unreadBadge = unreadInfo ? chalk.blue(isAccessible() ? ' (unread)' : ' *') : '';
188
- console.log(`${chalk.bold(title)}${unreadBadge}`);
189
- console.log(` ${colors.timestamp(`id:${conv.id}`)} ${colors.author(participants)}`);
190
- console.log(` ${colors.url(conv.url)}`);
191
- console.log('');
192
- }
193
- }
194
- async function viewConversation(ref, options) {
195
- const conversationId = resolveConversationId(ref);
196
- const client = await getTwistClient();
197
- const limit = options.limit ? parseInt(options.limit, 10) : 50;
198
- const [convResponse, messagesResponse] = await client.batch(client.conversations.getConversation(conversationId, { batch: true }), client.conversationMessages.getMessages({
199
- conversationId,
200
- limit,
201
- }, { batch: true }));
202
- const conversation = convResponse.data;
203
- const messages = messagesResponse.data;
204
- const userIds = new Set([...conversation.userIds, ...messages.map((m) => m.creator)]);
205
- const userCalls = [...userIds].map((id) => client.workspaceUsers.getUserById({ workspaceId: conversation.workspaceId, userId: id }, { batch: true }));
206
- const userResponses = await client.batch(...userCalls);
207
- const userMap = new Map(userResponses.map((r) => [r.data.id, r.data.name]));
208
- const conversationOutput = {
209
- ...conversation,
210
- participantNames: conversation.userIds.map((id) => userMap.get(id)),
211
- };
212
- const messageOutput = messages.map((m) => ({
213
- ...m,
214
- creatorName: userMap.get(m.creator),
215
- }));
216
- if (options.json) {
217
- const output = {
218
- conversation: filterEntityFields(conversationOutput, 'conversation', options.full),
219
- messages: filterEntityFields(messageOutput, 'message', options.full),
220
- };
221
- console.log(JSON.stringify(output, null, 2));
222
- return;
223
- }
224
- if (options.ndjson) {
225
- console.log(JSON.stringify({
226
- type: 'conversation',
227
- ...filterEntityFields(conversationOutput, 'conversation', options.full),
228
- }));
229
- const formattedMessages = filterEntityFields(messageOutput, 'message', options.full);
230
- for (const message of formattedMessages) {
231
- console.log(JSON.stringify({ type: 'message', ...message }));
232
- }
233
- return;
234
- }
235
- const title = buildConversationTitle(conversation, userMap);
236
- console.log(chalk.bold(title));
237
- console.log(colors.timestamp(`id:${conversation.id}`));
238
- console.log('');
239
- if (messages.length === 0) {
240
- console.log('No messages.');
241
- return;
242
- }
243
- for (const message of messages) {
244
- const author = colors.author(userMap.get(message.creator) || `user:${message.creator}`);
245
- const time = colors.timestamp(formatRelativeDate(message.posted));
246
- console.log(`${author} ${time} ${colors.timestamp(`id:${message.id}`)}`);
247
- console.log(options.raw ? message.content : renderMarkdown(message.content));
248
- console.log('');
249
- }
250
- }
251
- async function replyToConversation(ref, content, options) {
252
- const conversationId = resolveConversationId(ref);
253
- let replyContent = await readStdin();
254
- if (!replyContent && content) {
255
- replyContent = content;
256
- }
257
- if (!replyContent) {
258
- replyContent = await openEditor();
259
- }
260
- if (!replyContent || replyContent.trim() === '') {
261
- console.error('No content provided.');
262
- process.exit(1);
263
- }
264
- if (options.dryRun) {
265
- console.log('Dry run: would send message to conversation', conversationId);
266
- console.log('');
267
- console.log(replyContent);
268
- return;
269
- }
270
- const client = await getTwistClient();
271
- const message = await client.conversationMessages.createMessage({
272
- conversationId,
273
- content: replyContent,
274
- });
275
- if (options.json) {
276
- console.log(formatJson(message, 'message', options.full));
277
- return;
278
- }
279
- console.log(`Message sent: ${message.url}`);
280
- }
281
- async function markConversationDone(ref, options) {
282
- const conversationId = resolveConversationId(ref);
283
- if (options.dryRun) {
284
- console.log(`Dry run: would archive conversation ${conversationId}`);
285
- return;
286
- }
287
- const client = await getTwistClient();
288
- await client.conversations.archiveConversation(conversationId);
289
- if (options.json) {
290
- console.log(formatJson({ id: conversationId, archived: true }));
291
- return;
292
- }
293
- console.log(`Conversation ${conversationId} archived.`);
294
- }
295
- async function findConversationWithUser(userRef, workspaceRef, options) {
296
- try {
297
- if (workspaceRef && options.workspace) {
298
- throw new Error('Cannot specify workspace both as argument and --workspace flag');
299
- }
300
- let workspaceId;
301
- const ref = workspaceRef || options.workspace;
302
- if (ref) {
303
- const workspace = await resolveWorkspaceRef(ref);
304
- workspaceId = workspace.id;
305
- }
306
- else {
307
- workspaceId = await getCurrentWorkspaceId();
308
- }
309
- const userIds = await resolveUserRefs(userRef, workspaceId);
310
- if (userIds.length !== 1) {
311
- throw new Error('Expected a single user reference');
312
- }
313
- const targetUserId = userIds[0];
314
- const client = await getTwistClient();
315
- const [sessionUser, targetUser] = await Promise.all([
316
- getSessionUser(),
317
- client.workspaceUsers.getUserById({ workspaceId, userId: targetUserId }),
318
- ]);
319
- if (options.includeGroups) {
320
- const conversations = await getAllConversations(workspaceId);
321
- const matchingConversations = conversations.filter((conversation) => conversation.userIds.includes(targetUser.id));
322
- await listConversationsWithUser(matchingConversations, workspaceId, options);
323
- return;
324
- }
325
- const { directConversation, groupConversationCount } = await findDirectConversation(workspaceId, sessionUser.id, targetUser.id);
326
- if (!directConversation) {
327
- const suggestion = groupConversationCount > 0
328
- ? ` Found ${groupConversationCount} group conversation${groupConversationCount === 1 ? '' : 's'} with ${targetUser.name}. Use --include-groups to list them.`
329
- : '';
330
- console.log(`No 1:1 conversation found with ${targetUser.name}.${suggestion}`);
331
- return;
332
- }
333
- await listConversationsWithUser([directConversation], workspaceId, options);
334
- }
335
- catch (error) {
336
- console.error(error.message);
337
- process.exit(1);
338
- }
339
- }
340
- export function registerConversationCommand(program) {
341
- const conversation = program
342
- .command('conversation')
343
- .alias('convo')
344
- .description('Conversation (DM/group) operations');
345
- conversation
346
- .command('unread [workspace-ref]')
347
- .description('List unread conversations')
348
- .option('--workspace <ref>', 'Workspace ID or name')
349
- .option('--json', 'Output as JSON')
350
- .option('--ndjson', 'Output as newline-delimited JSON')
351
- .option('--full', 'Include all fields in JSON output')
352
- .action(showUnread);
353
- conversation
354
- .command('view [conversation-ref]', { isDefault: true })
355
- .description('Display a conversation with its messages')
356
- .option('--limit <n>', 'Max messages to show (default: 50)')
357
- .option('--since <date>', 'Messages newer than')
358
- .option('--until <date>', 'Messages older than')
359
- .option('--raw', 'Show raw markdown instead of rendered')
360
- .option('--json', 'Output as JSON')
361
- .option('--ndjson', 'Output as newline-delimited JSON')
362
- .option('--full', 'Include all fields in JSON output')
363
- .action((ref, options) => {
364
- if (!ref) {
365
- conversation.help();
366
- return;
367
- }
368
- return viewConversation(ref, options);
369
- });
370
- conversation
371
- .command('with <user-ref> [workspace-ref]')
372
- .description('Find your 1:1 conversation with a user')
373
- .option('--workspace <ref>', 'Workspace ID or name')
374
- .option('--include-groups', 'List any conversation that includes this user')
375
- .option('--snippet', 'Include the latest message snippet in text output')
376
- .option('--json', 'Output as JSON')
377
- .option('--ndjson', 'Output as newline-delimited JSON')
378
- .option('--full', 'Include all fields in JSON output')
379
- .action(findConversationWithUser);
380
- conversation
381
- .command('reply <conversation-ref> [content]')
382
- .description('Send a message in a conversation')
383
- .option('--dry-run', 'Show what would be sent without sending')
384
- .option('--json', 'Output sent message as JSON')
385
- .option('--full', 'Include all fields in JSON output')
386
- .action(replyToConversation);
387
- conversation
388
- .command('done <conversation-ref>')
389
- .description('Archive a conversation')
390
- .option('--dry-run', 'Show what would happen without executing')
391
- .option('--json', 'Output result as JSON')
392
- .action(markConversationDone);
393
- }
394
- //# sourceMappingURL=conversation.js.map