@wolfx/pi-magic-context 0.23.1 → 0.24.0

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/dist/index.js CHANGED
@@ -141961,6 +141961,7 @@ var SESSION_META_SELECT_COLUMNS = [
141961
141961
  "cached_m0_bytes",
141962
141962
  "cached_m1_bytes",
141963
141963
  "cached_m0_project_memory_epoch",
141964
+ "cached_m0_workspace_fingerprint",
141964
141965
  "cached_m0_project_user_profile_version",
141965
141966
  "cached_m0_max_compartment_seq",
141966
141967
  "cached_m0_max_memory_id",
@@ -142008,6 +142009,7 @@ var META_COLUMNS = {
142008
142009
  cachedM0Bytes: "cached_m0_bytes",
142009
142010
  cachedM1Bytes: "cached_m1_bytes",
142010
142011
  cachedM0ProjectMemoryEpoch: "cached_m0_project_memory_epoch",
142012
+ cachedM0WorkspaceFingerprint: "cached_m0_workspace_fingerprint",
142011
142013
  cachedM0ProjectUserProfileVersion: "cached_m0_project_user_profile_version",
142012
142014
  cachedM0MaxCompartmentSeq: "cached_m0_max_compartment_seq",
142013
142015
  cachedM0MaxMemoryId: "cached_m0_max_memory_id",
@@ -142037,6 +142039,7 @@ var NULL_BIND_META_KEYS = new Set([
142037
142039
  "cachedM0Bytes",
142038
142040
  "cachedM1Bytes",
142039
142041
  "cachedM0ProjectMemoryEpoch",
142042
+ "cachedM0WorkspaceFingerprint",
142040
142043
  "cachedM0ProjectUserProfileVersion",
142041
142044
  "cachedM0MaxCompartmentSeq",
142042
142045
  "cachedM0MaxMemoryId",
@@ -142070,7 +142073,7 @@ function isSessionMetaRow(row) {
142070
142073
  if (row === null || typeof row !== "object")
142071
142074
  return false;
142072
142075
  const r = row;
142073
- return typeof r.session_id === "string" && typeof r.last_response_time === "number" && isStringOrNull(r.cache_ttl) && typeof r.counter === "number" && typeof r.last_nudge_tokens === "number" && isStringOrNull(r.last_nudge_band) && isStringOrNull(r.last_transform_error) && typeof r.is_subagent === "number" && typeof r.last_context_percentage === "number" && typeof r.last_input_tokens === "number" && isNumberOrNull(r.observed_safe_input_tokens) && isNumberOrNull(r.cache_alert_sent) && isNumberOrNull(r.times_execute_threshold_reached) && isNumberOrNull(r.compartment_in_progress) && (r.system_prompt_hash === null || typeof r.system_prompt_hash === "string" || typeof r.system_prompt_hash === "number") && isNumberOrNull(r.system_prompt_tokens) && isNumberOrNull(r.conversation_tokens) && isNumberOrNull(r.tool_call_tokens) && isNumberOrNull(r.cleared_reasoning_through_tag) && isStringOrNull(r.last_todo_state) && isBlobOrNull(r.cached_m0_bytes) && isBlobOrNull(r.cached_m1_bytes) && isNumberOrNull(r.cached_m0_project_memory_epoch) && isNumberOrNull(r.cached_m0_project_user_profile_version) && isNumberOrNull(r.cached_m0_max_compartment_seq) && isNumberOrNull(r.cached_m0_max_memory_id) && isNumberOrNull(r.cached_m0_max_mutation_id) && isNumberOrNull(r.cached_m0_max_memory_mutation_id) && isStringOrNull(r.cached_m0_project_docs_hash) && isNumberOrNull(r.cached_m0_materialized_at) && isNumberOrNull(r.cached_m0_session_facts_version) && isStringOrNull(r.cached_m0_upgrade_state) && isStringOrNull(r.cached_m0_system_hash) && isStringOrNull(r.cached_m0_tool_set_hash) && isStringOrNull(r.cached_m0_model_key) && isStringOrNull(r.last_observed_model_key) && isNumberOrNull(r.last_usage_context_limit) && isNumberOrNull(r.prior_boundary_ordinal) && isNumberOrNull(r.protected_tail_policy_version) && isNumberOrNull(r.protected_tail_drain_window_started_at) && isNumberOrNull(r.protected_tail_drain_tokens) && isNumberOrNull(r.recovery_no_eligible_head_count) && isNumberOrNull(r.force_emergency_bypass_window_start) && isNumberOrNull(r.force_emergency_bypass_used) && isNumberOrNull(r.upgrade_reminded_at) && isNumberOrNull(r.pi_stable_id_scheme);
142076
+ return typeof r.session_id === "string" && typeof r.last_response_time === "number" && isStringOrNull(r.cache_ttl) && typeof r.counter === "number" && typeof r.last_nudge_tokens === "number" && isStringOrNull(r.last_nudge_band) && isStringOrNull(r.last_transform_error) && typeof r.is_subagent === "number" && typeof r.last_context_percentage === "number" && typeof r.last_input_tokens === "number" && isNumberOrNull(r.observed_safe_input_tokens) && isNumberOrNull(r.cache_alert_sent) && isNumberOrNull(r.times_execute_threshold_reached) && isNumberOrNull(r.compartment_in_progress) && (r.system_prompt_hash === null || typeof r.system_prompt_hash === "string" || typeof r.system_prompt_hash === "number") && isNumberOrNull(r.system_prompt_tokens) && isNumberOrNull(r.conversation_tokens) && isNumberOrNull(r.tool_call_tokens) && isNumberOrNull(r.cleared_reasoning_through_tag) && isStringOrNull(r.last_todo_state) && isBlobOrNull(r.cached_m0_bytes) && isBlobOrNull(r.cached_m1_bytes) && isNumberOrNull(r.cached_m0_project_memory_epoch) && isStringOrNull(r.cached_m0_workspace_fingerprint) && isNumberOrNull(r.cached_m0_project_user_profile_version) && isNumberOrNull(r.cached_m0_max_compartment_seq) && isNumberOrNull(r.cached_m0_max_memory_id) && isNumberOrNull(r.cached_m0_max_mutation_id) && isNumberOrNull(r.cached_m0_max_memory_mutation_id) && isStringOrNull(r.cached_m0_project_docs_hash) && isNumberOrNull(r.cached_m0_materialized_at) && isNumberOrNull(r.cached_m0_session_facts_version) && isStringOrNull(r.cached_m0_upgrade_state) && isStringOrNull(r.cached_m0_system_hash) && isStringOrNull(r.cached_m0_tool_set_hash) && isStringOrNull(r.cached_m0_model_key) && isStringOrNull(r.last_observed_model_key) && isNumberOrNull(r.last_usage_context_limit) && isNumberOrNull(r.prior_boundary_ordinal) && isNumberOrNull(r.protected_tail_policy_version) && isNumberOrNull(r.protected_tail_drain_window_started_at) && isNumberOrNull(r.protected_tail_drain_tokens) && isNumberOrNull(r.recovery_no_eligible_head_count) && isNumberOrNull(r.force_emergency_bypass_window_start) && isNumberOrNull(r.force_emergency_bypass_used) && isNumberOrNull(r.upgrade_reminded_at) && isNumberOrNull(r.pi_stable_id_scheme);
142074
142077
  }
142075
142078
  function getDefaultSessionMeta(sessionId) {
142076
142079
  return {
@@ -142097,6 +142100,7 @@ function getDefaultSessionMeta(sessionId) {
142097
142100
  cachedM0Bytes: null,
142098
142101
  cachedM1Bytes: null,
142099
142102
  cachedM0ProjectMemoryEpoch: null,
142103
+ cachedM0WorkspaceFingerprint: null,
142100
142104
  cachedM0ProjectUserProfileVersion: null,
142101
142105
  cachedM0MaxCompartmentSeq: null,
142102
142106
  cachedM0MaxMemoryId: null,
@@ -142159,6 +142163,7 @@ function toSessionMeta(row) {
142159
142163
  cachedM0Bytes: toBufferOrNull(row.cached_m0_bytes),
142160
142164
  cachedM1Bytes: toBufferOrNull(row.cached_m1_bytes),
142161
142165
  cachedM0ProjectMemoryEpoch: numOrNull(row.cached_m0_project_memory_epoch),
142166
+ cachedM0WorkspaceFingerprint: stringOrNull(row.cached_m0_workspace_fingerprint),
142162
142167
  cachedM0ProjectUserProfileVersion: numOrNull(row.cached_m0_project_user_profile_version),
142163
142168
  cachedM0MaxCompartmentSeq: numOrNull(row.cached_m0_max_compartment_seq),
142164
142169
  cachedM0MaxMemoryId: numOrNull(row.cached_m0_max_memory_id),
@@ -142189,6 +142194,7 @@ function persistCachedM0(db, sessionId, payload) {
142189
142194
  db.prepare(`UPDATE session_meta SET
142190
142195
  cached_m0_bytes = ?,
142191
142196
  cached_m0_project_memory_epoch = ?,
142197
+ cached_m0_workspace_fingerprint = ?,
142192
142198
  cached_m0_project_user_profile_version = ?,
142193
142199
  cached_m0_max_compartment_seq = ?,
142194
142200
  cached_m0_max_memory_id = ?,
@@ -142201,7 +142207,7 @@ function persistCachedM0(db, sessionId, payload) {
142201
142207
  cached_m0_upgrade_state = ?,
142202
142208
  cached_m0_system_hash = ?,
142203
142209
  cached_m0_model_key = ?
142204
- WHERE session_id = ?`).run(Buffer2.from(payload.m0Bytes), payload.projectMemoryEpoch, payload.projectUserProfileVersion, payload.maxCompartmentSeq, payload.maxMemoryId, payload.maxMutationId, payload.maxMemoryMutationId ?? null, payload.m1Bytes ? Buffer2.from(payload.m1Bytes) : null, payload.projectDocsHash, payload.materializedAt, payload.sessionFactsVersion, payload.upgradeState, payload.systemHash ?? "", payload.modelKey ?? "", sessionId);
142210
+ WHERE session_id = ?`).run(Buffer2.from(payload.m0Bytes), payload.projectMemoryEpoch, payload.workspaceFingerprint ?? null, payload.projectUserProfileVersion, payload.maxCompartmentSeq, payload.maxMemoryId, payload.maxMutationId, payload.maxMemoryMutationId ?? null, payload.m1Bytes ? Buffer2.from(payload.m1Bytes) : null, payload.projectDocsHash, payload.materializedAt, payload.sessionFactsVersion, payload.upgradeState, payload.systemHash ?? "", payload.modelKey ?? "", sessionId);
142205
142211
  }
142206
142212
  function clearCachedM0M1(db, sessionId) {
142207
142213
  ensureSessionMetaRow(db, sessionId);
@@ -142210,6 +142216,7 @@ function clearCachedM0M1(db, sessionId) {
142210
142216
  ["cached_m0_bytes", null],
142211
142217
  ["cached_m1_bytes", null],
142212
142218
  ["cached_m0_project_memory_epoch", null],
142219
+ ["cached_m0_workspace_fingerprint", null],
142213
142220
  ["cached_m0_project_user_profile_version", null],
142214
142221
  ["cached_m0_max_compartment_seq", null],
142215
142222
  ["cached_m0_max_memory_id", null],
@@ -142924,7 +142931,7 @@ var databases = new Map;
142924
142931
  var persistenceByDatabase = new WeakMap;
142925
142932
  var persistenceErrorByDatabase = new WeakMap;
142926
142933
  var lastSchemaFenceRejection = null;
142927
- var LATEST_SUPPORTED_VERSION = 32;
142934
+ var LATEST_SUPPORTED_VERSION = 36;
142928
142935
  function resolveDatabasePath(dbPathOverride) {
142929
142936
  if (dbPathOverride) {
142930
142937
  return { dbDir: dirname2(dbPathOverride), dbPath: dbPathOverride };
@@ -143090,6 +143097,35 @@ function initializeDatabase(db) {
143090
143097
  );
143091
143098
  CREATE INDEX IF NOT EXISTS idx_compartments_session ON compartments(session_id);
143092
143099
 
143100
+ CREATE TABLE IF NOT EXISTS compartment_chunk_embeddings (
143101
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
143102
+ compartment_id INTEGER NOT NULL REFERENCES compartments(id) ON DELETE CASCADE,
143103
+ session_id TEXT NOT NULL,
143104
+ project_path TEXT NOT NULL,
143105
+ harness TEXT NOT NULL DEFAULT 'opencode',
143106
+ window_index INTEGER NOT NULL DEFAULT 0,
143107
+ start_ordinal INTEGER NOT NULL,
143108
+ end_ordinal INTEGER NOT NULL,
143109
+ chunk_hash TEXT NOT NULL,
143110
+ model_id TEXT NOT NULL,
143111
+ dims INTEGER NOT NULL,
143112
+ vector BLOB NOT NULL,
143113
+ created_at INTEGER NOT NULL,
143114
+ UNIQUE(compartment_id, window_index)
143115
+ );
143116
+ CREATE INDEX IF NOT EXISTS idx_cce_session ON compartment_chunk_embeddings(session_id);
143117
+ CREATE INDEX IF NOT EXISTS idx_cce_project_model ON compartment_chunk_embeddings(project_path, model_id);
143118
+
143119
+ CREATE TABLE IF NOT EXISTS session_projects (
143120
+ session_id TEXT NOT NULL,
143121
+ harness TEXT NOT NULL DEFAULT 'opencode',
143122
+ project_path TEXT NOT NULL,
143123
+ updated_at INTEGER NOT NULL,
143124
+ PRIMARY KEY(session_id, harness)
143125
+ );
143126
+ CREATE INDEX IF NOT EXISTS idx_session_projects_project
143127
+ ON session_projects(project_path);
143128
+
143093
143129
  CREATE TABLE IF NOT EXISTS compartment_events (
143094
143130
  id INTEGER PRIMARY KEY AUTOINCREMENT,
143095
143131
  session_id TEXT NOT NULL,
@@ -143275,6 +143311,25 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143275
143311
  rekeyed_at INTEGER NOT NULL
143276
143312
  );
143277
143313
 
143314
+ CREATE TABLE IF NOT EXISTS workspaces (
143315
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
143316
+ name TEXT NOT NULL UNIQUE,
143317
+ created_at INTEGER NOT NULL,
143318
+ updated_at INTEGER NOT NULL,
143319
+ share_categories TEXT NOT NULL DEFAULT '["CONSTRAINTS"]'
143320
+ );
143321
+
143322
+ CREATE TABLE IF NOT EXISTS workspace_members (
143323
+ workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE,
143324
+ project_path TEXT NOT NULL,
143325
+ display_name TEXT NOT NULL,
143326
+ display_path TEXT NOT NULL,
143327
+ added_at INTEGER NOT NULL,
143328
+ PRIMARY KEY (workspace_id, project_path)
143329
+ );
143330
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_unique ON workspace_members(project_path);
143331
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_name ON workspace_members(workspace_id, display_name);
143332
+
143278
143333
  CREATE TABLE IF NOT EXISTS v22_backfill_failures (
143279
143334
  id INTEGER PRIMARY KEY AUTOINCREMENT,
143280
143335
  table_name TEXT NOT NULL,
@@ -143386,6 +143441,7 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143386
143441
  deferred_execute_state TEXT,
143387
143442
  cached_m0_bytes BLOB,
143388
143443
  cached_m0_project_memory_epoch INTEGER,
143444
+ cached_m0_workspace_fingerprint TEXT,
143389
143445
  cached_m0_project_user_profile_version INTEGER,
143390
143446
  cached_m0_max_compartment_seq INTEGER,
143391
143447
  cached_m0_max_memory_id INTEGER,
@@ -143544,6 +143600,7 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143544
143600
  ensureColumn(db, "session_meta", "cleared_reasoning_through_tag", "INTEGER DEFAULT 0");
143545
143601
  ensureColumn(db, "session_meta", "stripped_placeholder_ids", "TEXT DEFAULT ''");
143546
143602
  ensureColumn(db, "session_meta", "stale_reduce_stripped_ids", "TEXT DEFAULT ''");
143603
+ ensureColumn(db, "session_meta", "processed_image_stripped_ids", "TEXT DEFAULT ''");
143547
143604
  ensureColumn(db, "compartments", "start_message_id", "TEXT DEFAULT ''");
143548
143605
  ensureColumn(db, "compartments", "end_message_id", "TEXT DEFAULT ''");
143549
143606
  ensureColumn(db, "memory_embeddings", "model_id", "TEXT");
@@ -143597,6 +143654,7 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143597
143654
  ensureColumn(db, "memories", "importance", "INTEGER");
143598
143655
  ensureColumn(db, "session_meta", "cached_m0_bytes", "BLOB");
143599
143656
  ensureColumn(db, "session_meta", "cached_m0_project_memory_epoch", "INTEGER");
143657
+ ensureColumn(db, "session_meta", "cached_m0_workspace_fingerprint", "TEXT");
143600
143658
  ensureColumn(db, "session_meta", "cached_m0_project_user_profile_version", "INTEGER");
143601
143659
  ensureColumn(db, "session_meta", "cached_m0_max_compartment_seq", "INTEGER");
143602
143660
  ensureColumn(db, "session_meta", "cached_m0_max_memory_id", "INTEGER");
@@ -143628,6 +143686,15 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143628
143686
  project_user_profile_version INTEGER NOT NULL DEFAULT 0,
143629
143687
  updated_at INTEGER NOT NULL DEFAULT 0
143630
143688
  );
143689
+ CREATE TABLE IF NOT EXISTS session_projects (
143690
+ session_id TEXT NOT NULL,
143691
+ harness TEXT NOT NULL DEFAULT 'opencode',
143692
+ project_path TEXT NOT NULL,
143693
+ updated_at INTEGER NOT NULL,
143694
+ PRIMARY KEY(session_id, harness)
143695
+ );
143696
+ CREATE INDEX IF NOT EXISTS idx_session_projects_project
143697
+ ON session_projects(project_path);
143631
143698
  CREATE TABLE IF NOT EXISTS m0_mutation_log (
143632
143699
  id INTEGER PRIMARY KEY AUTOINCREMENT,
143633
143700
  session_id TEXT NOT NULL,
@@ -143655,6 +143722,23 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143655
143722
  new_project_path TEXT NOT NULL,
143656
143723
  rekeyed_at INTEGER NOT NULL
143657
143724
  );
143725
+ CREATE TABLE IF NOT EXISTS workspaces (
143726
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
143727
+ name TEXT NOT NULL UNIQUE,
143728
+ created_at INTEGER NOT NULL,
143729
+ updated_at INTEGER NOT NULL,
143730
+ share_categories TEXT NOT NULL DEFAULT '["CONSTRAINTS"]'
143731
+ );
143732
+ CREATE TABLE IF NOT EXISTS workspace_members (
143733
+ workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE,
143734
+ project_path TEXT NOT NULL,
143735
+ display_name TEXT NOT NULL,
143736
+ display_path TEXT NOT NULL,
143737
+ added_at INTEGER NOT NULL,
143738
+ PRIMARY KEY (workspace_id, project_path)
143739
+ );
143740
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_unique ON workspace_members(project_path);
143741
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_name ON workspace_members(workspace_id, display_name);
143658
143742
  CREATE TABLE IF NOT EXISTS v22_backfill_failures (
143659
143743
  id INTEGER PRIMARY KEY AUTOINCREMENT,
143660
143744
  table_name TEXT NOT NULL,
@@ -143676,6 +143760,7 @@ CREATE INDEX IF NOT EXISTS idx_dream_queue_pending ON dream_queue(started_at, en
143676
143760
  ensureColumn(db, "recomp_compartments", "harness", "TEXT NOT NULL DEFAULT 'opencode'");
143677
143761
  ensureColumn(db, "recomp_facts", "harness", "TEXT NOT NULL DEFAULT 'opencode'");
143678
143762
  ensureColumn(db, "message_history_index", "harness", "TEXT NOT NULL DEFAULT 'opencode'");
143763
+ ensureColumn(db, "workspaces", "share_categories", `TEXT NOT NULL DEFAULT '["CONSTRAINTS"]'`);
143679
143764
  }
143680
143765
  function healAllNullColumns(db) {
143681
143766
  healNullTextColumns(db);
@@ -143714,6 +143799,7 @@ function healNullTextColumns(db) {
143714
143799
  ["system_prompt_hash", ""],
143715
143800
  ["stripped_placeholder_ids", ""],
143716
143801
  ["stale_reduce_stripped_ids", ""],
143802
+ ["processed_image_stripped_ids", ""],
143717
143803
  ["memory_block_cache", ""],
143718
143804
  ["memory_block_ids", ""],
143719
143805
  ["compaction_marker_state", ""],
@@ -143758,7 +143844,7 @@ function healNullIntegerColumns(db) {
143758
143844
  }
143759
143845
  }
143760
143846
  function ensureColumn(db, table, column, definition) {
143761
- if (!/^[a-z][a-z0-9_]*$/.test(table) || !/^[a-z][a-z0-9_]*$/.test(column) || !/^[A-Z0-9_'(),[\]\s]+$/i.test(definition)) {
143847
+ if (!/^[a-z][a-z0-9_]*$/.test(table) || !/^[a-z][a-z0-9_]*$/.test(column) || !/^[A-Z0-9_"'(),[\]\s]+$/i.test(definition)) {
143762
143848
  throw new Error(`Unsafe schema identifier: ${table}.${column} ${definition}`);
143763
143849
  }
143764
143850
  const rows = db.prepare(`PRAGMA table_info(${table})`).all();
@@ -143835,10 +143921,291 @@ function openDatabase(dbPathOrOptions) {
143835
143921
  }
143836
143922
  }
143837
143923
 
143924
+ // ../plugin/src/features/magic-context/workspaces.ts
143925
+ import { createHash as createHash3 } from "node:crypto";
143926
+
143927
+ // ../plugin/src/features/magic-context/memory/constants.ts
143928
+ var V2_MEMORY_CATEGORIES = [
143929
+ "PROJECT_RULES",
143930
+ "ARCHITECTURE",
143931
+ "CONSTRAINTS",
143932
+ "CONFIG_VALUES",
143933
+ "NAMING"
143934
+ ];
143935
+ var PROMOTABLE_CATEGORIES = [
143936
+ "PROJECT_RULES",
143937
+ "ARCHITECTURE",
143938
+ "CONSTRAINTS",
143939
+ "CONFIG_VALUES",
143940
+ "NAMING",
143941
+ "ARCHITECTURE_DECISIONS",
143942
+ "CONFIG_DEFAULTS",
143943
+ "USER_PREFERENCES",
143944
+ "USER_DIRECTIVES",
143945
+ "ENVIRONMENT",
143946
+ "WORKFLOW_RULES",
143947
+ "KNOWN_ISSUES"
143948
+ ];
143949
+ var CATEGORY_PRIORITY = [
143950
+ "PROJECT_RULES",
143951
+ "ARCHITECTURE",
143952
+ "CONSTRAINTS",
143953
+ "CONFIG_VALUES",
143954
+ "NAMING",
143955
+ "USER_DIRECTIVES",
143956
+ "USER_PREFERENCES",
143957
+ "CONFIG_DEFAULTS",
143958
+ "ARCHITECTURE_DECISIONS",
143959
+ "ENVIRONMENT",
143960
+ "WORKFLOW_RULES",
143961
+ "KNOWN_ISSUES"
143962
+ ];
143963
+ var MEMORY_CATEGORY_ORDER_UNKNOWN = 99;
143964
+ var MEMORY_CATEGORY_ORDER_PRIORITY = CATEGORY_PRIORITY.reduce((acc, category, index) => {
143965
+ acc[category] = index;
143966
+ return acc;
143967
+ }, {});
143968
+ var MEMORY_CATEGORY_ORDER_SQL = `CASE category ${CATEGORY_PRIORITY.map((category, index) => `WHEN '${category}' THEN ${index}`).join(" ")} ELSE ${MEMORY_CATEGORY_ORDER_UNKNOWN} END`;
143969
+ var CATEGORY_DEFAULT_TTL = {
143970
+ WORKFLOW_RULES: 90 * 24 * 60 * 60 * 1000,
143971
+ KNOWN_ISSUES: 30 * 24 * 60 * 60 * 1000
143972
+ };
143973
+ // ../plugin/src/features/magic-context/workspaces.ts
143974
+ var VALID_SHARE_CATEGORIES = new Set(V2_MEMORY_CATEGORIES);
143975
+ function tableExists(db, tableName) {
143976
+ const row = db.prepare("SELECT 1 FROM sqlite_master WHERE type='table' AND name = ? LIMIT 1").get(tableName);
143977
+ return Boolean(row);
143978
+ }
143979
+ function columnExists(db, tableName, columnName) {
143980
+ const rows = db.prepare(`PRAGMA table_info(${tableName})`).all();
143981
+ return rows.some((row) => row.name === columnName);
143982
+ }
143983
+ function uniqueSorted(values) {
143984
+ return [...new Set(values)].sort((left, right) => left.localeCompare(right));
143985
+ }
143986
+ function placeholders(values) {
143987
+ return values.map(() => "?").join(", ");
143988
+ }
143989
+ function normalizeShareCategories(raw) {
143990
+ if (raw === null || raw === undefined)
143991
+ return null;
143992
+ if (typeof raw !== "string")
143993
+ return null;
143994
+ let parsed;
143995
+ try {
143996
+ parsed = JSON.parse(raw);
143997
+ } catch {
143998
+ return null;
143999
+ }
144000
+ if (!Array.isArray(parsed))
144001
+ return null;
144002
+ const categories = [];
144003
+ for (const value of parsed) {
144004
+ if (typeof value !== "string" || !VALID_SHARE_CATEGORIES.has(value)) {
144005
+ return null;
144006
+ }
144007
+ if (!categories.includes(value))
144008
+ categories.push(value);
144009
+ }
144010
+ return categories.sort((left, right) => left.localeCompare(right));
144011
+ }
144012
+ function selectWorkspaceShareCategories(db, identities) {
144013
+ const candidates = uniqueSorted(identities.filter((identity) => identity.length > 0));
144014
+ if (candidates.length === 0 || !tableExists(db, "workspace_members") || !tableExists(db, "workspaces") || !columnExists(db, "workspaces", "share_categories")) {
144015
+ return null;
144016
+ }
144017
+ const row = db.prepare(`SELECT workspace.share_categories AS shareCategories
144018
+ FROM workspace_members AS member
144019
+ JOIN workspaces AS workspace ON workspace.id = member.workspace_id
144020
+ WHERE member.project_path IN (${placeholders(candidates)})
144021
+ ORDER BY workspace.id ASC
144022
+ LIMIT 1`).get(...candidates);
144023
+ return normalizeShareCategories(row?.shareCategories ?? null);
144024
+ }
144025
+ function resolveWorkspaceShareCategories(db, projectIdentity) {
144026
+ return selectWorkspaceShareCategories(db, [projectIdentity]);
144027
+ }
144028
+ function resolveWorkspaceIdentitySet(db, projectIdentity) {
144029
+ if (!tableExists(db, "workspace_members")) {
144030
+ return { identities: [projectIdentity], namesByIdentity: new Map };
144031
+ }
144032
+ const rows = db.prepare(`SELECT member.project_path AS identity, member.display_name AS displayName
144033
+ FROM workspace_members AS anchor
144034
+ JOIN workspace_members AS member ON member.workspace_id = anchor.workspace_id
144035
+ WHERE anchor.project_path = ?
144036
+ ORDER BY member.display_name ASC, member.project_path ASC`).all(projectIdentity);
144037
+ if (rows.length === 0) {
144038
+ return { identities: [projectIdentity], namesByIdentity: new Map };
144039
+ }
144040
+ const namesByIdentity = new Map;
144041
+ const identities = [];
144042
+ for (const row of rows) {
144043
+ if (typeof row.identity !== "string" || row.identity.length === 0)
144044
+ continue;
144045
+ if (identities.includes(row.identity))
144046
+ continue;
144047
+ identities.push(row.identity);
144048
+ if (typeof row.displayName === "string" && row.displayName.length > 0) {
144049
+ namesByIdentity.set(row.identity, row.displayName);
144050
+ }
144051
+ }
144052
+ return identities.length > 0 ? { identities, namesByIdentity } : { identities: [projectIdentity], namesByIdentity: new Map };
144053
+ }
144054
+ function expandWorkspaceIdentitySetWithAliases(db, identities) {
144055
+ const canonical = uniqueSorted(identities.filter((identity) => identity.length > 0));
144056
+ const expanded = new Set(canonical);
144057
+ const canonicalIdentityByStoredPath = new Map;
144058
+ for (const identity of canonical) {
144059
+ canonicalIdentityByStoredPath.set(identity, identity);
144060
+ }
144061
+ if (canonical.length === 0 || !tableExists(db, "v22_identity_rekey_map")) {
144062
+ return { expandedIdentities: [...expanded], canonicalIdentityByStoredPath };
144063
+ }
144064
+ const rows = db.prepare(`SELECT old_project_path AS oldProjectPath, new_project_path AS newProjectPath
144065
+ FROM v22_identity_rekey_map
144066
+ WHERE new_project_path IN (${placeholders(canonical)})
144067
+ ORDER BY old_project_path ASC`).all(...canonical);
144068
+ for (const row of rows) {
144069
+ if (typeof row.oldProjectPath !== "string" || typeof row.newProjectPath !== "string") {
144070
+ continue;
144071
+ }
144072
+ if (!canonicalIdentityByStoredPath.has(row.newProjectPath))
144073
+ continue;
144074
+ expanded.add(row.oldProjectPath);
144075
+ canonicalIdentityByStoredPath.set(row.oldProjectPath, row.newProjectPath);
144076
+ }
144077
+ return { expandedIdentities: [...expanded], canonicalIdentityByStoredPath };
144078
+ }
144079
+ function resolveStoredPathWorkspaceIdentity(storedProjectPath, memberIdentities, canonicalIdentityByStoredPath) {
144080
+ const direct = canonicalIdentityByStoredPath.get(storedProjectPath);
144081
+ if (direct)
144082
+ return direct;
144083
+ const normalized = normalizeStoredProjectPath(storedProjectPath);
144084
+ const normalizedDirect = canonicalIdentityByStoredPath.get(normalized);
144085
+ if (normalizedDirect)
144086
+ return normalizedDirect;
144087
+ if (memberIdentities.includes(normalized))
144088
+ return normalized;
144089
+ for (const identity of memberIdentities) {
144090
+ if (storedPathBelongsToIdentity(storedProjectPath, identity)) {
144091
+ return identity;
144092
+ }
144093
+ }
144094
+ return null;
144095
+ }
144096
+ function storedPathBelongsToWorkspace(storedProjectPath, memberIdentities, expandedIdentities, canonicalIdentityByStoredPath) {
144097
+ if (expandedIdentities.includes(storedProjectPath))
144098
+ return true;
144099
+ return resolveStoredPathWorkspaceIdentity(storedProjectPath, memberIdentities, canonicalIdentityByStoredPath) !== null;
144100
+ }
144101
+ function sourceNameForMemory(storedProjectPath, ownIdentity, memberIdentities, namesByIdentity, canonicalIdentityByStoredPath) {
144102
+ const canonicalIdentity = resolveStoredPathWorkspaceIdentity(storedProjectPath, memberIdentities, canonicalIdentityByStoredPath);
144103
+ if (!canonicalIdentity || canonicalIdentity === ownIdentity)
144104
+ return;
144105
+ return namesByIdentity.get(canonicalIdentity);
144106
+ }
144107
+ function getEpochMap(db, identities) {
144108
+ if (identities.length === 0)
144109
+ return new Map;
144110
+ const rows = db.prepare(`SELECT project_path AS projectPath, project_memory_epoch AS epoch
144111
+ FROM project_state
144112
+ WHERE project_path IN (${placeholders(identities)})`).all(...identities);
144113
+ const epochs = new Map;
144114
+ for (const row of rows) {
144115
+ if (typeof row.projectPath !== "string" || typeof row.epoch !== "number")
144116
+ continue;
144117
+ epochs.set(row.projectPath, row.epoch);
144118
+ }
144119
+ return epochs;
144120
+ }
144121
+ function computeWorkspaceEpochFingerprint(db, identities) {
144122
+ const canonical = uniqueSorted(identities.filter((identity) => identity.length > 0));
144123
+ const epochs = getEpochMap(db, canonical);
144124
+ const shareCategories = selectWorkspaceShareCategories(db, canonical);
144125
+ const hash = createHash3("sha256");
144126
+ hash.update("share_categories", "utf8");
144127
+ hash.update("\x00");
144128
+ hash.update(shareCategories === null ? "ALL" : JSON.stringify(shareCategories), "utf8");
144129
+ hash.update(`
144130
+ `);
144131
+ for (const identity of canonical) {
144132
+ hash.update(identity, "utf8");
144133
+ hash.update("\x00");
144134
+ hash.update(String(epochs.get(identity) ?? 0), "utf8");
144135
+ hash.update(`
144136
+ `);
144137
+ }
144138
+ return hash.digest("hex");
144139
+ }
144140
+ function isInTransaction(db) {
144141
+ const candidate = db;
144142
+ return candidate.inTransaction === true || candidate.isTransaction === true;
144143
+ }
144144
+ function workspaceMembersForIdentity(db, identity) {
144145
+ if (!tableExists(db, "workspace_members"))
144146
+ return [identity];
144147
+ const rows = db.prepare(`SELECT member.project_path AS identity
144148
+ FROM workspace_members AS anchor
144149
+ JOIN workspace_members AS member ON member.workspace_id = anchor.workspace_id
144150
+ WHERE anchor.project_path = ?
144151
+ ORDER BY member.project_path ASC`).all(identity);
144152
+ const identities = rows.map((row) => typeof row.identity === "string" ? row.identity : "").filter((value) => value.length > 0);
144153
+ return identities.length > 0 ? uniqueSorted(identities) : [identity];
144154
+ }
144155
+ function bumpEpochRows(db, identities, now) {
144156
+ const stmt = db.prepare(`INSERT INTO project_state
144157
+ (project_path, project_memory_epoch, project_user_profile_version, updated_at)
144158
+ VALUES (?, 1, 0, ?)
144159
+ ON CONFLICT(project_path) DO UPDATE SET
144160
+ project_memory_epoch = project_memory_epoch + 1,
144161
+ updated_at = excluded.updated_at`);
144162
+ for (const identity of uniqueSorted(identities)) {
144163
+ stmt.run(identity, now);
144164
+ }
144165
+ }
144166
+ function bumpEpochsForWorkspaceMembers(db, identity, now = Date.now()) {
144167
+ const run = () => bumpEpochRows(db, workspaceMembersForIdentity(db, identity), now);
144168
+ if (isInTransaction(db)) {
144169
+ run();
144170
+ return;
144171
+ }
144172
+ db.exec("BEGIN IMMEDIATE");
144173
+ try {
144174
+ run();
144175
+ db.exec("COMMIT");
144176
+ } catch (error) {
144177
+ try {
144178
+ db.exec("ROLLBACK");
144179
+ } catch {}
144180
+ throw error;
144181
+ }
144182
+ }
144183
+ function bumpEpochsForWorkspaceMemberSet(db, identities, now = Date.now()) {
144184
+ const run = () => bumpEpochRows(db, identities, now);
144185
+ if (isInTransaction(db)) {
144186
+ run();
144187
+ return;
144188
+ }
144189
+ db.exec("BEGIN IMMEDIATE");
144190
+ try {
144191
+ run();
144192
+ db.exec("COMMIT");
144193
+ } catch (error) {
144194
+ try {
144195
+ db.exec("ROLLBACK");
144196
+ } catch {}
144197
+ throw error;
144198
+ }
144199
+ }
144200
+
143838
144201
  // ../plugin/src/features/magic-context/migrations.ts
143839
- function tableExists(db, name2) {
144202
+ function tableExists2(db, name2) {
143840
144203
  return Boolean(db.prepare("SELECT name FROM sqlite_master WHERE type='table' AND name = ?").get(name2));
143841
144204
  }
144205
+ function columnExists2(db, table, column) {
144206
+ const rows = db.prepare(`PRAGMA table_info(${table})`).all();
144207
+ return rows.some((row) => row.name === column);
144208
+ }
143842
144209
  var MIGRATIONS = [
143843
144210
  {
143844
144211
  version: 1,
@@ -144297,9 +144664,9 @@ var MIGRATIONS = [
144297
144664
  version: 22,
144298
144665
  description: "v2.0 cache architecture schema foundation",
144299
144666
  up: (db) => {
144300
- const hasSessionMetaTable = tableExists(db, "session_meta");
144301
- const hasCompartmentsTable = tableExists(db, "compartments");
144302
- const hasMemoriesTable = tableExists(db, "memories");
144667
+ const hasSessionMetaTable = tableExists2(db, "session_meta");
144668
+ const hasCompartmentsTable = tableExists2(db, "compartments");
144669
+ const hasMemoriesTable = tableExists2(db, "memories");
144303
144670
  if (hasSessionMetaTable) {
144304
144671
  ensureColumn(db, "session_meta", "cached_m0_bytes", "BLOB");
144305
144672
  ensureColumn(db, "session_meta", "cached_m0_project_memory_epoch", "INTEGER");
@@ -144324,7 +144691,7 @@ var MIGRATIONS = [
144324
144691
  ensureColumn(db, "compartments", "p1_embedding_model_id", "TEXT");
144325
144692
  ensureColumn(db, "compartments", "legacy", "INTEGER NOT NULL DEFAULT 0");
144326
144693
  }
144327
- const hasRecompCompartmentsTable = tableExists(db, "recomp_compartments");
144694
+ const hasRecompCompartmentsTable = tableExists2(db, "recomp_compartments");
144328
144695
  if (hasRecompCompartmentsTable) {
144329
144696
  ensureColumn(db, "recomp_compartments", "p1", "TEXT");
144330
144697
  ensureColumn(db, "recomp_compartments", "p2", "TEXT");
@@ -144635,6 +145002,156 @@ var MIGRATIONS = [
144635
145002
  db.prepare("UPDATE session_meta SET force_emergency_bypass_used = 0 WHERE force_emergency_bypass_used IS NULL").run();
144636
145003
  db.prepare("UPDATE session_meta SET last_usage_context_limit = 0 WHERE last_usage_context_limit IS NULL").run();
144637
145004
  }
145005
+ },
145006
+ {
145007
+ version: 33,
145008
+ description: "Compartment chunk embeddings for semantic message-history search",
145009
+ up: (db) => {
145010
+ db.exec(`
145011
+ CREATE TABLE IF NOT EXISTS compartment_chunk_embeddings (
145012
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
145013
+ compartment_id INTEGER NOT NULL REFERENCES compartments(id) ON DELETE CASCADE,
145014
+ session_id TEXT NOT NULL,
145015
+ project_path TEXT NOT NULL,
145016
+ harness TEXT NOT NULL DEFAULT 'opencode',
145017
+ window_index INTEGER NOT NULL DEFAULT 0,
145018
+ start_ordinal INTEGER NOT NULL,
145019
+ end_ordinal INTEGER NOT NULL,
145020
+ chunk_hash TEXT NOT NULL,
145021
+ model_id TEXT NOT NULL,
145022
+ dims INTEGER NOT NULL,
145023
+ vector BLOB NOT NULL,
145024
+ created_at INTEGER NOT NULL,
145025
+ UNIQUE(compartment_id, window_index)
145026
+ );
145027
+ CREATE INDEX IF NOT EXISTS idx_cce_session
145028
+ ON compartment_chunk_embeddings(session_id);
145029
+ CREATE INDEX IF NOT EXISTS idx_cce_project_model
145030
+ ON compartment_chunk_embeddings(project_path, model_id);
145031
+ `);
145032
+ }
145033
+ },
145034
+ {
145035
+ version: 34,
145036
+ description: "workspace tables and m[0] workspace fingerprint cache reset",
145037
+ up: (db) => {
145038
+ db.exec(`
145039
+ CREATE TABLE IF NOT EXISTS workspaces (
145040
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
145041
+ name TEXT NOT NULL UNIQUE,
145042
+ created_at INTEGER NOT NULL,
145043
+ updated_at INTEGER NOT NULL
145044
+ );
145045
+ CREATE TABLE IF NOT EXISTS workspace_members (
145046
+ workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE,
145047
+ project_path TEXT NOT NULL,
145048
+ display_name TEXT NOT NULL,
145049
+ display_path TEXT NOT NULL,
145050
+ added_at INTEGER NOT NULL,
145051
+ PRIMARY KEY (workspace_id, project_path)
145052
+ );
145053
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_unique
145054
+ ON workspace_members(project_path);
145055
+ CREATE UNIQUE INDEX IF NOT EXISTS idx_workspace_member_name
145056
+ ON workspace_members(workspace_id, display_name);
145057
+ `);
145058
+ const hasSessionMeta = db.prepare("SELECT 1 FROM sqlite_master WHERE type='table' AND name='session_meta' LIMIT 1").get();
145059
+ if (!hasSessionMeta)
145060
+ return;
145061
+ ensureColumn(db, "session_meta", "cached_m0_workspace_fingerprint", "TEXT");
145062
+ const columns = new Set(db.prepare("PRAGMA table_info(session_meta)").all().map((column) => column.name));
145063
+ const clears = [
145064
+ ["cached_m0_bytes", null],
145065
+ ["cached_m1_bytes", null],
145066
+ ["cached_m0_project_memory_epoch", null],
145067
+ ["cached_m0_workspace_fingerprint", null],
145068
+ ["cached_m0_project_user_profile_version", null],
145069
+ ["cached_m0_max_compartment_seq", null],
145070
+ ["cached_m0_max_memory_id", null],
145071
+ ["cached_m0_max_mutation_id", null],
145072
+ ["cached_m0_max_memory_mutation_id", null],
145073
+ ["cached_m0_project_docs_hash", null],
145074
+ ["cached_m0_materialized_at", null],
145075
+ ["cached_m0_session_facts_version", null],
145076
+ ["cached_m0_upgrade_state", null],
145077
+ ["cached_m0_system_hash", null],
145078
+ ["cached_m0_tool_set_hash", null],
145079
+ ["cached_m0_model_key", null],
145080
+ ["cached_m0_last_baseline_end_message_id", null],
145081
+ ["memory_block_cache", ""],
145082
+ ["memory_block_ids", ""],
145083
+ ["memory_block_count", 0]
145084
+ ];
145085
+ const setClauses = [];
145086
+ const values = [];
145087
+ for (const [column, value] of clears) {
145088
+ if (!columns.has(column))
145089
+ continue;
145090
+ setClauses.push(`${column} = ?`);
145091
+ values.push(value);
145092
+ }
145093
+ if (setClauses.length > 0) {
145094
+ db.prepare(`UPDATE session_meta SET ${setClauses.join(", ")}`).run(...values);
145095
+ }
145096
+ }
145097
+ },
145098
+ {
145099
+ version: 35,
145100
+ description: "workspace per-category share defaults and epoch refresh",
145101
+ up: (db) => {
145102
+ db.exec(`
145103
+ CREATE TABLE IF NOT EXISTS workspaces (
145104
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
145105
+ name TEXT NOT NULL UNIQUE,
145106
+ created_at INTEGER NOT NULL,
145107
+ updated_at INTEGER NOT NULL,
145108
+ share_categories TEXT NOT NULL DEFAULT '["CONSTRAINTS"]'
145109
+ );
145110
+ `);
145111
+ if (!columnExists2(db, "workspaces", "share_categories")) {
145112
+ db.exec(`ALTER TABLE workspaces ADD COLUMN share_categories TEXT NOT NULL DEFAULT '["CONSTRAINTS"]'`);
145113
+ }
145114
+ db.prepare(`UPDATE workspaces
145115
+ SET share_categories = '["CONSTRAINTS"]'
145116
+ WHERE share_categories IS NULL OR share_categories = ''`).run();
145117
+ if (!tableExists2(db, "workspace_members"))
145118
+ return;
145119
+ const rows = db.prepare(`SELECT DISTINCT project_path AS identity
145120
+ FROM workspace_members
145121
+ WHERE project_path IS NOT NULL AND project_path <> ''
145122
+ ORDER BY project_path ASC`).all();
145123
+ const identities = rows.map((row) => typeof row.identity === "string" ? row.identity : "").filter((identity) => identity.length > 0);
145124
+ if (identities.length > 0) {
145125
+ bumpEpochsForWorkspaceMemberSet(db, identities, Date.now());
145126
+ }
145127
+ }
145128
+ },
145129
+ {
145130
+ version: 36,
145131
+ description: "session project ownership map for compartment chunk backfill scoping",
145132
+ up: (db) => {
145133
+ db.exec(`
145134
+ CREATE TABLE IF NOT EXISTS session_projects (
145135
+ session_id TEXT NOT NULL,
145136
+ harness TEXT NOT NULL DEFAULT 'opencode',
145137
+ project_path TEXT NOT NULL,
145138
+ updated_at INTEGER NOT NULL,
145139
+ PRIMARY KEY(session_id, harness)
145140
+ );
145141
+ CREATE INDEX IF NOT EXISTS idx_session_projects_project
145142
+ ON session_projects(project_path);
145143
+ `);
145144
+ const hasChunkTable = db.prepare("SELECT 1 FROM sqlite_master WHERE type='table' AND name='compartment_chunk_embeddings'").get();
145145
+ if (hasChunkTable) {
145146
+ db.exec(`
145147
+ INSERT OR IGNORE INTO session_projects (session_id, harness, project_path, updated_at)
145148
+ SELECT session_id, harness, MIN(project_path), 0
145149
+ FROM compartment_chunk_embeddings
145150
+ GROUP BY session_id, harness
145151
+ HAVING COUNT(DISTINCT project_path) = 1;
145152
+ `);
145153
+ }
145154
+ }
144638
145155
  }
144639
145156
  ];
144640
145157
  var LATEST_MIGRATION_VERSION = MIGRATIONS.reduce((max, m) => Math.max(max, m.version), 0);
@@ -144704,7 +145221,7 @@ function runMigrations(db) {
144704
145221
  log(`[migrations] schema version now: ${MIGRATIONS[MIGRATIONS.length - 1].version}`);
144705
145222
  }
144706
145223
  // ../plugin/src/features/magic-context/project-docs-hash.ts
144707
- import { createHash as createHash3 } from "node:crypto";
145224
+ import { createHash as createHash4 } from "node:crypto";
144708
145225
  import { lstatSync, readFileSync as readFileSync2, statSync as statSync2 } from "node:fs";
144709
145226
  import path4 from "node:path";
144710
145227
  var PROJECT_DOC_FILES = ["ARCHITECTURE.md", "STRUCTURE.md"];
@@ -144802,7 +145319,7 @@ function hashCanonicalPieces(hashPieces) {
144802
145319
  if (hashPieces.length === 0) {
144803
145320
  return "";
144804
145321
  }
144805
- return createHash3("sha256").update(hashPieces.join(PROJECT_DOCS_DELIMITER), "utf8").digest("hex");
145322
+ return createHash4("sha256").update(hashPieces.join(PROJECT_DOCS_DELIMITER), "utf8").digest("hex");
144806
145323
  }
144807
145324
  function readProjectDocsCanonical(projectDirectory) {
144808
145325
  const canonicalDirectory = path4.resolve(projectDirectory);
@@ -144908,18 +145425,13 @@ function getMemoryMutation(db, id) {
144908
145425
  WHERE id = ?`).get(id);
144909
145426
  return row ? toMemoryMutation(row) : null;
144910
145427
  }
144911
- function getMemoryMutationsForRender(db, projectPath, afterId, renderedMemoryIds) {
144912
- if (renderedMemoryIds.length === 0)
144913
- return [];
144914
- const uniqueIds = [...new Set(renderedMemoryIds)].sort((left, right) => left - right);
144915
- const placeholders = uniqueIds.map(() => "?").join(", ");
144916
- const rows = db.prepare(`SELECT id, project_path, mutation_type, target_memory_id,
144917
- superseded_by_id, category, new_content, queued_at
144918
- FROM memory_mutation_log
144919
- WHERE project_path = ?
144920
- AND id > ?
144921
- AND target_memory_id IN (${placeholders})
144922
- ORDER BY id ASC`).all(projectPath, afterId ?? 0, ...uniqueIds);
145428
+ function uniqueProjectPaths(projectPaths) {
145429
+ return [...new Set(projectPaths.filter((path5) => path5.length > 0))];
145430
+ }
145431
+ function placeholders2(values) {
145432
+ return values.map(() => "?").join(", ");
145433
+ }
145434
+ function coalesceMutations(rows) {
144923
145435
  const chosenByTarget = new Map;
144924
145436
  for (const dbRow of rows) {
144925
145437
  const candidate = toMemoryMutation(dbRow);
@@ -144940,10 +145452,54 @@ function getMemoryMutationsForRender(db, projectPath, afterId, renderedMemoryIds
144940
145452
  }
144941
145453
  return [...chosenByTarget.values()].sort((left, right) => left.id - right.id);
144942
145454
  }
145455
+ function getMemoryMutationsForRender(db, projectPath, afterId, renderedMemoryIds) {
145456
+ if (renderedMemoryIds.length === 0)
145457
+ return [];
145458
+ const uniqueIds = [...new Set(renderedMemoryIds)].sort((left, right) => left - right);
145459
+ const placeholders3 = uniqueIds.map(() => "?").join(", ");
145460
+ const rows = db.prepare(`SELECT id, project_path, mutation_type, target_memory_id,
145461
+ superseded_by_id, category, new_content, queued_at
145462
+ FROM memory_mutation_log
145463
+ WHERE project_path = ?
145464
+ AND id > ?
145465
+ AND target_memory_id IN (${placeholders3})
145466
+ ORDER BY id ASC`).all(projectPath, afterId ?? 0, ...uniqueIds);
145467
+ return coalesceMutations(rows);
145468
+ }
145469
+ function getMemoryMutationsForRenderByProjects(db, projectPaths, afterId, renderedMemoryIds) {
145470
+ if (renderedMemoryIds.length === 0)
145471
+ return [];
145472
+ const identities = uniqueProjectPaths(projectPaths);
145473
+ if (identities.length === 0)
145474
+ return [];
145475
+ if (identities.length === 1) {
145476
+ return getMemoryMutationsForRender(db, identities[0], afterId, renderedMemoryIds);
145477
+ }
145478
+ const uniqueIds = [...new Set(renderedMemoryIds)].sort((left, right) => left - right);
145479
+ const rows = db.prepare(`SELECT id, project_path, mutation_type, target_memory_id,
145480
+ superseded_by_id, category, new_content, queued_at
145481
+ FROM memory_mutation_log
145482
+ WHERE project_path IN (${placeholders2(identities)})
145483
+ AND id > ?
145484
+ AND target_memory_id IN (${placeholders2(uniqueIds)})
145485
+ ORDER BY id ASC`).all(...identities, afterId ?? 0, ...uniqueIds);
145486
+ return coalesceMutations(rows);
145487
+ }
144943
145488
  function getMaxMemoryMutationId(db, projectPath) {
144944
145489
  const row = db.prepare("SELECT MAX(id) AS max_id FROM memory_mutation_log WHERE project_path = ?").get(projectPath);
144945
145490
  return row?.max_id ?? null;
144946
145491
  }
145492
+ function getMaxMemoryMutationIdForProjects(db, projectPaths) {
145493
+ const identities = uniqueProjectPaths(projectPaths);
145494
+ if (identities.length === 0)
145495
+ return null;
145496
+ if (identities.length === 1)
145497
+ return getMaxMemoryMutationId(db, identities[0]);
145498
+ const row = db.prepare(`SELECT MAX(id) AS max_id
145499
+ FROM memory_mutation_log
145500
+ WHERE project_path IN (${placeholders2(identities)})`).get(...identities);
145501
+ return row?.max_id ?? null;
145502
+ }
144947
145503
  // ../plugin/src/features/magic-context/storage-meta-persisted.ts
144948
145504
  init_logger();
144949
145505
  var CAS_RETRY_LIMIT = 5;
@@ -145789,9 +146345,9 @@ function buildStatusClause(status) {
145789
146345
  if (statuses.length === 0) {
145790
146346
  return null;
145791
146347
  }
145792
- const placeholders = statuses.map(() => "?").join(", ");
146348
+ const placeholders3 = statuses.map(() => "?").join(", ");
145793
146349
  return {
145794
- sql: `status IN (${placeholders})`,
146350
+ sql: `status IN (${placeholders3})`,
145795
146351
  params: statuses
145796
146352
  };
145797
146353
  }
@@ -145987,19 +146543,6 @@ function getProjectState(db, projectPath) {
145987
146543
  WHERE project_path = ?`).get(projectPath);
145988
146544
  return row ? toProjectState(row) : null;
145989
146545
  }
145990
- function bumpProjectMemoryEpoch(db, projectPath, now = Date.now()) {
145991
- db.prepare(`INSERT INTO project_state
145992
- (project_path, project_memory_epoch, project_user_profile_version, updated_at)
145993
- VALUES (?, 1, 0, ?)
145994
- ON CONFLICT(project_path) DO UPDATE SET
145995
- project_memory_epoch = project_memory_epoch + 1,
145996
- updated_at = excluded.updated_at`).run(projectPath, now);
145997
- const state = getProjectState(db, projectPath);
145998
- if (!state) {
145999
- throw new Error(`Failed to bump project memory epoch for ${projectPath}`);
146000
- }
146001
- return state;
146002
- }
146003
146546
  function bumpProjectUserProfileVersion(db, projectPath = GLOBAL_USER_PROFILE_PROJECT_PATH, now = Date.now()) {
146004
146547
  db.prepare(`INSERT INTO project_state
146005
146548
  (project_path, project_memory_epoch, project_user_profile_version, updated_at)
@@ -146033,8 +146576,8 @@ function getSourceContents(db, sessionId, tagIds) {
146033
146576
  if (tagIds.length === 0) {
146034
146577
  return new Map;
146035
146578
  }
146036
- const placeholders = tagIds.map(() => "?").join(", ");
146037
- const rows = db.prepare(`SELECT tag_id, content FROM source_contents WHERE session_id = ? AND tag_id IN (${placeholders})`).all(sessionId, ...tagIds).filter(isSourceContentRow);
146579
+ const placeholders3 = tagIds.map(() => "?").join(", ");
146580
+ const rows = db.prepare(`SELECT tag_id, content FROM source_contents WHERE session_id = ? AND tag_id IN (${placeholders3})`).all(sessionId, ...tagIds).filter(isSourceContentRow);
146038
146581
  const sources = new Map;
146039
146582
  for (const row of rows) {
146040
146583
  sources.set(row.tag_id, row.content);
@@ -146324,8 +146867,8 @@ function getTagsByNumbers(db, sessionId, tagNumbers) {
146324
146867
  }
146325
146868
  return all;
146326
146869
  }
146327
- const placeholders = tagNumbers.map(() => "?").join(",");
146328
- const rows = db.prepare(`SELECT ${TAG_SELECT_COLUMNS} FROM tags WHERE session_id = ? AND tag_number IN (${placeholders}) ORDER BY tag_number ASC, id ASC`).all(sessionId, ...tagNumbers).filter(isTagRow);
146870
+ const placeholders3 = tagNumbers.map(() => "?").join(",");
146871
+ const rows = db.prepare(`SELECT ${TAG_SELECT_COLUMNS} FROM tags WHERE session_id = ? AND tag_number IN (${placeholders3}) ORDER BY tag_number ASC, id ASC`).all(sessionId, ...tagNumbers).filter(isTagRow);
146329
146872
  return rows.map(toTagEntry);
146330
146873
  }
146331
146874
  var getToolTagNumberByOwnerStatements = new WeakMap;
@@ -146395,7 +146938,7 @@ var ERROR_CLASSES = new Set([
146395
146938
  ]);
146396
146939
  // ../plugin/src/features/magic-context/v22-deferred-backfill.ts
146397
146940
  init_logger();
146398
- import { createHash as createHash4 } from "node:crypto";
146941
+ import { createHash as createHash5 } from "node:crypto";
146399
146942
  import { realpathSync as realpathSync2 } from "node:fs";
146400
146943
  import path5 from "node:path";
146401
146944
  var BATCH_SIZE = 25;
@@ -146449,7 +146992,7 @@ function computeLegacyRustDirIdentity(rawProjectPath) {
146449
146992
  } catch {
146450
146993
  canonical = path5.isAbsolute(rawProjectPath) ? rawProjectPath : path5.join(process.cwd(), rawProjectPath);
146451
146994
  }
146452
- return `dir:${createHash4("sha256").update(canonical, "utf8").digest("hex")}`;
146995
+ return `dir:${createHash5("sha256").update(canonical, "utf8").digest("hex")}`;
146453
146996
  }
146454
146997
  function upsertRekeyMap(db, oldProjectPath, newProjectPath, rekeyedAt) {
146455
146998
  db.prepare(`INSERT INTO v22_identity_rekey_map (old_project_path, new_project_path, rekeyed_at)
@@ -147007,7 +147550,7 @@ function clearNoteNudgeTriggerOnly(db, sessionId) {
147007
147550
  }
147008
147551
 
147009
147552
  // ../plugin/src/hooks/magic-context/todo-view.ts
147010
- import { createHash as createHash5 } from "node:crypto";
147553
+ import { createHash as createHash6 } from "node:crypto";
147011
147554
  var TERMINAL_STATUSES = new Set(["completed", "cancelled"]);
147012
147555
  var TITLE_DONE_STATUSES = new Set(["completed"]);
147013
147556
  var SYNTHETIC_CALL_ID_PREFIX = "mc_synthetic_todo_";
@@ -147052,7 +147595,7 @@ function buildSyntheticTodoPart(stateJson) {
147052
147595
  };
147053
147596
  }
147054
147597
  function computeSyntheticCallId(stateJson) {
147055
- const hash = createHash5("sha256").update(stateJson).digest("hex").slice(0, 16);
147598
+ const hash = createHash6("sha256").update(stateJson).digest("hex").slice(0, 16);
147056
147599
  return `${SYNTHETIC_CALL_ID_PREFIX}${hash}`;
147057
147600
  }
147058
147601
  function parseTodoState(stateJson) {
@@ -147193,13 +147736,13 @@ async function maybeSendUpgradeReminder(deps, sessionId) {
147193
147736
  init_data_path();
147194
147737
  import * as fs2 from "node:fs";
147195
147738
  import * as path6 from "node:path";
147196
- var ANNOUNCEMENT_VERSION = "0.23.0";
147739
+ var ANNOUNCEMENT_VERSION = "0.24.0";
147197
147740
  var ANNOUNCEMENT_FEATURES = [
147198
- "Smarter context nudges: gentle <system-reminder> notes on tool outputs replace the old chat nudgesquieter, cache-safe, and they now also help subagents manage their own context.",
147199
- "The agent can now maintain its own project memories: update, archive (batch), and merge the memories it sees not just write new ones. Memory categories are schema-enforced.",
147200
- "Big-session performance: historian trigger and token math moved off the database hot pathmulti-second stalls on large sessions are gone (measured 250ms → 2.4ms per pass).",
147201
- "History compaction unstuck for sparse sessions: the protected-tail boundary is now size-based, so sessions with few user turns compact reliably instead of growing forever (issue #132).",
147202
- "Fixed: session titles no longer fail to generate in fresh directories (issue #129), plus 20+ correctness fixes from three audit rounds."
147741
+ "Searchable session history: ctx_search can now find older discussion by meaning, not just keywords. New history is embedded automatically to backfill an EXISTING session's older history, run /ctx-embed-history once (it works in the background).",
147742
+ "Cross-project workspaces: group related repos and share project memories across them, with per-category control over what's shared. Set them up in the dashboard's Workspaces panel.",
147743
+ "Pi: fixed sessions overflowing the model context while still showing moderate usagePi now sheds context before a tool-heavy turn overflows.",
147744
+ "Fewer prompt-cache busts: doc edits, processed screenshots, and a rebuild-then-bust-again case no longer re-bill large prompt prefixes.",
147745
+ "Setup wizard now lists your actual models with type-ahead instead of fixed recommendations, and explains the historian/dreamer roles (issue #144). Plus a GitHub Copilot tool-pairing fix (#135)."
147203
147746
  ];
147204
147747
  var ANNOUNCEMENT_FOOTER = "Join us on Discord: https://discord.gg/F2uWxjGnU";
147205
147748
  var STATE_FILENAME = "last_announced_version";
@@ -149082,7 +149625,7 @@ function isEmbeddingRow(row) {
149082
149625
  if (row === null || typeof row !== "object")
149083
149626
  return false;
149084
149627
  const candidate = row;
149085
- return typeof candidate.memoryId === "number" && isEmbeddingBlob(candidate.embedding);
149628
+ return typeof candidate.memoryId === "number" && isEmbeddingBlob(candidate.embedding) && (candidate.modelId === null || typeof candidate.modelId === "string");
149086
149629
  }
149087
149630
  function toFloat32Array(blob) {
149088
149631
  if (blob instanceof Uint8Array) {
@@ -149102,7 +149645,7 @@ function getSaveEmbeddingStatement(db) {
149102
149645
  function getLoadAllEmbeddingsStatement(db) {
149103
149646
  let stmt = loadAllEmbeddingsStatements.get(db);
149104
149647
  if (!stmt) {
149105
- stmt = db.prepare("SELECT memory_embeddings.memory_id AS memoryId, memory_embeddings.embedding AS embedding FROM memory_embeddings INNER JOIN memories ON memories.id = memory_embeddings.memory_id WHERE memories.project_path = ? ORDER BY memory_embeddings.memory_id ASC");
149648
+ stmt = db.prepare("SELECT memory_embeddings.memory_id AS memoryId, memory_embeddings.embedding AS embedding, memory_embeddings.model_id AS modelId FROM memory_embeddings INNER JOIN memories ON memories.id = memory_embeddings.memory_id WHERE memories.project_path = ? ORDER BY memory_embeddings.memory_id ASC");
149106
149649
  loadAllEmbeddingsStatements.set(db, stmt);
149107
149650
  }
149108
149651
  return stmt;
@@ -149131,7 +149674,10 @@ function loadAllEmbeddings(db, projectPath) {
149131
149674
  const rows = getLoadAllEmbeddingsStatement(db).all(projectPath).filter(isEmbeddingRow);
149132
149675
  const embeddings = new Map;
149133
149676
  for (const row of rows) {
149134
- embeddings.set(row.memoryId, toFloat32Array(row.embedding));
149677
+ embeddings.set(row.memoryId, {
149678
+ embedding: toFloat32Array(row.embedding),
149679
+ modelId: row.modelId
149680
+ });
149135
149681
  }
149136
149682
  return embeddings;
149137
149683
  }
@@ -149182,13 +149728,13 @@ function invalidateMemory(projectPath, memoryId) {
149182
149728
  }
149183
149729
 
149184
149730
  // ../plugin/src/features/magic-context/memory/normalize-hash.ts
149185
- import { createHash as createHash6 } from "node:crypto";
149731
+ import { createHash as createHash7 } from "node:crypto";
149186
149732
  function normalizeMemoryContent(content) {
149187
149733
  return content.toLowerCase().replace(/\s+/g, " ").trim();
149188
149734
  }
149189
149735
  function computeNormalizedHash(content) {
149190
149736
  const normalized = normalizeMemoryContent(content);
149191
- return createHash6("md5").update(normalized).digest("hex");
149737
+ return createHash7("md5").update(normalized).digest("hex");
149192
149738
  }
149193
149739
 
149194
149740
  // ../plugin/src/features/magic-context/memory/storage-memory.ts
@@ -149376,8 +149922,8 @@ function getMemoriesByProjectStatement(db, statuses) {
149376
149922
  }
149377
149923
  let stmt = statements.get(db);
149378
149924
  if (!stmt) {
149379
- const placeholders = statuses.map(() => "?").join(", ");
149380
- stmt = db.prepare(`SELECT ${getMemorySelectColumns(db)} FROM memories WHERE project_path = ? AND status IN (${placeholders}) AND (expires_at IS NULL OR expires_at > ?) ORDER BY category ASC, updated_at DESC, id ASC`);
149925
+ const placeholders3 = statuses.map(() => "?").join(", ");
149926
+ stmt = db.prepare(`SELECT ${getMemorySelectColumns(db)} FROM memories WHERE project_path = ? AND status IN (${placeholders3}) AND (expires_at IS NULL OR expires_at > ?) ORDER BY category ASC, updated_at DESC, id ASC`);
149381
149927
  statements.set(db, stmt);
149382
149928
  }
149383
149929
  return stmt;
@@ -149522,6 +150068,97 @@ function getMemoriesByProject(db, projectPath, statuses = ["active", "permanent"
149522
150068
  const rows = getMemoriesByProjectStatement(db, statuses).all(projectPath, ...statuses, expiryCutoff).filter(isMemoryRow);
149523
150069
  return rows.map(toMemory);
149524
150070
  }
150071
+ function sqlPlaceholders(values) {
150072
+ return values.map(() => "?").join(", ");
150073
+ }
150074
+ function uniqueValues(values) {
150075
+ return [...new Set(values.filter((value) => value.length > 0))];
150076
+ }
150077
+ function buildWorkspaceMemorySqlFilter(args) {
150078
+ if (args.shareCategories === null || args.shareCategories === undefined) {
150079
+ return { clause: "", params: [], active: false };
150080
+ }
150081
+ const identities = uniqueValues(args.identities);
150082
+ const identitySet = new Set(identities);
150083
+ const ownSet = new Set(uniqueValues(args.ownIdentities ?? []).filter((identity) => identitySet.has(identity)));
150084
+ const foreignIdentities = identities.filter((identity) => !ownSet.has(identity));
150085
+ if (foreignIdentities.length === 0) {
150086
+ return { clause: "", params: [], active: false };
150087
+ }
150088
+ const ownIdentities = identities.filter((identity) => ownSet.has(identity));
150089
+ const shareCategories = uniqueValues([...args.shareCategories]);
150090
+ const qualifier = args.tableName ? `${args.tableName}.` : "";
150091
+ const predicates = [];
150092
+ const params = [];
150093
+ if (ownIdentities.length > 0) {
150094
+ predicates.push(`${qualifier}project_path IN (${sqlPlaceholders(ownIdentities)})`);
150095
+ params.push(...ownIdentities);
150096
+ }
150097
+ if (foreignIdentities.length > 0 && shareCategories.length > 0) {
150098
+ predicates.push(`(${qualifier}project_path IN (${sqlPlaceholders(foreignIdentities)}) AND ${qualifier}category IN (${sqlPlaceholders(shareCategories)}))`);
150099
+ params.push(...foreignIdentities, ...shareCategories);
150100
+ }
150101
+ if (predicates.length === 0) {
150102
+ return { clause: " AND 0 = 1", params: [], active: true };
150103
+ }
150104
+ return { clause: ` AND (${predicates.join(" OR ")})`, params, active: true };
150105
+ }
150106
+ function getMemoriesByProjects(db, projectPaths, statuses = ["active", "permanent"], expiryCutoff = Date.now(), ownIdentities, shareCategories) {
150107
+ const identities = uniqueValues(projectPaths);
150108
+ if (identities.length === 0 || statuses.length === 0)
150109
+ return [];
150110
+ const sharingFilter = buildWorkspaceMemorySqlFilter({
150111
+ identities,
150112
+ ownIdentities,
150113
+ shareCategories
150114
+ });
150115
+ if (identities.length === 1 && !sharingFilter.active) {
150116
+ return getMemoriesByProject(db, identities[0], statuses, expiryCutoff);
150117
+ }
150118
+ const rows = db.prepare(`SELECT ${getMemorySelectColumns(db)}
150119
+ FROM memories
150120
+ WHERE project_path IN (${sqlPlaceholders(identities)})
150121
+ AND status IN (${sqlPlaceholders(statuses)})
150122
+ AND (expires_at IS NULL OR expires_at > ?)${sharingFilter.clause}
150123
+ ORDER BY category ASC, updated_at DESC, id ASC`).all(...identities, ...statuses, expiryCutoff, ...sharingFilter.params).filter(isMemoryRow);
150124
+ return rows.map(toMemory);
150125
+ }
150126
+ function getMaxMemoryIdForProjects(db, projectPaths, ownIdentities, shareCategories) {
150127
+ const identities = uniqueValues(projectPaths);
150128
+ if (identities.length === 0)
150129
+ return 0;
150130
+ const sharingFilter = buildWorkspaceMemorySqlFilter({
150131
+ identities,
150132
+ ownIdentities,
150133
+ shareCategories
150134
+ });
150135
+ if (identities.length === 1 && !sharingFilter.active) {
150136
+ const row2 = db.prepare("SELECT COALESCE(MAX(id), 0) AS max_id FROM memories WHERE project_path = ?").get(identities[0]);
150137
+ return typeof row2?.max_id === "number" ? row2.max_id : 0;
150138
+ }
150139
+ const row = db.prepare(`SELECT COALESCE(MAX(id), 0) AS max_id
150140
+ FROM memories
150141
+ WHERE project_path IN (${sqlPlaceholders(identities)})${sharingFilter.clause}`).get(...identities, ...sharingFilter.params);
150142
+ return typeof row?.max_id === "number" ? row.max_id : 0;
150143
+ }
150144
+ function readNewMemoriesForM1Union(db, projectPaths, afterId, expiryCutoff, ownIdentities, shareCategories) {
150145
+ const identities = uniqueValues(projectPaths);
150146
+ if (identities.length === 0)
150147
+ return [];
150148
+ const sharingFilter = buildWorkspaceMemorySqlFilter({
150149
+ identities,
150150
+ ownIdentities,
150151
+ shareCategories
150152
+ });
150153
+ const rows = db.prepare(`SELECT ${getMemorySelectColumns(db)}
150154
+ FROM memories
150155
+ WHERE project_path IN (${sqlPlaceholders(identities)})
150156
+ AND id > ?
150157
+ AND status IN ('active', 'permanent')
150158
+ AND (expires_at IS NULL OR expires_at > ?)${sharingFilter.clause}
150159
+ ORDER BY ${MEMORY_CATEGORY_ORDER_SQL}, id ASC`).all(...identities, afterId, expiryCutoff, ...sharingFilter.params).filter(isMemoryRow);
150160
+ return rows.map(toMemory);
150161
+ }
149525
150162
  function getAllActiveMemoriesForMigration(db, projectPath) {
149526
150163
  const rows = getActiveMemoriesNoExpiryStatement(db).all(projectPath).filter(isMemoryRow);
149527
150164
  return rows.map(toMemory);
@@ -150043,8 +150680,8 @@ function getUserMemoryCandidates(db) {
150043
150680
  function deleteUserMemoryCandidates(db, ids) {
150044
150681
  if (ids.length === 0)
150045
150682
  return;
150046
- const placeholders = ids.map(() => "?").join(",");
150047
- db.prepare(`DELETE FROM user_memory_candidates WHERE id IN (${placeholders})`).run(...ids);
150683
+ const placeholders3 = ids.map(() => "?").join(",");
150684
+ db.prepare(`DELETE FROM user_memory_candidates WHERE id IN (${placeholders3})`).run(...ids);
150048
150685
  }
150049
150686
  function insertUserMemory(db, content, sourceCandidateIds) {
150050
150687
  const now = Date.now();
@@ -165534,7 +166171,8 @@ var BaseEmbeddingConfigSchema = exports_external.object({
165534
166171
  endpoint: exports_external.string().optional().describe("API endpoint URL. Required when provider is openai-compatible."),
165535
166172
  api_key: exports_external.string().optional().describe("API key for remote embedding provider (optional)"),
165536
166173
  input_type: exports_external.string().optional().describe("Optional input_type sent in the embedding request body. Required by some openai-compatible providers (e.g. NVIDIA NIM expects 'query' or 'passage'). Omitted from the request when unset."),
165537
- truncate: exports_external.string().optional().describe("Optional truncate mode sent in the embedding request body (e.g. NVIDIA NIM accepts 'NONE' | 'START' | 'END'). Omitted from the request when unset.")
166174
+ truncate: exports_external.string().optional().describe("Optional truncate mode sent in the embedding request body (e.g. NVIDIA NIM accepts 'NONE' | 'START' | 'END'). Omitted from the request when unset."),
166175
+ max_input_tokens: exports_external.number().int().positive().optional().describe("Optional maximum input tokens for chunk embeddings. Defaults conservatively to 512 when omitted.")
165538
166176
  }).superRefine((data, ctx) => {
165539
166177
  if (data.provider === "openai-compatible" && !data.endpoint?.trim()) {
165540
166178
  ctx.addIssue({
@@ -165555,7 +166193,8 @@ var EmbeddingConfigSchema = BaseEmbeddingConfigSchema.transform((data) => {
165555
166193
  if (data.provider === "local") {
165556
166194
  return {
165557
166195
  provider: "local",
165558
- model: data.model?.trim() || DEFAULT_LOCAL_EMBEDDING_MODEL
166196
+ model: data.model?.trim() || DEFAULT_LOCAL_EMBEDDING_MODEL,
166197
+ ...data.max_input_tokens ? { max_input_tokens: data.max_input_tokens } : {}
165559
166198
  };
165560
166199
  }
165561
166200
  if (data.provider === "openai-compatible") {
@@ -165568,7 +166207,8 @@ var EmbeddingConfigSchema = BaseEmbeddingConfigSchema.transform((data) => {
165568
166207
  endpoint: data.endpoint?.trim() ?? "",
165569
166208
  ...apiKey ? { api_key: apiKey } : {},
165570
166209
  ...inputType ? { input_type: inputType } : {},
165571
- ...truncate ? { truncate } : {}
166210
+ ...truncate ? { truncate } : {},
166211
+ ...data.max_input_tokens ? { max_input_tokens: data.max_input_tokens } : {}
165572
166212
  };
165573
166213
  }
165574
166214
  return { provider: "off" };
@@ -165653,6 +166293,441 @@ var MagicContextConfigSchema = exports_external.object({
165653
166293
  // ../plugin/src/features/magic-context/memory/embedding.ts
165654
166294
  init_logger();
165655
166295
 
166296
+ // ../plugin/src/features/magic-context/compartment-chunk-embedding.ts
166297
+ import { createHash as createHash8 } from "node:crypto";
166298
+ var DEFAULT_COMPARTMENT_CHUNK_MAX_INPUT_TOKENS = 512;
166299
+ var loadFtsRowsStatements = new WeakMap;
166300
+ var existingHashStatements = new WeakMap;
166301
+ var existingHashByProjectStatements = new WeakMap;
166302
+ var deleteByCompartmentStatements = new WeakMap;
166303
+ var insertEmbeddingStatements = new WeakMap;
166304
+ var distinctModelStatements = new WeakMap;
166305
+ var clearProjectStatements = new WeakMap;
166306
+ var clearProjectModelStatements = new WeakMap;
166307
+ var searchRowsStatements = new WeakMap;
166308
+ var searchRowsByModelStatements = new WeakMap;
166309
+ var backfillCandidateStatements = new WeakMap;
166310
+ function getLoadFtsRowsStatement(db) {
166311
+ let stmt = loadFtsRowsStatements.get(db);
166312
+ if (!stmt) {
166313
+ stmt = db.prepare(`SELECT message_ordinal AS messageOrdinal, role, content
166314
+ FROM message_history_fts
166315
+ WHERE session_id = ?
166316
+ AND message_ordinal >= ?
166317
+ AND message_ordinal <= ?
166318
+ AND role IN ('user', 'assistant')
166319
+ ORDER BY message_ordinal ASC`);
166320
+ loadFtsRowsStatements.set(db, stmt);
166321
+ }
166322
+ return stmt;
166323
+ }
166324
+ function getExistingHashStatement(db, scopedToProject) {
166325
+ const map2 = scopedToProject ? existingHashByProjectStatements : existingHashStatements;
166326
+ let stmt = map2.get(db);
166327
+ if (!stmt) {
166328
+ stmt = db.prepare(`SELECT window_index AS windowIndex, chunk_hash AS chunkHash
166329
+ FROM compartment_chunk_embeddings
166330
+ WHERE compartment_id = ?
166331
+ AND model_id = ?
166332
+ ${scopedToProject ? "AND project_path = ?" : ""}
166333
+ ORDER BY window_index ASC`);
166334
+ map2.set(db, stmt);
166335
+ }
166336
+ return stmt;
166337
+ }
166338
+ function getDeleteByCompartmentStatement(db) {
166339
+ let stmt = deleteByCompartmentStatements.get(db);
166340
+ if (!stmt) {
166341
+ stmt = db.prepare("DELETE FROM compartment_chunk_embeddings WHERE compartment_id = ?");
166342
+ deleteByCompartmentStatements.set(db, stmt);
166343
+ }
166344
+ return stmt;
166345
+ }
166346
+ function getInsertEmbeddingStatement(db) {
166347
+ let stmt = insertEmbeddingStatements.get(db);
166348
+ if (!stmt) {
166349
+ stmt = db.prepare(`INSERT INTO compartment_chunk_embeddings (
166350
+ compartment_id, session_id, project_path, harness, window_index,
166351
+ start_ordinal, end_ordinal, chunk_hash, model_id, dims, vector, created_at
166352
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`);
166353
+ insertEmbeddingStatements.set(db, stmt);
166354
+ }
166355
+ return stmt;
166356
+ }
166357
+ function getDistinctModelStatement(db) {
166358
+ let stmt = distinctModelStatements.get(db);
166359
+ if (!stmt) {
166360
+ stmt = db.prepare(`SELECT DISTINCT model_id AS modelId
166361
+ FROM compartment_chunk_embeddings
166362
+ WHERE project_path = ?`);
166363
+ distinctModelStatements.set(db, stmt);
166364
+ }
166365
+ return stmt;
166366
+ }
166367
+ function getClearProjectStatement(db) {
166368
+ let stmt = clearProjectStatements.get(db);
166369
+ if (!stmt) {
166370
+ stmt = db.prepare("DELETE FROM compartment_chunk_embeddings WHERE project_path = ?");
166371
+ clearProjectStatements.set(db, stmt);
166372
+ }
166373
+ return stmt;
166374
+ }
166375
+ function getClearProjectModelStatement(db) {
166376
+ let stmt = clearProjectModelStatements.get(db);
166377
+ if (!stmt) {
166378
+ stmt = db.prepare("DELETE FROM compartment_chunk_embeddings WHERE project_path = ? AND model_id = ?");
166379
+ clearProjectModelStatements.set(db, stmt);
166380
+ }
166381
+ return stmt;
166382
+ }
166383
+ function getSearchRowsStatement(db, withModel) {
166384
+ const map2 = withModel ? searchRowsByModelStatements : searchRowsStatements;
166385
+ let stmt = map2.get(db);
166386
+ if (!stmt) {
166387
+ stmt = db.prepare(`SELECT e.compartment_id AS compartmentId,
166388
+ e.session_id AS sessionId,
166389
+ c.title AS title,
166390
+ c.start_message AS compartmentStart,
166391
+ c.end_message AS compartmentEnd,
166392
+ e.window_index AS windowIndex,
166393
+ e.start_ordinal AS windowStart,
166394
+ e.end_ordinal AS windowEnd,
166395
+ e.chunk_hash AS chunkHash,
166396
+ e.model_id AS modelId,
166397
+ e.dims AS dims,
166398
+ e.vector AS vector
166399
+ FROM compartment_chunk_embeddings e
166400
+ JOIN compartments c ON c.id = e.compartment_id
166401
+ WHERE e.session_id = ?
166402
+ AND e.project_path = ?
166403
+ ${withModel ? "AND e.model_id = ?" : ""}
166404
+ ORDER BY e.compartment_id ASC, e.window_index ASC`);
166405
+ map2.set(db, stmt);
166406
+ }
166407
+ return stmt;
166408
+ }
166409
+ function isFinitePositiveInteger(value) {
166410
+ return typeof value === "number" && Number.isFinite(value) && value > 0;
166411
+ }
166412
+ function normalizeCompartmentChunkMaxInputTokens(value) {
166413
+ if (!isFinitePositiveInteger(value)) {
166414
+ return DEFAULT_COMPARTMENT_CHUNK_MAX_INPUT_TOKENS;
166415
+ }
166416
+ return Math.max(1, Math.floor(value));
166417
+ }
166418
+ function normalizeContent(text) {
166419
+ return text.replace(/\s+/g, " ").trim();
166420
+ }
166421
+ function formatOrdinalRange(start, end) {
166422
+ return start === end ? `[${start}]` : `[${start}-${end}]`;
166423
+ }
166424
+ function rolePrefix(role) {
166425
+ if (role === "user")
166426
+ return "U";
166427
+ if (role === "assistant")
166428
+ return "A";
166429
+ return null;
166430
+ }
166431
+ function parseOrdinal(value) {
166432
+ const parsed = typeof value === "number" ? value : Number.parseInt(String(value ?? ""), 10);
166433
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : null;
166434
+ }
166435
+ function parseCanonicalLineRange(line) {
166436
+ const match = /^\[(\d+)(?:-(\d+))?\]\s+[UA]:/.exec(line.trim());
166437
+ if (!match)
166438
+ return null;
166439
+ const start = Number.parseInt(match[1], 10);
166440
+ const end = match[2] ? Number.parseInt(match[2], 10) : start;
166441
+ if (!Number.isFinite(start) || !Number.isFinite(end))
166442
+ return null;
166443
+ return { start, end };
166444
+ }
166445
+ function hashChunkText(text) {
166446
+ return createHash8("sha256").update(text).digest("hex");
166447
+ }
166448
+ function vectorBlob(vector) {
166449
+ return new Uint8Array(vector.buffer, vector.byteOffset, vector.byteLength);
166450
+ }
166451
+ function toFloat32Array2(blob) {
166452
+ if (blob instanceof Uint8Array) {
166453
+ const buffer2 = blob.buffer.slice(blob.byteOffset, blob.byteOffset + blob.byteLength);
166454
+ return new Float32Array(buffer2);
166455
+ }
166456
+ return new Float32Array(blob.slice(0));
166457
+ }
166458
+ function buildCanonicalChunkTextFromFts(db, sessionId, startOrdinal, endOrdinal) {
166459
+ if (endOrdinal < startOrdinal)
166460
+ return "";
166461
+ const rows = getLoadFtsRowsStatement(db).all(sessionId, startOrdinal, endOrdinal).map((row) => row);
166462
+ const lines = [];
166463
+ let current = null;
166464
+ const flush2 = () => {
166465
+ if (!current || current.parts.length === 0)
166466
+ return;
166467
+ lines.push(`${formatOrdinalRange(current.start, current.end)} ${current.role}: ${current.parts.join(" / ")}`);
166468
+ current = null;
166469
+ };
166470
+ for (const row of rows) {
166471
+ const ordinal = parseOrdinal(row.messageOrdinal);
166472
+ const prefix = rolePrefix(row.role);
166473
+ const content = typeof row.content === "string" ? normalizeContent(row.content) : "";
166474
+ if (ordinal === null || prefix === null || content.length === 0)
166475
+ continue;
166476
+ if (current && current.role === prefix) {
166477
+ current.end = ordinal;
166478
+ current.parts.push(content);
166479
+ continue;
166480
+ }
166481
+ flush2();
166482
+ current = { role: prefix, start: ordinal, end: ordinal, parts: [content] };
166483
+ }
166484
+ flush2();
166485
+ return lines.join(`
166486
+ `);
166487
+ }
166488
+ function canonicalizeInMemoryChunkTextForEmbedding(chunkText, startOrdinal, endOrdinal) {
166489
+ const lines = [];
166490
+ for (const rawLine of chunkText.split(/\r?\n/)) {
166491
+ const line = rawLine.trim();
166492
+ const match = /^(\[(\d+)(?:-(\d+))?\]\s+[UA]:)\s*(.*)$/.exec(line);
166493
+ if (!match)
166494
+ continue;
166495
+ const lineStart = Number.parseInt(match[2], 10);
166496
+ const lineEnd = match[3] ? Number.parseInt(match[3], 10) : lineStart;
166497
+ if (startOrdinal != null && lineEnd < startOrdinal)
166498
+ continue;
166499
+ if (endOrdinal != null && lineStart > endOrdinal)
166500
+ continue;
166501
+ const rawParts = match[4].split(" / ").map((part) => normalizeContent(part)).filter((part) => part.length > 0);
166502
+ const ordinalSpan = lineEnd - lineStart + 1;
166503
+ const roleLabel = match[1].slice(match[1].indexOf("]") + 2);
166504
+ if (ordinalSpan === rawParts.length) {
166505
+ const retained = rawParts.map((part, index) => ({ ordinal: lineStart + index, part })).filter(({ ordinal, part }) => {
166506
+ if (part.startsWith("TC:"))
166507
+ return false;
166508
+ if (startOrdinal != null && ordinal < startOrdinal)
166509
+ return false;
166510
+ if (endOrdinal != null && ordinal > endOrdinal)
166511
+ return false;
166512
+ return true;
166513
+ });
166514
+ if (retained.length === 0)
166515
+ continue;
166516
+ const retainedStart = retained[0].ordinal;
166517
+ const retainedEnd = retained[retained.length - 1].ordinal;
166518
+ lines.push(`${formatOrdinalRange(retainedStart, retainedEnd)} ${roleLabel} ${retained.map(({ part }) => part).join(" / ")}`);
166519
+ continue;
166520
+ }
166521
+ const parts = rawParts.filter((part) => !part.startsWith("TC:"));
166522
+ if (parts.length === 0)
166523
+ continue;
166524
+ lines.push(`${match[1]} ${parts.join(" / ")}`);
166525
+ }
166526
+ return lines.join(`
166527
+ `);
166528
+ }
166529
+ function chunkCanonicalText(canonicalText, startOrdinal, endOrdinal, maxInputTokens) {
166530
+ const lines = canonicalText.split(/\r?\n/).map((line) => line.trim()).filter((line) => line.length > 0);
166531
+ if (lines.length === 0 || endOrdinal < startOrdinal)
166532
+ return [];
166533
+ const normalizedMax = normalizeCompartmentChunkMaxInputTokens(maxInputTokens);
166534
+ const fullText = lines.join(`
166535
+ `);
166536
+ if (estimateTokens(fullText) <= normalizedMax) {
166537
+ return [
166538
+ {
166539
+ windowIndex: 0,
166540
+ startOrdinal,
166541
+ endOrdinal,
166542
+ text: fullText,
166543
+ chunkHash: hashChunkText(fullText)
166544
+ }
166545
+ ];
166546
+ }
166547
+ const windows = [];
166548
+ let currentLines = [];
166549
+ let currentStart = null;
166550
+ let currentEnd = null;
166551
+ let currentTokens = 0;
166552
+ const flush2 = () => {
166553
+ if (currentLines.length === 0 || currentStart === null || currentEnd === null)
166554
+ return;
166555
+ const text = currentLines.join(`
166556
+ `);
166557
+ windows.push({
166558
+ windowIndex: windows.length + 1,
166559
+ startOrdinal: currentStart,
166560
+ endOrdinal: currentEnd,
166561
+ text,
166562
+ chunkHash: hashChunkText(text)
166563
+ });
166564
+ currentLines = [];
166565
+ currentStart = null;
166566
+ currentEnd = null;
166567
+ currentTokens = 0;
166568
+ };
166569
+ for (const line of lines) {
166570
+ const range = parseCanonicalLineRange(line);
166571
+ const lineStart = range?.start ?? startOrdinal;
166572
+ const lineEnd = range?.end ?? lineStart;
166573
+ const lineTokens = estimateTokens(line);
166574
+ if (currentLines.length > 0 && currentTokens + lineTokens > normalizedMax) {
166575
+ flush2();
166576
+ }
166577
+ if (currentLines.length === 0) {
166578
+ currentStart = lineStart;
166579
+ }
166580
+ currentLines.push(line);
166581
+ currentEnd = lineEnd;
166582
+ currentTokens += lineTokens;
166583
+ }
166584
+ flush2();
166585
+ return windows;
166586
+ }
166587
+ function getExistingChunkHashes(db, compartmentId, modelId, projectPath) {
166588
+ const scoped = typeof projectPath === "string" && projectPath.length > 0;
166589
+ const rows = scoped ? getExistingHashStatement(db, true).all(compartmentId, modelId, projectPath) : getExistingHashStatement(db, false).all(compartmentId, modelId);
166590
+ return new Map(rows.filter((row) => typeof row.windowIndex === "number" && typeof row.chunkHash === "string").map((row) => [row.windowIndex, row.chunkHash]));
166591
+ }
166592
+ function chunkEmbeddingWindowsAreCurrent(db, compartmentId, modelId, windows, projectPath) {
166593
+ const existing = getExistingChunkHashes(db, compartmentId, modelId, projectPath);
166594
+ if (existing.size !== windows.length)
166595
+ return false;
166596
+ return windows.every((window) => existing.get(window.windowIndex) === window.chunkHash);
166597
+ }
166598
+ function replaceCompartmentChunkEmbeddings(db, rows) {
166599
+ if (rows.length === 0)
166600
+ return;
166601
+ const compartmentId = rows[0].compartmentId;
166602
+ const now = Date.now();
166603
+ db.transaction(() => {
166604
+ getDeleteByCompartmentStatement(db).run(compartmentId);
166605
+ const insert = getInsertEmbeddingStatement(db);
166606
+ for (const row of rows) {
166607
+ insert.run(row.compartmentId, row.sessionId, row.projectPath, getHarness(), row.window.windowIndex, row.window.startOrdinal, row.window.endOrdinal, row.window.chunkHash, row.modelId, row.vector.length, vectorBlob(row.vector), row.createdAt ?? now);
166608
+ }
166609
+ })();
166610
+ }
166611
+ function getDistinctChunkEmbeddingModelIds(db, projectPath) {
166612
+ const rows = getDistinctModelStatement(db).all(projectPath);
166613
+ return new Set(rows.map((row) => typeof row.modelId === "string" ? row.modelId : null));
166614
+ }
166615
+ function clearChunkEmbeddingsForProject(db, projectPath, modelId) {
166616
+ if (modelId) {
166617
+ return getClearProjectModelStatement(db).run(projectPath, modelId).changes;
166618
+ }
166619
+ return getClearProjectStatement(db).run(projectPath).changes;
166620
+ }
166621
+ function loadCompartmentChunkEmbeddingsForSearch(db, sessionId, projectPath, modelId) {
166622
+ const rows = modelId ? getSearchRowsStatement(db, true).all(sessionId, projectPath, modelId) : getSearchRowsStatement(db, false).all(sessionId, projectPath);
166623
+ return rows.filter((row) => typeof row.compartmentId === "number" && typeof row.sessionId === "string" && typeof row.title === "string" && typeof row.compartmentStart === "number" && typeof row.compartmentEnd === "number" && typeof row.windowIndex === "number" && typeof row.windowStart === "number" && typeof row.windowEnd === "number" && typeof row.chunkHash === "string" && typeof row.modelId === "string" && typeof row.dims === "number" && (row.vector instanceof Uint8Array || row.vector instanceof ArrayBuffer)).map((row) => ({
166624
+ compartmentId: row.compartmentId,
166625
+ sessionId: row.sessionId,
166626
+ title: row.title,
166627
+ startOrdinal: row.compartmentStart,
166628
+ endOrdinal: row.compartmentEnd,
166629
+ windowIndex: row.windowIndex,
166630
+ windowStartOrdinal: row.windowStart,
166631
+ windowEndOrdinal: row.windowEnd,
166632
+ chunkHash: row.chunkHash,
166633
+ modelId: row.modelId,
166634
+ dims: row.dims,
166635
+ vector: toFloat32Array2(row.vector)
166636
+ }));
166637
+ }
166638
+ function mapBackfillCandidateRows(rows) {
166639
+ return rows.filter((row) => {
166640
+ if (row === null || typeof row !== "object")
166641
+ return false;
166642
+ const candidate = row;
166643
+ return typeof candidate.id === "number" && typeof candidate.sessionId === "string" && typeof candidate.startMessage === "number" && typeof candidate.endMessage === "number" && typeof candidate.title === "string";
166644
+ }).map((row) => ({
166645
+ id: row.id,
166646
+ sessionId: row.sessionId,
166647
+ startMessage: row.startMessage,
166648
+ endMessage: row.endMessage,
166649
+ title: row.title
166650
+ }));
166651
+ }
166652
+ var sessionBackfillCandidateStatements = new WeakMap;
166653
+ function loadUnembeddedSessionChunkCandidates(db, projectPath, sessionId, modelId, limit, excludeIds) {
166654
+ if (excludeIds && excludeIds.length > 0) {
166655
+ const placeholders3 = excludeIds.map(() => "?").join(", ");
166656
+ const stmt2 = db.prepare(`SELECT c.id AS id,
166657
+ c.session_id AS sessionId,
166658
+ c.start_message AS startMessage,
166659
+ c.end_message AS endMessage,
166660
+ c.title AS title
166661
+ FROM compartments c
166662
+ JOIN session_projects sp
166663
+ ON sp.session_id = c.session_id
166664
+ AND sp.harness = c.harness
166665
+ AND sp.project_path = ?
166666
+ WHERE c.session_id = ?
166667
+ AND c.start_message IS NOT NULL
166668
+ AND c.end_message IS NOT NULL
166669
+ AND c.id NOT IN (${placeholders3})
166670
+ AND NOT EXISTS (
166671
+ SELECT 1
166672
+ FROM compartment_chunk_embeddings current
166673
+ WHERE current.compartment_id = c.id
166674
+ AND current.project_path = ?
166675
+ AND current.model_id = ?
166676
+ )
166677
+ ORDER BY c.start_message ASC, c.id ASC
166678
+ LIMIT ?`);
166679
+ const rows2 = stmt2.all(projectPath, sessionId, ...excludeIds, projectPath, modelId, Math.max(1, limit));
166680
+ return mapBackfillCandidateRows(rows2);
166681
+ }
166682
+ let stmt = sessionBackfillCandidateStatements.get(db);
166683
+ if (!stmt) {
166684
+ stmt = db.prepare(`SELECT c.id AS id,
166685
+ c.session_id AS sessionId,
166686
+ c.start_message AS startMessage,
166687
+ c.end_message AS endMessage,
166688
+ c.title AS title
166689
+ FROM compartments c
166690
+ JOIN session_projects sp
166691
+ ON sp.session_id = c.session_id
166692
+ AND sp.harness = c.harness
166693
+ AND sp.project_path = ?
166694
+ WHERE c.session_id = ?
166695
+ AND c.start_message IS NOT NULL
166696
+ AND c.end_message IS NOT NULL
166697
+ AND NOT EXISTS (
166698
+ SELECT 1
166699
+ FROM compartment_chunk_embeddings current
166700
+ WHERE current.compartment_id = c.id
166701
+ AND current.project_path = ?
166702
+ AND current.model_id = ?
166703
+ )
166704
+ ORDER BY c.start_message ASC, c.id ASC
166705
+ LIMIT ?`);
166706
+ sessionBackfillCandidateStatements.set(db, stmt);
166707
+ }
166708
+ const rows = stmt.all(projectPath, sessionId, projectPath, modelId, Math.max(1, limit));
166709
+ return mapBackfillCandidateRows(rows);
166710
+ }
166711
+ function countUnembeddedSessionCompartments(db, projectPath, sessionId, modelId) {
166712
+ const row = db.prepare(`SELECT COUNT(*) AS n
166713
+ FROM compartments c
166714
+ JOIN session_projects sp
166715
+ ON sp.session_id = c.session_id
166716
+ AND sp.harness = c.harness
166717
+ AND sp.project_path = ?
166718
+ WHERE c.session_id = ?
166719
+ AND c.start_message IS NOT NULL
166720
+ AND c.end_message IS NOT NULL
166721
+ AND NOT EXISTS (
166722
+ SELECT 1
166723
+ FROM compartment_chunk_embeddings current
166724
+ WHERE current.compartment_id = c.id
166725
+ AND current.project_path = ?
166726
+ AND current.model_id = ?
166727
+ )`).get(projectPath, sessionId, projectPath, modelId);
166728
+ return typeof row?.n === "number" ? row.n : 0;
166729
+ }
166730
+
165656
166731
  // ../plugin/src/features/magic-context/memory/cosine-similarity.ts
165657
166732
  function cosineSimilarity(a, b) {
165658
166733
  if (a.length !== b.length) {
@@ -165811,19 +166886,19 @@ function isArrayLikeNumber(value) {
165811
166886
  }
165812
166887
  return arr.length === 0 || typeof arr[0] === "number";
165813
166888
  }
165814
- function toFloat32Array2(values) {
166889
+ function toFloat32Array3(values) {
165815
166890
  return values instanceof Float32Array ? new Float32Array(values) : Float32Array.from(Array.from(values));
165816
166891
  }
165817
166892
  function extractBatchEmbeddings(result, expectedCount) {
165818
166893
  const { data } = result;
165819
166894
  if (Array.isArray(data) && data.length === expectedCount && data.every((entry) => typeof entry !== "number" && isArrayLikeNumber(entry))) {
165820
- return data.map((entry) => toFloat32Array2(entry));
166895
+ return data.map((entry) => toFloat32Array3(entry));
165821
166896
  }
165822
166897
  if (!isArrayLikeNumber(data)) {
165823
166898
  log("[magic-context] embedding batch returned unexpected data shape");
165824
166899
  return Array.from({ length: expectedCount }, () => null);
165825
166900
  }
165826
- const flatData = toFloat32Array2(data);
166901
+ const flatData = toFloat32Array3(data);
165827
166902
  const dimension = result.dims?.at(-1) ?? flatData.length / expectedCount;
165828
166903
  if (!Number.isInteger(dimension) || dimension <= 0 || flatData.length !== expectedCount * dimension) {
165829
166904
  log("[magic-context] embedding batch returned invalid dimensions");
@@ -165838,6 +166913,7 @@ function extractBatchEmbeddings(result, expectedCount) {
165838
166913
 
165839
166914
  class LocalEmbeddingProvider {
165840
166915
  modelId;
166916
+ maxInputTokens;
165841
166917
  model;
165842
166918
  pipeline = null;
165843
166919
  initPromise = null;
@@ -165845,8 +166921,9 @@ class LocalEmbeddingProvider {
165845
166921
  disposing = false;
165846
166922
  disposePromise = null;
165847
166923
  inFlightWaiters = [];
165848
- constructor(model = DEFAULT_LOCAL_EMBEDDING_MODEL) {
166924
+ constructor(model = DEFAULT_LOCAL_EMBEDDING_MODEL, maxInputTokens = 512) {
165849
166925
  this.model = model;
166926
+ this.maxInputTokens = maxInputTokens;
165850
166927
  this.modelId = getEmbeddingProviderIdentity({ provider: "local", model });
165851
166928
  }
165852
166929
  async initialize() {
@@ -166101,6 +167178,7 @@ var FETCH_TIMEOUT_MS = 30000;
166101
167178
 
166102
167179
  class OpenAICompatibleEmbeddingProvider {
166103
167180
  modelId;
167181
+ maxInputTokens;
166104
167182
  endpoint;
166105
167183
  model;
166106
167184
  apiKey;
@@ -166117,11 +167195,13 @@ class OpenAICompatibleEmbeddingProvider {
166117
167195
  this.apiKey = options.apiKey?.trim() ?? "";
166118
167196
  this.inputType = options.inputType?.trim() ?? "";
166119
167197
  this.truncate = options.truncate?.trim() ?? "";
167198
+ this.maxInputTokens = typeof options.maxInputTokens === "number" && Number.isFinite(options.maxInputTokens) ? Math.max(1, Math.floor(options.maxInputTokens)) : 512;
166120
167199
  this.modelId = getEmbeddingProviderIdentity({
166121
167200
  provider: "openai-compatible",
166122
167201
  endpoint: this.endpoint,
166123
167202
  model: this.model,
166124
- ...this.apiKey ? { api_key: this.apiKey } : {}
167203
+ ...this.apiKey ? { api_key: this.apiKey } : {},
167204
+ ...this.inputType ? { input_type: this.inputType } : {}
166125
167205
  });
166126
167206
  }
166127
167207
  async initialize() {
@@ -166312,7 +167392,7 @@ class OpenAICompatibleEmbeddingProvider {
166312
167392
  }
166313
167393
 
166314
167394
  // ../plugin/src/features/magic-context/project-embedding-registry.ts
166315
- import { createHash as createHash7, randomUUID } from "node:crypto";
167395
+ import { createHash as createHash9, randomUUID } from "node:crypto";
166316
167396
  init_logger();
166317
167397
 
166318
167398
  // ../plugin/src/features/magic-context/git-commits/storage-git-commit-embeddings.ts
@@ -166320,7 +167400,7 @@ var saveStatements = new WeakMap;
166320
167400
  var loadProjectStatements = new WeakMap;
166321
167401
  var loadUnembeddedStatements = new WeakMap;
166322
167402
  var countEmbeddedStatements = new WeakMap;
166323
- var clearProjectStatements = new WeakMap;
167403
+ var clearProjectStatements2 = new WeakMap;
166324
167404
  var distinctModelIdStatements = new WeakMap;
166325
167405
  function getSaveStatement(db) {
166326
167406
  let stmt = saveStatements.get(db);
@@ -166368,12 +167448,12 @@ function getCountEmbeddedStatement(db) {
166368
167448
  }
166369
167449
  return stmt;
166370
167450
  }
166371
- function getClearProjectStatement(db) {
166372
- let stmt = clearProjectStatements.get(db);
167451
+ function getClearProjectStatement2(db) {
167452
+ let stmt = clearProjectStatements2.get(db);
166373
167453
  if (!stmt) {
166374
167454
  stmt = db.prepare(`DELETE FROM git_commit_embeddings
166375
167455
  WHERE sha IN (SELECT sha FROM git_commits WHERE project_path = ?)`);
166376
- clearProjectStatements.set(db, stmt);
167456
+ clearProjectStatements2.set(db, stmt);
166377
167457
  }
166378
167458
  return stmt;
166379
167459
  }
@@ -166409,7 +167489,7 @@ function countEmbeddedCommits(db, projectPath) {
166409
167489
  return row?.count ?? 0;
166410
167490
  }
166411
167491
  function clearProjectCommitEmbeddings(db, projectPath) {
166412
- return getClearProjectStatement(db).run(projectPath).changes;
167492
+ return getClearProjectStatement2(db).run(projectPath).changes;
166413
167493
  }
166414
167494
  function getDistinctCommitEmbeddingModelIds(db, projectPath) {
166415
167495
  const rows = getDistinctModelIdStatement(db).all(projectPath);
@@ -166535,9 +167615,83 @@ function releaseGitSweepLease(db, projectPath, holderId) {
166535
167615
  });
166536
167616
  }
166537
167617
 
167618
+ // ../plugin/src/features/magic-context/session-project-storage.ts
167619
+ var upsertSessionProjectStatements = new WeakMap;
167620
+ var repairSessionChunkProjectStatements = new WeakMap;
167621
+ var repairProjectChunkProjectStatements = new WeakMap;
167622
+ function getUpsertSessionProjectStatement(db) {
167623
+ let stmt = upsertSessionProjectStatements.get(db);
167624
+ if (!stmt) {
167625
+ stmt = db.prepare(`INSERT INTO session_projects (session_id, harness, project_path, updated_at)
167626
+ VALUES (?, ?, ?, ?)
167627
+ ON CONFLICT(session_id, harness) DO UPDATE SET
167628
+ project_path = excluded.project_path,
167629
+ updated_at = excluded.updated_at
167630
+ WHERE session_projects.project_path <> excluded.project_path`);
167631
+ upsertSessionProjectStatements.set(db, stmt);
167632
+ }
167633
+ return stmt;
167634
+ }
167635
+ function getRepairSessionChunkProjectStatement(db) {
167636
+ let stmt = repairSessionChunkProjectStatements.get(db);
167637
+ if (!stmt) {
167638
+ stmt = db.prepare(`UPDATE compartment_chunk_embeddings
167639
+ SET project_path = ?
167640
+ WHERE session_id = ?
167641
+ AND harness = ?
167642
+ AND project_path <> ?`);
167643
+ repairSessionChunkProjectStatements.set(db, stmt);
167644
+ }
167645
+ return stmt;
167646
+ }
167647
+ function getRepairProjectChunkProjectStatement(db) {
167648
+ let stmt = repairProjectChunkProjectStatements.get(db);
167649
+ if (!stmt) {
167650
+ stmt = db.prepare(`UPDATE compartment_chunk_embeddings
167651
+ SET project_path = (
167652
+ SELECT sp.project_path
167653
+ FROM session_projects sp
167654
+ WHERE sp.session_id = compartment_chunk_embeddings.session_id
167655
+ AND sp.harness = compartment_chunk_embeddings.harness
167656
+ LIMIT 1
167657
+ )
167658
+ WHERE EXISTS (
167659
+ SELECT 1
167660
+ FROM session_projects sp
167661
+ WHERE sp.session_id = compartment_chunk_embeddings.session_id
167662
+ AND sp.harness = compartment_chunk_embeddings.harness
167663
+ AND sp.project_path <> compartment_chunk_embeddings.project_path
167664
+ AND (
167665
+ sp.project_path = ?
167666
+ OR compartment_chunk_embeddings.project_path = ?
167667
+ )
167668
+ )`);
167669
+ repairProjectChunkProjectStatements.set(db, stmt);
167670
+ }
167671
+ return stmt;
167672
+ }
167673
+ function recordSessionProjectIdentity(db, sessionId, projectPath) {
167674
+ if (!sessionId || !projectPath)
167675
+ return;
167676
+ const harness = getHarness();
167677
+ const now = Date.now();
167678
+ db.transaction(() => {
167679
+ getUpsertSessionProjectStatement(db).run(sessionId, harness, projectPath, now);
167680
+ getRepairSessionChunkProjectStatement(db).run(projectPath, sessionId, harness, projectPath);
167681
+ })();
167682
+ }
167683
+ function repairMisScopedCompartmentChunkEmbeddingsForProject(db, projectPath) {
167684
+ if (!projectPath)
167685
+ return 0;
167686
+ return getRepairProjectChunkProjectStatement(db).run(projectPath, projectPath).changes;
167687
+ }
167688
+
166538
167689
  // ../plugin/src/features/magic-context/project-embedding-registry.ts
166539
167690
  var OFF_PROVIDER_IDENTITY = "embedding-provider:off";
166540
167691
  var SWEEP_MAX_WALL_CLOCK_MS = 10 * 60 * 1000;
167692
+ var CHUNK_DRAIN_BATCH_SIZE = 8;
167693
+ var MAX_WINDOWS_PER_EMBED_CALL = 16;
167694
+ var SESSION_EMBED_LEASE_RENEWAL_MS = 60 * 1000;
166541
167695
  var projectRegistrations = new Map;
166542
167696
  var loadUnembeddedMemoriesStatements = new WeakMap;
166543
167697
  var globalRegistrationGeneration = 0;
@@ -166546,7 +167700,10 @@ function resolveEmbeddingConfig(config2) {
166546
167700
  if (!config2 || config2.provider === "local") {
166547
167701
  return {
166548
167702
  provider: "local",
166549
- model: config2?.model?.trim() || DEFAULT_LOCAL_EMBEDDING_MODEL
167703
+ model: config2?.model?.trim() || DEFAULT_LOCAL_EMBEDDING_MODEL,
167704
+ ...config2?.max_input_tokens ? {
167705
+ max_input_tokens: normalizeCompartmentChunkMaxInputTokens(config2.max_input_tokens)
167706
+ } : {}
166550
167707
  };
166551
167708
  }
166552
167709
  if (config2.provider === "openai-compatible") {
@@ -166559,7 +167716,10 @@ function resolveEmbeddingConfig(config2) {
166559
167716
  endpoint: config2.endpoint.trim(),
166560
167717
  ...apiKey ? { api_key: apiKey } : {},
166561
167718
  ...inputType ? { input_type: inputType } : {},
166562
- ...truncate ? { truncate } : {}
167719
+ ...truncate ? { truncate } : {},
167720
+ ...config2.max_input_tokens ? {
167721
+ max_input_tokens: normalizeCompartmentChunkMaxInputTokens(config2.max_input_tokens)
167722
+ } : {}
166563
167723
  };
166564
167724
  }
166565
167725
  return { provider: "off" };
@@ -166577,10 +167737,11 @@ function createProvider(config2) {
166577
167737
  model: config2.model,
166578
167738
  apiKey: config2.api_key,
166579
167739
  inputType: config2.input_type,
166580
- truncate: config2.truncate
167740
+ truncate: config2.truncate,
167741
+ maxInputTokens: config2.max_input_tokens
166581
167742
  });
166582
167743
  }
166583
- return new LocalEmbeddingProvider(config2.model);
167744
+ return new LocalEmbeddingProvider(config2.model, config2.max_input_tokens);
166584
167745
  }
166585
167746
  function stableStringify2(value) {
166586
167747
  if (Array.isArray(value)) {
@@ -166593,7 +167754,7 @@ function stableStringify2(value) {
166593
167754
  return JSON.stringify(value);
166594
167755
  }
166595
167756
  function sha256Prefix(value, length = 16) {
166596
- return createHash7("sha256").update(value).digest("hex").slice(0, length);
167757
+ return createHash9("sha256").update(value).digest("hex").slice(0, length);
166597
167758
  }
166598
167759
  function getRuntimeFingerprint(config2) {
166599
167760
  if (config2.provider === "off") {
@@ -166601,6 +167762,18 @@ function getRuntimeFingerprint(config2) {
166601
167762
  }
166602
167763
  return `${getEmbeddingProviderIdentity(config2)}:${sha256Prefix(stableStringify2(config2))}`;
166603
167764
  }
167765
+ function getChunkEmbeddingModelId(config2, providerIdentity) {
167766
+ if (config2.provider === "off") {
167767
+ return OFF_PROVIDER_IDENTITY;
167768
+ }
167769
+ const chunkIdentity = {
167770
+ providerIdentity,
167771
+ chunkerVersion: 1,
167772
+ maxInputTokens: normalizeCompartmentChunkMaxInputTokens("max_input_tokens" in config2 ? config2.max_input_tokens : undefined),
167773
+ truncate: config2.provider === "openai-compatible" ? config2.truncate ?? "" : ""
167774
+ };
167775
+ return `${providerIdentity}:chunk:${sha256Prefix(stableStringify2(chunkIdentity))}`;
167776
+ }
166604
167777
  function sameFeatures(a, b) {
166605
167778
  return a.memoryEnabled === b.memoryEnabled && a.gitCommitEnabled === b.gitCommitEnabled;
166606
167779
  }
@@ -166617,7 +167790,8 @@ function snapshotFor(registration) {
166617
167790
  features: { ...registration.features },
166618
167791
  enabled,
166619
167792
  gitCommitEnabled,
166620
- modelId: registration.observationMode || !providerIsOn ? "off" : registration.modelId
167793
+ modelId: registration.observationMode || !providerIsOn ? "off" : registration.modelId,
167794
+ chunkModelId: registration.observationMode || !providerIsOn ? "off" : registration.chunkModelId
166621
167795
  };
166622
167796
  }
166623
167797
  function disposeProvider(provider) {
@@ -166637,7 +167811,7 @@ function anyStoredModelIdIsStale(storedIds, currentId) {
166637
167811
  }
166638
167812
  return false;
166639
167813
  }
166640
- function maybeWipeStaleEmbeddings(db, projectIdentity, currentProviderIdentity, features) {
167814
+ function maybeWipeStaleEmbeddings(db, projectIdentity, currentProviderIdentity, currentChunkIdentity, features) {
166641
167815
  if (currentProviderIdentity === OFF_PROVIDER_IDENTITY) {
166642
167816
  return false;
166643
167817
  }
@@ -166658,6 +167832,14 @@ function maybeWipeStaleEmbeddings(db, projectIdentity, currentProviderIdentity,
166658
167832
  wiped = true;
166659
167833
  }
166660
167834
  }
167835
+ if (features.memoryEnabled) {
167836
+ repairMisScopedCompartmentChunkEmbeddingsForProject(db, projectIdentity);
167837
+ const chunkIds = getDistinctChunkEmbeddingModelIds(db, projectIdentity);
167838
+ if (anyStoredModelIdIsStale(chunkIds, currentChunkIdentity)) {
167839
+ clearChunkEmbeddingsForProject(db, projectIdentity);
167840
+ wiped = true;
167841
+ }
167842
+ }
166661
167843
  })();
166662
167844
  return wiped;
166663
167845
  }
@@ -166665,10 +167847,11 @@ function registerProjectEmbeddingAndMaybeWipe(db, projectIdentity, config2, feat
166665
167847
  const resolvedConfig = resolveEmbeddingConfig(config2);
166666
167848
  const providerIdentity = getEmbeddingProviderIdentity(resolvedConfig);
166667
167849
  const runtimeFingerprint = getRuntimeFingerprint(resolvedConfig);
167850
+ const chunkModelId = getChunkEmbeddingModelId(resolvedConfig, providerIdentity);
166668
167851
  const prior = projectRegistrations.get(projectIdentity);
166669
167852
  const canReuseProvider = prior !== undefined && !prior.observationMode && prior.runtimeFingerprint === runtimeFingerprint && prior.providerIdentity === providerIdentity;
166670
- const wiped = maybeWipeStaleEmbeddings(db, projectIdentity, providerIdentity, features);
166671
- const generationChanged = prior === undefined || prior.observationMode || prior.runtimeFingerprint !== runtimeFingerprint || !sameFeatures(prior.features, features) || wiped;
167853
+ const wiped = maybeWipeStaleEmbeddings(db, projectIdentity, providerIdentity, chunkModelId, features);
167854
+ const generationChanged = prior === undefined || prior.observationMode || prior.runtimeFingerprint !== runtimeFingerprint || prior.chunkModelId !== chunkModelId || !sameFeatures(prior.features, features) || wiped;
166672
167855
  const generation = generationChanged ? ++globalRegistrationGeneration : prior.generation;
166673
167856
  const registration = {
166674
167857
  projectIdentity,
@@ -166680,6 +167863,7 @@ function registerProjectEmbeddingAndMaybeWipe(db, projectIdentity, config2, feat
166680
167863
  generation,
166681
167864
  features: { ...features },
166682
167865
  modelId: providerIdentity === OFF_PROVIDER_IDENTITY ? "off" : providerIdentity,
167866
+ chunkModelId: providerIdentity === OFF_PROVIDER_IDENTITY ? "off" : chunkModelId,
166683
167867
  observationMode: false
166684
167868
  };
166685
167869
  projectRegistrations.set(projectIdentity, registration);
@@ -166702,6 +167886,7 @@ function registerProjectInObservationMode(db, projectIdentity, sourceDirectory,
166702
167886
  generation,
166703
167887
  features: { memoryEnabled: false, gitCommitEnabled: false },
166704
167888
  modelId: "off",
167889
+ chunkModelId: "off",
166705
167890
  observationMode: true
166706
167891
  };
166707
167892
  projectRegistrations.set(projectIdentity, registration);
@@ -166712,6 +167897,15 @@ function getProjectEmbeddingSnapshot(projectIdentity) {
166712
167897
  const registration = projectRegistrations.get(projectIdentity);
166713
167898
  return registration ? snapshotFor(registration) : null;
166714
167899
  }
167900
+ function getProjectChunkEmbeddingModelId(projectIdentity) {
167901
+ const registration = projectRegistrations.get(projectIdentity);
167902
+ return registration && !registration.observationMode ? registration.chunkModelId : "off";
167903
+ }
167904
+ function getProjectEmbeddingMaxInputTokens(projectIdentity) {
167905
+ const registration = projectRegistrations.get(projectIdentity);
167906
+ const configMax = registration?.config && "max_input_tokens" in registration.config ? registration.config.max_input_tokens : undefined;
167907
+ return normalizeCompartmentChunkMaxInputTokens(registration?.provider?.maxInputTokens ?? configMax);
167908
+ }
166715
167909
  function getOrCreateProjectProvider(registration) {
166716
167910
  if (registration.providerIdentity === OFF_PROVIDER_IDENTITY || registration.observationMode) {
166717
167911
  return null;
@@ -166806,6 +168000,131 @@ async function embedUnembeddedMemoriesForProject(db, projectIdentity, batchSize
166806
168000
  return 0;
166807
168001
  }
166808
168002
  }
168003
+ async function embedCandidateChunkBatch(db, projectIdentity, modelId, candidates, signal) {
168004
+ const noWork = [];
168005
+ if (candidates.length === 0)
168006
+ return { embedded: 0, noWork };
168007
+ const maxInputTokens = getProjectEmbeddingMaxInputTokens(projectIdentity);
168008
+ const prepared = [];
168009
+ for (const candidate of candidates) {
168010
+ const canonicalText = buildCanonicalChunkTextFromFts(db, candidate.sessionId, candidate.startMessage, candidate.endMessage);
168011
+ if (canonicalText.length === 0) {
168012
+ noWork.push(candidate.id);
168013
+ continue;
168014
+ }
168015
+ const windows = chunkCanonicalText(canonicalText, candidate.startMessage, candidate.endMessage, maxInputTokens);
168016
+ if (windows.length === 0 || chunkEmbeddingWindowsAreCurrent(db, candidate.id, modelId, windows, projectIdentity)) {
168017
+ noWork.push(candidate.id);
168018
+ continue;
168019
+ }
168020
+ prepared.push({ candidate, windows });
168021
+ }
168022
+ if (prepared.length === 0)
168023
+ return { embedded: 0, noWork };
168024
+ let embedded = 0;
168025
+ let i = 0;
168026
+ while (i < prepared.length) {
168027
+ if (signal?.aborted)
168028
+ break;
168029
+ const slice = [];
168030
+ let windowCount = 0;
168031
+ do {
168032
+ const item = prepared[i];
168033
+ slice.push(item);
168034
+ windowCount += item.windows.length;
168035
+ i += 1;
168036
+ } while (i < prepared.length && windowCount + prepared[i].windows.length <= MAX_WINDOWS_PER_EMBED_CALL);
168037
+ const texts = [];
168038
+ for (const item of slice)
168039
+ texts.push(...item.windows.map((w) => w.text));
168040
+ try {
168041
+ const result = await embedBatchForProject(projectIdentity, texts, signal);
168042
+ if (!result)
168043
+ continue;
168044
+ if (signal?.aborted)
168045
+ break;
168046
+ let offset = 0;
168047
+ for (const item of slice) {
168048
+ const vectors = result.vectors.slice(offset, offset + item.windows.length);
168049
+ offset += item.windows.length;
168050
+ if (vectors.length !== item.windows.length || vectors.some((v) => !v)) {
168051
+ continue;
168052
+ }
168053
+ const rows = item.windows.map((window, index) => ({
168054
+ compartmentId: item.candidate.id,
168055
+ sessionId: item.candidate.sessionId,
168056
+ projectPath: projectIdentity,
168057
+ window,
168058
+ modelId,
168059
+ vector: vectors[index]
168060
+ }));
168061
+ replaceCompartmentChunkEmbeddings(db, rows);
168062
+ embedded += 1;
168063
+ }
168064
+ } catch (error51) {
168065
+ log("[magic-context] failed to proactively embed compartment chunks:", error51);
168066
+ }
168067
+ }
168068
+ return { embedded, noWork };
168069
+ }
168070
+ async function embedSessionCompartmentChunks(db, projectIdentity, sessionId, options) {
168071
+ const snapshot = getProjectEmbeddingSnapshot(projectIdentity);
168072
+ if (!snapshot?.enabled || snapshot.chunkModelId === "off") {
168073
+ return { status: "disabled", embedded: 0, total: 0 };
168074
+ }
168075
+ recordSessionProjectIdentity(db, sessionId, projectIdentity);
168076
+ const total = countUnembeddedSessionCompartments(db, projectIdentity, sessionId, snapshot.chunkModelId);
168077
+ if (total === 0)
168078
+ return { status: "nothing", embedded: 0, total: 0 };
168079
+ const holderId = `session-embed-${randomUUID()}`;
168080
+ const lease2 = acquireGitSweepLease(db, projectIdentity, holderId, { ignoreCooldown: true });
168081
+ if (!lease2.acquired)
168082
+ return { status: "busy", embedded: 0, total };
168083
+ const renewal = setInterval(() => {
168084
+ try {
168085
+ renewGitSweepLease(db, projectIdentity, holderId);
168086
+ } catch {}
168087
+ }, SESSION_EMBED_LEASE_RENEWAL_MS);
168088
+ renewal.unref?.();
168089
+ const batchSize = Math.max(1, options?.batchSize ?? CHUNK_DRAIN_BATCH_SIZE);
168090
+ const skipIds = [];
168091
+ let embedded = 0;
168092
+ let aborted2 = false;
168093
+ let providerStalled = false;
168094
+ try {
168095
+ options?.onProgress?.({ embedded, total });
168096
+ for (;; ) {
168097
+ if (options?.signal?.aborted) {
168098
+ aborted2 = true;
168099
+ break;
168100
+ }
168101
+ const candidates = loadUnembeddedSessionChunkCandidates(db, projectIdentity, sessionId, snapshot.chunkModelId, batchSize, skipIds);
168102
+ if (candidates.length === 0)
168103
+ break;
168104
+ const { embedded: n, noWork } = await embedCandidateChunkBatch(db, projectIdentity, snapshot.chunkModelId, candidates, options?.signal);
168105
+ for (const id of noWork)
168106
+ skipIds.push(id);
168107
+ if (n === 0 && noWork.length === 0) {
168108
+ providerStalled = true;
168109
+ break;
168110
+ }
168111
+ embedded += n;
168112
+ options?.onProgress?.({ embedded: Math.min(embedded, total), total });
168113
+ await new Promise((resolve4) => setTimeout(resolve4, 0));
168114
+ }
168115
+ } finally {
168116
+ clearInterval(renewal);
168117
+ releaseGitSweepLease(db, projectIdentity, holderId);
168118
+ }
168119
+ if (aborted2)
168120
+ return { status: "aborted", embedded, total };
168121
+ if (providerStalled) {
168122
+ const remaining = Math.max(0, countUnembeddedSessionCompartments(db, projectIdentity, sessionId, snapshot.chunkModelId) - skipIds.length);
168123
+ if (remaining > 0)
168124
+ return { status: "stalled", embedded, total, remaining };
168125
+ }
168126
+ return { status: "done", embedded, total };
168127
+ }
166809
168128
 
166810
168129
  // ../plugin/src/features/magic-context/memory/embedding.ts
166811
168130
  var DEFAULT_EMBEDDING_CONFIG = {
@@ -166825,10 +168144,11 @@ function createProvider2(config2) {
166825
168144
  model: config2.model,
166826
168145
  apiKey: config2.api_key,
166827
168146
  inputType: config2.input_type,
166828
- truncate: config2.truncate
168147
+ truncate: config2.truncate,
168148
+ maxInputTokens: config2.max_input_tokens
166829
168149
  });
166830
168150
  }
166831
- return new LocalEmbeddingProvider(config2.model);
168151
+ return new LocalEmbeddingProvider(config2.model, config2.max_input_tokens);
166832
168152
  }
166833
168153
  function getOrCreateProvider() {
166834
168154
  if (provider) {
@@ -167071,6 +168391,7 @@ init_logger();
167071
168391
  // ../plugin/src/features/magic-context/memory/storage-memory-fts.ts
167072
168392
  var DEFAULT_SEARCH_LIMIT = 10;
167073
168393
  var searchStatements = new WeakMap;
168394
+ var unionSearchStatements = new Map;
167074
168395
  function getSearchStatement(db) {
167075
168396
  let stmt = searchStatements.get(db);
167076
168397
  if (!stmt) {
@@ -167079,6 +168400,23 @@ function getSearchStatement(db) {
167079
168400
  }
167080
168401
  return stmt;
167081
168402
  }
168403
+ function getUnionSearchStatement(db, arity) {
168404
+ let statements = unionSearchStatements.get(arity);
168405
+ if (!statements) {
168406
+ statements = new WeakMap;
168407
+ unionSearchStatements.set(arity, statements);
168408
+ }
168409
+ let stmt = statements.get(db);
168410
+ if (!stmt) {
168411
+ const placeholders3 = Array.from({ length: arity }, () => "?").join(", ");
168412
+ stmt = db.prepare(`SELECT ${getMemorySelectColumns(db)} FROM memories_fts INNER JOIN memories ON memories.id = memories_fts.rowid WHERE memories.project_path IN (${placeholders3}) AND memories.status IN ('active', 'permanent') AND (memories.expires_at IS NULL OR memories.expires_at > ?) AND memories_fts MATCH ? ORDER BY bm25(memories_fts), memories.updated_at DESC, memories.id ASC LIMIT ?`);
168413
+ statements.set(db, stmt);
168414
+ }
168415
+ return stmt;
168416
+ }
168417
+ function uniqueProjectPaths2(projectPaths) {
168418
+ return [...new Set(projectPaths.filter((path7) => path7.length > 0))];
168419
+ }
167082
168420
  function sanitizeFtsQuery(query) {
167083
168421
  const tokens = query.split(/\s+/).filter((token) => token.length > 0);
167084
168422
  if (tokens.length === 0)
@@ -167097,6 +168435,28 @@ function searchMemoriesFTS(db, projectPath, query, limit = DEFAULT_SEARCH_LIMIT)
167097
168435
  const rows = getSearchStatement(db).all(projectPath, Date.now(), sanitized, limit).filter(isMemoryRow);
167098
168436
  return rows.map(toMemory);
167099
168437
  }
168438
+ function searchMemoriesFTSUnion(db, projectPaths, query, limit = DEFAULT_SEARCH_LIMIT, ownIdentities, shareCategories) {
168439
+ const identities = uniqueProjectPaths2(projectPaths);
168440
+ if (identities.length === 0)
168441
+ return [];
168442
+ const sharingFilter = buildWorkspaceMemorySqlFilter({
168443
+ identities,
168444
+ ownIdentities,
168445
+ shareCategories,
168446
+ tableName: "memories"
168447
+ });
168448
+ if (identities.length === 1 && !sharingFilter.active) {
168449
+ return searchMemoriesFTS(db, identities[0], query, limit);
168450
+ }
168451
+ const trimmedQuery = query.trim();
168452
+ if (trimmedQuery.length === 0 || limit <= 0)
168453
+ return [];
168454
+ const sanitized = sanitizeFtsQuery(trimmedQuery);
168455
+ if (sanitized.length === 0)
168456
+ return [];
168457
+ const rows = sharingFilter.active ? db.prepare(`SELECT ${getMemorySelectColumns(db)} FROM memories_fts INNER JOIN memories ON memories.id = memories_fts.rowid WHERE memories.project_path IN (${identities.map(() => "?").join(", ")}) AND memories.status IN ('active', 'permanent') AND (memories.expires_at IS NULL OR memories.expires_at > ?) AND memories_fts MATCH ?${sharingFilter.clause} ORDER BY bm25(memories_fts), memories.updated_at DESC, memories.id ASC LIMIT ?`).all(...identities, Date.now(), sanitized, ...sharingFilter.params, limit).filter(isMemoryRow) : getUnionSearchStatement(db, identities.length).all(...identities, Date.now(), sanitized, limit).filter(isMemoryRow);
168458
+ return rows.map(toMemory);
168459
+ }
167100
168460
 
167101
168461
  // ../plugin/src/features/magic-context/git-commits/search-git-commits.ts
167102
168462
  var ftsStatements = new WeakMap;
@@ -167389,7 +168749,7 @@ async function sweepGitCommits(args) {
167389
168749
  }
167390
168750
 
167391
168751
  // ../plugin/src/plugin/embedding-bootstrap-helpers.ts
167392
- import { createHash as createHash8 } from "node:crypto";
168752
+ import { createHash as createHash10 } from "node:crypto";
167393
168753
  init_logger();
167394
168754
  var EMBEDDING_AFFECTING_KEYS = new Set([
167395
168755
  "embedding.api_key",
@@ -167411,7 +168771,7 @@ var EMBEDDING_WARNING_TERMS = [
167411
168771
  ];
167412
168772
  var loggedFailureSignatures = new Map;
167413
168773
  function sha256Prefix2(value, length = 16) {
167414
- return createHash8("sha256").update(value).digest("hex").slice(0, length);
168774
+ return createHash10("sha256").update(value).digest("hex").slice(0, length);
167415
168775
  }
167416
168776
  function warningLooksEmbeddingRelated(message) {
167417
168777
  const lower = message.toLowerCase();
@@ -168457,6 +169817,88 @@ function registerCtxDreamCommand(pi, deps) {
168457
169817
  });
168458
169818
  }
168459
169819
 
169820
+ // src/commands/ctx-embed-history.ts
169821
+ function registerCtxEmbedHistoryCommand(pi, deps) {
169822
+ pi.registerCommand("ctx-embed-history", {
169823
+ description: "Embed all of this session's history compartments for semantic search, in one pass",
169824
+ handler: async (_args, ctx) => {
169825
+ const sessionId = resolveSessionId(ctx);
169826
+ if (!sessionId) {
169827
+ sendCtxStatusMessage(pi, {
169828
+ title: "/ctx-embed-history",
169829
+ text: `## /ctx-embed-history
169830
+
169831
+ No active Pi session is available.`,
169832
+ level: "error"
169833
+ });
169834
+ return;
169835
+ }
169836
+ if (deps.memoryEnabled === false) {
169837
+ sendCtxStatusMessage(pi, {
169838
+ title: "/ctx-embed-history",
169839
+ text: `## /ctx-embed-history
169840
+
169841
+ Memory is disabled for this project, so there is no semantic embedding to backfill.`,
169842
+ level: "info"
169843
+ });
169844
+ return;
169845
+ }
169846
+ const project = deps.resolveProject?.(ctx) ?? {
169847
+ projectDir: deps.projectDir,
169848
+ projectIdentity: deps.projectIdentity
169849
+ };
169850
+ await ensureProjectRegisteredFromPiDirectory(project.projectDir, deps.db);
169851
+ const outcome = await embedSessionCompartmentChunks(deps.db, project.projectIdentity, sessionId);
169852
+ const { text, level } = (() => {
169853
+ switch (outcome.status) {
169854
+ case "nothing":
169855
+ return {
169856
+ text: `## /ctx-embed-history
169857
+
169858
+ All of this session's history is already embedded.`,
169859
+ level: "info"
169860
+ };
169861
+ case "disabled":
169862
+ return {
169863
+ text: `## /ctx-embed-history
169864
+
169865
+ No embedding provider is configured, so there is nothing to embed.`,
169866
+ level: "info"
169867
+ };
169868
+ case "busy":
169869
+ return {
169870
+ text: `## /ctx-embed-history
169871
+
169872
+ Embedding is already running for this project — ${outcome.total} compartment${outcome.total === 1 ? "" : "s"} still pending. Try again shortly.`,
169873
+ level: "info"
169874
+ };
169875
+ case "stalled":
169876
+ return {
169877
+ text: `## /ctx-embed-history
169878
+
169879
+ Embedded ${outcome.embedded} compartment${outcome.embedded === 1 ? "" : "s"}; ${outcome.remaining} could not be embedded (the provider returned no result). Run /ctx-embed-history again to retry them.`,
169880
+ level: "info"
169881
+ };
169882
+ default:
169883
+ return {
169884
+ text: `## /ctx-embed-history
169885
+
169886
+ Embedded ${outcome.embedded} compartment${outcome.embedded === 1 ? "" : "s"} of history for semantic search.`,
169887
+ level: "success"
169888
+ };
169889
+ }
169890
+ })();
169891
+ sendCtxStatusMessage(pi, { title: "/ctx-embed-history", text, level }, {
169892
+ sessionId,
169893
+ projectIdentity: project.projectIdentity,
169894
+ status: outcome.status,
169895
+ embedded: outcome.embedded,
169896
+ total: outcome.total
169897
+ });
169898
+ }
169899
+ });
169900
+ }
169901
+
168460
169902
  // ../plugin/src/hooks/magic-context/execute-flush.ts
168461
169903
  init_logger();
168462
169904
  function executeFlush(db, sessionId) {
@@ -170592,53 +172034,6 @@ ${body}
170592
172034
  </system-reminder>`;
170593
172035
  }
170594
172036
 
170595
- // ../plugin/src/features/magic-context/memory/constants.ts
170596
- var V2_MEMORY_CATEGORIES = [
170597
- "PROJECT_RULES",
170598
- "ARCHITECTURE",
170599
- "CONSTRAINTS",
170600
- "CONFIG_VALUES",
170601
- "NAMING"
170602
- ];
170603
- var PROMOTABLE_CATEGORIES = [
170604
- "PROJECT_RULES",
170605
- "ARCHITECTURE",
170606
- "CONSTRAINTS",
170607
- "CONFIG_VALUES",
170608
- "NAMING",
170609
- "ARCHITECTURE_DECISIONS",
170610
- "CONFIG_DEFAULTS",
170611
- "USER_PREFERENCES",
170612
- "USER_DIRECTIVES",
170613
- "ENVIRONMENT",
170614
- "WORKFLOW_RULES",
170615
- "KNOWN_ISSUES"
170616
- ];
170617
- var CATEGORY_PRIORITY = [
170618
- "PROJECT_RULES",
170619
- "ARCHITECTURE",
170620
- "CONSTRAINTS",
170621
- "CONFIG_VALUES",
170622
- "NAMING",
170623
- "USER_DIRECTIVES",
170624
- "USER_PREFERENCES",
170625
- "CONFIG_DEFAULTS",
170626
- "ARCHITECTURE_DECISIONS",
170627
- "ENVIRONMENT",
170628
- "WORKFLOW_RULES",
170629
- "KNOWN_ISSUES"
170630
- ];
170631
- var MEMORY_CATEGORY_ORDER_UNKNOWN = 99;
170632
- var MEMORY_CATEGORY_ORDER_PRIORITY = CATEGORY_PRIORITY.reduce((acc, category, index) => {
170633
- acc[category] = index;
170634
- return acc;
170635
- }, {});
170636
- var MEMORY_CATEGORY_ORDER_SQL = `CASE category ${CATEGORY_PRIORITY.map((category, index) => `WHEN '${category}' THEN ${index}`).join(" ")} ELSE ${MEMORY_CATEGORY_ORDER_UNKNOWN} END`;
170637
- var CATEGORY_DEFAULT_TTL = {
170638
- WORKFLOW_RULES: 90 * 24 * 60 * 60 * 1000,
170639
- KNOWN_ISSUES: 30 * 24 * 60 * 60 * 1000
170640
- };
170641
-
170642
172037
  // ../plugin/src/shared/bounded-session-map.ts
170643
172038
  class BoundedSessionMap {
170644
172039
  maxEntries;
@@ -171049,26 +172444,40 @@ ${sections.join(`
171049
172444
  var DEFAULT_MEMORY_BUDGET_TOKENS = 8000;
171050
172445
  var MEMORY_BLOCK_WRAPPER_TOKENS = 6;
171051
172446
  var DEFAULT_USER_PROFILE_BUDGET_TOKENS = 4000;
172447
+ function memoryCanonicalIdentity(memory, workspace) {
172448
+ return resolveStoredPathWorkspaceIdentity(memory.projectPath, workspace.identities, workspace.canonicalIdentityByStoredPath);
172449
+ }
172450
+ function memorySelectionOrder(left, right) {
172451
+ if (left.status === "permanent" && right.status !== "permanent")
172452
+ return -1;
172453
+ if (right.status === "permanent" && left.status !== "permanent")
172454
+ return 1;
172455
+ const leftImportance = left.importance ?? Number.NEGATIVE_INFINITY;
172456
+ const rightImportance = right.importance ?? Number.NEGATIVE_INFINITY;
172457
+ const importanceDiff = rightImportance - leftImportance;
172458
+ if (importanceDiff !== 0)
172459
+ return importanceDiff;
172460
+ return left.id - right.id;
172461
+ }
172462
+ function memoryRenderOrder(left, right) {
172463
+ const aPriority = MEMORY_CATEGORY_ORDER_PRIORITY[left.category] ?? MEMORY_CATEGORY_ORDER_UNKNOWN;
172464
+ const bPriority = MEMORY_CATEGORY_ORDER_PRIORITY[right.category] ?? MEMORY_CATEGORY_ORDER_UNKNOWN;
172465
+ const categoryDiff = aPriority - bPriority;
172466
+ if (categoryDiff !== 0)
172467
+ return categoryDiff;
172468
+ return left.id - right.id;
172469
+ }
171052
172470
  var maxCompartmentSeqStatements = new WeakMap;
171053
172471
  var maxMemoryIdStatements = new WeakMap;
171054
172472
  var legacyCompartmentCountStatements = new WeakMap;
171055
172473
  var m0CompartmentStatements = new WeakMap;
171056
172474
  var newCompartmentStatements = new WeakMap;
171057
- function trimMemoriesToBudgetV2(sessionId, memories, budgetTokens) {
171058
- const selectionOrder = [...memories].sort((a, b) => {
171059
- if (a.status === "permanent" && b.status !== "permanent")
171060
- return -1;
171061
- if (b.status === "permanent" && a.status !== "permanent")
171062
- return 1;
171063
- const importanceDiff = (b.importance ?? 50) - (a.importance ?? 50);
171064
- if (importanceDiff !== 0)
171065
- return importanceDiff;
171066
- return a.id - b.id;
171067
- });
172475
+ function trimMemoriesToBudgetV2(sessionId, memories, budgetTokens, renderOptions = {}) {
172476
+ const selectionOrder = [...memories].sort(memorySelectionOrder);
171068
172477
  const selected = [];
171069
172478
  let usedTokens = MEMORY_BLOCK_WRAPPER_TOKENS;
171070
172479
  for (const memory of selectionOrder) {
171071
- const memoryTokens = estimateTokens(renderMemoryLineV2(memory));
172480
+ const memoryTokens = estimateTokens(renderMemoryLineV2(memory, renderOptions.sourceNameByMemoryId?.get(memory.id)));
171072
172481
  if (usedTokens + memoryTokens > budgetTokens)
171073
172482
  continue;
171074
172483
  selected.push(memory);
@@ -171077,16 +172486,70 @@ function trimMemoriesToBudgetV2(sessionId, memories, budgetTokens) {
171077
172486
  if (selected.length < memories.length) {
171078
172487
  sessionLog(sessionId, `v2 trimmed memories from ${memories.length} to ${selected.length} to fit injection budget of ${budgetTokens} tokens`);
171079
172488
  }
171080
- const renderOrder = [...selected].sort((a, b) => {
171081
- const aPriority = MEMORY_CATEGORY_ORDER_PRIORITY[a.category] ?? MEMORY_CATEGORY_ORDER_UNKNOWN;
171082
- const bPriority = MEMORY_CATEGORY_ORDER_PRIORITY[b.category] ?? MEMORY_CATEGORY_ORDER_UNKNOWN;
171083
- const categoryDiff = aPriority - bPriority;
171084
- if (categoryDiff !== 0)
171085
- return categoryDiff;
171086
- return a.id - b.id;
171087
- });
172489
+ const renderOrder = [...selected].sort(memoryRenderOrder);
171088
172490
  return { selected, renderOrder };
171089
172491
  }
172492
+ function trimWorkspaceMemoriesToBudgetV2(sessionId, memories, budgetTokens, workspace, renderOptions = {}) {
172493
+ if (!workspace.isWorkspaced) {
172494
+ return trimMemoriesToBudgetV2(sessionId, memories, budgetTokens, renderOptions);
172495
+ }
172496
+ const selected = [];
172497
+ const selectedIds = new Set;
172498
+ let usedTokens = MEMORY_BLOCK_WRAPPER_TOKENS;
172499
+ const tokenCost = (memory) => estimateTokens(renderMemoryLineV2(memory, renderOptions.sourceNameByMemoryId?.get(memory.id)));
172500
+ const trySelect = (memory) => {
172501
+ if (selectedIds.has(memory.id))
172502
+ return false;
172503
+ const tokens = tokenCost(memory);
172504
+ if (usedTokens + tokens > budgetTokens)
172505
+ return false;
172506
+ selected.push(memory);
172507
+ selectedIds.add(memory.id);
172508
+ usedTokens += tokens;
172509
+ return true;
172510
+ };
172511
+ for (const memory of memories.filter((candidate) => candidate.status === "permanent").sort(memorySelectionOrder)) {
172512
+ trySelect(memory);
172513
+ }
172514
+ const remainingAfterPermanent = Math.max(0, budgetTokens - usedTokens);
172515
+ const floorTokens = remainingAfterPermanent / Math.max(1, workspace.identities.length);
172516
+ const byIdentity = new Map;
172517
+ for (const memory of memories) {
172518
+ if (memory.status === "permanent")
172519
+ continue;
172520
+ const identity = memoryCanonicalIdentity(memory, workspace);
172521
+ if (!identity)
172522
+ continue;
172523
+ const list = byIdentity.get(identity) ?? [];
172524
+ list.push(memory);
172525
+ byIdentity.set(identity, list);
172526
+ }
172527
+ for (const identity of workspace.identities) {
172528
+ let memberTokens = 0;
172529
+ const candidates = (byIdentity.get(identity) ?? []).sort(memorySelectionOrder);
172530
+ for (const memory of candidates) {
172531
+ if (selectedIds.has(memory.id))
172532
+ continue;
172533
+ const tokens = tokenCost(memory);
172534
+ if (memberTokens + tokens > floorTokens)
172535
+ continue;
172536
+ if (usedTokens + tokens > budgetTokens)
172537
+ continue;
172538
+ selected.push(memory);
172539
+ selectedIds.add(memory.id);
172540
+ usedTokens += tokens;
172541
+ memberTokens += tokens;
172542
+ }
172543
+ }
172544
+ const remaining = memories.filter((memory) => !selectedIds.has(memory.id)).sort(memorySelectionOrder);
172545
+ for (const memory of remaining) {
172546
+ trySelect(memory);
172547
+ }
172548
+ if (selected.length < memories.length) {
172549
+ sessionLog(sessionId, `v2 trimmed memories from ${memories.length} to ${selected.length} to fit injection budget of ${budgetTokens} tokens`);
172550
+ }
172551
+ return { selected, renderOrder: [...selected].sort(memoryRenderOrder) };
172552
+ }
171090
172553
  function trimUserMemoriesToBudget(memories, budgetTokens) {
171091
172554
  const selected = [];
171092
172555
  let usedTokens = 0;
@@ -171099,15 +172562,16 @@ function trimUserMemoriesToBudget(memories, budgetTokens) {
171099
172562
  }
171100
172563
  return selected;
171101
172564
  }
171102
- function renderMemoryLineV2(memory) {
171103
- return ` <memory id="${memory.id}" category="${escapeXmlAttr(memory.category)}" importance="${memory.importance ?? 50}">${escapeXmlContent(memory.content)}</memory>`;
172565
+ function renderMemoryLineV2(memory, sourceName) {
172566
+ const sourceAttr = sourceName ? ` source="${escapeXmlAttr(sourceName)}"` : "";
172567
+ return ` <memory id="${memory.id}" category="${escapeXmlAttr(memory.category)}"${sourceAttr} importance="${memory.importance ?? 50}">${escapeXmlContent(memory.content)}</memory>`;
171104
172568
  }
171105
- function renderMemoryBlockV2(memories, wrapper = "project-memory") {
172569
+ function renderMemoryBlockV2(memories, wrapper = "project-memory", renderOptions = {}) {
171106
172570
  if (memories.length === 0)
171107
172571
  return "";
171108
172572
  const lines = [`<${wrapper}>`];
171109
172573
  for (const memory of memories) {
171110
- lines.push(renderMemoryLineV2(memory));
172574
+ lines.push(renderMemoryLineV2(memory, renderOptions.sourceNameByMemoryId?.get(memory.id)));
171111
172575
  }
171112
172576
  lines.push(`</${wrapper}>`);
171113
172577
  return lines.join(`
@@ -171701,7 +173165,7 @@ async function ensureMemoryEmbeddings(args) {
171701
173165
  continue;
171702
173166
  }
171703
173167
  saveEmbedding(args.db, memory.id, embedding, result.modelId);
171704
- staged.set(memory.id, embedding);
173168
+ staged.set(memory.id, { embedding, modelId: result.modelId });
171705
173169
  }
171706
173170
  })();
171707
173171
  const currentSnapshot = getProjectEmbeddingSnapshot(args.projectIdentity);
@@ -171792,6 +173256,37 @@ function previewText(text) {
171792
173256
  }
171793
173257
  return `${normalized.slice(0, RESULT_PREVIEW_LIMIT - 1).trimEnd()}…`;
171794
173258
  }
173259
+ function resolveSearchWorkspaceContext(db, projectPath, identitySet) {
173260
+ const resolved = identitySet ?? resolveWorkspaceIdentitySet(db, projectPath);
173261
+ const isWorkspaced = resolved.identities.length > 1;
173262
+ const expanded = expandWorkspaceIdentitySetWithAliases(db, resolved.identities);
173263
+ const expandedIdentities = isWorkspaced ? expanded.expandedIdentities : resolved.identities;
173264
+ const canonicalIdentityByStoredPath = isWorkspaced ? expanded.canonicalIdentityByStoredPath : new Map(resolved.identities.map((identity) => [identity, identity]));
173265
+ const ownIdentities = expandedIdentities.filter((identity) => canonicalIdentityByStoredPath.get(identity) === projectPath);
173266
+ return {
173267
+ identities: resolved.identities,
173268
+ expandedIdentities,
173269
+ ownIdentities,
173270
+ shareCategories: isWorkspaced ? resolveWorkspaceShareCategories(db, projectPath) : null,
173271
+ namesByIdentity: resolved.namesByIdentity,
173272
+ canonicalIdentityByStoredPath,
173273
+ isWorkspaced
173274
+ };
173275
+ }
173276
+ function memoryWorkspaceIdentity(memory, workspace) {
173277
+ return resolveStoredPathWorkspaceIdentity(memory.projectPath, workspace.identities, workspace.canonicalIdentityByStoredPath);
173278
+ }
173279
+ function sourceNamesForSearchMemories(args) {
173280
+ if (!args.workspace.isWorkspaced)
173281
+ return;
173282
+ const sourceNames = new Map;
173283
+ for (const memory of args.memories) {
173284
+ const source = sourceNameForMemory(memory.projectPath, args.projectPath, args.workspace.identities, args.workspace.namesByIdentity, args.workspace.canonicalIdentityByStoredPath);
173285
+ if (source)
173286
+ sourceNames.set(memory.id, source);
173287
+ }
173288
+ return sourceNames.size > 0 ? sourceNames : undefined;
173289
+ }
171795
173290
  function getMessageSearchStatement(db) {
171796
173291
  let stmt = messageSearchStatements.get(db);
171797
173292
  if (!stmt) {
@@ -171800,6 +173295,30 @@ function getMessageSearchStatement(db) {
171800
173295
  }
171801
173296
  return stmt;
171802
173297
  }
173298
+ var ftsRowCountStatements = new WeakMap;
173299
+ var ftsMatchCountStatements = new WeakMap;
173300
+ function getSessionFtsRowCount(db, sessionId) {
173301
+ let stmt = ftsRowCountStatements.get(db);
173302
+ if (!stmt) {
173303
+ stmt = db.prepare("SELECT COUNT(*) AS n FROM message_history_fts WHERE session_id = ?");
173304
+ ftsRowCountStatements.set(db, stmt);
173305
+ }
173306
+ const row = stmt.get(sessionId);
173307
+ return typeof row?.n === "number" ? row.n : 0;
173308
+ }
173309
+ function countSessionFtsMatches(db, sessionId, ftsQuery) {
173310
+ let stmt = ftsMatchCountStatements.get(db);
173311
+ if (!stmt) {
173312
+ stmt = db.prepare("SELECT COUNT(*) AS n FROM message_history_fts WHERE session_id = ? AND message_history_fts MATCH ?");
173313
+ ftsMatchCountStatements.set(db, stmt);
173314
+ }
173315
+ try {
173316
+ const row = stmt.get(sessionId, ftsQuery);
173317
+ return typeof row?.n === "number" ? row.n : 0;
173318
+ } catch {
173319
+ return 0;
173320
+ }
173321
+ }
171803
173322
  function getMessageOrdinal(value) {
171804
173323
  if (typeof value === "number" && Number.isFinite(value)) {
171805
173324
  return value;
@@ -171815,25 +173334,63 @@ async function getSemanticScores(args) {
171815
173334
  if (!args.queryEmbedding || args.memories.length === 0) {
171816
173335
  return semanticScores;
171817
173336
  }
171818
- const cachedEmbeddings = getProjectEmbeddings(args.db, args.projectPath);
171819
- const embeddings = await ensureMemoryEmbeddings({
171820
- db: args.db,
171821
- projectIdentity: args.projectPath,
171822
- memories: args.memories,
171823
- existingEmbeddings: cachedEmbeddings
171824
- });
173337
+ if (!args.workspace?.isWorkspaced) {
173338
+ const cachedEmbeddings = getProjectEmbeddings(args.db, args.projectPath);
173339
+ const embeddings = await ensureMemoryEmbeddings({
173340
+ db: args.db,
173341
+ projectIdentity: args.projectPath,
173342
+ memories: args.memories,
173343
+ existingEmbeddings: cachedEmbeddings
173344
+ });
173345
+ for (const memory of args.memories) {
173346
+ const memoryEmbedding = embeddings.get(memory.id);
173347
+ if (!memoryEmbedding) {
173348
+ continue;
173349
+ }
173350
+ semanticScores.set(memory.id, normalizeCosineScore(cosineSimilarity(args.queryEmbedding, memoryEmbedding.embedding)));
173351
+ }
173352
+ return semanticScores;
173353
+ }
173354
+ if (!args.queryModelId || args.queryModelId === "off") {
173355
+ return semanticScores;
173356
+ }
173357
+ const workspace = args.workspace;
173358
+ const memoriesByIdentity = new Map;
171825
173359
  for (const memory of args.memories) {
171826
- const memoryEmbedding = embeddings.get(memory.id);
171827
- if (!memoryEmbedding) {
173360
+ const identity = memoryWorkspaceIdentity(memory, workspace);
173361
+ if (!identity)
171828
173362
  continue;
173363
+ const list = memoriesByIdentity.get(identity) ?? [];
173364
+ list.push(memory);
173365
+ memoriesByIdentity.set(identity, list);
173366
+ }
173367
+ const ownMemories = memoriesByIdentity.get(args.projectPath) ?? [];
173368
+ if (ownMemories.length > 0) {
173369
+ const ownEmbeddings = getProjectEmbeddings(args.db, args.projectPath);
173370
+ await ensureMemoryEmbeddings({
173371
+ db: args.db,
173372
+ projectIdentity: args.projectPath,
173373
+ memories: ownMemories,
173374
+ existingEmbeddings: ownEmbeddings
173375
+ });
173376
+ }
173377
+ for (const identity of workspace.identities) {
173378
+ const memberMemories = memoriesByIdentity.get(identity) ?? [];
173379
+ if (memberMemories.length === 0)
173380
+ continue;
173381
+ const cachedEmbeddings = getProjectEmbeddings(args.db, identity);
173382
+ for (const memory of memberMemories) {
173383
+ const memoryEmbedding = cachedEmbeddings.get(memory.id);
173384
+ if (!memoryEmbedding || memoryEmbedding.modelId !== args.queryModelId)
173385
+ continue;
173386
+ semanticScores.set(memory.id, normalizeCosineScore(cosineSimilarity(args.queryEmbedding, memoryEmbedding.embedding)));
171829
173387
  }
171830
- semanticScores.set(memory.id, normalizeCosineScore(cosineSimilarity(args.queryEmbedding, memoryEmbedding)));
171831
173388
  }
171832
173389
  return semanticScores;
171833
173390
  }
171834
173391
  function getFtsMatches(args) {
171835
173392
  try {
171836
- return searchMemoriesFTS(args.db, args.projectPath, args.query, args.limit);
173393
+ return args.workspace?.isWorkspaced ? searchMemoriesFTSUnion(args.db, args.workspace.expandedIdentities, args.query, args.limit, args.workspace.ownIdentities, args.workspace.shareCategories) : searchMemoriesFTS(args.db, args.projectPath, args.query, args.limit);
171837
173394
  } catch (error51) {
171838
173395
  log(`[search] FTS query failed for "${args.query}": ${error51 instanceof Error ? error51.message : String(error51)}`);
171839
173396
  return [];
@@ -171847,8 +173404,11 @@ function selectSemanticCandidates(args) {
171847
173404
  return args.memories;
171848
173405
  }
171849
173406
  const candidateIds = new Set(args.ftsMatches.map((memory) => memory.id));
171850
- const cachedEmbeddings = peekProjectEmbeddings(args.projectPath);
171851
- if (cachedEmbeddings) {
173407
+ const embeddingProjects = args.workspace?.isWorkspaced ? args.workspace.identities : [args.projectPath];
173408
+ for (const projectPath of embeddingProjects) {
173409
+ const cachedEmbeddings = peekProjectEmbeddings(projectPath);
173410
+ if (!cachedEmbeddings)
173411
+ continue;
171852
173412
  for (const memoryId of cachedEmbeddings.keys()) {
171853
173413
  candidateIds.add(memoryId);
171854
173414
  }
@@ -171890,7 +173450,8 @@ function mergeMemoryResults(args) {
171890
173450
  score,
171891
173451
  memoryId: memory.id,
171892
173452
  category: memory.category,
171893
- matchType
173453
+ matchType,
173454
+ sourceName: args.sourceNameByMemoryId?.get(memory.id)
171894
173455
  });
171895
173456
  }
171896
173457
  return results.sort((left, right) => {
@@ -171904,7 +173465,7 @@ async function searchMemories(args) {
171904
173465
  if (!args.memoryEnabled) {
171905
173466
  return [];
171906
173467
  }
171907
- const memories = getMemoriesByProject(args.db, args.projectPath);
173468
+ const memories = args.workspace?.isWorkspaced ? getMemoriesByProjects(args.db, args.workspace.expandedIdentities, ["active", "permanent"], Date.now(), args.workspace.ownIdentities, args.workspace.shareCategories) : getMemoriesByProject(args.db, args.projectPath);
171908
173469
  if (memories.length === 0) {
171909
173470
  return [];
171910
173471
  }
@@ -171912,26 +173473,43 @@ async function searchMemories(args) {
171912
173473
  db: args.db,
171913
173474
  projectPath: args.projectPath,
171914
173475
  query: args.query,
171915
- limit: FTS_SEMANTIC_CANDIDATE_LIMIT
173476
+ limit: FTS_SEMANTIC_CANDIDATE_LIMIT,
173477
+ workspace: args.workspace
171916
173478
  });
171917
173479
  const ftsScores = getFtsScores(ftsMatches);
171918
173480
  const semanticCandidates = selectSemanticCandidates({
171919
173481
  memories,
171920
173482
  projectPath: args.projectPath,
171921
- ftsMatches
173483
+ ftsMatches,
173484
+ workspace: args.workspace
171922
173485
  });
171923
173486
  const semanticScores = await getSemanticScores({
171924
173487
  db: args.db,
171925
173488
  projectPath: args.projectPath,
171926
173489
  memories: semanticCandidates,
171927
- queryEmbedding: args.queryEmbedding
173490
+ queryEmbedding: args.queryEmbedding,
173491
+ queryModelId: args.queryModelId,
173492
+ workspace: args.workspace
171928
173493
  });
171929
173494
  return mergeMemoryResults({
171930
173495
  memories,
171931
173496
  semanticScores,
171932
173497
  ftsScores,
171933
173498
  limit: args.limit,
171934
- visibleMemoryIds: args.visibleMemoryIds
173499
+ visibleMemoryIds: args.visibleMemoryIds,
173500
+ sourceNameByMemoryId: sourceNamesForSearchMemories({
173501
+ memories,
173502
+ projectPath: args.projectPath,
173503
+ workspace: args.workspace ?? {
173504
+ identities: [args.projectPath],
173505
+ expandedIdentities: [args.projectPath],
173506
+ namesByIdentity: new Map,
173507
+ canonicalIdentityByStoredPath: new Map([[args.projectPath, args.projectPath]]),
173508
+ ownIdentities: [args.projectPath],
173509
+ shareCategories: null,
173510
+ isWorkspaced: false
173511
+ }
173512
+ })
171935
173513
  });
171936
173514
  }
171937
173515
  function linearDecayScore(rank, total) {
@@ -171962,7 +173540,13 @@ function runMessageFtsQuery(db, sessionId, ftsQuery, fetchLimit, cutoff) {
171962
173540
  return result;
171963
173541
  }
171964
173542
  var RRF_K = 60;
171965
- var VERBATIM_PROBE_BONUS = 0.5;
173543
+ var VERBATIM_RANK_BONUS = 1 / RRF_K;
173544
+ var IDF_FALLOFF = 100;
173545
+ function probeDiscriminationWeight(df, corpusSize) {
173546
+ if (corpusSize <= 0 || df <= 0)
173547
+ return 1;
173548
+ return 1 / (1 + IDF_FALLOFF * df / corpusSize);
173549
+ }
171966
173550
  function searchMessages(args) {
171967
173551
  const cutoff = args.maxOrdinal != null && args.maxOrdinal >= 0 ? args.maxOrdinal : null;
171968
173552
  const fetchLimit = args.maxOrdinal != null && args.maxOrdinal >= 0 ? args.limit * 3 : args.limit;
@@ -171979,20 +173563,31 @@ function searchMessages(args) {
171979
173563
  role: row.role
171980
173564
  }));
171981
173565
  }
173566
+ const corpusSize = getSessionFtsRowCount(args.db, args.sessionId);
171982
173567
  const queryLists = [];
171983
173568
  if (baseQuery.length > 0) {
171984
- queryLists.push(runMessageFtsQuery(args.db, args.sessionId, baseQuery, fetchLimit, cutoff));
173569
+ queryLists.push({
173570
+ rows: runMessageFtsQuery(args.db, args.sessionId, baseQuery, fetchLimit, cutoff),
173571
+ weight: 1
173572
+ });
171985
173573
  }
173574
+ const probeWeights = new Map;
171986
173575
  for (const probe of probes) {
171987
173576
  const probeQuery = sanitizeFtsQuery(probe);
171988
173577
  if (probeQuery.length === 0)
171989
173578
  continue;
171990
- queryLists.push(runMessageFtsQuery(args.db, args.sessionId, probeQuery, fetchLimit, cutoff));
173579
+ const df = countSessionFtsMatches(args.db, args.sessionId, probeQuery);
173580
+ const weight = probeDiscriminationWeight(df, corpusSize);
173581
+ probeWeights.set(probe, weight);
173582
+ queryLists.push({
173583
+ rows: runMessageFtsQuery(args.db, args.sessionId, probeQuery, fetchLimit, cutoff),
173584
+ weight
173585
+ });
171991
173586
  }
171992
173587
  const fused = new Map;
171993
173588
  for (const list of queryLists) {
171994
- list.forEach((row, rank) => {
171995
- const rrf = 1 / (RRF_K + rank);
173589
+ list.rows.forEach((row, rank) => {
173590
+ const rrf = list.weight / (RRF_K + rank);
171996
173591
  const existing = fused.get(row.messageId);
171997
173592
  if (existing) {
171998
173593
  existing.score += rrf;
@@ -172002,26 +173597,107 @@ function searchMessages(args) {
172002
173597
  });
172003
173598
  }
172004
173599
  for (const entry of fused.values()) {
172005
- if (containsProbeVerbatim(entry.row.content, probes)) {
172006
- entry.score += VERBATIM_PROBE_BONUS;
173600
+ let best = 0;
173601
+ for (const probe of probes) {
173602
+ const weight = probeWeights.get(probe) ?? 0;
173603
+ if (weight > best && containsProbeVerbatim(entry.row.content, [probe])) {
173604
+ best = weight;
173605
+ }
173606
+ }
173607
+ if (best > 0) {
173608
+ entry.score += best * VERBATIM_RANK_BONUS;
172007
173609
  }
172008
173610
  }
172009
173611
  const ranked = [...fused.values()].sort((a, b) => b.score !== a.score ? b.score - a.score : a.row.messageOrdinal - b.row.messageOrdinal).slice(0, args.limit);
172010
- const maxScore = ranked.length > 0 ? ranked[0].score : 1;
172011
- return ranked.map((entry) => ({
173612
+ return ranked.map((entry, rank) => ({
172012
173613
  source: "message",
172013
173614
  content: previewText(entry.row.content),
172014
- score: maxScore > 0 ? entry.score / maxScore : 0,
173615
+ score: linearDecayScore(rank, ranked.length),
172015
173616
  messageOrdinal: entry.row.messageOrdinal,
172016
173617
  messageId: entry.row.messageId,
172017
173618
  role: entry.row.role
172018
173619
  }));
172019
173620
  }
173621
+ function searchCompartmentChunks(args) {
173622
+ if (!args.queryEmbedding || args.limit <= 0)
173623
+ return [];
173624
+ const cutoff = args.maxOrdinal != null && args.maxOrdinal >= 0 ? args.maxOrdinal : null;
173625
+ const rows = loadCompartmentChunkEmbeddingsForSearch(args.db, args.sessionId, args.projectPath, args.modelId);
173626
+ if (rows.length === 0)
173627
+ return [];
173628
+ const byCompartment = new Map;
173629
+ for (const row of rows) {
173630
+ if (cutoff !== null && row.endOrdinal > cutoff) {
173631
+ continue;
173632
+ }
173633
+ const score = normalizeCosineScore(cosineSimilarity(args.queryEmbedding, row.vector));
173634
+ if (score <= 0)
173635
+ continue;
173636
+ const existing = byCompartment.get(row.compartmentId);
173637
+ if (!existing || score > existing.score) {
173638
+ byCompartment.set(row.compartmentId, { row, score });
173639
+ }
173640
+ }
173641
+ return [...byCompartment.values()].sort((left, right) => right.score !== left.score ? right.score - left.score : left.row.startOrdinal - right.row.startOrdinal).slice(0, args.limit).map(({ row, score }) => ({
173642
+ source: "compartment",
173643
+ content: previewText(row.title),
173644
+ score: score * SINGLE_SOURCE_PENALTY,
173645
+ compartmentId: row.compartmentId,
173646
+ sessionId: row.sessionId,
173647
+ title: row.title,
173648
+ startOrdinal: row.startOrdinal,
173649
+ endOrdinal: row.endOrdinal,
173650
+ matchType: "semantic"
173651
+ }));
173652
+ }
173653
+ function mergeMessageAndCompartmentResults(args) {
173654
+ if (args.compartments.length === 0)
173655
+ return args.messages;
173656
+ if (args.messages.length === 0)
173657
+ return args.compartments;
173658
+ const fused = new Map;
173659
+ const add = (key, result, score, tieOrdinal) => {
173660
+ const existing = fused.get(key);
173661
+ if (existing) {
173662
+ existing.score += score;
173663
+ return existing;
173664
+ }
173665
+ const entry = { result, score, tieOrdinal, snippetScore: -1 };
173666
+ fused.set(key, entry);
173667
+ return entry;
173668
+ };
173669
+ args.compartments.forEach((compartment, rank) => {
173670
+ add(`compartment:${compartment.compartmentId}`, compartment, 1 / (RRF_K + rank), compartment.startOrdinal);
173671
+ });
173672
+ for (const [rank, message] of args.messages.entries()) {
173673
+ const containing = args.compartments.find((compartment) => message.messageOrdinal >= compartment.startOrdinal && message.messageOrdinal <= compartment.endOrdinal);
173674
+ const contribution = 1 / (RRF_K + rank);
173675
+ if (!containing) {
173676
+ add(`message:${message.messageId}`, message, contribution, message.messageOrdinal);
173677
+ continue;
173678
+ }
173679
+ const entry = add(`compartment:${containing.compartmentId}`, containing, contribution, containing.startOrdinal);
173680
+ if (message.score > entry.snippetScore && entry.result.source === "compartment") {
173681
+ entry.snippetScore = message.score;
173682
+ entry.result = {
173683
+ ...entry.result,
173684
+ matchType: "hybrid",
173685
+ snippet: message.content
173686
+ };
173687
+ }
173688
+ }
173689
+ const ranked = [...fused.values()].sort((left, right) => right.score !== left.score ? right.score - left.score : left.tieOrdinal - right.tieOrdinal).slice(0, args.limit);
173690
+ return ranked.map((entry, rank) => ({
173691
+ ...entry.result,
173692
+ score: linearDecayScore(rank, ranked.length)
173693
+ }));
173694
+ }
172020
173695
  function getSourceBoost(result) {
172021
173696
  switch (result.source) {
172022
173697
  case "memory":
172023
173698
  return MEMORY_SOURCE_BOOST;
172024
173699
  case "message":
173700
+ case "compartment":
172025
173701
  return MESSAGE_SOURCE_BOOST;
172026
173702
  case "git_commit":
172027
173703
  return GIT_COMMIT_SOURCE_BOOST;
@@ -172039,6 +173715,9 @@ function compareUnifiedResults(left, right) {
172039
173715
  if (left.source === "message" && right.source === "message") {
172040
173716
  return left.messageOrdinal - right.messageOrdinal;
172041
173717
  }
173718
+ if (left.source === "compartment" && right.source === "compartment") {
173719
+ return left.startOrdinal - right.startOrdinal;
173720
+ }
172042
173721
  if (left.source === "git_commit" && right.source === "git_commit") {
172043
173722
  return right.committedAtMs - left.committedAtMs;
172044
173723
  }
@@ -172089,10 +173768,12 @@ async function unifiedSearch(db, sessionId, projectPath, query, options = {}) {
172089
173768
  const isEmbeddingRuntimeEnabled = options.isEmbeddingRuntimeEnabled ?? isEmbeddingEnabled;
172090
173769
  const gitCommitsEnabled = options.gitCommitsEnabled ?? false;
172091
173770
  const activeSources = resolveSources(options.sources);
172092
- const runMemory = activeSources.has("memory") && (options.memoryEnabled ?? true);
173771
+ const memoryFeatureEnabled = options.memoryEnabled ?? true;
173772
+ const runMemory = activeSources.has("memory") && memoryFeatureEnabled;
172093
173773
  const runMessages = activeSources.has("message");
172094
173774
  const runGitCommits = activeSources.has("git_commit") && gitCommitsEnabled;
172095
- const needsEmbedding = (runMemory || runGitCommits) && embeddingEnabled && isEmbeddingRuntimeEnabled();
173775
+ const runCompartmentChunks = runMessages && memoryFeatureEnabled && embeddingEnabled;
173776
+ const needsEmbedding = (runMemory || runGitCommits || runCompartmentChunks) && embeddingEnabled && isEmbeddingRuntimeEnabled();
172096
173777
  const queryEmbeddingPromise = needsEmbedding ? embedQuery(trimmedQuery, options.signal).catch((error51) => {
172097
173778
  log(`[search] query embedding failed: ${error51 instanceof Error ? error51.message : String(error51)}`);
172098
173779
  return null;
@@ -172108,6 +173789,24 @@ async function unifiedSearch(db, sessionId, projectPath, query, options = {}) {
172108
173789
  probes: messageProbes
172109
173790
  }) : [];
172110
173791
  const queryEmbedding = await queryEmbeddingPromise;
173792
+ const workspace = resolveSearchWorkspaceContext(db, projectPath);
173793
+ const embeddingSnapshot = getProjectEmbeddingSnapshot(projectPath);
173794
+ const embeddingModelId = embeddingSnapshot?.modelId;
173795
+ const chunkModelId = embeddingSnapshot?.chunkModelId;
173796
+ const compartmentResults = runCompartmentChunks ? searchCompartmentChunks({
173797
+ db,
173798
+ sessionId,
173799
+ projectPath,
173800
+ queryEmbedding,
173801
+ limit: tierLimit,
173802
+ maxOrdinal: options.maxMessageOrdinal,
173803
+ modelId: chunkModelId && chunkModelId !== "off" ? chunkModelId : null
173804
+ }) : [];
173805
+ const messageLikeResults = mergeMessageAndCompartmentResults({
173806
+ messages: messageResults,
173807
+ compartments: compartmentResults,
173808
+ limit: tierLimit
173809
+ });
172111
173810
  const [memoryResults, gitCommitResults] = await Promise.all([
172112
173811
  runMemory ? searchMemories({
172113
173812
  db,
@@ -172116,6 +173815,8 @@ async function unifiedSearch(db, sessionId, projectPath, query, options = {}) {
172116
173815
  limit: tierLimit,
172117
173816
  memoryEnabled: true,
172118
173817
  queryEmbedding,
173818
+ queryModelId: embeddingModelId && embeddingModelId !== "off" ? embeddingModelId : null,
173819
+ workspace,
172119
173820
  visibleMemoryIds: options.visibleMemoryIds
172120
173821
  }) : Promise.resolve([]),
172121
173822
  runGitCommits ? Promise.resolve(searchGitCommits({
@@ -172126,7 +173827,7 @@ async function unifiedSearch(db, sessionId, projectPath, query, options = {}) {
172126
173827
  queryEmbedding
172127
173828
  })) : Promise.resolve([])
172128
173829
  ]);
172129
- const results = [...memoryResults, ...messageResults, ...gitCommitResults].sort(compareUnifiedResults).slice(0, limit);
173830
+ const results = [...memoryResults, ...messageLikeResults, ...gitCommitResults].sort(compareUnifiedResults).slice(0, limit);
172130
173831
  const countRetrievals = options.countRetrievals ?? true;
172131
173832
  if (countRetrievals) {
172132
173833
  const memoryIds = results.filter((result) => result.source === "memory").map((result) => result.memoryId);
@@ -172186,6 +173887,11 @@ function renderFragment(result, charCap) {
172186
173887
  const compressed = cavemanCompress(result.content, "ultra");
172187
173888
  return truncate(compressed, charCap);
172188
173889
  }
173890
+ case "compartment": {
173891
+ const source = result.snippet ?? result.title;
173892
+ const compressed = cavemanCompress(source, "ultra");
173893
+ return truncate(compressed, charCap);
173894
+ }
172189
173895
  }
172190
173896
  }
172191
173897
  function buildAutoSearchHint(results, options = {}) {
@@ -173609,6 +175315,49 @@ function safeGetActiveUserMemoriesPi(db) {
173609
175315
  function memoryProjectPath(state) {
173610
175316
  return state.memoryEnabled === false ? undefined : state.projectIdentity;
173611
175317
  }
175318
+ function resolveWorkspaceRenderContextPi(state, db) {
175319
+ const memPath = memoryProjectPath(state);
175320
+ if (!memPath) {
175321
+ return {
175322
+ identities: [],
175323
+ expandedIdentities: [],
175324
+ ownIdentities: [],
175325
+ shareCategories: null,
175326
+ namesByIdentity: new Map,
175327
+ canonicalIdentityByStoredPath: new Map,
175328
+ isWorkspaced: false
175329
+ };
175330
+ }
175331
+ const identitySet = resolveWorkspaceIdentitySet(db, memPath);
175332
+ const isWorkspaced = identitySet.identities.length > 1;
175333
+ const expanded = expandWorkspaceIdentitySetWithAliases(db, identitySet.identities);
175334
+ const expandedIdentities = isWorkspaced ? expanded.expandedIdentities : identitySet.identities;
175335
+ const canonicalIdentityByStoredPath = isWorkspaced ? expanded.canonicalIdentityByStoredPath : new Map(identitySet.identities.map((identity) => [identity, identity]));
175336
+ let ownIdentities = expandedIdentities.filter((identity) => canonicalIdentityByStoredPath.get(identity) === memPath);
175337
+ if (ownIdentities.length === 0 && expandedIdentities.includes(memPath)) {
175338
+ ownIdentities = [memPath];
175339
+ }
175340
+ return {
175341
+ identities: identitySet.identities,
175342
+ expandedIdentities,
175343
+ ownIdentities,
175344
+ shareCategories: isWorkspaced ? resolveWorkspaceShareCategories(db, memPath) : null,
175345
+ namesByIdentity: identitySet.namesByIdentity,
175346
+ canonicalIdentityByStoredPath,
175347
+ isWorkspaced
175348
+ };
175349
+ }
175350
+ function sourceNamesForPiMemories(args) {
175351
+ if (!args.projectPath || !args.workspace.isWorkspaced)
175352
+ return;
175353
+ const names = new Map;
175354
+ for (const memory of args.memories) {
175355
+ const source = sourceNameForMemory(memory.projectPath, args.projectPath, args.workspace.identities, args.workspace.namesByIdentity, args.workspace.canonicalIdentityByStoredPath);
175356
+ if (source)
175357
+ names.set(memory.id, source);
175358
+ }
175359
+ return names.size > 0 ? names : undefined;
175360
+ }
173612
175361
  var EMPTY_PI_HARD_SIGNALS = {
173613
175362
  systemHash: "",
173614
175363
  modelKey: "",
@@ -173656,6 +175405,7 @@ function getCachedMarkers(db, state, compartmentsForNormalization) {
173656
175405
  maxMutationId: meta3.cachedM0MaxMutationId,
173657
175406
  maxMemoryMutationId: meta3.cachedM0MaxMemoryMutationId,
173658
175407
  projectMemoryEpoch: meta3.cachedM0ProjectMemoryEpoch,
175408
+ workspaceFingerprint: meta3.cachedM0WorkspaceFingerprint,
173659
175409
  projectUserProfileVersion: meta3.cachedM0ProjectUserProfileVersion,
173660
175410
  projectDocsHash: meta3.cachedM0ProjectDocsHash,
173661
175411
  sessionFactsVersion: meta3.cachedM0SessionFactsVersion,
@@ -173675,15 +175425,17 @@ function readCurrentMarkers(db, state, projectDocsHash) {
173675
175425
  }
173676
175426
  function readCurrentMarkersFromCompartments(db, state, compartments, projectDocsHash) {
173677
175427
  const memPath = memoryProjectPath(state);
173678
- const memories = memPath ? getMemoriesByProject(db, memPath, ["active", "permanent"]) : [];
175428
+ const workspace = resolveWorkspaceRenderContextPi(state, db);
175429
+ const maxMemoryId = memPath ? workspace.isWorkspaced ? getMaxMemoryIdForProjects(db, workspace.expandedIdentities, workspace.ownIdentities, workspace.shareCategories) : getMaxMemoryIdForProjects(db, [memPath]) : 0;
173679
175430
  const projectState = memPath ? getProjectState(db, memPath) : undefined;
173680
175431
  const globalState = getProjectState(db, GLOBAL_USER_PROFILE_PROJECT_PATH);
173681
175432
  return {
173682
175433
  maxCompartmentSeq: compartments.length > 0 ? compartments.reduce((max, compartment) => compartment.sequence > max ? compartment.sequence : max, EMPTY_MAX_COMPARTMENT_SEQ) : EMPTY_MAX_COMPARTMENT_SEQ,
173683
- maxMemoryId: memories.length > 0 ? memories.reduce((max, memory) => memory.id > max ? memory.id : max, 0) : 0,
175434
+ maxMemoryId,
173684
175435
  maxMutationId: getMaxM0MutationId(db, state.sessionId) ?? 0,
173685
- maxMemoryMutationId: memPath ? getMaxMemoryMutationId(db, memPath) ?? 0 : 0,
175436
+ maxMemoryMutationId: memPath ? workspace.isWorkspaced ? getMaxMemoryMutationIdForProjects(db, workspace.expandedIdentities) ?? 0 : getMaxMemoryMutationId(db, memPath) ?? 0 : 0,
173686
175437
  projectMemoryEpoch: projectState?.projectMemoryEpoch ?? 0,
175438
+ workspaceFingerprint: workspace.isWorkspaced ? computeWorkspaceEpochFingerprint(db, workspace.identities) : null,
173687
175439
  projectUserProfileVersion: globalState?.projectUserProfileVersion ?? 0,
173688
175440
  projectDocsHash: projectDocsHash ?? readProjectDocsCanonical(state.projectDirectory).canonicalHash,
173689
175441
  sessionFactsVersion: getSessionFactsVersion(db, state.sessionId),
@@ -173721,10 +175473,11 @@ function mustMaterializePi(state, db, currentCompartmentsOverride) {
173721
175473
  if (meta3.cachedM0UpgradeState !== current.upgradeState) {
173722
175474
  return { value: true, reason: "renderer_upgrade" };
173723
175475
  }
173724
- if (current.projectDocsHash !== (meta3.cachedM0ProjectDocsHash ?? "")) {
173725
- return { value: true, reason: "project_docs_change" };
173726
- }
173727
- if (current.projectMemoryEpoch !== (meta3.cachedM0ProjectMemoryEpoch ?? 0)) {
175476
+ if (current.workspaceFingerprint !== null || (meta3.cachedM0WorkspaceFingerprint ?? null) !== null) {
175477
+ if (current.workspaceFingerprint !== (meta3.cachedM0WorkspaceFingerprint ?? null)) {
175478
+ return { value: true, reason: "project_memory_change" };
175479
+ }
175480
+ } else if (current.projectMemoryEpoch !== (meta3.cachedM0ProjectMemoryEpoch ?? 0)) {
173728
175481
  return { value: true, reason: "project_memory_change" };
173729
175482
  }
173730
175483
  if (current.maxMutationId !== (meta3.cachedM0MaxMutationId ?? 0)) {
@@ -173741,11 +175494,19 @@ ${memories.map((memory) => `- ${escapeXmlContent(memory.content)}`).join(`
173741
175494
  `)}
173742
175495
  </${wrapper}>`;
173743
175496
  }
173744
- function renderM0Pi(state, db, projectDocs = readProjectDocsCanonical(state.projectDirectory).renderedBlock, decayPressureMultiplier = 1, memoriesOverride, compartmentsOverride, userProfileOverride) {
175497
+ function renderM0Pi(state, db, projectDocs = readProjectDocsCanonical(state.projectDirectory).renderedBlock, decayPressureMultiplier = 1, memoriesOverride, compartmentsOverride, userProfileOverride, workspaceOverride) {
173745
175498
  const memPath = memoryProjectPath(state);
173746
- const allMemories = memoriesOverride ?? (memPath ? getMemoriesByProject(db, memPath, ["active", "permanent"]) : []);
173747
- const memories = allMemories.length > 0 ? trimMemoriesToBudgetV2(state.sessionId, allMemories, state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS).renderOrder : allMemories;
173748
- const memoryBlock = memories.length > 0 ? renderMemoryBlockV2(memories) : undefined;
175499
+ const workspace = workspaceOverride ?? resolveWorkspaceRenderContextPi(state, db);
175500
+ const allMemories = memoriesOverride ?? (memPath ? workspace.isWorkspaced ? getMemoriesByProjects(db, workspace.expandedIdentities, ["active", "permanent"], Date.now(), workspace.ownIdentities, workspace.shareCategories) : getMemoriesByProject(db, memPath, ["active", "permanent"]) : []);
175501
+ const memoryRenderOptions = {
175502
+ sourceNameByMemoryId: sourceNamesForPiMemories({
175503
+ memories: allMemories,
175504
+ projectPath: memPath,
175505
+ workspace
175506
+ })
175507
+ };
175508
+ const memories = allMemories.length > 0 ? workspace.isWorkspaced ? trimWorkspaceMemoriesToBudgetV2(state.sessionId, allMemories, state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS, workspace, memoryRenderOptions).renderOrder : trimMemoriesToBudgetV2(state.sessionId, allMemories, state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS).renderOrder : allMemories;
175509
+ const memoryBlock = memories.length > 0 ? renderMemoryBlockV2(memories, "project-memory", memoryRenderOptions) : undefined;
173749
175510
  const baseHistoryBudget = state.historyBudgetTokens ?? DEFAULT_HISTORY_BUDGET_TOKENS;
173750
175511
  const decayed = renderDecayedCompartments({
173751
175512
  compartments: compartmentsOverride ?? getCompartments(db, state.sessionId),
@@ -173767,10 +175528,19 @@ ${decayed}
173767
175528
 
173768
175529
  `).trim();
173769
175530
  }
173770
- function renderedMemoryIdsForPi(state, memories) {
175531
+ function renderedMemoryIdsForPi(state, memories, workspace, db) {
173771
175532
  if (memories.length === 0)
173772
175533
  return [];
173773
- return trimMemoriesToBudgetV2(state.sessionId, [...memories], state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS).renderOrder.map((memory) => memory.id);
175534
+ const resolvedWorkspace = workspace ?? (db ? resolveWorkspaceRenderContextPi(state, db) : undefined);
175535
+ const renderOptions = resolvedWorkspace ? {
175536
+ sourceNameByMemoryId: sourceNamesForPiMemories({
175537
+ memories,
175538
+ projectPath: memoryProjectPath(state),
175539
+ workspace: resolvedWorkspace
175540
+ })
175541
+ } : {};
175542
+ const trimmed = resolvedWorkspace?.isWorkspaced ? trimWorkspaceMemoriesToBudgetV2(state.sessionId, [...memories], state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS, resolvedWorkspace, renderOptions) : trimMemoriesToBudgetV2(state.sessionId, [...memories], state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS);
175543
+ return trimmed.renderOrder.map((memory) => memory.id);
173774
175544
  }
173775
175545
  function isTransientSqliteLockError(error51) {
173776
175546
  if (!error51 || typeof error51 !== "object")
@@ -173795,17 +175565,19 @@ class PiMaterializeContentionError extends Error {
173795
175565
  function readFrozenM0InputsPi(state, db, docs = readProjectDocsCanonical(state.projectDirectory), memoryCutoff) {
173796
175566
  const memPath = memoryProjectPath(state);
173797
175567
  const read = db.transaction(() => {
175568
+ const workspace = resolveWorkspaceRenderContextPi(state, db);
173798
175569
  const compartments = getCompartments(db, state.sessionId);
173799
- const memories = memPath ? getMemoriesByProject(db, memPath, ["active", "permanent"], memoryCutoff) : [];
175570
+ const memories = memPath ? workspace.isWorkspaced ? getMemoriesByProjects(db, workspace.expandedIdentities, ["active", "permanent"], memoryCutoff, workspace.ownIdentities, workspace.shareCategories) : getMemoriesByProject(db, memPath, ["active", "permanent"], memoryCutoff) : [];
173800
175571
  const userProfile = safeGetActiveUserMemoriesPi(db);
173801
175572
  const projectState = memPath ? getProjectState(db, memPath) : undefined;
173802
175573
  const globalState = getProjectState(db, GLOBAL_USER_PROFILE_PROJECT_PATH);
173803
175574
  const markers = {
173804
175575
  maxCompartmentSeq: compartments.reduce((max, compartment) => compartment.sequence > max ? compartment.sequence : max, EMPTY_MAX_COMPARTMENT_SEQ),
173805
- maxMemoryId: memories.reduce((max, memory) => memory.id > max ? memory.id : max, 0),
175576
+ maxMemoryId: memPath ? workspace.isWorkspaced ? getMaxMemoryIdForProjects(db, workspace.expandedIdentities, workspace.ownIdentities, workspace.shareCategories) : getMaxMemoryIdForProjects(db, [memPath]) : 0,
173806
175577
  maxMutationId: getMaxM0MutationId(db, state.sessionId) ?? 0,
173807
- maxMemoryMutationId: memPath ? getMaxMemoryMutationId(db, memPath) ?? 0 : 0,
175578
+ maxMemoryMutationId: memPath ? workspace.isWorkspaced ? getMaxMemoryMutationIdForProjects(db, workspace.expandedIdentities) ?? 0 : getMaxMemoryMutationId(db, memPath) ?? 0 : 0,
173808
175579
  projectMemoryEpoch: projectState?.projectMemoryEpoch ?? 0,
175580
+ workspaceFingerprint: workspace.isWorkspaced ? computeWorkspaceEpochFingerprint(db, workspace.identities) : null,
173809
175581
  projectUserProfileVersion: globalState?.projectUserProfileVersion ?? 0,
173810
175582
  projectDocsHash: docs.canonicalHash,
173811
175583
  sessionFactsVersion: getSessionFactsVersion(db, state.sessionId),
@@ -173815,7 +175587,7 @@ function readFrozenM0InputsPi(state, db, docs = readProjectDocsCanonical(state.p
173815
175587
  systemHash: (state.hardSignals ?? EMPTY_PI_HARD_SIGNALS).systemHash,
173816
175588
  modelKey: (state.hardSignals ?? EMPTY_PI_HARD_SIGNALS).modelKey
173817
175589
  };
173818
- return { docs, markers, compartments, memories, userProfile };
175590
+ return { docs, markers, compartments, memories, userProfile, workspace };
173819
175591
  });
173820
175592
  return read();
173821
175593
  }
@@ -173826,17 +175598,17 @@ function renderFreshM0PiNonPersisted(state, db) {
173826
175598
  frozen.markers.materializedAt = cachedMaterializedAt;
173827
175599
  const historyBudget = state.historyBudgetTokens ?? DEFAULT_HISTORY_BUDGET_TOKENS;
173828
175600
  let dpm = 1;
173829
- let m0 = renderM0Pi(state, db, docs.renderedBlock, dpm, frozen.memories, frozen.compartments, frozen.userProfile);
175601
+ let m0 = renderM0Pi(state, db, docs.renderedBlock, dpm, frozen.memories, frozen.compartments, frozen.userProfile, frozen.workspace);
173830
175602
  let attempts = 0;
173831
175603
  while (historyBudget > 0 && historySliceTokensPi(m0) > historyBudget * 1.05 && attempts < 3) {
173832
175604
  dpm *= 1.15;
173833
- m0 = renderM0Pi(state, db, docs.renderedBlock, dpm, frozen.memories, frozen.compartments, frozen.userProfile);
175605
+ m0 = renderM0Pi(state, db, docs.renderedBlock, dpm, frozen.memories, frozen.compartments, frozen.userProfile, frozen.workspace);
173834
175606
  attempts += 1;
173835
175607
  }
173836
175608
  return {
173837
175609
  m0,
173838
175610
  snapshotMarkers: frozen.markers,
173839
- renderedMemoryIds: renderedMemoryIdsForPi(state, frozen.memories)
175611
+ renderedMemoryIds: renderedMemoryIdsForPi(state, frozen.memories, frozen.workspace, db)
173840
175612
  };
173841
175613
  }
173842
175614
  function materializeM0Pi(state, db) {
@@ -173846,14 +175618,14 @@ function materializeM0Pi(state, db) {
173846
175618
  const snapshotMemories = frozen.memories;
173847
175619
  const snapshotCompartments = frozen.compartments;
173848
175620
  const snapshotUserProfile = frozen.userProfile;
173849
- const renderedMemoryIds = renderedMemoryIdsForPi(state, snapshotMemories);
175621
+ const renderedMemoryIds = renderedMemoryIdsForPi(state, snapshotMemories, frozen.workspace, db);
173850
175622
  let decayPressureMultiplier = 1;
173851
- let m0 = renderM0Pi(state, db, docs.renderedBlock, decayPressureMultiplier, snapshotMemories, snapshotCompartments, snapshotUserProfile);
175623
+ let m0 = renderM0Pi(state, db, docs.renderedBlock, decayPressureMultiplier, snapshotMemories, snapshotCompartments, snapshotUserProfile, frozen.workspace);
173852
175624
  const historyBudget = state.historyBudgetTokens ?? DEFAULT_HISTORY_BUDGET_TOKENS;
173853
175625
  let attempts = 0;
173854
175626
  while (historyBudget > 0 && historySliceTokensPi(m0) > historyBudget * 1.05 && attempts < 3) {
173855
175627
  decayPressureMultiplier *= 1.15;
173856
- m0 = renderM0Pi(state, db, docs.renderedBlock, decayPressureMultiplier, snapshotMemories, snapshotCompartments, snapshotUserProfile);
175628
+ m0 = renderM0Pi(state, db, docs.renderedBlock, decayPressureMultiplier, snapshotMemories, snapshotCompartments, snapshotUserProfile, frozen.workspace);
173857
175629
  attempts += 1;
173858
175630
  }
173859
175631
  const m0Bytes = Buffer.from(m0, "utf8");
@@ -173869,7 +175641,8 @@ function materializeM0Pi(state, db) {
173869
175641
  }
173870
175642
  try {
173871
175643
  const current = readCurrentMarkers(db, state, phase3ProjectDocsHash);
173872
- const stale = current.projectMemoryEpoch !== snapshotMarkers.projectMemoryEpoch || current.projectUserProfileVersion !== snapshotMarkers.projectUserProfileVersion || current.maxCompartmentSeq !== snapshotMarkers.maxCompartmentSeq || current.maxMutationId !== snapshotMarkers.maxMutationId || current.maxMemoryMutationId !== snapshotMarkers.maxMemoryMutationId || current.projectDocsHash !== snapshotMarkers.projectDocsHash || current.sessionFactsVersion !== snapshotMarkers.sessionFactsVersion || current.upgradeState !== snapshotMarkers.upgradeState;
175644
+ const memoryEpochStale = current.workspaceFingerprint !== null || snapshotMarkers.workspaceFingerprint !== null ? current.workspaceFingerprint !== snapshotMarkers.workspaceFingerprint : current.projectMemoryEpoch !== snapshotMarkers.projectMemoryEpoch;
175645
+ const stale = memoryEpochStale || current.projectUserProfileVersion !== snapshotMarkers.projectUserProfileVersion || current.maxCompartmentSeq !== snapshotMarkers.maxCompartmentSeq || current.maxMutationId !== snapshotMarkers.maxMutationId || current.maxMemoryMutationId !== snapshotMarkers.maxMemoryMutationId || current.sessionFactsVersion !== snapshotMarkers.sessionFactsVersion || current.upgradeState !== snapshotMarkers.upgradeState;
173873
175646
  if (stale) {
173874
175647
  db.exec("ROLLBACK");
173875
175648
  throw new PiMaterializeContentionError("snapshot changed before persist");
@@ -173880,6 +175653,7 @@ function materializeM0Pi(state, db) {
173880
175653
  persistCachedM0(db, state.sessionId, {
173881
175654
  m0Bytes,
173882
175655
  projectMemoryEpoch: snapshotMarkers.projectMemoryEpoch,
175656
+ workspaceFingerprint: snapshotMarkers.workspaceFingerprint,
173883
175657
  projectUserProfileVersion: snapshotMarkers.projectUserProfileVersion,
173884
175658
  maxCompartmentSeq: snapshotMarkers.maxCompartmentSeq,
173885
175659
  maxMemoryId: snapshotMarkers.maxMemoryId,
@@ -173944,7 +175718,7 @@ function renderMemoryUpdatesBlockPi(args) {
173944
175718
  if (args.renderedMemoryIds.length === 0)
173945
175719
  return { block: "", count: 0 };
173946
175720
  const renderedIds = new Set(args.renderedMemoryIds);
173947
- const mutations = getMemoryMutationsForRender(args.db, args.projectPath, args.afterId, args.renderedMemoryIds);
175721
+ const mutations = args.workspace.isWorkspaced ? getMemoryMutationsForRenderByProjects(args.db, args.workspace.expandedIdentities, args.afterId, args.renderedMemoryIds) : getMemoryMutationsForRender(args.db, args.projectPath, args.afterId, args.renderedMemoryIds);
173948
175722
  if (mutations.length === 0)
173949
175723
  return { block: "", count: 0 };
173950
175724
  const lines = [
@@ -173975,6 +175749,7 @@ ${lines.join(`
173975
175749
  }
173976
175750
  function renderM1PiWithMetadata(state, db, markers, renderedMemoryIds, preRenderedKeyFilesBlock, compartmentsOverride) {
173977
175751
  const sections = [];
175752
+ const workspace = resolveWorkspaceRenderContextPi(state, db);
173978
175753
  const keyFiles = renderedKeyFilesBlockPi(state, db, preRenderedKeyFilesBlock);
173979
175754
  if (keyFiles)
173980
175755
  sections.push(keyFiles);
@@ -173982,6 +175757,7 @@ function renderM1PiWithMetadata(state, db, markers, renderedMemoryIds, preRender
173982
175757
  const memoryUpdates = memPath ? renderMemoryUpdatesBlockPi({
173983
175758
  db,
173984
175759
  projectPath: memPath,
175760
+ workspace,
173985
175761
  afterId: markers.maxMemoryMutationId,
173986
175762
  renderedMemoryIds
173987
175763
  }) : { block: undefined, count: 0 };
@@ -173996,11 +175772,18 @@ function renderM1PiWithMetadata(state, db, markers, renderedMemoryIds, preRender
173996
175772
  ${body}
173997
175773
  </new-compartments>`);
173998
175774
  }
173999
- const newMemories = memPath ? getMemoriesByProject(db, memPath, ["active", "permanent"], markers.materializedAt).filter((memory) => memory.id > markers.maxMemoryId) : [];
175775
+ const newMemories = memPath ? workspace.isWorkspaced ? readNewMemoriesForM1Union(db, workspace.expandedIdentities, markers.maxMemoryId, markers.materializedAt, workspace.ownIdentities, workspace.shareCategories) : getMemoriesByProject(db, memPath, ["active", "permanent"], markers.materializedAt).filter((memory) => memory.id > markers.maxMemoryId) : [];
174000
175776
  if (newMemories.length > 0) {
174001
175777
  const memoryBudget = state.injectionBudgetTokens ?? DEFAULT_MEMORY_BUDGET_TOKENS;
174002
- const trimmedNewMemories = trimMemoriesToBudgetV2(state.sessionId, newMemories, Math.max(1, Math.floor(memoryBudget * 0.25))).renderOrder;
174003
- const newMemoriesBlock = renderMemoryBlockV2(trimmedNewMemories, "new-memories");
175778
+ const memoryRenderOptions = {
175779
+ sourceNameByMemoryId: sourceNamesForPiMemories({
175780
+ memories: newMemories,
175781
+ projectPath: memPath,
175782
+ workspace
175783
+ })
175784
+ };
175785
+ const trimmedNewMemories = trimMemoriesToBudgetV2(state.sessionId, newMemories, Math.max(1, Math.floor(memoryBudget * 0.25)), memoryRenderOptions).renderOrder;
175786
+ const newMemoriesBlock = renderMemoryBlockV2(trimmedNewMemories, "new-memories", memoryRenderOptions);
174004
175787
  if (newMemoriesBlock)
174005
175788
  sections.push(newMemoriesBlock);
174006
175789
  }
@@ -174049,6 +175832,7 @@ function parseMemoryBlockIds(raw) {
174049
175832
  function readCachedPiM0M1Row(db, sessionId) {
174050
175833
  return db.prepare(`SELECT cached_m0_bytes, cached_m1_bytes,
174051
175834
  cached_m0_project_memory_epoch,
175835
+ cached_m0_workspace_fingerprint,
174052
175836
  cached_m0_project_user_profile_version,
174053
175837
  cached_m0_max_compartment_seq,
174054
175838
  cached_m0_max_memory_id,
@@ -174092,6 +175876,7 @@ function markersFromCachedPiRow(row, compartmentsForNormalization) {
174092
175876
  maxMutationId: row.cached_m0_max_mutation_id,
174093
175877
  maxMemoryMutationId: row.cached_m0_max_memory_mutation_id,
174094
175878
  projectMemoryEpoch: row.cached_m0_project_memory_epoch,
175879
+ workspaceFingerprint: row.cached_m0_workspace_fingerprint,
174095
175880
  projectUserProfileVersion: row.cached_m0_project_user_profile_version,
174096
175881
  projectDocsHash: row.cached_m0_project_docs_hash ?? "",
174097
175882
  materializedAt: row.cached_m0_materialized_at,
@@ -174106,7 +175891,7 @@ function cachedPiRowMatchesSnapshot(args) {
174106
175891
  const rowMarkers = markersFromCachedPiRow(args.row, args.compartmentsForNormalization);
174107
175892
  if (!rowMarkers)
174108
175893
  return false;
174109
- return bufferEqualsNullable(args.row.cached_m0_bytes, args.m0Bytes) && rowMarkers.projectMemoryEpoch === args.markers.projectMemoryEpoch && rowMarkers.projectUserProfileVersion === args.markers.projectUserProfileVersion && rowMarkers.maxCompartmentSeq === args.markers.maxCompartmentSeq && rowMarkers.maxMemoryId === args.markers.maxMemoryId && rowMarkers.maxMutationId === args.markers.maxMutationId && rowMarkers.maxMemoryMutationId === args.markers.maxMemoryMutationId && (rowMarkers.projectDocsHash ?? "") === (args.markers.projectDocsHash ?? "") && rowMarkers.materializedAt === args.markers.materializedAt && rowMarkers.sessionFactsVersion === args.markers.sessionFactsVersion && (rowMarkers.upgradeState ?? null) === (args.markers.upgradeState ?? null) && (rowMarkers.systemHash ?? "") === (args.markers.systemHash ?? "") && (rowMarkers.modelKey ?? "") === (args.markers.modelKey ?? "");
175894
+ return bufferEqualsNullable(args.row.cached_m0_bytes, args.m0Bytes) && rowMarkers.projectMemoryEpoch === args.markers.projectMemoryEpoch && rowMarkers.projectUserProfileVersion === args.markers.projectUserProfileVersion && rowMarkers.maxCompartmentSeq === args.markers.maxCompartmentSeq && rowMarkers.maxMemoryId === args.markers.maxMemoryId && rowMarkers.maxMutationId === args.markers.maxMutationId && rowMarkers.maxMemoryMutationId === args.markers.maxMemoryMutationId && rowMarkers.materializedAt === args.markers.materializedAt && rowMarkers.sessionFactsVersion === args.markers.sessionFactsVersion && (rowMarkers.upgradeState ?? null) === (args.markers.upgradeState ?? null) && (rowMarkers.systemHash ?? "") === (args.markers.systemHash ?? "") && (rowMarkers.modelKey ?? "") === (args.markers.modelKey ?? "") && (rowMarkers.workspaceFingerprint ?? null) === (args.markers.workspaceFingerprint ?? null);
174110
175895
  }
174111
175896
  function decodeCachedM1(row, sessionId) {
174112
175897
  if (!row.cached_m1_bytes) {
@@ -174317,14 +176102,14 @@ function injectM0M1Pi(state, db, piMessages, entryIds, recomputeM1ThisPass = fal
174317
176102
  const skippedVisibleMessages = boundaryId ? trimPiMessagesToBoundary(piMessages, entryIds, boundaryId) : 0;
174318
176103
  prependM0M1Messages(piMessages, m0, m1);
174319
176104
  sessionLog(state.sessionId, `injected m[0]/m[1] into Pi messages (${m0.length} + ${m1.length} bytes, materialized=${materialized}${decision.reason ? ` reason=${decision.reason}` : ""})`);
176105
+ const memPath = memoryProjectPath(state);
176106
+ const workspace = resolveWorkspaceRenderContextPi(state, db);
176107
+ const memoryCount = memPath ? workspace.isWorkspaced ? getMemoriesByProjects(db, workspace.expandedIdentities, ["active", "permanent"], Date.now(), workspace.ownIdentities, workspace.shareCategories).length : getMemoriesByProject(db, memPath, ["active", "permanent"]).length : 0;
174320
176108
  return {
174321
176109
  injected: true,
174322
176110
  compartmentCount: getCompartments(db, state.sessionId).length,
174323
176111
  factCount: 0,
174324
- memoryCount: memoryProjectPath(state) ? getMemoriesByProject(db, memoryProjectPath(state), [
174325
- "active",
174326
- "permanent"
174327
- ]).length : 0,
176112
+ memoryCount,
174328
176113
  skippedVisibleMessages,
174329
176114
  m0Materialized: materialized,
174330
176115
  m0Reason: decision.reason,
@@ -174385,21 +176170,48 @@ import * as crypto2 from "node:crypto";
174385
176170
 
174386
176171
  // ../plugin/src/features/magic-context/compartment-embedding.ts
174387
176172
  init_logger();
174388
- async function embedAndStoreCompartments(db, sessionId, projectPath, compartments) {
176173
+ async function embedAndStoreCompartmentChunks(db, sessionId, projectPath, compartments) {
174389
176174
  if (compartments.length === 0)
174390
176175
  return;
174391
- const update = db.prepare("UPDATE compartments SET p1_embedding = ?, p1_embedding_model_id = ? WHERE id = ?");
174392
- for (const c of compartments) {
174393
- if (!c.p1 || c.p1.length === 0)
174394
- continue;
176176
+ const maxInputTokens = getProjectEmbeddingMaxInputTokens(projectPath);
176177
+ for (const compartment of compartments) {
174395
176178
  try {
174396
- const result = await embedTextForProject(projectPath, c.p1);
174397
- if (result) {
174398
- const blob = Buffer.from(result.vector.buffer, result.vector.byteOffset, result.vector.byteLength);
174399
- update.run(blob, result.modelId, c.id);
176179
+ const fromMemory = compartment.sourceChunkText ? canonicalizeInMemoryChunkTextForEmbedding(compartment.sourceChunkText, compartment.startMessage, compartment.endMessage) : "";
176180
+ const canonicalText = fromMemory || buildCanonicalChunkTextFromFts(db, sessionId, compartment.startMessage, compartment.endMessage);
176181
+ if (canonicalText.length === 0)
176182
+ continue;
176183
+ const windows = chunkCanonicalText(canonicalText, compartment.startMessage, compartment.endMessage, maxInputTokens);
176184
+ if (windows.length === 0)
176185
+ continue;
176186
+ const currentModelId = getProjectChunkEmbeddingModelId(projectPath);
176187
+ if (currentModelId !== "off" && chunkEmbeddingWindowsAreCurrent(db, compartment.id, currentModelId, windows, projectPath)) {
176188
+ continue;
176189
+ }
176190
+ const result = await embedBatchForProject(projectPath, windows.map((window) => window.text));
176191
+ if (!result)
176192
+ continue;
176193
+ if (chunkEmbeddingWindowsAreCurrent(db, compartment.id, currentModelId, windows, projectPath)) {
176194
+ continue;
176195
+ }
176196
+ const rows = [];
176197
+ for (const [index, window] of windows.entries()) {
176198
+ const vector = result.vectors[index];
176199
+ if (!vector)
176200
+ continue;
176201
+ rows.push({
176202
+ compartmentId: compartment.id,
176203
+ sessionId,
176204
+ projectPath,
176205
+ window,
176206
+ modelId: currentModelId,
176207
+ vector
176208
+ });
176209
+ }
176210
+ if (rows.length === windows.length) {
176211
+ replaceCompartmentChunkEmbeddings(db, rows);
174400
176212
  }
174401
176213
  } catch (error51) {
174402
- sessionLog(sessionId, `compartment embedding failed for compartment ${c.id}:`, error51);
176214
+ sessionLog(sessionId, `compartment chunk embedding failed for compartment ${compartment.id}:`, error51);
174403
176215
  }
174404
176216
  }
174405
176217
  }
@@ -177468,6 +179280,7 @@ async function runPiHistorian(deps) {
177468
179280
  fallbackModels,
177469
179281
  historianChunkTokens,
177470
179282
  boundarySnapshot: providedBoundarySnapshot,
179283
+ refreshBoundarySnapshot,
177471
179284
  currentContextLimit,
177472
179285
  historianTimeoutMs = DEFAULT_HISTORIAN_TIMEOUT_MS2,
177473
179286
  twoPass,
@@ -177521,16 +179334,29 @@ async function runPiHistorian(deps) {
177521
179334
  return;
177522
179335
  }
177523
179336
  const offset = priorCompartments.length > 0 ? priorCompartments[priorCompartments.length - 1].endMessage + 1 : 1;
177524
- const boundarySnapshot = providedBoundarySnapshot ?? null;
179337
+ let boundarySnapshot = providedBoundarySnapshot ?? null;
177525
179338
  if (!boundarySnapshot) {
177526
179339
  sessionLog(sessionId, "historian no-op: missing protected-tail boundary snapshot from Pi trigger decision");
177527
179340
  return;
177528
179341
  }
177529
- const validation = boundarySnapshot.rawRangeFingerprint.length > 0 ? validateBoundarySnapshot({
179342
+ let validation = boundarySnapshot.rawRangeFingerprint.length > 0 ? validateBoundarySnapshot({
177530
179343
  db,
177531
179344
  snapshot: boundarySnapshot,
177532
179345
  currentContextLimit: currentContextLimit ?? boundarySnapshot.contextLimit
177533
179346
  }) : { ok: true };
179347
+ if (!validation.ok && validation.reason === "stale_snapshot" && refreshBoundarySnapshot) {
179348
+ try {
179349
+ const refreshed = refreshBoundarySnapshot();
179350
+ if (hasRunnableCompartmentWindow(refreshed)) {
179351
+ sessionLog(sessionId, `historian: refreshed stale protected-tail snapshot at run time (was: ${validation.detail ?? "stale"}) — eligible head ${refreshed.offset}-${refreshed.eligibleEndOrdinal - 1}`);
179352
+ boundarySnapshot = refreshed;
179353
+ validation = { ok: true };
179354
+ }
179355
+ } catch (error51) {
179356
+ const desc = describeError(error51);
179357
+ sessionLog(sessionId, `historian: failed to refresh stale protected-tail snapshot at run time (${validation.detail ?? "stale"}): ${desc.brief}`);
179358
+ }
179359
+ }
177534
179360
  if (!validation.ok) {
177535
179361
  sessionLog(sessionId, `historian no-op: stale protected-tail snapshot (${validation.detail ?? validation.reason ?? "unknown"})`);
177536
179362
  return;
@@ -177859,8 +179685,13 @@ ${chunkText}`,
177859
179685
  }
177860
179686
  }
177861
179687
  if (embeddingActive) {
177862
- const toEmbed = newCompartments.map((c, i) => ({ id: persistedIds[i], p1: c.p1 ?? c.content })).filter((c) => typeof c.id === "number" && c.p1.length > 0);
177863
- embedAndStoreCompartments(db, sessionId, projectPath, toEmbed);
179688
+ const chunksToEmbed = newCompartments.map((c, i) => ({
179689
+ id: persistedIds[i],
179690
+ startMessage: c.startMessage,
179691
+ endMessage: c.endMessage,
179692
+ sourceChunkText: chunk.text
179693
+ })).filter((c) => typeof c.id === "number");
179694
+ embedAndStoreCompartmentChunks(db, sessionId, projectPath, chunksToEmbed);
177864
179695
  }
177865
179696
  onPublished?.();
177866
179697
  completedSuccessfully = true;
@@ -178408,7 +180239,7 @@ function stripPiDroppedPlaceholderMessages(args) {
178408
180239
  }
178409
180240
 
178410
180241
  // src/system-prompt.ts
178411
- import { createHash as createHash9 } from "node:crypto";
180242
+ import { createHash as createHash11 } from "node:crypto";
178412
180243
 
178413
180244
  // ../plugin/src/agents/magic-context-prompt.ts
178414
180245
  var LONG_TERM_PARTNER_FRAME = `### You are the user's long-term partner on this project — not a one-off hire
@@ -178559,7 +180390,7 @@ function processSystemPromptForCache(args) {
178559
180390
  sessionLog(sessionId, `system prompt date frozen: real=${liveDate}, using=${stickyDate} (cache-stable pass)`);
178560
180391
  }
178561
180392
  }
178562
- const currentHash = createHash9("md5").update(frozenPrompt).digest("hex");
180393
+ const currentHash = createHash11("md5").update(frozenPrompt).digest("hex");
178563
180394
  const hashChanged = !isFirstHash && currentHash !== previousHash;
178564
180395
  if (hashChanged) {
178565
180396
  sessionLog(sessionId, `system prompt hash changed: ${previousHash} → ${currentHash} (len=${frozenPrompt.length})`);
@@ -179186,6 +181017,19 @@ function extractStableId(msg, index, entryIds) {
179186
181017
  // src/context-handler.ts
179187
181018
  var FORCE_MATERIALIZATION_PERCENTAGE = 85;
179188
181019
  var EMERGENCY_BLOCK_PERCENTAGE = 95;
181020
+ var FORWARD_PRESSURE_LIMIT_FACTOR = 0.85;
181021
+ function applyForwardPressureFloor(trailingPercentage, trailingInputTokens, piUsageTokens, correctedLimit) {
181022
+ const forwardTokens = typeof piUsageTokens === "number" && piUsageTokens > 0 ? piUsageTokens : 0;
181023
+ if (forwardTokens === 0 || !isSaneLimit(correctedLimit)) {
181024
+ return { percentage: trailingPercentage, inputTokens: trailingInputTokens };
181025
+ }
181026
+ const forwardPressureLimit = correctedLimit * FORWARD_PRESSURE_LIMIT_FACTOR;
181027
+ const forwardPercentage = forwardTokens / forwardPressureLimit * 100;
181028
+ return forwardPercentage > trailingPercentage ? {
181029
+ percentage: forwardPercentage,
181030
+ inputTokens: Math.max(trailingInputTokens, forwardTokens)
181031
+ } : { percentage: trailingPercentage, inputTokens: trailingInputTokens };
181032
+ }
179189
181033
  var DEFAULT_CLEAR_REASONING_AGE = 50;
179190
181034
  var PI_STABLE_ID_SCHEME = 1;
179191
181035
  var lastEmergencyNotificationAtMs = new Map;
@@ -179270,7 +181114,7 @@ function trackSessionForProject(projectIdentity, sessionId) {
179270
181114
  function isContextHandlerSessionActive(sessionId) {
179271
181115
  return activeContextHandlerSessions.has(sessionId);
179272
181116
  }
179273
- function updateSessionProjectTracking(sessionId, projectIdentity) {
181117
+ function updateSessionProjectTracking(sessionId, projectIdentity, db) {
179274
181118
  const prev = lastSeenProjectIdentityBySession.get(sessionId);
179275
181119
  if (prev && prev !== projectIdentity) {
179276
181120
  const prevSessions = sessionsByProject.get(prev);
@@ -179279,6 +181123,11 @@ function updateSessionProjectTracking(sessionId, projectIdentity) {
179279
181123
  sessionsByProject.delete(prev);
179280
181124
  clearPiSystemPromptSession(sessionId);
179281
181125
  }
181126
+ if (db && prev !== projectIdentity) {
181127
+ try {
181128
+ recordSessionProjectIdentity(db, sessionId, projectIdentity);
181129
+ } catch {}
181130
+ }
179282
181131
  trackSessionForProject(projectIdentity, sessionId);
179283
181132
  lastSeenProjectIdentityBySession.set(sessionId, projectIdentity);
179284
181133
  }
@@ -179559,7 +181408,7 @@ function registerPiContextHandler(pi, baseOptions) {
179559
181408
  const schedulerConfig = options.scheduler ?? DEFAULT_SCHEDULER_CONFIG;
179560
181409
  const scheduler2 = schedulerFor(options);
179561
181410
  const projectIdentity = resolveProjectIdentity(projectDirectory);
179562
- updateSessionProjectTracking(sessionId, projectIdentity);
181411
+ updateSessionProjectTracking(sessionId, projectIdentity, options.db);
179563
181412
  logTransformTiming(sessionId, "findSessionId", tFindSession, `messages=${event.messages.length}`);
179564
181413
  const branchEntries = readPiBranchEntriesForContext(ctx, sessionId);
179565
181414
  const rawMessageProvider = {
@@ -179668,9 +181517,10 @@ function registerPiContextHandler(pi, baseOptions) {
179668
181517
  if (!usedPersistedUsage && isSaneLimit(usageContextLimit) && usageInputTokens > 0) {
179669
181518
  usagePercentage = usageInputTokens / usageContextLimit * 100;
179670
181519
  }
181520
+ ({ percentage: usagePercentage, inputTokens: usageInputTokens } = applyForwardPressureFloor(usagePercentage, usageInputTokens, piUsage?.tokens, usageContextLimit));
179671
181521
  if (needsEmergencyBump) {
179672
181522
  sessionLog(sessionId, `transform: overflow recovery flag set — bumping percentage to 95% (detectedLimit=${usageContextLimit ?? "unknown"})`);
179673
- usagePercentage = 95;
181523
+ usagePercentage = Math.max(usagePercentage, 95);
179674
181524
  }
179675
181525
  let schedulerDecision;
179676
181526
  const tScheduler = performance.now();
@@ -180054,6 +181904,7 @@ function spawnPiHistorianRun(args) {
180054
181904
  provider: provider2,
180055
181905
  unregister,
180056
181906
  boundarySnapshot,
181907
+ refreshBoundarySnapshot,
180057
181908
  currentContextLimit
180058
181909
  } = args;
180059
181910
  const holderId = crypto3.randomUUID();
@@ -180077,6 +181928,7 @@ function spawnPiHistorianRun(args) {
180077
181928
  fallbackModels: historian.fallbackModels,
180078
181929
  historianChunkTokens: historian.historianChunkTokens,
180079
181930
  boundarySnapshot,
181931
+ refreshBoundarySnapshot,
180080
181932
  currentContextLimit,
180081
181933
  historianTimeoutMs: historian.timeoutMs,
180082
181934
  twoPass: historian.twoPass,
@@ -180143,6 +181995,7 @@ function maybeFireHistorian(args) {
180143
181995
  let usageContextLimit;
180144
181996
  try {
180145
181997
  const piUsage = ctx.getContextUsage?.();
181998
+ let usageSource;
180146
181999
  usageContextLimit = isSaneLimit(piUsage?.contextWindow) ? piUsage.contextWindow : undefined;
180147
182000
  if (usageContextLimit === undefined && isSaneLimit(ctx.model?.contextWindow)) {
180148
182001
  usageContextLimit = ctx.model.contextWindow;
@@ -180159,7 +182012,7 @@ function maybeFireHistorian(args) {
180159
182012
  percentage: sessionMetaForUsage.lastContextPercentage,
180160
182013
  inputTokens: sessionMetaForUsage.lastInputTokens
180161
182014
  };
180162
- sessionLog(sessionId, `historian trigger eval: usage=${usage.percentage.toFixed(1)}% (${usage.inputTokens} tokens) [from session_meta], checking trigger...`);
182015
+ usageSource = "session_meta";
180163
182016
  } else {
180164
182017
  if (!piUsage || piUsage.tokens === null || piUsage.percent === null || piUsage.contextWindow === 0) {
180165
182018
  sessionLog(sessionId, `historian trigger eval: no usage info yet (tokens=${piUsage?.tokens ?? "<no piUsage>"}, percent=${piUsage?.percent ?? "<no piUsage>"}, contextWindow=${piUsage?.contextWindow ?? "<no piUsage>"})`);
@@ -180170,8 +182023,10 @@ function maybeFireHistorian(args) {
180170
182023
  percentage: fallbackPercentage,
180171
182024
  inputTokens: piUsage.tokens
180172
182025
  };
180173
- sessionLog(sessionId, `historian trigger eval: usage=${usage.percentage.toFixed(1)}% (${usage.inputTokens} tokens) [piUsage fallback], checking trigger...`);
182026
+ usageSource = "piUsage fallback";
180174
182027
  }
182028
+ usage = applyForwardPressureFloor(usage.percentage, usage.inputTokens, piUsage?.tokens, usageContextLimit);
182029
+ sessionLog(sessionId, `historian trigger eval: usage=${usage.percentage.toFixed(1)}% (${usage.inputTokens} tokens) [${usageSource}], checking trigger...`);
180175
182030
  } catch (err) {
180176
182031
  sessionLog(sessionId, `historian trigger eval: getContextUsage threw: ${err instanceof Error ? err.message : String(err)}`);
180177
182032
  return;
@@ -180202,10 +182057,14 @@ function maybeFireHistorian(args) {
180202
182057
  cacheNamespace: `pi:${sessionId}`,
180203
182058
  emergencyTailScale
180204
182059
  }));
180205
- let boundarySnapshot = ensureRunnablePiBoundaryForTests(resolvePiBoundarySnapshot());
180206
- if (!hasRunnableCompartmentWindow(boundarySnapshot) && usage.percentage >= 80) {
180207
- boundarySnapshot = ensureRunnablePiBoundaryForTests(resolvePiBoundarySnapshot(usage.percentage >= 95 ? 0.25 : 0.5));
180208
- }
182060
+ const resolveRunnablePiBoundarySnapshot = () => {
182061
+ let snapshot = ensureRunnablePiBoundaryForTests(resolvePiBoundarySnapshot());
182062
+ if (!hasRunnableCompartmentWindow(snapshot) && usage.percentage >= 80) {
182063
+ snapshot = ensureRunnablePiBoundaryForTests(resolvePiBoundarySnapshot(usage.percentage >= 95 ? 0.25 : 0.5));
182064
+ }
182065
+ return snapshot;
182066
+ };
182067
+ const boundarySnapshot = resolveRunnablePiBoundarySnapshot();
180209
182068
  let triggered = false;
180210
182069
  try {
180211
182070
  if (isFirstContextPassForSession) {
@@ -180230,6 +182089,7 @@ Historian previously failed ${failureState.failureCount} time(s), so Magic Conte
180230
182089
  provider: provider2,
180231
182090
  unregister,
180232
182091
  boundarySnapshot,
182092
+ refreshBoundarySnapshot: resolveRunnablePiBoundarySnapshot,
180233
182093
  currentContextLimit: boundaryContextLimit
180234
182094
  });
180235
182095
  return;
@@ -180253,6 +182113,7 @@ Historian previously failed ${failureState.failureCount} time(s), so Magic Conte
180253
182113
  resolvedBoundarySnapshot: boundarySnapshot,
180254
182114
  triggerBoundarySnapshot: trigger.boundarySnapshot
180255
182115
  }),
182116
+ refreshBoundarySnapshot: resolveRunnablePiBoundarySnapshot,
180256
182117
  currentContextLimit: boundaryContextLimit
180257
182118
  });
180258
182119
  } catch (err) {
@@ -180293,7 +182154,31 @@ async function runPipeline(args) {
180293
182154
  const alreadyRanHeuristicsThisTurn = currentTurnId !== null && lastHeuristicsTurnIdBySession.get(args.sessionId) === currentTurnId;
180294
182155
  const canConsumeDeferredLate = args.schedulerDecision === "execute" || args.forceMaterialization === true || args.contextUsage.percentage >= FORCE_MATERIALIZATION_PERCENTAGE;
180295
182156
  const deferredMaterializeEligible = canConsumeDeferredLate && deferredMaterializationSessions.has(args.sessionId);
180296
- const shouldRunHeuristics = args.heuristics !== undefined && (args.forceMaterialization === true || hasPendingMaterialization(args.sessionId) || deferredMaterializeEligible || args.schedulerDecision === "execute" && !alreadyRanHeuristicsThisTurn);
182157
+ const piHardSignals = args.injection ? (() => {
182158
+ const hardMeta = getOrCreateSessionMeta(args.db, args.sessionId);
182159
+ let piTtlMs = 5 * 60 * 1000;
182160
+ try {
182161
+ piTtlMs = parseCacheTtl(hardMeta.cacheTtl);
182162
+ } catch {}
182163
+ return {
182164
+ systemHash: typeof hardMeta.systemPromptHash === "string" ? hardMeta.systemPromptHash : "",
182165
+ modelKey: liveModelBySession.get(args.sessionId) ?? "",
182166
+ cacheExpired: hardMeta.lastResponseTime > 0 && Date.now() - hardMeta.lastResponseTime >= piTtlMs,
182167
+ lastResponseTime: hardMeta.lastResponseTime
182168
+ };
182169
+ })() : undefined;
182170
+ const m0HardFoldThisPass = args.injection && piHardSignals ? mustMaterializePi({
182171
+ sessionId: args.sessionId,
182172
+ projectIdentity: args.projectIdentity,
182173
+ projectDirectory: args.projectDirectory,
182174
+ memoryEnabled: args.injection.memoryEnabled,
182175
+ injectionBudgetTokens: args.injection.injectionBudgetTokens,
182176
+ historyBudgetTokens: args.injection.historyBudgetTokens,
182177
+ keyFilesEnabled: args.injection.keyFilesEnabled,
182178
+ keyFilesTokenBudget: args.injection.keyFilesTokenBudget,
182179
+ hardSignals: piHardSignals
182180
+ }, args.db, getCompartments(args.db, args.sessionId)).value : false;
182181
+ const shouldRunHeuristics = args.heuristics !== undefined && (args.forceMaterialization === true || hasPendingMaterialization(args.sessionId) || deferredMaterializeEligible || m0HardFoldThisPass || args.schedulerDecision === "execute" && !alreadyRanHeuristicsThisTurn);
180297
182182
  const entryFingerprintByMessageId = buildEntryFingerprintMap(args.messages, stableIdResolver);
180298
182183
  adoptPiFallbackTags(args.db, args.sessionId, args.tagger, entryFingerprintByMessageId);
180299
182184
  const tTag = performance.now();
@@ -180320,12 +182205,12 @@ async function runPipeline(args) {
180320
182205
  const deferredMaterializationWasPending = deferredMaterializationSessions.has(args.sessionId);
180321
182206
  const deferredHistoryRefreshWasPending = deferredHistoryWasPendingAtPassStart;
180322
182207
  const pendingOps = getPendingOps(args.db, args.sessionId);
180323
- const baseShouldApplyPendingOps = args.schedulerDecision === "execute" || args.forceMaterialization || hasPendingMaterializeSignal;
182208
+ const baseShouldApplyPendingOps = args.schedulerDecision === "execute" || args.forceMaterialization || hasPendingMaterializeSignal || m0HardFoldThisPass;
180324
182209
  const deferredMaterialize = canConsumeDeferredLate && deferredMaterializationWasPending;
180325
182210
  const deferredHistoryRefresh = canConsumeDeferredLate && deferredHistoryRefreshWasPending;
180326
182211
  const shouldApplyPendingOps = baseShouldApplyPendingOps || deferredMaterialize;
180327
182212
  if (shouldApplyPendingOps) {
180328
- const applyReason = hasPendingMaterializeSignal ? "explicit_flush" : deferredMaterialize ? "deferred_publication" : args.forceMaterialization ? "force_materialization" : `scheduler_execute (scheduler=${args.schedulerDecision})`;
182213
+ const applyReason = hasPendingMaterializeSignal ? "explicit_flush" : deferredMaterialize ? "deferred_publication" : args.forceMaterialization ? "force_materialization" : m0HardFoldThisPass && args.schedulerDecision !== "execute" ? `m0_hard_fold (drain folded into known m[0] bust, scheduler=${args.schedulerDecision})` : `scheduler_execute (scheduler=${args.schedulerDecision})`;
180329
182214
  sessionLog(args.sessionId, `pending ops WILL APPLY — reason=${applyReason}, pendingOps=${pendingOps.length}, context=${args.contextUsage.percentage.toFixed(1)}%`);
180330
182215
  try {
180331
182216
  const tApplyPending = performance.now();
@@ -180396,7 +182281,7 @@ async function runPipeline(args) {
180396
182281
  const activeTags = getActiveTagsBySession(args.db, args.sessionId);
180397
182282
  logTransformTiming(args.sessionId, "getActiveTagsBySession", tActiveTags, `count=${activeTags.length}`);
180398
182283
  if (shouldRunHeuristics) {
180399
- const reason = args.forceMaterialization ? "force_materialization" : `scheduler_execute (pendingOps=${pendingOps.length}, scheduler=${args.schedulerDecision})`;
182284
+ const reason = args.forceMaterialization ? "force_materialization" : m0HardFoldThisPass && args.schedulerDecision !== "execute" ? `m0_hard_fold (drain folded into known m[0] bust, scheduler=${args.schedulerDecision})` : `scheduler_execute (pendingOps=${pendingOps.length}, scheduler=${args.schedulerDecision})`;
180400
182285
  sessionLog(args.sessionId, `heuristics WILL RUN — reason=${reason}, context=${args.contextUsage.percentage.toFixed(1)}%, turn=n/a`);
180401
182286
  } else {
180402
182287
  const reason = args.heuristics === undefined ? "disabled" : "scheduler_defer";
@@ -180484,17 +182369,6 @@ async function runPipeline(args) {
180484
182369
  if (args.injection) {
180485
182370
  try {
180486
182371
  const tInjection = performance.now();
180487
- const hardMeta = getOrCreateSessionMeta(args.db, args.sessionId);
180488
- let piTtlMs = 5 * 60 * 1000;
180489
- try {
180490
- piTtlMs = parseCacheTtl(hardMeta.cacheTtl);
180491
- } catch {}
180492
- const piHardSignals = {
180493
- systemHash: typeof hardMeta.systemPromptHash === "string" ? hardMeta.systemPromptHash : "",
180494
- modelKey: liveModelBySession.get(args.sessionId) ?? "",
180495
- cacheExpired: hardMeta.lastResponseTime > 0 && Date.now() - hardMeta.lastResponseTime >= piTtlMs,
180496
- lastResponseTime: hardMeta.lastResponseTime
180497
- };
180498
182372
  injectionResult = injectM0M1Pi({
180499
182373
  sessionId: args.sessionId,
180500
182374
  projectIdentity: args.projectIdentity,
@@ -181584,8 +183458,12 @@ Found ${existingStaging.compartments.length} staged compartment(s) from ${existi
181584
183458
  const projectIdentity = resolveProjectIdentity(sessionDirectory);
181585
183459
  await deps.ensureProjectRegistered?.(sessionDirectory, db);
181586
183460
  const liveCompartments = getCompartments(db, sessionId);
181587
- const toEmbed = liveCompartments.map((c) => ({ id: c.id, p1: c.p1 ?? c.content })).filter((c) => typeof c.id === "number" && c.p1.length > 0);
181588
- embedAndStoreCompartments(db, sessionId, projectIdentity, toEmbed);
183461
+ const chunksToEmbed = liveCompartments.map((c) => ({
183462
+ id: c.id,
183463
+ startMessage: c.startMessage,
183464
+ endMessage: c.endMessage
183465
+ }));
183466
+ embedAndStoreCompartmentChunks(db, sessionId, projectIdentity, chunksToEmbed);
181589
183467
  }
181590
183468
  const lastCompartmentEnd2 = promoted2.compartments[promoted2.compartments.length - 1]?.endMessage ?? 0;
181591
183469
  if (lastCompartmentEnd2 > 0) {
@@ -181798,8 +183676,12 @@ Another process acquired the compartment-state lease before recomp could publish
181798
183676
  const projectIdentity = resolveProjectIdentity(sessionDirectory);
181799
183677
  await deps.ensureProjectRegistered?.(sessionDirectory, db);
181800
183678
  const liveCompartments = getCompartments(db, sessionId);
181801
- const toEmbed = liveCompartments.map((c) => ({ id: c.id, p1: c.p1 ?? c.content })).filter((c) => typeof c.id === "number" && c.p1.length > 0);
181802
- embedAndStoreCompartments(db, sessionId, projectIdentity, toEmbed);
183679
+ const chunksToEmbed = liveCompartments.map((c) => ({
183680
+ id: c.id,
183681
+ startMessage: c.startMessage,
183682
+ endMessage: c.endMessage
183683
+ }));
183684
+ embedAndStoreCompartmentChunks(db, sessionId, projectIdentity, chunksToEmbed);
181803
183685
  }
181804
183686
  if (lastCompartmentEnd > 0) {
181805
183687
  const markerUpdated = updateCompactionMarkerAfterPublication(db, sessionId, lastCompartmentEnd, deps.directory);
@@ -181956,8 +183838,12 @@ Could not acquire the compartment-state lease for this session.`;
181956
183838
  if (deps.memoryEnabled !== false) {
181957
183839
  const projectIdentity = resolveProjectIdentity(sessionDirectory);
181958
183840
  const liveCompartments = getCompartments(db, sessionId);
181959
- const toEmbed = liveCompartments.map((c) => ({ id: c.id, p1: c.p1 ?? c.content })).filter((c) => typeof c.id === "number" && c.p1.length > 0);
181960
- Promise.resolve(deps.ensureProjectRegistered?.(sessionDirectory, db)).then(() => embedAndStoreCompartments(db, sessionId, projectIdentity, toEmbed));
183841
+ const chunksToEmbed = liveCompartments.map((c) => ({
183842
+ id: c.id,
183843
+ startMessage: c.startMessage,
183844
+ endMessage: c.endMessage
183845
+ }));
183846
+ Promise.resolve(deps.ensureProjectRegistered?.(sessionDirectory, db)).then(() => embedAndStoreCompartmentChunks(db, sessionId, projectIdentity, chunksToEmbed));
181961
183847
  }
181962
183848
  const lastEnd = merged[merged.length - 1]?.endMessage ?? snapEnd;
181963
183849
  if (lastEnd > 0) {
@@ -182403,7 +184289,7 @@ function renderStatusText(ctx, db, sessionId) {
182403
184289
  const pct = typeof usage?.percent === "number" ? usage.percent : undefined;
182404
184290
  const meta3 = readSessionMetaStatus(db, sessionId);
182405
184291
  const state = renderHistorianState(meta3, recompSessions.has(sessionId));
182406
- return `mc: ${inputTokens === undefined ? "--" : fmt(inputTokens)} (${pct === undefined ? "--" : `${Math.round(pct)}%`}) · ${state}`;
184292
+ return `MC: ${inputTokens === undefined ? "--" : fmt(inputTokens)} (${pct === undefined ? "--" : `${Math.round(pct)}%`}) · ${state}`;
182407
184293
  }
182408
184294
  function renderHistorianState(meta3, recompActive) {
182409
184295
  const failureCount = meta3?.historian_failure_count ?? 0;
@@ -182769,7 +184655,7 @@ function applyMemoryMigration(db, projectPath, result) {
182769
184655
  inserted++;
182770
184656
  }
182771
184657
  if (removed > 0 || inserted > 0) {
182772
- bumpProjectMemoryEpoch(db, projectPath);
184658
+ bumpEpochsForWorkspaceMembers(db, projectPath);
182773
184659
  }
182774
184660
  })();
182775
184661
  return { removed, inserted };
@@ -183245,7 +185131,7 @@ function formatThresholdPercent(value) {
183245
185131
  // package.json
183246
185132
  var package_default = {
183247
185133
  name: "@wolfx/pi-magic-context",
183248
- version: "0.23.1",
185134
+ version: "0.24.0",
183249
185135
  type: "module",
183250
185136
  description: "Pi coding agent extension for Magic Context — cross-session memory and context management",
183251
185137
  main: "dist/index.js",
@@ -188136,6 +190022,23 @@ function createCtxMemoryTool(deps) {
188136
190022
  }
188137
190023
  const projectIdentity = resolveProjectIdentity(ctx.cwd);
188138
190024
  await deps.ensureProjectRegistered?.(ctx.cwd, deps.db);
190025
+ const workspaceIdentitySet = resolveWorkspaceIdentitySet(deps.db, projectIdentity);
190026
+ const expandedWorkspace = expandWorkspaceIdentitySetWithAliases(deps.db, workspaceIdentitySet.identities);
190027
+ const workspaceVisibleIdentities = workspaceIdentitySet.identities.length > 1 ? expandedWorkspace.expandedIdentities : workspaceIdentitySet.identities;
190028
+ const targetIdentityForStoredPath = (rawProjectPath) => workspaceIdentitySet.identities.length > 1 ? resolveStoredPathWorkspaceIdentity(rawProjectPath, workspaceIdentitySet.identities, expandedWorkspace.canonicalIdentityByStoredPath) ?? normalizeStoredProjectPath(rawProjectPath) : normalizeStoredProjectPath(rawProjectPath);
190029
+ const toolShareCategories = workspaceIdentitySet.identities.length > 1 ? resolveWorkspaceShareCategories(deps.db, projectIdentity) : null;
190030
+ const memoryVisibleToTool = (memory2) => {
190031
+ if (workspaceIdentitySet.identities.length <= 1) {
190032
+ return storedPathBelongsToIdentity(memory2.projectPath, projectIdentity);
190033
+ }
190034
+ if (!storedPathBelongsToWorkspace(memory2.projectPath, workspaceIdentitySet.identities, workspaceVisibleIdentities, expandedWorkspace.canonicalIdentityByStoredPath)) {
190035
+ return false;
190036
+ }
190037
+ const isOwn = targetIdentityForStoredPath(memory2.projectPath) === projectIdentity;
190038
+ if (isOwn)
190039
+ return true;
190040
+ return toolShareCategories === null || toolShareCategories.includes(memory2.category);
190041
+ };
188139
190042
  const snapshot = getProjectEmbeddingSnapshot(projectIdentity);
188140
190043
  if (snapshot ? !snapshot.features.memoryEnabled : deps.memoryEnabled === false) {
188141
190044
  return err2("Cross-session memory is disabled for this project.");
@@ -188182,28 +190085,34 @@ function createCtxMemoryTool(deps) {
188182
190085
  return err2("Error: 'content' is required when action is 'update'.");
188183
190086
  }
188184
190087
  const memory2 = getMemoryById(deps.db, updateId);
188185
- if (!memory2 || !storedPathBelongsToIdentity(memory2.projectPath, projectIdentity)) {
190088
+ if (!memory2 || !memoryVisibleToTool(memory2)) {
188186
190089
  return err2(`Error: Memory with ID ${updateId} was not found.`);
188187
190090
  }
188188
190091
  if (!dreamerAllowed && !isPrimaryMutableMemory(memory2)) {
188189
190092
  return err2(inactiveMemoryError(updateId, "updating"));
188190
190093
  }
188191
190094
  const normalizedHash = computeNormalizedHash(content);
188192
- const duplicate = getMemoryByHash(deps.db, projectIdentity, memory2.category, normalizedHash);
190095
+ const targetIdentity = targetIdentityForStoredPath(memory2.projectPath);
190096
+ const duplicate = getMemoryByHash(deps.db, targetIdentity, memory2.category, normalizedHash);
188193
190097
  if (duplicate && duplicate.id !== memory2.id) {
188194
190098
  return err2(`Error: Memory content already exists as ID ${duplicate.id}; merge or archive duplicates instead.`);
188195
190099
  }
188196
190100
  deps.db.transaction(() => {
188197
190101
  updateMemoryContent(deps.db, memory2.id, content, normalizedHash);
188198
190102
  queueMemoryMutation(deps.db, {
188199
- projectPath: projectIdentity,
190103
+ projectPath: targetIdentity,
188200
190104
  mutationType: "update",
188201
190105
  targetMemoryId: memory2.id,
188202
190106
  category: memory2.category,
188203
190107
  newContent: content
188204
190108
  });
188205
190109
  })();
188206
- queueEmbedding({ deps, projectIdentity, memoryId: memory2.id, content });
190110
+ queueEmbedding({
190111
+ deps,
190112
+ projectIdentity: targetIdentity,
190113
+ memoryId: memory2.id,
190114
+ content
190115
+ });
188207
190116
  return ok2(`Updated memory [ID: ${memory2.id}] in ${memory2.category}.`);
188208
190117
  }
188209
190118
  if (params.action === "merge") {
@@ -188223,7 +190132,7 @@ function createCtxMemoryTool(deps) {
188223
190132
  return err2("Error: One or more source memories were not found.");
188224
190133
  }
188225
190134
  if (!dreamerAllowed) {
188226
- const foreign = sourceMemories.find((memory2) => !storedPathBelongsToIdentity(memory2.projectPath, projectIdentity));
190135
+ const foreign = sourceMemories.find((memory2) => !memoryVisibleToTool(memory2));
188227
190136
  if (foreign) {
188228
190137
  return err2(`Error: Memory with ID ${foreign.id} was not found.`);
188229
190138
  }
@@ -188312,26 +190221,36 @@ function createCtxMemoryTool(deps) {
188312
190221
  return ok2(`Merged memories [${ids.join(", ")}] into canonical memory [ID: ${canonicalMemory.id}] in ${category}; superseded [${supersededIds.join(", ")}].`);
188313
190222
  }
188314
190223
  if (params.action === "archive") {
188315
- const archiveIds = params.ids;
188316
- if (!archiveIds || archiveIds.length === 0 || !archiveIds.every(Number.isInteger)) {
190224
+ const rawArchiveIds = params.ids;
190225
+ if (!rawArchiveIds || rawArchiveIds.length === 0 || !rawArchiveIds.every(Number.isInteger)) {
188317
190226
  return err2("Error: 'ids' must contain at least one integer memory ID when action is 'archive'.");
188318
190227
  }
190228
+ const archiveIds = [...new Set(rawArchiveIds)];
188319
190229
  for (const memoryId of archiveIds) {
188320
190230
  const memory2 = getMemoryById(deps.db, memoryId);
188321
- if (!memory2 || !storedPathBelongsToIdentity(memory2.projectPath, projectIdentity)) {
190231
+ if (!memory2 || !memoryVisibleToTool(memory2)) {
188322
190232
  return err2(`Error: Memory with ID ${memoryId} was not found.`);
188323
190233
  }
188324
190234
  if (!dreamerAllowed && !isPrimaryMutableMemory(memory2)) {
188325
190235
  return err2(inactiveMemoryError(memoryId, "archiving"));
188326
190236
  }
188327
190237
  }
190238
+ const targets = archiveIds.map((memoryId) => {
190239
+ const memory2 = getMemoryById(deps.db, memoryId);
190240
+ if (!memory2)
190241
+ throw new Error(`validated memory ${memoryId} disappeared`);
190242
+ return {
190243
+ memoryId,
190244
+ projectIdentity: targetIdentityForStoredPath(memory2.projectPath)
190245
+ };
190246
+ });
188328
190247
  deps.db.transaction(() => {
188329
- for (const memoryId of archiveIds) {
188330
- archiveMemory(deps.db, memoryId, params.reason);
190248
+ for (const target2 of targets) {
190249
+ archiveMemory(deps.db, target2.memoryId, params.reason);
188331
190250
  queueMemoryMutation(deps.db, {
188332
- projectPath: projectIdentity,
190251
+ projectPath: target2.projectIdentity,
188333
190252
  mutationType: "archive",
188334
- targetMemoryId: memoryId
190253
+ targetMemoryId: target2.memoryId
188335
190254
  });
188336
190255
  }
188337
190256
  })();
@@ -188839,8 +190758,9 @@ function formatAge2(committedAtMs) {
188839
190758
  }
188840
190759
  function formatResult(result, index) {
188841
190760
  if (result.source === "memory") {
190761
+ const source = result.sourceName ? ` source=${result.sourceName}` : "";
188842
190762
  return [
188843
- `[${index}] [memory] score=${result.score.toFixed(2)} id=${result.memoryId} category=${result.category} match=${result.matchType}`,
190763
+ `[${index}] [memory] score=${result.score.toFixed(2)} id=${result.memoryId} category=${result.category}${source} match=${result.matchType}`,
188844
190764
  result.content
188845
190765
  ].join(`
188846
190766
  `);
@@ -188850,6 +190770,13 @@ function formatResult(result, index) {
188850
190770
  `[${index}] [git_commit] score=${result.score.toFixed(2)} sha=${result.shortSha} ${formatAge2(result.committedAtMs)} match=${result.matchType}`,
188851
190771
  result.content
188852
190772
  ].join(`
190773
+ `);
190774
+ }
190775
+ if (result.source === "compartment") {
190776
+ return [
190777
+ `[${index}] [message] score=${result.score.toFixed(2)} compartment_id=${result.compartmentId} range=${result.startOrdinal}-${result.endOrdinal} match=${result.matchType} title=${result.title}`,
190778
+ result.snippet ? `Snippet: ${result.snippet}` : result.content
190779
+ ].join(`
188853
190780
  `);
188854
190781
  }
188855
190782
  const expandStart = Math.max(1, result.messageOrdinal - 3);
@@ -188865,7 +190792,7 @@ function formatSearchResults(query, results) {
188865
190792
  return `No results found for "${query}" across memories, git commits, or message history.`;
188866
190793
  }
188867
190794
  const bodyParts = results.map((result, index) => formatResult(result, index + 1));
188868
- if (results.some((result) => result.source === "message")) {
190795
+ if (results.some((result) => result.source === "message" || result.source === "compartment")) {
188869
190796
  bodyParts.push("Use ctx_expand(start, end) with the range from any message result above to read the full conversation context.");
188870
190797
  }
188871
190798
  const body = bodyParts.join(`
@@ -189361,6 +191288,14 @@ async function src_default2(pi) {
189361
191288
  onProjectSeen: (identity) => seenDreamerProjectIdentities.add(identity)
189362
191289
  });
189363
191290
  info("registered /ctx-dream");
191291
+ registerCtxEmbedHistoryCommand(pi, {
191292
+ db,
191293
+ projectDir,
191294
+ projectIdentity,
191295
+ memoryEnabled: config2.memory.enabled,
191296
+ resolveProject: resolveCurrentProject
191297
+ });
191298
+ info("registered /ctx-embed-history");
189364
191299
  const dreamerConfig = resolveDreamerFromConfig(config2);
189365
191300
  if (dreamerConfig) {
189366
191301
  registerPiDreamerProject({