clawvault 2.2.1 → 2.3.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.
- package/bin/clawvault.js +12 -0
- package/bin/register-tailscale-commands.js +106 -0
- package/bin/register-task-commands.js +257 -0
- package/dist/{chunk-2HM7ZI4X.js → chunk-2AYPFUGX.js} +1 -1
- package/dist/chunk-4GBPTBFJ.js +628 -0
- package/dist/{chunk-LB6P4CD5.js → chunk-6AQZIPLV.js} +7 -7
- package/dist/chunk-CLE2HHNT.js +513 -0
- package/dist/{chunk-VR5NE7PZ.js → chunk-HVTTYDCJ.js} +1 -1
- package/dist/{chunk-GQVYQCY5.js → chunk-JVAWKNIZ.js} +2 -2
- package/dist/chunk-MDIH26GC.js +183 -0
- package/dist/{chunk-Z2XBWN7A.js → chunk-NAMFB7ZA.js} +2 -0
- package/dist/chunk-NGVAEFT2.js +352 -0
- package/dist/chunk-NZ4ZZNSR.js +373 -0
- package/dist/{chunk-MQUJNOHK.js → chunk-QALB2V3E.js} +1 -1
- package/dist/{chunk-L6NB43WV.js → chunk-RARDNTUP.js} +4 -4
- package/dist/{chunk-I5X6J4FX.js → chunk-TB3BM2PQ.js} +6 -6
- package/dist/{chunk-GJEGPO7U.js → chunk-TT3FXYCN.js} +1 -1
- package/dist/{chunk-73P7XCQM.js → chunk-USZU5CBB.js} +5 -5
- package/dist/{chunk-WZI3OAE5.js → chunk-VBVEXNI5.js} +4 -4
- package/dist/commands/archive.js +3 -3
- package/dist/commands/backlog.d.ts +53 -0
- package/dist/commands/backlog.js +119 -0
- package/dist/commands/blocked.d.ts +25 -0
- package/dist/commands/blocked.js +43 -0
- package/dist/commands/canvas.d.ts +20 -0
- package/dist/commands/canvas.js +683 -0
- package/dist/commands/context.js +3 -3
- package/dist/commands/doctor.js +1 -1
- package/dist/commands/graph.js +2 -2
- package/dist/commands/link.js +3 -3
- package/dist/commands/migrate-observations.js +3 -3
- package/dist/commands/observe.js +5 -5
- package/dist/commands/rebuild.js +4 -4
- package/dist/commands/recover.js +2 -2
- package/dist/commands/reflect.js +5 -5
- package/dist/commands/replay.js +6 -6
- package/dist/commands/sleep.js +7 -7
- package/dist/commands/status.js +1 -1
- package/dist/commands/tailscale.d.ts +52 -0
- package/dist/commands/tailscale.js +25 -0
- package/dist/commands/task.d.ts +71 -0
- package/dist/commands/task.js +189 -0
- package/dist/commands/wake.js +8 -8
- package/dist/index.d.ts +4 -0
- package/dist/index.js +104 -42
- package/dist/lib/canvas-layout.d.ts +115 -0
- package/dist/lib/canvas-layout.js +34 -0
- package/dist/lib/tailscale.d.ts +225 -0
- package/dist/lib/tailscale.js +49 -0
- package/dist/lib/task-utils.d.ts +159 -0
- package/dist/lib/task-utils.js +46 -0
- package/dist/lib/webdav.d.ts +109 -0
- package/dist/lib/webdav.js +34 -0
- package/package.json +2 -2
- package/dist/{chunk-MILVYUPK.js → chunk-IWYZAXKJ.js} +3 -3
- package/dist/{chunk-H7JW4L7H.js → chunk-OZ7RIXTO.js} +3 -3
package/bin/clawvault.js
CHANGED
|
@@ -16,6 +16,10 @@ import { registerResilienceCommands } from './register-resilience-commands.js';
|
|
|
16
16
|
import { registerSessionLifecycleCommands } from './register-session-lifecycle-commands.js';
|
|
17
17
|
import { registerTemplateCommands } from './register-template-commands.js';
|
|
18
18
|
import { registerVaultOperationsCommands } from './register-vault-operations-commands.js';
|
|
19
|
+
|
|
20
|
+
import { registerTaskCommands } from './register-task-commands.js';
|
|
21
|
+
|
|
22
|
+
import { registerTailscaleCommands } from './register-tailscale-commands.js';
|
|
19
23
|
import {
|
|
20
24
|
getVault,
|
|
21
25
|
resolveVaultPath,
|
|
@@ -82,5 +86,13 @@ registerVaultOperationsCommands(program, {
|
|
|
82
86
|
path
|
|
83
87
|
});
|
|
84
88
|
|
|
89
|
+
|
|
90
|
+
registerTaskCommands(program, {
|
|
91
|
+
chalk,
|
|
92
|
+
resolveVaultPath
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
registerTailscaleCommands(program, { chalk });
|
|
96
|
+
|
|
85
97
|
// Parse and run
|
|
86
98
|
program.parse();
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tailscale networking command registrations.
|
|
3
|
+
* Provides vault synchronization over Tailscale networks.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export function registerTailscaleCommands(program, { chalk }) {
|
|
7
|
+
// === TAILSCALE-STATUS ===
|
|
8
|
+
program
|
|
9
|
+
.command('tailscale-status')
|
|
10
|
+
.alias('ts-status')
|
|
11
|
+
.description('Show Tailscale connection status and peers')
|
|
12
|
+
.option('--json', 'Output as JSON')
|
|
13
|
+
.option('--peers', 'Show all peers including offline')
|
|
14
|
+
.action(async (options) => {
|
|
15
|
+
try {
|
|
16
|
+
const { tailscaleStatusCommand } = await import('../dist/commands/tailscale.js');
|
|
17
|
+
await tailscaleStatusCommand({
|
|
18
|
+
json: options.json,
|
|
19
|
+
peers: options.peers
|
|
20
|
+
});
|
|
21
|
+
} catch (err) {
|
|
22
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
23
|
+
process.exit(1);
|
|
24
|
+
}
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
// === TAILSCALE-SYNC ===
|
|
28
|
+
program
|
|
29
|
+
.command('tailscale-sync')
|
|
30
|
+
.alias('ts-sync')
|
|
31
|
+
.description('Sync vault with a peer on the Tailscale network')
|
|
32
|
+
.requiredOption('--peer <hostname>', 'Peer hostname or IP to sync with')
|
|
33
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
34
|
+
.option('--port <number>', 'Port on the peer (default: 8384)', parseInt)
|
|
35
|
+
.option('--direction <dir>', 'Sync direction: push, pull, or bidirectional', 'bidirectional')
|
|
36
|
+
.option('--dry-run', 'Show what would be synced without making changes')
|
|
37
|
+
.option('--delete-orphans', 'Delete files that exist locally but not on peer (pull only)')
|
|
38
|
+
.option('--categories <list>', 'Comma-separated list of categories to sync')
|
|
39
|
+
.option('--https', 'Use HTTPS for connection')
|
|
40
|
+
.option('--json', 'Output as JSON')
|
|
41
|
+
.action(async (options) => {
|
|
42
|
+
try {
|
|
43
|
+
const { tailscaleSyncCommand } = await import('../dist/commands/tailscale.js');
|
|
44
|
+
await tailscaleSyncCommand({
|
|
45
|
+
peer: options.peer,
|
|
46
|
+
vaultPath: options.vault,
|
|
47
|
+
port: options.port,
|
|
48
|
+
direction: options.direction,
|
|
49
|
+
dryRun: options.dryRun,
|
|
50
|
+
deleteOrphans: options.deleteOrphans,
|
|
51
|
+
categories: options.categories?.split(',').map(c => c.trim()),
|
|
52
|
+
https: options.https,
|
|
53
|
+
json: options.json
|
|
54
|
+
});
|
|
55
|
+
} catch (err) {
|
|
56
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
57
|
+
process.exit(1);
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
// === TAILSCALE-SERVE ===
|
|
62
|
+
program
|
|
63
|
+
.command('tailscale-serve')
|
|
64
|
+
.alias('ts-serve')
|
|
65
|
+
.description('Serve vault for sync over Tailscale')
|
|
66
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
67
|
+
.option('--port <number>', 'Port to serve on (default: 8384)', parseInt)
|
|
68
|
+
.option('--funnel', 'Expose via Tailscale Funnel (public internet)')
|
|
69
|
+
.option('--background', 'Run in background')
|
|
70
|
+
.option('--stop', 'Stop serving')
|
|
71
|
+
.action(async (options) => {
|
|
72
|
+
try {
|
|
73
|
+
const { tailscaleServeCommand } = await import('../dist/commands/tailscale.js');
|
|
74
|
+
await tailscaleServeCommand({
|
|
75
|
+
vaultPath: options.vault,
|
|
76
|
+
port: options.port,
|
|
77
|
+
funnel: options.funnel,
|
|
78
|
+
background: options.background,
|
|
79
|
+
stop: options.stop
|
|
80
|
+
});
|
|
81
|
+
} catch (err) {
|
|
82
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
83
|
+
process.exit(1);
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
// === TAILSCALE-DISCOVER ===
|
|
88
|
+
program
|
|
89
|
+
.command('tailscale-discover')
|
|
90
|
+
.alias('ts-discover')
|
|
91
|
+
.description('Discover ClawVault peers on the Tailscale network')
|
|
92
|
+
.option('--port <number>', 'Port to check (default: 8384)', parseInt)
|
|
93
|
+
.option('--json', 'Output as JSON')
|
|
94
|
+
.action(async (options) => {
|
|
95
|
+
try {
|
|
96
|
+
const { tailscaleDiscoverCommand } = await import('../dist/commands/tailscale.js');
|
|
97
|
+
await tailscaleDiscoverCommand({
|
|
98
|
+
port: options.port,
|
|
99
|
+
json: options.json
|
|
100
|
+
});
|
|
101
|
+
} catch (err) {
|
|
102
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
103
|
+
process.exit(1);
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
}
|
|
@@ -0,0 +1,257 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Task tracking command registrations for ClawVault
|
|
3
|
+
* Registers task, backlog, blocked, and canvas commands
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export function registerTaskCommands(
|
|
7
|
+
program,
|
|
8
|
+
{ chalk, resolveVaultPath }
|
|
9
|
+
) {
|
|
10
|
+
// === TASK ===
|
|
11
|
+
const taskCmd = program
|
|
12
|
+
.command('task')
|
|
13
|
+
.description('Task management');
|
|
14
|
+
|
|
15
|
+
// task add
|
|
16
|
+
taskCmd
|
|
17
|
+
.command('add <title>')
|
|
18
|
+
.description('Add a new task')
|
|
19
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
20
|
+
.option('--owner <owner>', 'Task owner')
|
|
21
|
+
.option('--project <project>', 'Project name')
|
|
22
|
+
.option('--priority <priority>', 'Priority (critical, high, medium, low)')
|
|
23
|
+
.option('--due <date>', 'Due date (YYYY-MM-DD)')
|
|
24
|
+
.action(async (title, options) => {
|
|
25
|
+
try {
|
|
26
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
27
|
+
const { taskCommand } = await import('../dist/commands/task.js');
|
|
28
|
+
await taskCommand(vaultPath, 'add', {
|
|
29
|
+
title,
|
|
30
|
+
options: {
|
|
31
|
+
owner: options.owner,
|
|
32
|
+
project: options.project,
|
|
33
|
+
priority: options.priority,
|
|
34
|
+
due: options.due
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
} catch (err) {
|
|
38
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
39
|
+
process.exit(1);
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
// task list
|
|
44
|
+
taskCmd
|
|
45
|
+
.command('list')
|
|
46
|
+
.description('List tasks')
|
|
47
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
48
|
+
.option('--owner <owner>', 'Filter by owner')
|
|
49
|
+
.option('--project <project>', 'Filter by project')
|
|
50
|
+
.option('--status <status>', 'Filter by status (open, in-progress, blocked, done)')
|
|
51
|
+
.option('--priority <priority>', 'Filter by priority')
|
|
52
|
+
.option('--json', 'Output as JSON')
|
|
53
|
+
.action(async (options) => {
|
|
54
|
+
try {
|
|
55
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
56
|
+
const { taskCommand } = await import('../dist/commands/task.js');
|
|
57
|
+
await taskCommand(vaultPath, 'list', {
|
|
58
|
+
options: {
|
|
59
|
+
owner: options.owner,
|
|
60
|
+
project: options.project,
|
|
61
|
+
status: options.status,
|
|
62
|
+
priority: options.priority,
|
|
63
|
+
json: options.json
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
} catch (err) {
|
|
67
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
68
|
+
process.exit(1);
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
// task update
|
|
73
|
+
taskCmd
|
|
74
|
+
.command('update <slug>')
|
|
75
|
+
.description('Update a task')
|
|
76
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
77
|
+
.option('--status <status>', 'New status')
|
|
78
|
+
.option('--owner <owner>', 'New owner')
|
|
79
|
+
.option('--project <project>', 'New project')
|
|
80
|
+
.option('--priority <priority>', 'New priority')
|
|
81
|
+
.option('--blocked-by <blocker>', 'What is blocking this task')
|
|
82
|
+
.option('--due <date>', 'New due date')
|
|
83
|
+
.action(async (slug, options) => {
|
|
84
|
+
try {
|
|
85
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
86
|
+
const { taskCommand } = await import('../dist/commands/task.js');
|
|
87
|
+
await taskCommand(vaultPath, 'update', {
|
|
88
|
+
slug,
|
|
89
|
+
options: {
|
|
90
|
+
status: options.status,
|
|
91
|
+
owner: options.owner,
|
|
92
|
+
project: options.project,
|
|
93
|
+
priority: options.priority,
|
|
94
|
+
blockedBy: options.blockedBy,
|
|
95
|
+
due: options.due
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
} catch (err) {
|
|
99
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
100
|
+
process.exit(1);
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
// task done
|
|
105
|
+
taskCmd
|
|
106
|
+
.command('done <slug>')
|
|
107
|
+
.description('Mark a task as done')
|
|
108
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
109
|
+
.action(async (slug, options) => {
|
|
110
|
+
try {
|
|
111
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
112
|
+
const { taskCommand } = await import('../dist/commands/task.js');
|
|
113
|
+
await taskCommand(vaultPath, 'done', { slug });
|
|
114
|
+
} catch (err) {
|
|
115
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
116
|
+
process.exit(1);
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
// task show
|
|
121
|
+
taskCmd
|
|
122
|
+
.command('show <slug>')
|
|
123
|
+
.description('Show task details')
|
|
124
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
125
|
+
.option('--json', 'Output as JSON')
|
|
126
|
+
.action(async (slug, options) => {
|
|
127
|
+
try {
|
|
128
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
129
|
+
const { taskCommand } = await import('../dist/commands/task.js');
|
|
130
|
+
await taskCommand(vaultPath, 'show', {
|
|
131
|
+
slug,
|
|
132
|
+
options: { json: options.json }
|
|
133
|
+
});
|
|
134
|
+
} catch (err) {
|
|
135
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
136
|
+
process.exit(1);
|
|
137
|
+
}
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
// === BACKLOG ===
|
|
141
|
+
const backlogCmd = program
|
|
142
|
+
.command('backlog')
|
|
143
|
+
.description('Backlog management');
|
|
144
|
+
|
|
145
|
+
// backlog add (also supports "backlog <title>" shorthand)
|
|
146
|
+
backlogCmd
|
|
147
|
+
.command('add <title>')
|
|
148
|
+
.description('Add item to backlog')
|
|
149
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
150
|
+
.option('--source <source>', 'Source of the idea')
|
|
151
|
+
.option('--project <project>', 'Project name')
|
|
152
|
+
.action(async (title, options) => {
|
|
153
|
+
try {
|
|
154
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
155
|
+
const { backlogCommand } = await import('../dist/commands/backlog.js');
|
|
156
|
+
await backlogCommand(vaultPath, 'add', {
|
|
157
|
+
title,
|
|
158
|
+
options: {
|
|
159
|
+
source: options.source,
|
|
160
|
+
project: options.project
|
|
161
|
+
}
|
|
162
|
+
});
|
|
163
|
+
} catch (err) {
|
|
164
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
165
|
+
process.exit(1);
|
|
166
|
+
}
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
// backlog list
|
|
170
|
+
backlogCmd
|
|
171
|
+
.command('list')
|
|
172
|
+
.description('List backlog items')
|
|
173
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
174
|
+
.option('--project <project>', 'Filter by project')
|
|
175
|
+
.option('--json', 'Output as JSON')
|
|
176
|
+
.action(async (options) => {
|
|
177
|
+
try {
|
|
178
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
179
|
+
const { backlogCommand } = await import('../dist/commands/backlog.js');
|
|
180
|
+
await backlogCommand(vaultPath, 'list', {
|
|
181
|
+
options: {
|
|
182
|
+
project: options.project,
|
|
183
|
+
json: options.json
|
|
184
|
+
}
|
|
185
|
+
});
|
|
186
|
+
} catch (err) {
|
|
187
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
188
|
+
process.exit(1);
|
|
189
|
+
}
|
|
190
|
+
});
|
|
191
|
+
|
|
192
|
+
// backlog promote
|
|
193
|
+
backlogCmd
|
|
194
|
+
.command('promote <slug>')
|
|
195
|
+
.description('Promote backlog item to task')
|
|
196
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
197
|
+
.option('--owner <owner>', 'Task owner')
|
|
198
|
+
.option('--priority <priority>', 'Task priority')
|
|
199
|
+
.option('--due <date>', 'Due date')
|
|
200
|
+
.action(async (slug, options) => {
|
|
201
|
+
try {
|
|
202
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
203
|
+
const { backlogCommand } = await import('../dist/commands/backlog.js');
|
|
204
|
+
await backlogCommand(vaultPath, 'promote', {
|
|
205
|
+
slug,
|
|
206
|
+
options: {
|
|
207
|
+
owner: options.owner,
|
|
208
|
+
priority: options.priority,
|
|
209
|
+
due: options.due
|
|
210
|
+
}
|
|
211
|
+
});
|
|
212
|
+
} catch (err) {
|
|
213
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
214
|
+
process.exit(1);
|
|
215
|
+
}
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
// === BLOCKED ===
|
|
219
|
+
program
|
|
220
|
+
.command('blocked')
|
|
221
|
+
.description('View blocked tasks')
|
|
222
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
223
|
+
.option('--project <project>', 'Filter by project')
|
|
224
|
+
.option('--json', 'Output as JSON')
|
|
225
|
+
.action(async (options) => {
|
|
226
|
+
try {
|
|
227
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
228
|
+
const { blockedCommand } = await import('../dist/commands/blocked.js');
|
|
229
|
+
await blockedCommand(vaultPath, {
|
|
230
|
+
project: options.project,
|
|
231
|
+
json: options.json
|
|
232
|
+
});
|
|
233
|
+
} catch (err) {
|
|
234
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
235
|
+
process.exit(1);
|
|
236
|
+
}
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
// === CANVAS ===
|
|
240
|
+
program
|
|
241
|
+
.command('canvas')
|
|
242
|
+
.description('Generate Obsidian canvas dashboard')
|
|
243
|
+
.option('-v, --vault <path>', 'Vault path')
|
|
244
|
+
.option('--output <path>', 'Output file path (default: dashboard.canvas)')
|
|
245
|
+
.action(async (options) => {
|
|
246
|
+
try {
|
|
247
|
+
const vaultPath = resolveVaultPath(options.vault);
|
|
248
|
+
const { canvasCommand } = await import('../dist/commands/canvas.js');
|
|
249
|
+
await canvasCommand(vaultPath, {
|
|
250
|
+
output: options.output
|
|
251
|
+
});
|
|
252
|
+
} catch (err) {
|
|
253
|
+
console.error(chalk.red(`Error: ${err.message}`));
|
|
254
|
+
process.exit(1);
|
|
255
|
+
}
|
|
256
|
+
});
|
|
257
|
+
}
|
|
@@ -14,7 +14,7 @@ import {
|
|
|
14
14
|
getObservationPath,
|
|
15
15
|
getRawTranscriptPath,
|
|
16
16
|
toDateKey
|
|
17
|
-
} from "./chunk-
|
|
17
|
+
} from "./chunk-NAMFB7ZA.js";
|
|
18
18
|
|
|
19
19
|
// src/observer/compressor.ts
|
|
20
20
|
var CRITICAL_RE = /(?:\b(?:decision|decided|chose|chosen|selected|picked|opted|switched to)\s*:?|\bdecid(?:e|ed|ing|ion)\b|\berror\b|\bfail(?:ed|ure|ing)?\b|\bblock(?:ed|er)?\b|\bbreaking(?:\s+change)?s?\b|\bcritical\b|\b\w+\s+chosen\s+(?:for|over|as)\b|\bpublish(?:ed)?\b.*@?\d+\.\d+|\bmerge[d]?\s+(?:PR|pull\s+request)\b|\bshipped\b|\breleased?\b.*v?\d+\.\d+|\bsigned\b.*\b(?:contract|agreement|deal)\b|\bpricing\b.*\$|\bdemo\b.*\b(?:completed?|done|finished)\b|\bmeeting\b.*\b(?:completed?|done|finished)\b|\bstrategy\b.*\b(?:pivot|change|shift)\b)/i;
|