context-mode 1.0.13 → 1.0.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude-plugin/marketplace.json +2 -2
- package/.claude-plugin/plugin.json +1 -1
- package/build/server.js +1 -1
- package/build/session/extract.js +67 -4
- package/build/session/snapshot.d.ts +1 -1
- package/build/session/snapshot.js +7 -17
- package/cli.bundle.mjs +1 -1
- package/package.json +1 -1
- package/server.bundle.mjs +1 -1
|
@@ -6,14 +6,14 @@
|
|
|
6
6
|
},
|
|
7
7
|
"metadata": {
|
|
8
8
|
"description": "Claude Code plugins by Mert Koseoğlu",
|
|
9
|
-
"version": "1.0.
|
|
9
|
+
"version": "1.0.14"
|
|
10
10
|
},
|
|
11
11
|
"plugins": [
|
|
12
12
|
{
|
|
13
13
|
"name": "context-mode",
|
|
14
14
|
"source": "./",
|
|
15
15
|
"description": "Claude Code MCP plugin that saves 98% of your context window. Sandboxed code execution in 11 languages, FTS5 knowledge base with BM25 ranking, and intent-driven search.",
|
|
16
|
-
"version": "1.0.
|
|
16
|
+
"version": "1.0.14",
|
|
17
17
|
"author": {
|
|
18
18
|
"name": "Mert Koseoğlu"
|
|
19
19
|
},
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "context-mode",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.14",
|
|
4
4
|
"description": "MCP server that saves 98% of your context window with session continuity. Sandboxed code execution in 11 languages, FTS5 knowledge base with BM25 ranking, and automatic state restore across compactions.",
|
|
5
5
|
"author": {
|
|
6
6
|
"name": "Mert Koseoğlu",
|
package/build/server.js
CHANGED
|
@@ -13,7 +13,7 @@ import { ContentStore, cleanupStaleDBs } from "./store.js";
|
|
|
13
13
|
import { readBashPolicies, evaluateCommandDenyOnly, extractShellCommands, readToolDenyPatterns, evaluateFilePath, } from "./security.js";
|
|
14
14
|
import { detectRuntimes, getRuntimeSummary, getAvailableLanguages, hasBunRuntime, } from "./runtime.js";
|
|
15
15
|
import { classifyNonZeroExit } from "./exit-classify.js";
|
|
16
|
-
const VERSION = "1.0.
|
|
16
|
+
const VERSION = "1.0.14";
|
|
17
17
|
// Prevent silent server death from unhandled async errors
|
|
18
18
|
process.on("unhandledRejection", (err) => {
|
|
19
19
|
process.stderr.write(`[context-mode] unhandledRejection: ${err}\n`);
|
package/build/session/extract.js
CHANGED
|
@@ -66,13 +66,23 @@ function extractFileAndRule(input) {
|
|
|
66
66
|
if (tool_name === "Edit") {
|
|
67
67
|
const filePath = String(tool_input["file_path"] ?? "");
|
|
68
68
|
events.push({
|
|
69
|
-
type: "
|
|
69
|
+
type: "file_edit",
|
|
70
70
|
category: "file",
|
|
71
71
|
data: truncate(filePath),
|
|
72
72
|
priority: 1,
|
|
73
73
|
});
|
|
74
74
|
return events;
|
|
75
75
|
}
|
|
76
|
+
if (tool_name === "NotebookEdit") {
|
|
77
|
+
const notebookPath = String(tool_input["notebook_path"] ?? "");
|
|
78
|
+
events.push({
|
|
79
|
+
type: "file_edit",
|
|
80
|
+
category: "file",
|
|
81
|
+
data: truncate(notebookPath),
|
|
82
|
+
priority: 1,
|
|
83
|
+
});
|
|
84
|
+
return events;
|
|
85
|
+
}
|
|
76
86
|
if (tool_name === "Write") {
|
|
77
87
|
const filePath = String(tool_input["file_path"] ?? "");
|
|
78
88
|
events.push({
|
|
@@ -165,6 +175,12 @@ const GIT_PATTERNS = [
|
|
|
165
175
|
{ pattern: /\bgit\s+status\b/, operation: "status" },
|
|
166
176
|
{ pattern: /\bgit\s+branch\b/, operation: "branch" },
|
|
167
177
|
{ pattern: /\bgit\s+reset\b/, operation: "reset" },
|
|
178
|
+
{ pattern: /\bgit\s+add\b/, operation: "add" },
|
|
179
|
+
{ pattern: /\bgit\s+cherry-pick\b/, operation: "cherry-pick" },
|
|
180
|
+
{ pattern: /\bgit\s+tag\b/, operation: "tag" },
|
|
181
|
+
{ pattern: /\bgit\s+fetch\b/, operation: "fetch" },
|
|
182
|
+
{ pattern: /\bgit\s+clone\b/, operation: "clone" },
|
|
183
|
+
{ pattern: /\bgit\s+worktree\b/, operation: "worktree" },
|
|
168
184
|
];
|
|
169
185
|
function extractGit(input) {
|
|
170
186
|
if (input.tool_name !== "Bash")
|
|
@@ -287,6 +303,12 @@ const ENV_PATTERNS = [
|
|
|
287
303
|
/\bbun\s+install\b/,
|
|
288
304
|
/\byarn\s+(add|install)\b/,
|
|
289
305
|
/\bpnpm\s+(add|install)\b/,
|
|
306
|
+
/\bcargo\s+(install|add)\b/,
|
|
307
|
+
/\bgo\s+(install|get)\b/,
|
|
308
|
+
/\brustup\b/,
|
|
309
|
+
/\basdf\b/,
|
|
310
|
+
/\bvolta\b/,
|
|
311
|
+
/\bdeno\s+install\b/,
|
|
290
312
|
];
|
|
291
313
|
function extractEnv(input) {
|
|
292
314
|
if (input.tool_name !== "Bash")
|
|
@@ -295,10 +317,12 @@ function extractEnv(input) {
|
|
|
295
317
|
const isEnvCmd = ENV_PATTERNS.some(p => p.test(cmd));
|
|
296
318
|
if (!isEnvCmd)
|
|
297
319
|
return [];
|
|
320
|
+
// Sanitize export commands to prevent secret leakage
|
|
321
|
+
const sanitized = cmd.replace(/\bexport\s+(\w+)=\S*/g, "export $1=***");
|
|
298
322
|
return [{
|
|
299
323
|
type: "env",
|
|
300
324
|
category: "env",
|
|
301
|
-
data: truncate(
|
|
325
|
+
data: truncate(sanitized),
|
|
302
326
|
priority: 2,
|
|
303
327
|
}];
|
|
304
328
|
}
|
|
@@ -359,6 +383,43 @@ function extractMcp(input) {
|
|
|
359
383
|
priority: 3,
|
|
360
384
|
}];
|
|
361
385
|
}
|
|
386
|
+
/**
|
|
387
|
+
* Category 6 (tool-based): decision
|
|
388
|
+
* AskUserQuestion tool — tracks questions posed to user and their answers.
|
|
389
|
+
*/
|
|
390
|
+
function extractDecision(input) {
|
|
391
|
+
if (input.tool_name !== "AskUserQuestion")
|
|
392
|
+
return [];
|
|
393
|
+
const questions = input.tool_input["questions"];
|
|
394
|
+
const questionText = Array.isArray(questions) && questions.length > 0
|
|
395
|
+
? String(questions[0]["question"] ?? "")
|
|
396
|
+
: "";
|
|
397
|
+
const answer = truncate(String(input.tool_response ?? ""), 150);
|
|
398
|
+
const summary = questionText
|
|
399
|
+
? `Q: ${truncate(questionText, 120)} → A: ${answer}`
|
|
400
|
+
: `answer: ${answer}`;
|
|
401
|
+
return [{
|
|
402
|
+
type: "decision_question",
|
|
403
|
+
category: "decision",
|
|
404
|
+
data: truncate(summary),
|
|
405
|
+
priority: 2,
|
|
406
|
+
}];
|
|
407
|
+
}
|
|
408
|
+
/**
|
|
409
|
+
* Category 8: env (worktree)
|
|
410
|
+
* EnterWorktree tool — tracks worktree creation.
|
|
411
|
+
*/
|
|
412
|
+
function extractWorktree(input) {
|
|
413
|
+
if (input.tool_name !== "EnterWorktree")
|
|
414
|
+
return [];
|
|
415
|
+
const name = String(input.tool_input["name"] ?? "unnamed");
|
|
416
|
+
return [{
|
|
417
|
+
type: "worktree",
|
|
418
|
+
category: "env",
|
|
419
|
+
data: truncate(`entered worktree: ${name}`),
|
|
420
|
+
priority: 2,
|
|
421
|
+
}];
|
|
422
|
+
}
|
|
362
423
|
// ── User-message extractors ────────────────────────────────────────────────
|
|
363
424
|
/**
|
|
364
425
|
* Category 6: decision
|
|
@@ -371,7 +432,7 @@ const DECISION_PATTERNS = [
|
|
|
371
432
|
// Turkish patterns
|
|
372
433
|
/\b(hayır|hayir|evet|böyle|boyle|degil|değil|yerine|kullan)\b/i,
|
|
373
434
|
];
|
|
374
|
-
function
|
|
435
|
+
function extractUserDecision(message) {
|
|
375
436
|
const isDecision = DECISION_PATTERNS.some(p => p.test(message));
|
|
376
437
|
if (!isDecision)
|
|
377
438
|
return [];
|
|
@@ -461,6 +522,8 @@ export function extractEvents(input) {
|
|
|
461
522
|
events.push(...extractSkill(input));
|
|
462
523
|
events.push(...extractSubagent(input));
|
|
463
524
|
events.push(...extractMcp(input));
|
|
525
|
+
events.push(...extractDecision(input));
|
|
526
|
+
events.push(...extractWorktree(input));
|
|
464
527
|
return events;
|
|
465
528
|
}
|
|
466
529
|
catch {
|
|
@@ -477,7 +540,7 @@ export function extractEvents(input) {
|
|
|
477
540
|
export function extractUserEvents(message) {
|
|
478
541
|
try {
|
|
479
542
|
const events = [];
|
|
480
|
-
events.push(...
|
|
543
|
+
events.push(...extractUserDecision(message));
|
|
481
544
|
events.push(...extractRole(message));
|
|
482
545
|
events.push(...extractIntent(message));
|
|
483
546
|
events.push(...extractData(message));
|
|
@@ -50,7 +50,7 @@ export declare function renderDecisions(decisionEvents: StoredEvent[]): string;
|
|
|
50
50
|
*/
|
|
51
51
|
export declare function renderEnvironment(cwdEvent: StoredEvent | undefined, envEvents: StoredEvent[], gitEvent: StoredEvent | undefined): string;
|
|
52
52
|
/**
|
|
53
|
-
* Render <
|
|
53
|
+
* Render <errors_encountered> from error events.
|
|
54
54
|
*/
|
|
55
55
|
export declare function renderErrors(errorEvents: StoredEvent[]): string;
|
|
56
56
|
/**
|
|
@@ -41,8 +41,10 @@ export function renderActiveFiles(fileEvents) {
|
|
|
41
41
|
op = "write";
|
|
42
42
|
else if (ev.type === "file_read")
|
|
43
43
|
op = "read";
|
|
44
|
+
else if (ev.type === "file_edit")
|
|
45
|
+
op = "edit";
|
|
44
46
|
else
|
|
45
|
-
op =
|
|
47
|
+
op = ev.type;
|
|
46
48
|
entry.ops.set(op, (entry.ops.get(op) ?? 0) + 1);
|
|
47
49
|
entry.last = op;
|
|
48
50
|
}
|
|
@@ -173,16 +175,16 @@ export function renderEnvironment(cwdEvent, envEvents, gitEvent) {
|
|
|
173
175
|
return parts.join("\n");
|
|
174
176
|
}
|
|
175
177
|
/**
|
|
176
|
-
* Render <
|
|
178
|
+
* Render <errors_encountered> from error events.
|
|
177
179
|
*/
|
|
178
180
|
export function renderErrors(errorEvents) {
|
|
179
181
|
if (errorEvents.length === 0)
|
|
180
182
|
return "";
|
|
181
|
-
const lines = [" <
|
|
183
|
+
const lines = [" <errors_encountered>"];
|
|
182
184
|
for (const ev of errorEvents) {
|
|
183
185
|
lines.push(` - ${escapeXML(truncateString(ev.data, 150))}`);
|
|
184
186
|
}
|
|
185
|
-
lines.push(" </
|
|
187
|
+
lines.push(" </errors_encountered>");
|
|
186
188
|
return lines.join("\n");
|
|
187
189
|
}
|
|
188
190
|
/**
|
|
@@ -251,10 +253,7 @@ export function buildResumeSnapshot(events, opts) {
|
|
|
251
253
|
const errorEvents = [];
|
|
252
254
|
const envEvents = [];
|
|
253
255
|
const gitEvents = [];
|
|
254
|
-
const skillEvents = [];
|
|
255
256
|
const subagentEvents = [];
|
|
256
|
-
const roleEvents = [];
|
|
257
|
-
const dataEvents = [];
|
|
258
257
|
const intentEvents = [];
|
|
259
258
|
const mcpEvents = [];
|
|
260
259
|
const planEvents = [];
|
|
@@ -284,18 +283,9 @@ export function buildResumeSnapshot(events, opts) {
|
|
|
284
283
|
case "git":
|
|
285
284
|
gitEvents.push(ev);
|
|
286
285
|
break;
|
|
287
|
-
case "skill":
|
|
288
|
-
skillEvents.push(ev);
|
|
289
|
-
break;
|
|
290
286
|
case "subagent":
|
|
291
287
|
subagentEvents.push(ev);
|
|
292
288
|
break;
|
|
293
|
-
case "role":
|
|
294
|
-
roleEvents.push(ev);
|
|
295
|
-
break;
|
|
296
|
-
case "data":
|
|
297
|
-
dataEvents.push(ev);
|
|
298
|
-
break;
|
|
299
289
|
case "intent":
|
|
300
290
|
intentEvents.push(ev);
|
|
301
291
|
break;
|
|
@@ -319,7 +309,7 @@ export function buildResumeSnapshot(events, opts) {
|
|
|
319
309
|
const rules = renderRules(ruleEvents);
|
|
320
310
|
if (rules)
|
|
321
311
|
p1Sections.push(rules);
|
|
322
|
-
// P2 sections (35% budget): decisions, environment,
|
|
312
|
+
// P2 sections (35% budget): decisions, environment, errors_encountered, completed subagents
|
|
323
313
|
const p2Sections = [];
|
|
324
314
|
const decisions = renderDecisions(decisionEvents);
|
|
325
315
|
if (decisions)
|
package/cli.bundle.mjs
CHANGED
|
@@ -282,7 +282,7 @@ async function main() {
|
|
|
282
282
|
main();
|
|
283
283
|
`}async function Rz(){let t=r0();t>0&&console.error(`Cleaned up ${t} stale DB file(s) from previous sessions`);let e=()=>{Rs.cleanupBackgrounded(),co&&co.cleanup()};process.on("exit",e),process.on("SIGINT",()=>{e(),process.exit(0)}),process.on("SIGTERM",()=>{e(),process.exit(0)});let r=new tc;await Yt.connect(r);try{let{detectPlatform:n,getAdapter:o}=await Promise.resolve().then(()=>(Hc(),hh)),s=n(),i=await o(s.platform);if(!i.capabilities.sessionStart){let a=gz(Ps(km(import.meta.url)),".."),c=process.env.CLAUDE_PROJECT_DIR??process.env.CODEX_HOME??process.cwd(),u=i.writeRoutingInstructions(c,a);u&&console.error(`Wrote routing instructions: ${u}`)}}catch{}console.error(`Context Mode MCP server v${y0} running on stdio`),console.error(`Detected runtimes:
|
|
284
284
|
${qs(Tm)}`),po()||(console.error(`
|
|
285
|
-
Performance tip: Install Bun for 3-5x faster JS/TS execution`),console.error(" curl -fsSL https://bun.sh/install | bash"))}var y0,Tm,_z,Yt,Rs,co,Je,bz,$z,Sz,wz,sc,vn,bm,kz,f0,m0,$m,Sm,b0=$(()=>{"use strict";Mb();qb();pm();hm();n0();d0();Fs();p0();y0="1.0.
|
|
285
|
+
Performance tip: Install Bun for 3-5x faster JS/TS execution`),console.error(" curl -fsSL https://bun.sh/install | bash"))}var y0,Tm,_z,Yt,Rs,co,Je,bz,$z,Sz,wz,sc,vn,bm,kz,f0,m0,$m,Sm,b0=$(()=>{"use strict";Mb();qb();pm();hm();n0();d0();Fs();p0();y0="1.0.13";process.on("unhandledRejection",t=>{process.stderr.write(`[context-mode] unhandledRejection: ${t}
|
|
286
286
|
`)});process.on("uncaughtException",t=>{process.stderr.write(`[context-mode] uncaughtException: ${t?.message??t}
|
|
287
287
|
`)});Tm=$n(),_z=Us(Tm),Yt=new Qa({name:"context-mode",version:y0}),Rs=new Es({runtimes:Tm,projectRoot:process.env.CLAUDE_PROJECT_DIR}),co=null;Je={calls:{},bytesReturned:{},bytesIndexed:0,bytesSandboxed:0,sessionStart:Date.now()};bz=_z.join(", "),$z=po()?" (Bun detected \u2014 JS/TS runs 3-5x faster)":"",Sz="",wz="";Yt.registerTool("ctx_execute",{title:"Execute Code",description:`MANDATORY: Use for any command where output exceeds 20 lines. Execute code in a sandboxed subprocess. Only stdout enters context \u2014 raw data stays in the subprocess.${$z} Available: ${bz}.
|
|
288
288
|
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "context-mode",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.14",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "MCP plugin that saves 98% of your context window. Works with Claude Code, Gemini CLI, VS Code Copilot, OpenCode, and Codex CLI. Sandboxed code execution, FTS5 knowledge base, and intent-driven search.",
|
|
6
6
|
"author": "Mert Koseoğlu",
|
package/server.bundle.mjs
CHANGED
|
@@ -227,7 +227,7 @@ stdout:
|
|
|
227
227
|
${n}
|
|
228
228
|
|
|
229
229
|
stderr:
|
|
230
|
-
${o}`}}var lx="1.0.
|
|
230
|
+
${o}`}}var lx="1.0.13";process.on("unhandledRejection",t=>{process.stderr.write(`[context-mode] unhandledRejection: ${t}
|
|
231
231
|
`)});process.on("uncaughtException",t=>{process.stderr.write(`[context-mode] uncaughtException: ${t?.message??t}
|
|
232
232
|
`)});var hp=Ui(),uz=py(hp),Nt=new Zi({name:"context-mode",version:lx}),Wo=new Hi({runtimes:hp,projectRoot:process.env.CLAUDE_PROJECT_DIR}),Dn=null;function lz(t){try{let e=ra(ux(),".claude","context-mode","sessions");if(!cx(e))return;let r=oz(e).filter(n=>n.endsWith("-events.md"));for(let n of r){let o=ra(e,n);try{t.index({path:o,source:"session-events"}),nz(o)}catch{}}}catch{}}function Yo(){return Dn||(Dn=new Vi),lz(Dn),Dn}var qe={calls:{},bytesReturned:{},bytesIndexed:0,bytesSandboxed:0,sessionStart:Date.now()};function J(t,e){let r=e.content.reduce((n,o)=>n+Buffer.byteLength(o.text),0);return qe.calls[t]=(qe.calls[t]||0)+1,qe.bytesReturned[t]=(qe.bytesReturned[t]||0)+r,e}function gr(t){qe.bytesIndexed+=t}function gp(t,e){try{let r=Fd(process.env.CLAUDE_PROJECT_DIR),n=Hd(t,r);if(n.decision==="deny")return J(e,{content:[{type:"text",text:`Command blocked by security policy: matches deny pattern ${n.matchedPattern}`}],isError:!0})}catch{}return null}function dx(t,e,r){try{let n=Iy(t,e);if(n.length===0)return null;let o=Fd(process.env.CLAUDE_PROJECT_DIR);for(let s of n){let i=Hd(s,o);if(i.decision==="deny")return J(r,{content:[{type:"text",text:`Command blocked by security policy: embedded shell command "${s}" matches deny pattern ${i.matchedPattern}`}],isError:!0})}}catch{}return null}function dz(t,e){try{let r=zy("Read",process.env.CLAUDE_PROJECT_DIR),n=Oy(t,r);if(n.denied)return J(e,{content:[{type:"text",text:`File access blocked by security policy: path matches Read deny pattern ${n.matchedPattern}`}],isError:!0})}catch{}return null}var pz=uz.join(", "),fz=Zd()?" (Bun detected \u2014 JS/TS runs 3-5x faster)":"",mz="",hz="";function gz(t){let e=[],r=0,n=0;for(;n<t.length;)if(t[n]===mz){for(e.push(r),n++;n<t.length&&t[n]!==hz;)r++,n++;n<t.length&&n++}else r++,n++;return e}function px(t,e,r=1500,n){if(t.length<=r)return t;let o=[];if(n)for(let u of gz(n))o.push(u);if(o.length===0){let u=e.toLowerCase().split(/\s+/).filter(d=>d.length>2),l=t.toLowerCase();for(let d of u){let f=l.indexOf(d);for(;f!==-1;)o.push(f),f=l.indexOf(d,f+1)}}if(o.length===0)return t.slice(0,r)+`
|
|
233
233
|
\u2026`;o.sort((u,l)=>u-l);let s=300,i=[];for(let u of o){let l=Math.max(0,u-s),d=Math.min(t.length,u+s);i.length>0&&l<=i[i.length-1][1]?i[i.length-1][1]=d:i.push([l,d])}let a=[],c=0;for(let[u,l]of i){if(c>=r)break;let d=t.slice(u,Math.min(l,u+(r-c)));a.push((u>0?"\u2026":"")+d+(l<t.length?"\u2026":"")),c+=d.length}return a.join(`
|