Get started
Refactor dello stato database-first
Refactor dello stato con priorità al database
Decisione
Usa un layout SQLite a due livelli:
- Database globale:
~/.openclaw/state/openclaw.sqlite - Database dell'agente: un database SQLite per ogni agente per workspace, trascrizione, VFS, artefatti e stato di runtime per agente di grandi dimensioni di proprietà dell'agente
- La configurazione resta basata su file:
openclaw.jsonrimane fuori dal database. I profili di autenticazione di runtime passano a SQLite; i file di credenziali di provider esterni o della CLI rimangono gestiti dal proprietario fuori dal database di OpenClaw.
Il database globale è il database del control plane. Possiede il rilevamento degli agenti, lo stato condiviso del Gateway, pairing, stato di dispositivi/nodi, registri di attività e flussi, stato dei Plugin, stato di runtime dello scheduler, metadati dei backup e stato delle migrazioni.
Il database dell'agente è il database del data plane. Possiede i metadati di sessione dell'agente, lo stream di eventi della trascrizione, il workspace VFS o namespace scratch, gli artefatti degli strumenti, gli artefatti delle esecuzioni e i dati di cache locali all'agente ricercabili/indicizzabili.
Questo offre una vista globale durevole senza forzare workspace degli agenti, trascrizioni e dati scratch binari di grandi dimensioni nella corsia di scrittura condivisa del Gateway.
Contratto vincolante
Questa migrazione ha una sola forma canonica di runtime:
- Le righe di sessione persistono solo i metadati di sessione. Non devono persistere
transcriptLocator, percorsi dei file di trascrizione, percorsi JSONL sibling, percorsi di lock, metadati di pruning o puntatori di compatibilità dell'era file. - L'identità della trascrizione è sempre identità SQLite:
{agentId, sessionId}più metadati opzionali del topic dove il protocollo ne ha bisogno. sqlite-transcript://...non è un'identità di runtime o di protocollo. Il nuovo codice non deve derivare, persistere, passare, analizzare o migrare locator di trascrizione. Runtime e test non devono contenere pseudo-locator affatto; la documentazione può menzionare la stringa solo per vietarla.- I vecchi
sessions.json, JSONL di trascrizione,.jsonl.lock, pruning, troncamento e vecchia logica dei percorsi di sessione appartengono solo al percorso di migrazione/importazione doctor. - Gli alias di configurazione di sessione legacy appartengono solo alla migrazione doctor. Il runtime non
interpreta
session.idleMinutes,session.resetByType.dmo alias di sessione principale cross-agentagent:main:*per un altro agente configurato. - L'identità di routing della sessione è stato relazionale tipizzato. I percorsi di runtime hot e UI
dovrebbero leggere
sessions.session_scope,sessions.account_id,sessions.primary_conversation_id,conversationsesession_conversations; non devono analizzaresession_keyo estrarre dasession_entries.entry_jsonl'identità del provider, tranne come ombra di compatibilità mentre i vecchi call site vengono eliminati. - I marker di messaggio diretto a livello di canale, come
dmrispetto adirect, sono vocabolario di routing, non locator di trascrizione o handle di compatibilità del file store. - La configurazione legacy degli handler hook appartiene solo alle superfici di avviso/migrazione doctor.
Il runtime non deve caricare
hooks.internal.handlers; gli hook vengono eseguiti solo tramite directory hook rilevate e metadatiHOOK.md. - Avvio del runtime, percorsi hot di risposta, Compaction, reset, recovery, diagnostica,
TTS, hook di memoria, subagenti, routing dei comandi Plugin, confini di protocollo e
hook devono passare
{agentId, sessionId}attraverso il runtime. - I test dovrebbero inizializzare e verificare righe di trascrizione SQLite tramite
{agentId, sessionId}. I test che dimostrano solo inoltro di percorsi JSONL, preservazione di locator forniti dal chiamante o compatibilità dei file di trascrizione dovrebbero essere eliminati, a meno che coprano importazione doctor, materializzazione di supporto/debug non di sessione o forma del protocollo. runEmbeddedPiAgent(...), esecuzioni worker preparate e il tentativo embedded interno non devono accettare locator di trascrizione. Aprono il gestore delle trascrizioni SQLite tramite{agentId, sessionId}e passano quel gestore alla sessione agente compatibile con PI internalizzata, così chiamanti obsoleti non possono far scrivere al runner trascrizioni JSON/JSONL.- La diagnostica del runner deve archiviare record di trace runtime/cache/payload in SQLite. La diagnostica di runtime non deve esporre manopole di override dei file JSONL o helper generici di esportazione JSONL delle trascrizioni; le esportazioni visibili all'utente possono materializzare artefatti espliciti da righe del database senza reimmettere nomi di file nel runtime.
- Il logging dello stream grezzo usa
OPENCLAW_RAW_STREAM=1più righe diagnostiche SQLite. Il vecchio contratto del logger file pi-monoPI_RAW_STREAM,PI_RAW_STREAM_PATHeraw-openai-completions.jsonlnon fa parte del runtime o dei test OpenClaw. - L'indicizzazione memoria QMD non deve esportare trascrizioni SQLite in file markdown. QMD indicizza solo i file di memoria configurati; la ricerca delle trascrizioni di sessione resta basata su SQLite.
- Il sottopercorso SDK QMD è solo QMD per il nuovo codice. Gli helper di indicizzazione
delle trascrizioni di sessione SQLite vivono su
memory-core-host-engine-session-transcripts; qualsiasi riesportazione QMD è solo compatibilità e non deve essere usata dal codice di runtime. - Gli indici memoria integrati vivono nel database dell'agente proprietario. La configurazione di runtime e
i contratti di runtime risolti non devono esporre
memorySearch.store.path; doctor elimina quella chiave di configurazione legacy e il codice corrente passa internamente ildatabasePathdell'agente.
Il lavoro di implementazione dovrebbe continuare a eliminare codice finché queste affermazioni sono vere senza eccezioni al di fuori dei confini doctor/import/export/debug.
Stato obiettivo e avanzamento
Obiettivo vincolante
- Un database SQLite globale possiede lo stato del control plane:
state/openclaw.sqlite. - Un database SQLite per agente possiede lo stato del data plane:
agents/<agentId>/agent/openclaw-agent.sqlite. - La configurazione resta basata su file.
openclaw.jsonnon fa parte di questo refactor del database. - I file legacy sono solo input di migrazione doctor.
- Il runtime non scrive né legge mai JSONL di sessioni o trascrizioni come stato attivo.
Stati obiettivo
not-started: il codice di runtime dell'era file scrive ancora stato attivo.migrating: il codice doctor/import può spostare dati file in SQLite.dual-read: bridge temporaneo che legge sia SQLite sia file legacy. Questo stato è vietato per questo refactor, a meno che sia esplicitamente documentato come solo doctor.sqlite-runtime: il runtime legge e scrive solo SQLite.clean: le API e i test di runtime legacy sono rimossi e la guardia previene regressioni.done: documentazione, test, backup, migrazione doctor e controlli sulle modifiche dimostrano lo stato pulito.
Stato corrente
- Sessioni:
cleanper il runtime. Le righe di sessione vivono nel database per agente, le API di runtime usano{agentId, sessionId}o{agentId, sessionKey}esessions.jsonè input legacy solo doctor. - Trascrizioni:
cleanper il runtime. Eventi di trascrizione, identità, snapshot ed eventi di runtime delle traiettorie vivono nel database per agente. Il runtime non accetta più locator di trascrizione o percorsi JSONL di trascrizione. - Runner PI embedded:
clean. Esecuzioni PI embedded, worker preparati, Compaction e cicli di retry usano l'ambito di sessione SQLite e rifiutano handle di trascrizione obsoleti. - Cron:
cleanper il runtime. Il runtime usacron_jobsecron_run_logs; i test di runtime usano denominazione SQLitestoreKeye i percorsi Cron dell'era file rimangono solo nei test di migrazione legacy doctor. - Registro attività:
clean. Le righe di runtime per attività e flussi di attività vivono instate/openclaw.sqlite; gli importer SQLite sidecar non rilasciati sono eliminati. - Stato Plugin:
clean. Le righe stato/blob dei Plugin vivono nel database globale condiviso; i vecchi helper SQLite sidecar dello stato Plugin sono protetti da guardia. - Memoria:
sqlite-runtimeper memoria integrata e indicizzazione delle trascrizioni di sessione. Le tabelle degli indici memoria vivono nel database per agente, lo stato memoria dei Plugin usa righe stato Plugin condivise e i file memoria legacy sono input di migrazione doctor o contenuto del workspace utente. - Backup:
sqlite-runtime. Le fasi di backup compattano snapshot SQLite, omettono sidecar WAL/SHM live, verificano l'integrità SQLite e registrano le esecuzioni di backup nel database globale. - Migrazione doctor:
migrating, intenzionalmente. Doctor importa JSON legacy, JSONL e store sidecar ritirati in SQLite, registra esecuzioni/fonti di migrazione e rimuove le fonti riuscite. - Script E2E:
cleanper la copertura runtime. Il seeding Docker MCP scrive righe SQLite. Lo script Docker runtime-context crea JSONL legacy solo all'interno del seed di migrazione doctor e denomina esplicitamente il percorso dell'indice sessioni legacy.
Lavoro rimanente
- [x] Rinominare le variabili store dei test di runtime Cron eliminando
storePath, salvo quando sono input legacy doctor. File:src/cron/service.test-harness.ts,src/cron/service.runs-one-shot-main-job-disables-it.test.ts,src/cron/service/timer.regression.test.ts,src/cron/service/ops.test.ts,src/cron/service/store.test.ts,src/cron/service.heartbeat-ok-summary-suppressed.test.ts,src/cron/service.main-job-passes-heartbeat-target-last.test.ts,src/cron/store.test.ts. Prova:pnpm check:database-first-legacy-stores;rg -n 'storePath' src/cron --glob '!**/commands/doctor/**'. - [x] Rimuovere o rinominare i mock obsoleti dei test di esportazione dell'era file.
File:
src/auto-reply/reply/commands-export-test-mocks.ts. Prova:rg -n 'resolveSessionFilePath|sessionFile|storePath|transcriptLocator' src/auto-reply/reply. - [x] Rendere il seed JSONL legacy Docker runtime-context chiaramente solo doctor.
File:
scripts/e2e/session-runtime-context-docker-client.ts. Prova:rg -n 'sessions\\.json|sessionFile|\\.jsonl' scripts/e2e/session-runtime-context-docker-client.tsmostra soloseedBrokenLegacySessionForDoctorMigration. - [x] Mantenere allineati i tipi generati Kysely dopo qualsiasi modifica allo schema.
File:
src/state/openclaw-state-schema.sql,src/state/openclaw-agent-schema.sql,src/state/*generated*. Prova: nessuna modifica allo schema in questo passaggio;pnpm db:kysely:check;pnpm lint:kysely. - [x] Rieseguire test mirati per store, comandi e script toccati.
Prova:
pnpm test src/cron/service/store.test.ts src/cron/store.test.ts src/cron/service.heartbeat-ok-summary-suppressed.test.ts src/cron/service.main-job-passes-heartbeat-target-last.test.ts src/cron/service.every-jobs-fire.test.ts src/cron/service.persists-delivered-status.test.ts src/cron/service.runs-one-shot-main-job-disables-it.test.ts src/cron/service/ops.test.ts src/cron/service/timer.regression.test.ts src/auto-reply/reply/commands-export-trajectory.test.ts extensions/telegram/src/thread-bindings.test.ts extensions/slack/src/monitor/message-handler/prepare.test.ts src/acp/translator.session-lineage-meta.test.ts;git diff --check. - [x] Prima di dichiarare
done, eseguire il gate delle modifiche o una prova ampia remota. Prova:pnpm check:changed --timed -- <changed extension paths>superato su Hetzner Crabbox runrun_3f1cabf6b25cdopo configurazione temporanea di Node 24/pnpm e routing esplicito dei percorsi per il workspace sincronizzato senza.git.
Da non regredire
- Nessun locator di trascrizione.
- Nessun file di sessione attivo.
- Nessuna fixture di test JSONL fittizia eccetto i test di migrazione legacy doctor.
- Nessun accesso SQLite grezzo dove è atteso Kysely.
- Nessuna nuova migrazione DB legacy. Questo layout non è stato rilasciato; mantenere la versione dello schema
a
1salvo motivo forte.
Presupposti di lettura del codice
Nessuna decisione di prodotto successiva blocca questo piano. L'implementazione dovrebbe procedere con questi presupposti:
- Usa direttamente
node:sqlitee richiedi il runtime Node 22+ per questo percorso di archiviazione. - Mantieni esattamente un normale file di configurazione. Non spostare configurazione, manifest dei plugin o workspace Git in SQLite in questo refactor.
- I file di compatibilità runtime non sono richiesti. I file JSON e JSONL legacy sono solo input di migrazione. I sidecar SQLite locali al branch non sono mai stati rilasciati e vengono eliminati invece di essere importati.
openclaw doctor --fixpossiede il passaggio di migrazione legacy da file a database. L'avvio del runtime eopenclaw migratenon devono includere percorsi legacy di upgrade del database OpenClaw.- La compatibilità delle credenziali segue la stessa regola: le credenziali runtime vivono in
SQLite. I vecchi file
auth-profiles.json,auth.jsonper agente ecredentials/oauth.jsoncondiviso sono input di migrazione di doctor, quindi vengono rimossi dopo l'importazione. - Lo stato del catalogo dei modelli generato è supportato dal database. Il codice runtime non deve scrivere
agents/<agentId>/agent/models.json; i filemodels.jsonesistenti sono input legacy di doctor e vengono rimossi dopo l'importazione inagent_model_catalogs. - Il runtime non deve migrare, normalizzare o collegare locator di trascrizioni. L'identità
della trascrizione attiva è
{agentId, sessionId}in SQLite. I percorsi file sono solo input legacy di doctor, esqlite-transcript://...deve sparire dalle superfici runtime, protocollo, hook e Plugin invece di essere trattato come handle di confine. - Le letture runtime delle trascrizioni SQLite non eseguono vecchie migrazioni della forma delle voci JSONL né riscrivono intere trascrizioni per compatibilità. La normalizzazione delle voci legacy resta nelle utilità esplicite di doctor/importazione. Doctor normalizza i file di trascrizione JSONL legacy prima di inserire righe SQLite; le righe runtime attuali sono già scritte nello schema di trascrizione attuale. L'esportazione di traiettorie/sessioni legge quelle righe così come sono e non deve eseguire migrazioni legacy al momento dell'esportazione.
- Gli helper legacy di parsing/migrazione delle trascrizioni JSONL sono solo per doctor. Il codice runtime del formato delle trascrizioni costruisce solo il contesto attuale delle trascrizioni SQLite; doctor possiede gli upgrade delle vecchie voci JSONL prima dell'inserimento delle righe.
- Il vecchio helper di streaming delle trascrizioni JSONL posseduto dal runtime è stato eliminato. Il codice di importazione di doctor possiede le letture esplicite dei file legacy; la cronologia delle sessioni runtime legge righe SQLite.
- I binding app-server di Codex usano
sessionIddi OpenClaw come chiave canonica nel namespace di stato del Plugin Codex.sessionKeyè metadato per routing/visualizzazione e non deve sostituire l'id durevole della sessione né ripristinare l'identità basata su file di trascrizione. - I motori di contesto ricevono direttamente il contratto runtime attuale. Il registro
non deve avvolgere i motori con shim di retry che eliminano
sessionKey,transcriptScopeoprompt; i motori che non possono accettare i parametri attuali database-first devono fallire in modo evidente invece di essere collegati. - L'output di backup deve restare un unico file archivio. I contenuti del database devono entrare in quell'archivio come snapshot SQLite compatti, non come sidecar WAL live grezzi.
- La ricerca nelle trascrizioni è utile ma non richiesta per il primo passaggio database-first. Progetta lo schema in modo che FTS possa essere aggiunto in seguito.
- L'esecuzione dei worker deve restare sperimentale dietro impostazioni mentre il confine del database si stabilizza.
Risultati della Lettura del Codice
Il branch attuale ha già superato la fase di proof-of-concept. Il database condiviso
esiste, Node node:sqlite è collegato tramite un piccolo helper runtime, e
gli store precedenti ora scrivono in state/openclaw.sqlite o nel database
openclaw-agent.sqlite proprietario.
Il lavoro rimanente non è scegliere SQLite; è mantenere pulito il nuovo confine ed eliminare qualsiasi interfaccia dalla forma compatibile che assomigli ancora al vecchio mondo dei file:
storePathdi sessione non è più un'identità runtime, una forma di fixture di test o un campo del payload di stato. I test runtime e bridge non contengono più il nome di contrattostorePath; il codice doctor/migrazione possiede quel vocabolario legacy.- Le scritture di sessione non passano più attraverso la vecchia coda in-process
store-writer.ts. Le scritture di patch SQLite usano invece rilevamento dei conflitti e retry limitato. - La scoperta dei percorsi legacy ha ancora usi di migrazione validi, ma il codice runtime dovrebbe
smettere di trattare
sessions.jsone i file JSONL di trascrizione come possibili target di scrittura. - Le tabelle possedute dall'agente vivono nei database SQLite per agente. Il DB globale mantiene
righe di registro/control-plane; l'identità della trascrizione è
{agentId, sessionId}nelle righe di trascrizione per agente. Il codice runtime non deve persistere percorsi file di trascrizione né migrare locator di trascrizioni. - Doctor importa già diversi file legacy. La pulizia consiste nel renderla una singola implementazione di migrazione esplicita che doctor richiama, con un report di migrazione durevole.
Nessuna ulteriore domanda di prodotto blocca l'implementazione.
Forma Attuale del Codice
Il branch ha già una vera base SQLite condivisa:
- Il requisito minimo di runtime ora è Node 22+:
package.json, la guardia di runtime della CLI, i valori predefiniti dell'installer, il localizzatore del runtime macOS, la CI e la documentazione pubblica di installazione sono tutti allineati. La vecchia corsia di compatibilità con Node 22 è rimossa. src/state/openclaw-state-db.tsapreopenclaw.sqlite, imposta WAL,synchronous=NORMAL,busy_timeout=30000,foreign_keys=ONe applica il modulo di schema generato derivato dasrc/state/openclaw-state-schema.sql.- I tipi di tabella Kysely e i moduli di schema runtime sono generati da database
SQLite temporanei creati dai file
.sqlsottoposti a commit; il codice runtime non mantiene più stringhe di schema copiate e incollate per database globali, per agente o di acquisizione proxy. - Gli store runtime derivano i tipi di righe selezionate e inserite da quelle interfacce
Kysely
DBgenerate invece di duplicare manualmente le forme delle righe SQLite. L'SQL grezzo resta limitato all'applicazione dello schema, ai pragma e al DDL solo per migrazione. - Gli schemi SQLite sono ridotti a
user_version = 1perché questo layout di database non è ancora stato distribuito. Gli opener runtime creano solo lo schema corrente; l'importazione da file a database resta nel codice doctor e gli helper di upgrade database locali al ramo sono stati eliminati. - La proprietà relazionale è applicata dove il confine di proprietà è canonico:
le righe di migrazione sorgente propagano l'eliminazione da
migration_runs, lo stato di consegna dei task propaga l'eliminazione datask_runse le righe di identità transcript propagano l'eliminazione dagli eventi transcript. - Le tabelle condivise correnti includono
agent_databases,auth_profile_stores,auth_profile_state,plugin_state_entries,plugin_blob_entries,media_blobs,skill_uploads,capture_sessions,capture_events,capture_blobs,sandbox_registry_entries,cron_run_logs,cron_jobs,commitments,delivery_queue_entries,model_capability_cache,workspace_setup_state,native_hook_relay_bridges,current_conversation_bindings,plugin_binding_approvals,tui_last_sessions,acp_sessions,acp_replay_sessions,acp_replay_events,task_runs,task_delivery_state,flow_runs,subagent_runs,migration_runsebackup_runs. - Lo stato arbitrario di proprietà dei plugin non ottiene tabelle tipizzate di proprietà dell'host. I plugin
installati usano
plugin_state_entriesper payload JSON versionati eplugin_blob_entriesper byte, con proprietà di namespace/chiave, pulizia TTL, backup e record di migrazione plugin. Lo stato di orchestrazione dei plugin di proprietà dell'host può comunque avere tabelle tipizzate quando l'host possiede il contratto di query, comeplugin_binding_approvals. - Le migrazioni dei plugin sono migrazioni di dati su namespace di proprietà dei plugin, non migrazioni
dello schema host. Un plugin può migrare le proprie voci stato/blob versionate
tramite un provider di migrazione, e l'host registra stato sorgente/esecuzione nel
normale registro di migrazione. Le nuove installazioni di plugin non richiedono modifiche a
openclaw-state-schema.sql, a meno che l'host stesso non stia assumendo la proprietà di un nuovo contratto cross-plugin. src/state/openclaw-agent-db.tsapreagents/<agentId>/agent/openclaw-agent.sqlite, registra il database nel DB globale e possiede le tabelle locali all'agente per sessioni, transcript, VFS, artefatti, cache e indici di memoria. La discovery runtime condivisa ora legge il registryagent_databasestipizzato generato invece di reimplementare quella query in ogni call site.- I database globali e per agente registrano una riga
schema_metacon ruolo del database, versione dello schema, timestamp e id agente per i database agente. Il layout resta comunque auser_version = 1perché questo schema SQLite non è ancora stato distribuito. - L'identità di sessione per agente ora ha una tabella radice canonica
sessionsindicizzata dasession_id, consession_key,session_scope,account_id,primary_conversation_id, timestamp, campi di visualizzazione, metadati modello, id harness e collegamento parent/spawn come colonne interrogabili.session_routesè l'indice univoco delle route attive dasession_keyalsession_idcorrente, quindi una chiave di route può spostarsi a una nuova sessione durevole senza costringere le letture hot a scegliere tra righesessions.session_keyduplicate. Il vecchio payload con forma di compatibilitàsession_entries.entry_jsonè collegato alla radice durevolesession_idtramite chiave esterna; non è più l'unica rappresentazione a livello di schema di una sessione. - Anche l'identità delle conversazioni esterne per agente è relazionale:
conversationsarchivia l'identità normalizzata di provider/account/conversazione, esession_conversationscollega una sessione OpenClaw a una o più conversazioni esterne. Questo copre le sessioni DM main condivise in cui più peer possono intenzionalmente mappare a una sessione senza mentire insession_key. SQLite applica anche l'univocità per l'identità naturale del provider, così la stessa tupla channel/account/kind/peer/thread non può biforcarsi su id conversazione diversi. I peer diretti main condivisi sono collegati con un ruoloparticipant, quindi una sessione OpenClaw può rappresentare più peer DM esterni senza retrocedere i peer precedenti a righe correlate vaghe.sessions.primary_conversation_idpunta ancora all'attuale target di consegna tipizzato. Le colonne chiuse di routing/stato sono applicate con vincoli SQLiteCHECKinvece di affidarsi solo alle union TypeScript. La proiezione runtime della sessione elimina le ombre di routing di compatibilità dasession_entries.entry_jsonprima di applicare le colonne tipizzate sessione/conversazione, così i payload JSON obsoleti non possono riesumare target di consegna. Il routing di annuncio dei subagent richiede allo stesso modo il contesto di consegna SQLite tipizzato; non ricade più sui campi route di compatibilitàSessionEntry. L'ereditarietà della consegna esplicita Gatewaychat.sendlegge il contesto di consegna SQLite tipizzato invece dei campi di compatibilitàorigin/last*. Anchetools.effectivederiva il contesto provider/account/thread da righe SQLite tipizzate di consegna/routing, non da ombre obsoletelast*della session-entry. Il contesto prompt degli eventi di sistema ricostruisce i campi channel/to/account/thread da campi di consegna tipizzati invece che da ombreorigin. L'helper condivisodeliveryContextFromSessione il mapper session-to-conversation ora ignorano completamenteSessionEntry.origin; solo i campi di consegna tipizzati e le righe di conversazione relazionali possono creare identità di route hot. La normalizzazione runtime delle session entry rimuoveoriginprima di persistere o proiettareentry_json, e le scritture dei metadati in ingresso scrivono campi channel/chat tipizzati più righe di conversazione relazionali invece di creare nuove ombre origin. - Gli eventi transcript, gli snapshot transcript e gli eventi runtime di traiettoria ora
referenziano la radice canonica
sessionsper agente e propagano l'eliminazione alla cancellazione della sessione. Le righe di identità/idempotenza transcript continuano a propagare l'eliminazione dalla riga esatta dell'evento transcript. - Gli indici memory-core ora usano tabelle esplicite del database agente
memory_index_meta,memory_index_sources,memory_index_chunksememory_embedding_cache, conmemory_index_stateche traccia le modifiche di revisione. Gli indici laterali FTS/vector opzionali sono denominatimemory_index_chunks_ftsememory_index_chunks_vecinvece delle tabelle generichemeta,files,chunks,chunks_ftsochunks_vec. I nomi canonici conservano l'attuale forma delle righe path/source e la compatibilità degli embedding serializzati. Queste tabelle sono cache derivate/di ricerca, non archiviazione canonica dei transcript; possono essere eliminate e ricostruite dai file del workspace di memoria e dalle sorgenti configurate. L'apertura di un indice di memoria distribuito con nomi generici migra i suoi metadati, sorgenti, chunk e cache degli embedding nelle tabelle canoniche; le tabelle derivate FTS/vector sono ricostruite con i loro nomi canonici. - Lo stato di recovery delle esecuzioni subagent ora vive in righe condivise tipizzate
subagent_runscon chiavi di sessione figlio, richiedente e controller indicizzate. Il vecchio filesubagents/runs.jsonè solo input di migrazione doctor. - I binding della conversazione corrente ora vivono in righe condivise tipizzate
current_conversation_bindingsindicizzate per id conversazione normalizzato, con colonne target agent/session, tipo di conversazione, stato, scadenza e metadati archiviati come colonne relazionali invece di un record di binding opaco duplicato. La chiave di binding durevole include il tipo di conversazione normalizzato, così i riferimenti direct/group/channel non possono collidere, e SQLite rifiuta valori di tipo/stato binding non validi. Il vecchio filebindings/current-conversations.jsonè solo input di migrazione doctor. - La recovery della coda di consegna ora sovrappone al JSON di replay colonne tipizzate della coda
per channel, target, account, sessione, retry, errore, invio piattaforma e stato
di recovery.
entry_jsonconserva i payload di replay, hook e formattazione, ma le colonne tipizzate sono autorevoli per routing/stato hot della coda. - I puntatori di ripristino dell'ultima sessione TUI ora vivono in righe condivise tipizzate
tui_last_sessionsindicizzate dallo scope hashato della connessione/sessione TUI. Il vecchio file JSON TUI è solo input di migrazione doctor. - Le preferenze TTS predefinite ora vivono in righe SQLite di stato plugin condivise indicizzate sotto il
plugin
speech-core. Il vecchio filesettings/tts.jsonè solo input di migrazione doctor; il runtime non legge né scrive più file JSON di preferenze TTS, e il resolver del percorso legacy vive nel modulo di migrazione doctor. - I metadati dei target dei segreti ora parlano di store invece di fingere che ogni
target di credenziali sia un file di configurazione.
openclaw.jsonresta lo store di configurazione; i target auth-profile usano righe SQLite tipizzateauth_profile_storescon credenziali conformate al provider mantenute come payload JSON. - L'audit dei segreti non scansiona più i file
auth.jsonper agente ritirati. Doctor possiede gli avvisi, l'importazione e la rimozione di quel file legacy. - Gli helper legacy dei percorsi dei profili auth ora vivono nel codice legacy doctor. Gli helper dei percorsi
dei profili auth core espongono identità e posizioni di visualizzazione auth-store SQLite,
non percorsi runtime
auth-profiles.jsonoauth-state.json. - I moduli runtime di recovery delle esecuzioni subagent e della cache delle capacità modello OpenRouter
ora tengono separati i lettori/scrittori di snapshot SQLite dagli helper di import JSON legacy
solo doctor. Le capacità OpenRouter usano le righe generiche tipizzate
model_capability_cachesottoprovider_id = "openrouter"invece di un unico blob cache opaco o una tabella host specifica del provider. IltaskNamedelle esecuzioni subagent è archiviato nella colonna tipizzatasubagent_runs.task_name; la copiapayload_jsonè dato di replay/debug, non la sorgente per i campi hot di visualizzazione o lookup. src/agents/filesystem/virtual-agent-fs.sqlite.tsimplementa un VFS SQLite sopra la tabellavfs_entriesdel database agente. Letture di directory, esportazioni ricorsive, eliminazioni e rinomine usano intervalli di prefisso indicizzati(namespace, path)invece di scansionare un intero namespace o affidarsi al matching dei percorsi conLIKE.src/agents/runtime-worker.entry.tscrea per ogni esecuzione VFS SQLite, artefatti tool, artefatti run e store cache con scope per i worker.- I marker di completamento del bootstrap del workspace ora vivono in righe condivise tipizzate
workspace_setup_stateindicizzate per percorso workspace risolto invece di.openclaw/workspace-state.json; il runtime non legge né riscrive più il marker workspace legacy, e le API helper non si passano più un falso percorso.openclaw/setup-statesolo per derivare l'identità di archiviazione. - Le approvazioni exec ora vivono nella riga singleton tipizzata SQLite condivisa
exec_approvals_config. Doctor importa il legacy~/.openclaw/exec-approvals.json; le scritture runtime non creano, riscrivono o riportano più quel file come sua posizione di store attiva. Il companion macOS legge e scrive la stessa riga della tabellastate/openclaw.sqlite; mantiene su disco solo il socket del prompt Unix perché quello è IPC, non stato runtime durevole. - I moduli runtime di identità dispositivo, auth dispositivo e bootstrap ora tengono separati
i loro lettori/scrittori di snapshot SQLite dagli helper di import JSON legacy solo doctor.
L'identità dispositivo usa righe tipizzate
device_identitiese i token di auth dispositivo usano righe tipizzatedevice_auth_tokens. Le scritture di auth dispositivo riconciliano le righe per dispositivo/ruolo invece di troncare la tabella dei token, e il runtime non instrada più gli aggiornamenti di singolo token attraverso il vecchio adapter dell'intero store. Il legacy I payload JSON versione 1 esistono solo come forme di importazione/esportazione di doctor. - La cache di scambio token di GitHub Copilot usa la tabella SQLite condivisa dello stato dei Plugin
sotto
github-copilot/token-cache/default. È stato di cache di proprietà del provider, quindi intenzionalmente non aggiunge una tabella di schema host. - La Compaction di GitHub Copilot non scrive più sidecar dell'area di lavoro
openclaw-compaction-*.json. L'harness chiama l'RPC di Compaction della cronologia dell'SDK per la sessione SDK tracciata, e OpenClaw mantiene lo stato durevole di sessione/trascrizione in SQLite invece che in file marker di compatibilità. - Il runtime Swift condiviso (
OpenClawKit) usa le stesse righestate/openclaw.sqliteper identità dispositivo e autenticazione dispositivo. Gli helper dell'app macOS importano gli helper SQLite condivisi invece di possedere un secondo percorso JSON o SQLite. Un file legacy residuoidentity/device.jsonblocca la creazione dell'identità finché doctor non lo importa in SQLite, in linea con il gate di avvio TypeScript e Android. - L'identità dispositivo Android usa lo stesso materiale di chiavi compatibile con TypeScript
archiviato in righe tipizzate
state/openclaw.sqlite#table/device_identities. Non legge né scrive maiopenclaw/identity/device.json; un file legacy residuo blocca l'avvio finché doctor non lo importa in SQLite. - Anche i token di autenticazione dispositivo Android memorizzati nella cache usano righe tipizzate
state/openclaw.sqlite#table/device_auth_tokense condividono la stessa semantica dei token versione 1 di TypeScript e Swift. Il runtime non legge più le chiavi di compatibilitàSecurePrefsgateway.deviceToken*; queste appartengono solo alla logica di migrazione/doctor. - La cronologia dei pacchetti recenti delle notifiche Android usa righe tipizzate
android_notification_recent_packages. Il runtime non migra né legge più le vecchie chiavi CSV SharedPreferences. - La creazione dell'identità dispositivo fallisce in modo chiuso quando esiste il legacy
identity/device.json, quando la riga di identità SQLite non è valida o quando lo store di identità SQLite non può essere aperto. Doctor importa e rimuove prima quel file, quindi l'avvio del runtime non può ruotare silenziosamente l'identità di pairing prima della migrazione. - La selezione dell'identità dispositivo è una chiave di riga SQLite, non un localizzatore di file JSON. I test
e gli helper del Gateway passano chiavi di identità esplicite; solo la migrazione doctor e il
gate di avvio fail-closed conoscono il nome file ritirato
identity/device.json. - La compatibilità del reset di sessione ora vive nella migrazione di configurazione di doctor:
session.idleMinutesviene spostato insession.reset.idleMinutes,session.resetByType.dmviene spostato insession.resetByType.direct, e la policy di reset del runtime legge solo chiavi di reset canoniche. - La compatibilità della configurazione legacy ora vive sotto
src/commands/doctor/. La normale validazionereadConfigFileSnapshot()non importa i detector legacy di doctor né annota problemi legacy;runDoctorConfigPreflight()aggiunge questi problemi per la riparazione/reportistica di doctor. Il flusso di configurazione doctor importasrc/commands/doctor/legacy-config.ts, e la vecchia riparazione degli id profilo OAuth vive sottosrc/commands/doctor/legacy/oauth-profile-ids.ts. - I comandi non doctor non eseguono automaticamente la riparazione della configurazione legacy. Per esempio,
openclaw update --channelora fallisce su una configurazione legacy non valida e chiede all'utente di eseguire doctor, invece di importare silenziosamente codice di migrazione doctor. - Web push, APNs, Voice Wake, controlli di aggiornamento e salute della configurazione ora usano tabelle SQLite condivise tipizzate per sottoscrizioni, chiavi VAPID, registrazioni dei Node, righe trigger, righe di routing, stato delle notifiche di aggiornamento e voci di salute della configurazione invece di interi blob JSON opachi. Le scritture di snapshot Web push e APNs ora riconciliano sottoscrizioni/registrazioni per chiave primaria invece di svuotare le loro tabelle; la salute della configurazione fa lo stesso per percorso di configurazione. I loro moduli runtime mantengono lettori/scrittori di snapshot SQLite separati dagli helper di importazione JSON legacy riservati a doctor.
- La configurazione host Node ora usa una riga singleton tipizzata nel database SQLite condiviso;
doctor importa il vecchio file
node.jsonprima del normale uso del runtime. - Pairing dispositivo/Node, pairing dei canali, allowlist dei canali e stato bootstrap
ora usano righe SQLite tipizzate invece di interi blob JSON opachi. Le approvazioni di binding
dei Plugin e lo stato dei job Cron seguono la stessa separazione: i moduli runtime espongono
operazioni basate su SQLite e helper di snapshot neutrali, e le scritture di snapshot di pairing/bootstrap
più approvazioni di binding dei Plugin riconciliano le righe per chiave primaria
invece di troncare le tabelle, mentre doctor importa/rimuove i vecchi file JSON tramite
moduli
src/commands/doctor/legacy/*. - I record dei Plugin installati ora vivono nell'indice SQLite dei Plugin installati.
La lettura/scrittura della configurazione runtime non migra né preserva più i vecchi
dati di configurazione authored
plugins.installs; doctor importa quella forma di configurazione legacy in SQLite prima del normale uso del runtime. - Gli snapshot di recupero credenziali QQBot ora vivono nello stato Plugin SQLite sotto
qqbot/credential-backups. Il runtime non scrive piùqqbot/data/credential-backup*.json; doctor importa e rimuove quei file di backup legacy insieme agli altri input di stato QQBot. - La pianificazione di reload del Gateway confronta snapshot dell'indice SQLite dei Plugin installati sotto
uno spazio dei nomi di diff interno
installedPluginIndex.installRecords.*. Le decisioni di reload del runtime non racchiudono più quelle righe in falsi oggetti di configurazioneplugins.installs. - L'upgrade delle credenziali account nominati Matrix non avviene più durante le letture
del runtime. Doctor possiede la rinomina del vecchio
credentials/matrix/credentials.jsondi livello superiore quando è possibile risolvere un account Matrix singolo/predefinito. - I moduli runtime core di pairing e Cron non esportano più builder di percorsi JSON
legacy. I moduli legacy di proprietà di doctor costruiscono i percorsi sorgente
pending.json,paired.json,bootstrap.jsonecron/jobs.jsonsolo per test di importazione e migrazione. La normalizzazione legacy della forma dei job Cron e l'importazione del log di esecuzione Cron vivono sottosrc/commands/doctor/legacy/cron*.ts. src/commands/doctor/legacy/runtime-state.tsimporta file di stato JSON legacy, inclusa la configurazione host Node, in SQLite da doctor. I nuovi importer di file legacy restano sottosrc/commands/doctor/legacy/.src/commands/doctor/state-migrations.tsimporta le trascrizioni legacysessions.jsone*.jsonldirettamente in SQLite e rimuove le sorgenti riuscite. Non mette più in staging le trascrizioni legacy root tramiteagents/<agentId>/sessions/*.jsonlné crea un target JSONL canonico prima dell'importazione.- I controlli doctor di integrità dello stato non scansionano più directory di sessioni legacy né offrono l'eliminazione di JSONL orfani. I file di trascrizione legacy sono solo input di migrazione, e lo step di migrazione possiede importazione più rimozione della sorgente.
- L'importazione del registro sandbox legacy vive sotto
src/commands/doctor/legacy/sandbox-registry.ts; le letture e scritture del registro sandbox attivo restano solo SQLite. - La riparazione legacy di salute/importazione delle trascrizioni di sessione vive sotto
src/commands/doctor/legacy/session-transcript-health.ts; i moduli comando runtime non portano più parsing di trascrizioni JSONL né codice di riparazione del ramo attivo.
Elementi principali del consolidamento/dell'eliminazione completati:
- Lo stato dei Plugin ora usa il database condiviso
state/openclaw.sqlite. Il vecchio importatore sidecarplugin-state/state.sqlitelocale al branch è stato rimosso perché quel layout SQLite non è mai stato rilasciato. Gli helper di probe/test riportano ildatabasePathcondiviso invece di esporre un percorso SQLite specifico per lo stato dei plugin. - Le tabelle runtime di attività e Task Flow ora risiedono nel database condiviso
state/openclaw.sqliteinvece che intasks/runs.sqliteetasks/flows/registry.sqlite; i vecchi importatori sidecar sono stati rimossi per la stessa ragione di layout non rilasciato. src/config/sessions/store.tsnon richiede piùstorePathper i metadati in ingresso, gli aggiornamenti delle route o le letture updated-at. Persistenza dei comandi, pulizia sessione CLI, profondità subagent, override auth e identità della sessione del transcript usano API di riga agente/sessione. Le scritture sono applicate come patch di riga SQLite con nuovo tentativo su conflitto ottimistico.- La risoluzione del target di sessione ora espone target di database per agente, non percorsi
legacy
sessions.json. Gateway condiviso, metadati ACP, riparazione route doctor eopenclaw sessionsenumeranoagent_databasespiù gli agenti configurati. - Il routing delle sessioni Gateway ora usa
resolveGatewaySessionDatabaseTarget; il target restituito contienedatabasePathe chiavi di riga SQLite candidate invece di un percorso file legacy dello store sessioni. - I tipi runtime delle sessioni canale ora espongono
{agentId, sessionKey}per letture updated-at, metadati in ingresso e aggiornamenti last-route. Il vecchio tipo di compatibilitàsaveSessionStore(storePath, store)non esiste più. - Le superfici barrel di runtime Plugin, API estensione e
config/sessionsora indirizzano il codice plugin verso helper di riga sessione basati su SQLite. Gli export di compatibilità della libreria root (loadSessionStore,saveSessionStore,resolveStorePath) restano come shim deprecati per i consumatori esistenti. Il vecchio helperresolveLegacySessionStorePathnon esiste più; la costruzione del percorso legacysessions.jsonè ora locale alle migrazioni e alle fixture di test. src/config/sessions/session-entries.sqlite.tsora memorizza le voci sessione canoniche nel database per agente e supporta patch di lettura/upsert/eliminazione a livello di riga. Upsert/patch/eliminazione runtime non scansionano più varianti di maiuscole/minuscole né potano chiavi alias legacy; doctor possiede la canonicalizzazione. L'helper autonomo di importazione JSON non esiste più e la migrazione fonde le righe più nuove con upsert invece di sostituire l'intera tabella sessioni. Gli helper pubblici di lettura/lista/caricamento proiettano metadati di sessione hot da righe tipizzatesessionseconversations;entry_jsonè un'ombra di compatibilità/debug e può essere obsoleto o non valido senza perdere l'identità tipizzata della sessione o il contesto di consegna.src/config/sessions/delivery-info.tsora risolve il contesto di consegna dalle righe tipizzate per agentesessions+conversations+session_conversations. Non ricostruisce più l'identità di consegna runtime dasession_entries.entry_json; una riga conversazione tipizzata mancante è un problema di migrazione/riparazione doctor, non un fallback runtime.- Le decisioni di reset delle sessioni memorizzate ora preferiscono i metadati tipizzati
sessions.session_scope,sessions.chat_typeesessions.channel. Il parsing disessionKeyresta solo per suffissi thread/topic espliciti sui target di comando; la classificazione reset gruppo vs diretto non deriva più dalla forma della chiave. - La classificazione nella visualizzazione elenco/stato sessioni ora usa metadati chat tipizzati e
tipo di sessione Gateway. Non tratta più sottostringhe
:group:o:channel:dentrosession_keycome verità durevole gruppo/diretto. - La selezione della policy di risposta silenziosa ora usa solo tipo conversazione esplicito o
metadati di superficie. Non deduce più la policy diretto/gruppo da sottostringhe di
session_key. - La risoluzione del modello di visualizzazione sessione ora riceve l'id agente dal target
database sessione SQLite invece di estrarlo da
session_key. - L'idratazione del target annuncio agente-ad-agente ora usa solo il
deliveryContexttipizzato disessions.list. Non recupera più routing canale/account/thread daoriginlegacy, campilast*specchiati o forma disession_key. - Il rifiuto del target thread di
sessions_sendora legge metadati di routing SQLite tipizzati. Non rifiuta né accetta più target analizzando suffissi thread dalla chiave target. - La validazione della policy strumenti con ambito gruppo ora legge il routing conversazione SQLite
tipizzato per la sessione corrente o avviata. Non si fida più dell'identità gruppo/canale
decodificando
sessionKey; gli id gruppo forniti dal chiamante vengono scartati quando nessuna riga sessione tipizzata li garantisce. - Il matching degli override modello canale ora usa metadati espliciti di gruppo e conversazione
padre. Non decodifica più id conversazione padre da
parentSessionKey. - L'ereditarietà degli override modello memorizzati ora richiede una chiave sessione padre esplicita
dal contesto sessione tipizzato. Non deriva più override padre da suffissi
:thread:o:topic:insessionKey. - Il vecchio wrapper thread-info di sessione e il parser thread dei plugin caricati non esistono più;
nessun codice runtime importa
config/sessions/thread-info. - L'helper conversazione canale non espone più bridge di parsing full-session-key.
Il core normalizza ancora gli id conversazione raw posseduti dal provider tramite
resolveSessionConversation(...), ma non ricostruisce i fatti di route dasessionKey. - Consegna completamento, policy di invio e manutenzione attività non derivano più il tipo chat
dalla forma di
session_key. Il vecchio parser chiave chat-type è stato eliminato; questi percorsi richiedono metadati sessione tipizzati, contesto consegna tipizzato o vocabolario esplicito dei target di consegna. - Elenco/stato sessioni, diagnostica, binding account approvazione, filtro heartbeat TUI
e riepiloghi d'uso non estraggono più da
SessionEntry.originrouting provider/account/thread/display. Le uniche letture runtime residue dioriginsono concetti non-sessione o oggetti di consegna del turno corrente. - La ricerca conversazione nativa delle richieste di approvazione ora legge righe di routing sessione
tipizzate per agente. Non analizza più identità conversazione canale/gruppo/thread
da
sessionKey; metadati tipizzati mancanti sono un problema di migrazione/riparazione. - I payload evento Gateway session changed/chat/session non riecheggiano più
SessionEntry.origino ombre di routelast*; i client ricevonochannel,chatTypeedeliveryContexttipizzati. - La risoluzione della consegna Heartbeat ora può ricevere direttamente il
deliveryContextSQLite tipizzato, e il runtime heartbeat passa la riga consegna sessione per agente invece di affidarsi alle ombre di compatibilitàsession_entriesper il routing corrente. - Anche la risoluzione del target di consegna dell'agente isolato Cron idrata la sua route corrente dalla riga consegna sessione tipizzata per agente prima di ricadere sul payload voce di compatibilità.
- La risoluzione dell'origine annuncio subagent ora propaga il contesto di consegna
della sessione richiedente tipizzata tramite
loadRequesterSessionEntrye preferisce quella riga rispetto alle ombre di compatibilitàlast*/deliveryContext. - Gli aggiornamenti dei metadati sessione in ingresso ora fondono prima contro la riga consegna
tipizzata per agente; i vecchi campi consegna
SessionEntrysono solo il fallback quando non esiste alcuna riga conversazione tipizzata. - L'estrazione consegna restart/update ora lascia prevalere il
threadIddella consegna SQLite tipizzata sui frammenti topic/thread analizzati dasessionKey; il parsing è solo un fallback per chiavi legacy a forma di thread. - Gli id canale del contesto hook agent ora preferiscono l'identità conversazione SQLite tipizzata,
poi i metadati messaggio espliciti. Non analizzano più frammenti provider/gruppo/canale
da
sessionKey. - L'ereditarietà della route esterna Gateway
chat.sendora legge metadati di routing sessione SQLite tipizzati invece di inferire ambito canale/diretto/gruppo da parti disessionKey. Le sessioni con ambito canale ereditano solo quando il canale sessione tipizzato e il tipo chat corrispondono al contesto di consegna memorizzato; le sessioni shared-main mantengono la loro regola più rigorosa CLI/no-client-metadata. - Il wake del restart-sentinel e il routing di continuazione ora leggono righe SQLite di consegna/routing tipizzate prima di accodare wake heartbeat o continuazioni agent-turn instradate. Non ricostruiscono più il contesto di consegna dall'ombra JSON della voce sessione.
- La risoluzione contesto Gateway
tools.effectiveora legge righe SQLite di consegna/routing tipizzate per input provider, account, target, thread e reply-mode. Non recupera più quei campi di routing hot da ombreoriginobsolete disession_entries.entry_json. - Il routing della consulenza vocale realtime ora risolve consegna padre/chiamata da righe sessione
SQLite tipizzate per agente. Non ricade più su ombre di compatibilità
SessionEntry.deliveryContextquando sceglie la route messaggio dell'agente incorporato. - Il relay heartbeat dello spawn ACP e il routing parent-stream ora leggono la consegna padre da righe sessione SQLite tipizzate. Non ricostruiscono più il contesto di consegna padre da ombre di compatibilità della voce sessione.
- La conservazione della route di consegna sessione ora segue metadati chat tipizzati e
colonne di consegna persistite. Non estrae più suggerimenti canale, marker direct/main
o forma thread da
sessionKey; le route webchat interne ereditano un target esterno solo quando SQLite ha già identità consegna tipizzata/persistita per la sessione. - L'estrazione generica della consegna sessione ora legge solo l'esatta riga consegna sessione SQLite tipizzata. Non analizza più suffissi thread/topic né ricade da una chiave a forma di thread a una chiave sessione base.
- Dispatch risposta, recovery restart sentinel e routing della consulenza vocale realtime ora usano righe sessione/conversazione SQLite tipizzate esatte per il routing thread. Non recuperano più id thread o contesto di consegna della sessione base analizzando chiavi sessione a forma di thread.
- La limitazione della cronologia PI incorporata ora usa la proiezione di routing sessione SQLite
tipizzata (
sessions+conversationsprimaria) per provider, tipo chat e identità peer. Non analizza più provider, DM, gruppo o forma thread dasessionKey. - L'inferenza della consegna strumenti Cron ora usa solo consegna esplicita o il contesto di
consegna tipizzato corrente. Non decodifica più target canale, peer, account o thread
da
agentSessionKey. - Le righe sessione runtime non portano più il vecchio alias route
lastProvider. Helper e test usano i campi tipizzatilastChanneledeliveryContext; la migrazione doctor è l'unico punto che dovrebbe tradurre alias route più vecchi o ombreoriginpersistite. - Eventi transcript, righe VFS e righe artefatti strumento ora scrivono nel database per agente. La tabella globale non rilasciata di mapping transcript-file non esiste più; doctor registra invece i percorsi sorgente legacy in righe di migrazione durevoli.
- La ricerca runtime dei transcript non scansiona più offset byte JSONL né sonda file transcript legacy. I percorsi Gateway chat/media/history leggono righe transcript da SQLite; JSONL sessione è ora solo un input doctor legacy, non uno stato runtime o un formato di esportazione.
- Le relazioni padre e branch dei transcript usano metadati strutturati
parentTranscriptScope: {agentId, sessionId}negli header transcript SQLite, non stringhe locator simili a percorsiagent-db:...transcript_events.... - Il contratto del gestore transcript non espone più costruttori impliciti persistiti
create(cwd)ocontinueRecent(cwd). I gestori transcript persistiti si aprono con un ambito esplicito{agentId, sessionId}; solo i gestori in memoria restano senza ambito per test e trasformazioni transcript pure. - Le API runtime dello store transcript risolvono l'ambito SQLite, non percorsi filesystem. Il
vecchio helper
resolve...ForPathe le opzioni di scritturatranscriptPathinutilizzate non esistono più nei chiamanti runtime. - La risoluzione runtime sessione ora usa
{agentId, sessionId}e non deve derivare stringhesqlite-transcript://<agent>/<session>per confini esterni. I percorsi JSONL assoluti legacy sono solo input di migrazione doctor. - I record direct-bridge del relay hook nativo ora risiedono in righe condivise tipizzate
native_hook_relay_bridgesindicizzate per id relay. Il runtime non scrive più un registro JSON in/tmpné record generici opachi per quei record bridge di breve durata. runEmbeddedPiAgent(...)non ha più un parametro transcript-locator. I descrittori dei worker preparati omettono anche i localizzatori di trascrizione. Lo stato della sessione runtime e le esecuzioni successive in coda trasportano{agentId, sessionId}invece di handle di trascrizione derivati.- La Compaction incorporata ora prende l'ambito SQLite da
agentIdesessionId. Gli hook di Compaction, le chiamate al context-engine, la delega CLI e le risposte di protocollo non devono ricevere handle derivatisqlite-transcript://.... Il codice di esportazione/debug può materializzare artefatti utente espliciti dalle righe, ma non fornisce un percorso generico di esportazione JSONL della sessione né reinserisce nomi di file nell'identità runtime. /export-sessionlegge le righe di trascrizione da SQLite e scrive solo la vista HTML autonoma richiesta. Il visualizzatore incorporato non ricostruisce né scarica più il JSONL della sessione da quelle righe.- La delega del context-engine non analizza più un localizzatore di trascrizione per recuperare
l'identità dell'agente. Il contesto runtime preparato trasporta l'
agentIdrisolto nell'adattatore di Compaction integrato. - La riscrittura della trascrizione e il troncamento live dei risultati degli strumenti ora leggono e persistono
lo stato della trascrizione tramite
{agentId, sessionId}e non derivano localizzatori temporanei per i payload degli eventi di aggiornamento della trascrizione. - La superficie helper dello stato di trascrizione non ha più varianti basate su localizzatore
readTranscriptState,replaceTranscriptStateEventsopersistTranscriptStateMutation. I chiamanti runtime devono usare le API{agentId, sessionId}. L'importazione doctor legge i file legacy tramite percorso file esplicito e scrive righe SQLite; non migra stringhe di localizzatore. - Il contratto del session-manager runtime non espone più
open(locator),forkFrom(locator)osetTranscriptLocator(...). I session manager persistiti si aprono solo tramite{agentId, sessionId}; gli helper di lista/fork risiedono su API di sessione e checkpoint orientate alle righe invece che sulla facade del transcript manager. - Le API del lettore di trascrizioni del Gateway sono scope-first. Accettano
{agentId, sessionId}e non accettano un localizzatore di trascrizione posizionale che potrebbe diventare accidentalmente identità runtime. L'analisi del localizzatore della trascrizione attiva è stata rimossa; i percorsi sorgente legacy vengono letti solo dal codice di importazione doctor. - Anche gli eventi di aggiornamento della trascrizione sono scope-first.
emitSessionTranscriptUpdatenon accetta più una stringa di localizzatore nuda e i listener instradano tramite{agentId, sessionId}senza analizzare un handle. - Il broadcast session-message del Gateway risolve le chiavi di sessione dall'ambito agente/sessione, non da un localizzatore di trascrizione. Il vecchio resolver/cache da localizzatore di trascrizione a chiave di sessione è stato rimosso.
- Gli SSE di session-history del Gateway filtrano gli aggiornamenti live per ambito agente/sessione. Non canonicalizzano più candidati localizzatori di trascrizione, realpath o identità di trascrizione a forma di file per decidere se uno stream debba ricevere un aggiornamento.
- Gli hook del ciclo di vita della sessione non derivano né espongono più localizzatori di trascrizione su
session_end. I consumer degli hook ricevonosessionId,sessionKey, gli id della sessione successiva e il contesto dell'agente; i file di trascrizione non fanno parte del contratto del ciclo di vita. - Nemmeno gli hook di reset derivano o espongono più localizzatori di trascrizione. Il
payload
before_resettrasporta i messaggi SQLite recuperati più il motivo del reset, mentre l'identità della sessione rimane nel contesto dell'hook. - Il reset dell'harness dell'agente non accetta più un localizzatore di trascrizione. Il dispatch del reset è
scoped tramite
sessionId/sessionKeypiù il motivo. - I tipi di sessione delle estensioni agente non espongono più
transcriptLocator; le estensioni devono usare il contesto di sessione e le API runtime invece di accedere a un'identità di trascrizione a forma di file. - Gli hook di Compaction dei Plugin non espongono più localizzatori di trascrizione. Il contesto dell'hook trasporta già l'identità della sessione e le letture della trascrizione devono passare tramite API consapevoli dell'ambito SQLite invece che tramite handle a forma di file.
- Gli hook
before_agent_finalizenon espongono piùtranscriptPath, inclusi i payload di relay degli hook nativi. Gli hook di finalizzazione usano solo il contesto di sessione. - Le risposte di reset del Gateway non sintetizzano più un localizzatore di trascrizione sulla voce restituita. Il reset crea righe di trascrizione SQLite, restituisce la voce di sessione pulita e lascia l'accesso alla trascrizione ai lettori consapevoli dell'ambito.
- I risultati di esecuzione incorporata e Compaction non espongono più localizzatori di trascrizione per la
contabilità della sessione. La Compaction automatica aggiorna solo il
sessionIdattivo, i contatori di Compaction e i metadati dei token. - I risultati dei tentativi incorporati non restituiscono più
transcriptLocatorUsed, e i risultaticompact()del context-engine non restituiscono più localizzatori di trascrizione. I cicli di retry runtime accettano solo unsessionIdsuccessore. - I risultati di append della trascrizione delivery-mirror non restituiscono più localizzatori di
trascrizione. I chiamanti ricevono il
messageIdaggiunto; i segnali di aggiornamento della trascrizione usano l'ambito SQLite. - Gli helper di fork della sessione padre restituiscono solo il
sessionIdforkato. La preparazione dei subagenti passa agli engine l'ambito agente/sessione figlio. - I parametri del runner CLI e il reseeding della cronologia non accettano più localizzatori di trascrizione.
Le letture della cronologia CLI risolvono l'ambito della trascrizione SQLite da
{agentId, sessionId}e dal contesto della chiave di sessione. - Le fixture di test CLI e embedded-runner ora seminano e leggono righe di trascrizione SQLite
per id sessione invece di fingere che le sessioni attive siano file
*.jsonlo di passare una stringasqlite-transcript://...tramite i parametri runtime. - Gli eventi della guardia dei risultati degli strumenti di sessione emettono dall'ambito sessione noto anche quando un
manager in memoria non ha localizzatore derivato. I relativi test non simulano più file di trascrizione
attivi
/tmp/*.jsonl. - Gli helper BTW e compaction-checkpoint ora leggono e forkano righe di trascrizione per ambito SQLite. I metadati dei checkpoint ora memorizzano solo id di sessione e id leaf/entry; i localizzatori derivati non vengono più scritti nei payload dei checkpoint.
- La ricerca della transcript-key del Gateway usa l'ambito della trascrizione SQLite ai confini del protocollo e non esegue più realpath o stat sui nomi file delle trascrizioni.
- La rotazione automatica della trascrizione di Compaction scrive le righe di trascrizione successive direttamente tramite lo store di trascrizioni SQLite. Le righe di sessione conservano solo l'identità della sessione successiva, non un percorso JSONL durevole o un localizzatore persistito.
- La Compaction del context-engine incorporato usa helper di rotazione della trascrizione denominati SQLite. I test di rotazione non costruiscono più percorsi successori JSONL né modellano le sessioni attive come file.
- La conservazione delle immagini in uscita gestite indicizza la cache dei messaggi di trascrizione da statistiche di trascrizione SQLite invece che da chiamate stat del filesystem.
- I lock delle sessioni runtime e la lane doctor autonoma legacy
.jsonl.locksono stati rimossi. - Il barrel runtime Microsoft Teams e l'SDK pubblico dei Plugin non riesportano più il vecchio helper di file-lock; i percorsi di stato durevole dei Plugin sono supportati da SQLite.
- La rimozione per età/conteggio delle sessioni e la pulizia esplicita delle sessioni sono state rimosse. Doctor possiede l'importazione legacy; le sessioni obsolete vengono reimpostate o eliminate esplicitamente.
- I controlli di integrità doctor non contano più un file JSONL legacy come trascrizione attiva valida per una riga di sessione SQLite. La salute della trascrizione attiva è solo SQLite; i file JSONL legacy sono segnalati come input di migrazione/pulizia degli orfani.
- Doctor non tratta più
agents/<agent>/sessions/come stato runtime richiesto. Scansiona quella directory solo quando esiste già, come input di importazione legacy o pulizia degli orfani. sessions.resolvedel Gateway, i percorsi patch/reset/compact di sessione, lo spawning dei subagenti, l'interruzione rapida, i metadati ACP, le sessioni isolate da Heartbeat e il patching TUI non migrano né eliminano più chiavi di sessione legacy come effetto collaterale del normale lavoro runtime.- La risoluzione della sessione del comando CLI ora restituisce l'
agentIdproprietario invece di unostorePath, e non copia più righe legacy della sessione principale durante la normale risoluzione--too--session-id. La canonicalizzazione delle righe principali legacy appartiene solo a doctor. - La risoluzione della profondità dei subagenti runtime non legge più
sessions.jsono store di sessioni JSON5. Leggesession_entriesSQLite per id agente, e i metadati legacy di profondità/sessione possono entrare solo tramite il percorso di importazione doctor. - Gli override della sessione del profilo di autenticazione persistono tramite upsert diretti delle righe
{agentId, sessionKey}invece di caricare pigramente un runtime session-store a forma di file. - Il gating verbose dell'auto-reply e gli helper di aggiornamento della sessione ora leggono/upsertano righe di sessione SQLite per identità di sessione e non richiedono più un percorso store legacy prima di toccare lo stato persistito delle righe.
- Gli helper dei metadati di sessione command-run ora usano nomi e percorsi modulo orientati alle entry;
la vecchia superficie helper di comando
session-storeè stata rimossa. - Il seeding dell'header di bootstrap e l'irrobustimento dei confini di Compaction manuale ora mutano
direttamente le righe di trascrizione SQLite. I chiamanti runtime passano l'identità della sessione, non
percorsi
.jsonlscrivibili. - Il replay silenzioso della rotazione di sessione copia i turni utente/assistant recenti tramite
{agentId, sessionId}dalle righe di trascrizione SQLite. Non accetta più localizzatori di trascrizione sorgente o destinazione. - Le nuove righe di sessione runtime non memorizzano più localizzatori di trascrizione. I chiamanti usano
direttamente
{agentId, sessionId}; i comandi di esportazione/debug possono scegliere i nomi dei file di output quando materializzano le righe. - L'avvio di una nuova sessione di trascrizione persistita ora apre sempre righe SQLite per ambito. Il session manager non riusa più un precedente percorso o localizzatore di trascrizione dell'era dei file come identità per la nuova sessione.
- Le sessioni di trascrizione persistite usano l'API esplicita
openTranscriptSessionManagerForSession({agentId, sessionId}). Le vecchie facade staticheSessionManager.create/openForSession/list/forkFromSessionsono state rimosse così che test e codice runtime non possano ricreare accidentalmente la discovery delle sessioni dell'era dei file. - Il runtime dei Plugin non espone più
api.runtime.agent.session.resolveTranscriptLocatorPath; il codice dei Plugin usa helper di righe SQLite e valori di ambito. - La superficie SDK pubblica
session-store-runtimeora esporta solo helper per righe di sessione e righe di trascrizione. Gli helper mirati di schema/percorso/transazione SQLite risiedono insqlite-runtime; gli helper grezzi di apertura/chiusura/reset restano locali solo per test first-party. - I classificatori legacy dei nomi file
.jsonldi traiettoria/checkpoint ora risiedono nel modulo doctor legacy session-file. La validazione delle sessioni core non importa più helper di artefatti file per decidere i normali id di sessione SQLite. - Le esecuzioni di subagenti bloccanti Active Memory usano righe di trascrizione SQLite invece di
creare file
session.jsonltemporanei o persistiti sotto lo stato del Plugin. La vecchia opzionetranscriptDirè stata rimossa. - La generazione di slug una tantum e le esecuzioni del planner Crestodian usano righe di trascrizione SQLite
invece di creare file
session.jsonltemporanei. - Anche le esecuzioni dell'helper
llm-taske l'estrazione nascosta degli impegni usano righe di trascrizione SQLite, quindi queste sessioni helper solo modello non creano più file di trascrizione JSON/JSONL temporanei. TranscriptSessionManagerora è solo un ambito di trascrizione SQLite aperto. Il codice runtime lo apre conopenTranscriptSessionManagerForSession({agentId, sessionId}); i flussi di creazione, branch, continuazione, lista e fork risiedono nei rispettivi helper di righe SQLite proprietari invece che nelle facade statiche del manager. Il codice doctor/import/debug gestisce file sorgente legacy espliciti fuori dal session manager runtime.- I metodi facade obsoleti
SessionManager.newSession()eSessionManager.createBranchedSession()sono stati rimossi. Le nuove sessioni e i discendenti di trascrizione vengono creati dal workflow SQLite proprietario invece di mutare un manager già aperto in una diversa sessione persistita. - Le decisioni di fork della trascrizione padre e la creazione dei fork non accettano più
storePathosessionsDir; usano invece l'ambito di trascrizione SQLite{agentId, sessionId}al posto di metadati di percorso filesystem mantenuti. - Memory-host non esporta più helper no-op di classificazione della trascrizione della directory di sessione; il filtraggio della trascrizione ora deriva dai metadati delle righe SQLite durante la costruzione delle entry.
- I test di esportazione sessione di Memory-host e QMD usano ambiti di trascrizione SQLite. I vecchi
percorsi
agents/<agentId>/sessions/*.jsonlrestano coperti solo dove un test sta intenzionalmente provando la compatibilità doctor/import/export. - L'ispezione grezza delle sessioni di QA-lab ora usa
sessions.listtramite il Gateway invece di leggereagents/qa/sessions/sessions.json; il feedback MSteams viene aggiunto direttamente alle trascrizioni SQLite senza fabbricare un percorso JSONL. - I turni dei canali in ingresso condivisi ora trasportano
{agentId, sessionKey}invece di unstorePathlegacy. I percorsi di registrazione LINE, WhatsApp, Slack, Discord, Telegram, Matrix, Signal, iMessage, BlueBubbles, Feishu, Google Chat, IRC, Nextcloud Talk, Zalo, Zalo Personal, QA Channel, Microsoft Teams, Mattermost, Synology Chat, Tlon, Twitch e QQBot ora leggono i metadati updated-at e registrano le righe di sessione in ingresso tramite l’identità SQLite. - La persistenza del localizzatore della trascrizione viene rimossa dalle righe di sessione attive.
resolveSessionTranscriptTargetrestituisceagentId,sessionIde metadati opzionali dell’argomento; doctor è l’unico codice che importa i nomi dei file di trascrizione legacy. - Le intestazioni delle trascrizioni runtime partono dalla versione SQLite
1. Gli aggiornamenti delle vecchie forme JSONL V1/V2/V3 vivono solo nell’importazione doctor e normalizzano le intestazioni importate alla versione corrente della trascrizione SQLite prima che le righe vengano archiviate. - La guardia database-first ora vieta
SessionManager.listAlleSessionManager.forkFromSession; i flussi di elenco sessioni e fork/restore devono restare sulle API SQLite per righe/con ambito. - La guardia vieta anche i nomi degli helper legacy per parsing JSONL delle trascrizioni/riparazione active-branch fuori dal codice doctor/import, quindi il runtime non può sviluppare un secondo percorso di migrazione delle trascrizioni legacy.
- Le esecuzioni PI embedded rifiutano gli handle di trascrizione in ingresso. Usano l’identità SQLite
{agentId, sessionId}prima dell’avvio del worker e di nuovo prima che il tentativo tocchi lo stato della trascrizione. Un input obsoleto/tmp/*.jsonlnon può selezionare un target di scrittura runtime. - I record di cache trace, payload Anthropic, stream grezzo e timeline diagnostica
ora vengono scritti in righe SQLite tipizzate
diagnostic_events. I bundle di stabilità del Gateway ora vengono scritti in righe SQLite tipizzatediagnostic_stability_bundles. I vecchi percorsi di override JSONLdiagnostics.cacheTrace.filePath,OPENCLAW_CACHE_TRACE_FILE,OPENCLAW_ANTHROPIC_PAYLOAD_LOG_FILEeOPENCLAW_DIAGNOSTICS_TIMELINE_PATHsono rimossi, e la normale cattura della stabilità non scrive più filelogs/stability/*.json. - La persistenza Cron ora riconcilia le righe SQLite
cron_jobsinvece di eliminare e reinserire l’intera tabella dei job a ogni salvataggio. Le riscritture dei target Plugin aggiornano direttamente le righe cron corrispondenti e mantengono lo stato cron runtime nella stessa transazione del database di stato. - I chiamanti runtime Cron ora usano una chiave stabile dello store cron SQLite. I percorsi legacy
cron.storesono solo input di importazione doctor; i percorsi production gateway, manutenzione task, stato, run-log e riscrittura target Telegram usanoresolveCronStoreKeye non normalizzano più la chiave come percorso. Lo stato Cron ora riportastoreKeyinvece del vecchio campostorePathin forma di file. - Il caricamento e la pianificazione runtime Cron non normalizzano più forme di job persistite legacy
come
jobId,schedule.cron,atMsnumerico, booleani stringa osessionTargetmancante. L’importazione legacy doctor possiede queste riparazioni prima che le righe vengano inserite in SQLite. - Lo spawn ACP non risolve né persiste più percorsi di file JSONL delle trascrizioni. La configurazione di spawn e thread-bind persiste direttamente la riga di sessione SQLite e mantiene l’id sessione come identità di trascrizione conservata.
- Le API dei metadati di sessione ACP ora leggono/elencano/eseguono upsert di righe SQLite per
agentIde non espongono piùstorePathcome parte del contratto della voce di sessione ACP. - La contabilizzazione dell’uso delle sessioni e l’aggregazione dell’uso del gateway ora risolvono le trascrizioni
solo tramite
{agentId, sessionId}. La cache cost/usage e i riepiloghi delle sessioni scoperte non sintetizzano né restituiscono più stringhe di localizzatore della trascrizione. - L’append della chat Gateway, la persistenza abort-partial,
/sessions.sende le scritture delle trascrizioni media webchat aggiungono direttamente tramite l’ambito della trascrizione SQLite. L’helper di injection delle trascrizioni del gateway non accetta più un parametrotranscriptLocator. - La discovery delle trascrizioni SQLite ora elenca solo ambiti e statistiche delle trascrizioni:
{agentId, sessionId, updatedAt, eventCount}. L’helper di compatibilità mortolistSqliteSessionTranscriptLocatorse il campo per-rigalocatorsono spariti. - Il runtime di riparazione delle trascrizioni ora espone solo
repairTranscriptSessionStateIfNeeded({agentId, sessionId}). Il vecchio helper di riparazione basato su localizzatore è eliminato; il codice doctor/debug legge percorsi espliciti dei file sorgente e non migra mai stringhe di localizzatore. - Il runtime del ledger di replay ACP ora archivia righe di replay per sessione nel database di stato
SQLite condiviso invece di
acp/event-ledger.json; doctor importa e rimuove il file legacy. - Gli helper di lettura delle trascrizioni Gateway ora vivono in
src/gateway/session-transcript-readers.tsinvece del vecchio nome modulosession-utils.fs. Il controllo della cronologia retry di fallback è nominato per il contenuto della trascrizione SQLite invece della vecchia superficie dell’helper file. - Gli helper Gateway injected-chat e Compaction ora passano l’ambito della trascrizione SQLite tramite API helper interne invece di nominare i valori come percorsi di trascrizione o file sorgente.
- Il rilevamento della continuazione bootstrap ora controlla le righe di trascrizione SQLite tramite
hasCompletedBootstrapTranscriptTurn; non espone più un nome helper in forma di file. - I test embedded-runner ora usano l’identità della trascrizione SQLite, e l’apertura di un nuovo
transcript manager richiede sempre un
sessionIdesplicito. - Gli helper di indicizzazione della memoria ora usano la terminologia delle trascrizioni SQLite end to end:
l’host esporta
listSessionTranscriptScopesForAgentesessionTranscriptKeyForScope, le code di sync miratesessionTranscripts, gli hit di ricerca sessione pubblici espongono percorsi opachitranscript:<agent>:<session>, e la chiave sorgente DB interna èsession:<session>sottosource_kind='sessions'invece di un finto percorso file. - L’helper persistent-dedupe generico del Plugin SDK non espone più opzioni in forma di file. I chiamanti forniscono chiavi di ambito SQLite e le righe di dedupe durevoli vivono nello stato Plugin condiviso.
- I token SSO Microsoft Teams sono stati spostati da file JSON bloccati allo stato Plugin
SQLite. Doctor importa
msteams-sso-tokens.json, ricostruisce le chiavi canoniche dei token SSO dai payload e rimuove il file sorgente. I token OAuth delegati restano sul loro confine esistente dei file di credenziali private. - Lo stato della cache di sync Matrix è stato spostato da
bot-storage.jsonallo stato Plugin SQLite. Doctor importa payload di sync legacy grezzi o incapsulati e rimuove il file sorgente. I client Matrix e QA Matrix attivi passano una directory radice dello store di sync SQLite, non un finto percorsosync-store.jsonobot-storage.json. - Lo stato della migrazione crypto legacy Matrix è stato spostato da
legacy-crypto-migration.jsonallo stato Plugin SQLite. Doctor importa il vecchio file di stato; gli snapshot IndexedDB Matrix SDK sono stati spostati dacrypto-idb-snapshot.jsona blob Plugin SQLite. Le chiavi di recovery Matrix e le credenziali sono righe di stato Plugin SQLite; i loro vecchi file JSON sono solo input di migrazione doctor. - I log di attività Memory Wiki ora usano lo stato Plugin SQLite invece di
.openclaw-wiki/log.jsonl. Il provider di migrazione Memory Wiki importa i vecchi log JSONL; il markdown wiki e il contenuto del vault utente restano file-backed come contenuto workspace. - Memory Wiki non crea più
.openclaw-wiki/state.jsono la directory inutilizzata.openclaw-wiki/locks. Il provider di migrazione rimuove quei file di metadati Plugin ritirati se un vault più vecchio li ha ancora. - Le voci di audit Crestodian ora usano lo stato Plugin SQLite core invece di
audit/crestodian.jsonl. Doctor importa il log di audit JSONL legacy e lo rimuove dopo un’importazione riuscita. - Le voci di audit di scrittura/osservazione della configurazione ora usano lo stato Plugin SQLite core
invece di
logs/config-audit.jsonl. Doctor importa il log di audit JSONL legacy e lo rimuove dopo un’importazione riuscita. - Il companion macOS non scrive più sidecar app-local
logs/config-audit.jsonlologs/config-health.jsondurante la modifica diopenclaw.json. Il file di configurazione resta file-backed, gli snapshot di recovery restano accanto al file di configurazione, e lo stato durevole di audit/salute della configurazione appartiene allo store SQLite del Gateway. - Le approvazioni in sospeso del rescue Crestodian ora usano lo stato Plugin SQLite core invece
di
crestodian/rescue-pending/*.json. Doctor importa i file legacy di approvazione in sospeso e li rimuove dopo un’importazione riuscita. - Lo stato temporaneo arm di Phone Control ora usa lo stato Plugin SQLite invece di
plugins/phone-control/armed.json. Doctor importa il file legacy dello stato armed nel namespacephone-control/arm-statee rimuove il file. - Doctor non ripara più le trascrizioni JSONL sul posto né crea file JSONL di backup. Importa il branch attivo in SQLite e rimuove la sorgente legacy.
- La ricerca delle trascrizioni dell’hook session-memory usa letture SQLite solo per ambito
{agentId, sessionId}. Il suo helper non accetta né deriva più localizzatori di trascrizione, letture di file legacy o opzioni di riscrittura file. - I binding di conversazione del server app Codex ora indicizzano lo stato Plugin SQLite per
chiave sessione OpenClaw o ambito esplicito
{agentId, sessionId}. Non devono preservare binding di fallback basati su percorso di trascrizione. - Le letture della cronologia mirrored-history del server app Codex usano solo l’ambito della trascrizione SQLite; non devono recuperare l’identità dai percorsi dei file di trascrizione.
- I percorsi di role-ordering e reset Compaction non eliminano più i vecchi file di trascrizione; il reset ruota solo la riga di sessione SQLite e l’identità della trascrizione.
- Le risposte di reset e checkpoint del Gateway restituiscono righe di sessione pulite più id sessione. Non sintetizzano più localizzatori di trascrizione SQLite per i client.
- Il dreaming memory-core non elimina più righe di sessione sondando file JSONL mancanti.
La pulizia dei subagent passa tramite l’API runtime di sessione invece di
controlli di esistenza del filesystem. I suoi test di ingestione delle trascrizioni preparano direttamente righe SQLite
invece di creare fixture
agents/<id>/sessionso placeholder di localizzatore. - L’indicizzazione delle trascrizioni di memoria può esporre
transcript:<agentId>:<sessionId>come percorso virtuale dell’hit di ricerca per helper di citazione/lettura. La sorgente durevole dell’indice è relazionale (source_kind='sessions',source_key='session:<sessionId>',session_id=<sessionId>), quindi il valore non è un localizzatore di trascrizione runtime, non è un percorso filesystem e non deve mai essere ripassato alle API runtime di sessione. - Lo stato memoria doctor del Gateway legge i conteggi di short-term recall e phase-signal
dalle righe di stato Plugin SQLite invece di
memory/.dreams/*.json; l’output CLI e doctor ora etichetta quello storage come store SQLite, non come percorso. - Il runtime memory-core, lo stato CLI, i metodi doctor Gateway e le facciate Plugin SDK
non fanno più audit né archiviano file legacy
.dreams/session-corpus. Quei file sono solo input di migrazione; doctor li importa in SQLite e elimina la sorgente dopo la verifica. Le righe di evidenza active session-ingestion ora usano il percorso SQLite virtualememory/session-ingestion/<day>.txt; il runtime non scrive né deriva mai stato da.dreams/session-corpus. - Gli artefatti pubblici memory-core espongono gli eventi host SQLite come artefatto JSON virtuale
memory/events/memory-host-events.json; non riusano più il percorso sorgente legacy.dreams/events.jsonl. - I registri container/browser sandbox ora usano la tabella SQLite condivisa
sandbox_registry_entriescon colonne tipizzate per sessione, immagine, timestamp, backend/config e porta browser. Doctor importa i file registro JSON legacy monolitici e sharded e rimuove le sorgenti riuscite. Le letture runtime usano le colonne tipizzate delle righe come fonte di verità;entry_jsonè solo una copia di replay/debug. - I commitments ora usano una tabella condivisa tipizzata
commitmentsinvece di un blob JSON dell’intero store. I salvataggi snapshot eseguono upsert per id commitment ed eliminano solo le righe mancanti invece di svuotare e reinserire la tabella. Il runtime carica i commitments da colonne tipizzate di ambito, delivery-window, stato, tentativo e testo;record_jsonè solo una copia di replay/debug. Doctor importa il legacycommitments.jsone lo rimuove dopo un’importazione riuscita. - Le definizioni dei job Cron, lo stato della pianificazione e la cronologia di esecuzione non hanno più writer o reader JSON
runtime. Il runtime usa righe
cron_jobscon pianificazione tipizzata, payload, consegna, avviso di errore, sessione, stato e stato di runtime più metadaticron_run_logstipizzati per stato, riepilogo diagnostico, stato/errore di consegna, sessione/run, modello e totali dei token.job_jsonè solo una copia di replay/debug;state_jsonconserva diagnostica di runtime annidata che non ha ancora campi di query caldi, mentre il runtime reidrata i campi di stato caldi dalle colonne tipizzate. Doctor importa i file legacyjobs.json,jobs-state.jsoneruns/*.jsonle rimuove le sorgenti importate. I writeback dei target Plugin aggiornano le righecron_jobscorrispondenti invece di caricare e sostituire l'intero archivio cron. - L'avvio del Gateway ignora i marcatori legacy
notify: truenella proiezione di runtime. Doctor li traduce in consegna SQLite esplicita quandocron.webhookè valido, rimuove i marcatori inerti quando non è impostato e li conserva con un avviso quando il webhook configurato non è valido. - Le code di consegna in uscita e di sessione ora memorizzano stato della coda, tipo di voce,
chiave di sessione, canale, target, ID account, conteggio dei tentativi, ultimo tentativo/errore,
stato di recupero e marcatori di invio della piattaforma come colonne tipizzate nella tabella condivisa
delivery_queue_entries. Il recupero runtime legge questi campi caldi dalle colonne tipizzate, e le mutazioni di retry/recupero aggiornano direttamente quelle colonne senza riscrivere il JSON di replay. Il payload JSON completo resta solo come blob di replay/debug per i corpi dei messaggi e altri dati di replay freddi. - I record delle immagini in uscita gestite ora usano righe condivise tipizzate
managed_outgoing_image_recordscon i byte multimediali ancora archiviati inmedia_blobs. Il record JSON resta solo come copia di replay/debug. - Le preferenze del selettore modello di Discord, gli hash di distribuzione dei comandi e i binding dei thread ora usano lo stato Plugin SQLite condiviso. I loro piani di importazione JSON legacy vivono nella superficie di migrazione setup/doctor del plugin Discord, non nel codice di migrazione core.
- I rilevatori di importazione legacy dei Plugin usano moduli denominati per doctor come
doctor-legacy-state.tsodoctor-state-imports.ts; i normali moduli runtime dei canali non devono importare rilevatori JSON legacy. - I cursori di catchup BlueBubbles e i marcatori di deduplicazione in ingresso ora usano lo stato Plugin SQLite condiviso. I loro piani di importazione JSON legacy vivono nella superficie di migrazione setup/doctor del plugin BlueBubbles, non nel codice di migrazione core.
- Offset degli aggiornamenti Telegram, righe cache degli sticker, righe cache dei messaggi inviati, righe cache dei nomi topic e binding dei thread ora usano lo stato Plugin SQLite condiviso. I loro piani di importazione JSON legacy vivono nella superficie di migrazione setup/doctor del plugin Telegram, non nel codice di migrazione core.
- I cursori di catchup iMessage, le mappature reply short-id e le righe di deduplicazione sent-echo
ora usano lo stato Plugin SQLite condiviso. I vecchi file
imessage/catchup/*.json,imessage/reply-cache.jsonleimessage/sent-echoes.jsonlsono solo input di doctor. - Le righe di deduplicazione dei messaggi Feishu ora usano lo stato Plugin SQLite condiviso invece dei
file
feishu/dedup/*.json. Il suo piano di importazione JSON legacy vive nella superficie di migrazione setup/doctor del plugin Feishu, non nel codice di migrazione core. - Conversazioni, sondaggi, buffer di upload in sospeso e apprendimenti di feedback di Microsoft Teams
ora usano tabelle di stato/blob Plugin SQLite condivise. Il percorso di upload in sospeso
usa
plugin_blob_entriescosì i buffer multimediali sono archiviati come BLOB SQLite invece che come JSON base64. I nomi degli helper runtime ora usano la nomenclatura SQLite/stato invece della nomenclatura file-store*-fs, e il vecchio shimstorePathè stato rimosso da questi archivi. Il suo piano di importazione JSON legacy vive nella superficie di migrazione setup/doctor del plugin Microsoft Teams. - I media in uscita ospitati da Zalo ora usano
plugin_blob_entriesSQLite condiviso invece dei sidecar temporanei JSON/binopenclaw-zalo-outbound-media. - HTML e metadati del visualizzatore diff ora usano
plugin_blob_entriesSQLite condiviso invece dei file temporaneimeta.json/viewer.html. Gli output PNG/PDF renderizzati restano materializzazioni temporanee perché la consegna del canale richiede ancora un percorso file. - I documenti gestiti Canvas ora usano
plugin_blob_entriesSQLite condiviso invece di una directory predefinitastate/canvas/documents. L'host Canvas serve questi blob direttamente; i file locali vengono creati solo per contenuti operatorehost.rootespliciti o per materializzazione temporanea quando un lettore multimediale downstream richiede un percorso. - Le decisioni di audit File Transfer ora usano
plugin_state_entriesSQLite condiviso invece del log runtime illimitatoaudit/file-transfer.jsonl. Doctor importa il file di audit JSONL legacy nello stato Plugin e rimuove la sorgente dopo un'importazione pulita. - I lease di processo ACPX e l'identità dell'istanza Gateway ora usano lo stato Plugin SQLite condiviso.
Doctor importa il file legacy
gateway-instance-idnello stato Plugin e rimuove la sorgente. - Gli script wrapper generati da ACPX e la home Codex isolata sono materializzazione
temporanea sotto la root temporanea OpenClaw, non stato OpenClaw durevole. I record runtime
ACPX durevoli sono il lease SQLite e le righe gateway-instance;
la vecchia superficie di configurazione ACPX
stateDirè rimossa perché nessuno stato runtime viene più scritto lì. - Gli allegati multimediali del Gateway ora usano la tabella SQLite condivisa
media_blobscome archivio canonico dei byte. I percorsi locali restituiti alle superfici di compatibilità canale e sandbox sono materializzazioni temporanee della riga del database, non l'archivio multimediale durevole. Le allowlist multimediali runtime non includono più le root legacy$OPENCLAW_STATE_DIR/mediaomediadella directory di configurazione; queste directory sono solo sorgenti di importazione doctor. - Il completamento shell non scrive più file cache
$OPENCLAW_STATE_DIR/completions/*. I percorsi smoke di installazione, doctor, update e release usano output di completamento generato o sourcing del profilo invece di file cache di completamento durevoli. - Lo staging degli upload Skills del Gateway ora usa righe
skill_uploadscondivise. Metadati di upload, chiavi di idempotenza e byte dell'archivio vivono in SQLite; l'installer riceve solo un percorso di archivio materializzato temporaneo mentre un'installazione è in esecuzione. - Gli allegati inline dei subagent non vengono più materializzati sotto
.openclaw/attachments/*del workspace. Il percorso spawn prepara voci seed SQLite VFS, i run inline inseriscono queste voci nello spazio dei nomi scratch runtime per-agent, e gli strumenti basati su disco sovrappongono quello scratch SQLite per i percorsi degli allegati. Le vecchie colonne di registro attachment-dir dei run subagent e gli hook di cleanup sono stati rimossi. - L'idratazione immagini CLI non mantiene più file cache stabili
openclaw-cli-images. I backend CLI esterni ricevono ancora percorsi file, ma quei percorsi sono materializzazioni temporanee per-run con cleanup. - Diagnostica cache-trace, diagnostica payload Anthropic, diagnostica stream modello grezzo,
eventi timeline diagnostici e bundle di stabilità Gateway ora
scrivono righe SQLite invece di file
logs/*.jsonlologs/stability/*.json. Flag di override dei percorsi runtime e variabili env sono stati rimossi; i comandi export/debug possono materializzare file esplicitamente dalle righe del database. - Il companion macOS non ha più uno writer continuo
diagnostics.jsonl. I log dell'app vanno al logging unificato, e la diagnostica Gateway durevole resta supportata da SQLite. - L'elenco dei record macOS port-guardian ora usa righe SQLite condivise tipizzate
macos_port_guardian_recordsinvece di un file JSON Application Support o di un blob singleton opaco. - I lock singleton del Gateway ora usano righe SQLite condivise tipizzate
state_leasessotto lo scopegateway_locksinvece di file lock nella directory temporanea. La documentazione di troubleshooting Fly e OAuth ora punta al lock di lease/auth refresh SQLite invece che a una pulizia obsoleta dei file lock. - Lo stato sentinel di riavvio del Gateway ora usa righe SQLite condivise tipizzate
gateway_restart_sentinelinvece direstart-sentinel.json; il runtime legge tipo sentinel, stato, routing, messaggio, continuation e statistiche da colonne tipizzate.payload_jsonè solo una copia di replay/debug. Il codice runtime cancella direttamente la riga SQLite e non porta più con sé plumbing di cleanup dei file. - Intento di riavvio del Gateway e stato di handoff del supervisor ora usano righe SQLite condivise tipizzate
gateway_restart_intentegateway_restart_handoffinvece dei sidecargateway-restart-intent.jsonegateway-supervisor-restart-handoff.json. - Il coordinamento singleton del Gateway ora usa righe tipizzate
state_leasessottogateway_locksinvece di scrivere filegateway.<hash>.lock. La riga lease possiede proprietario del lock, scadenza, Heartbeat e payload di debug; SQLite possiede il confine atomico acquire/release. L'opzione ritirata della directory dei file lock è rimossa; i test usano direttamente l'identità della riga SQLite. - Il vecchio helper non referenziato di report di utilizzo cron che scandiva i file
cron/runs/*.jsonlè stato eliminato. I report della cronologia dei run Cron devono leggere le righe SQLite tipizzatecron_run_logs. - Il recupero del riavvio della sessione principale ora scopre gli agenti candidati tramite il
registro SQLite
agent_databasesinvece di scandire directoryagents/*/sessions. - Il recupero della corruzione sessione Gemini ora elimina solo la riga sessione SQLite;
non ha più bisogno di un gate legacy
storePathné prova a scollegare un percorso transcript JSONL derivato. - La gestione degli override dei percorsi ora tratta i valori ambiente letterali
undefined/nullcome non impostati, evitando database accidentaliundefined/state/*.sqlitenella root del repo durante test o handoff shell. - Le impronte di integrità config ora usano righe SQLite condivise tipizzate
config_health_entriesinvece dilogs/config-health.json, mantenendo il normale file config come unico documento di configurazione non credenziale. Il companion macOS mantiene solo stato di integrità process-local e non ricrea il vecchio sidecar JSON. - Il runtime dei profili auth non importa né scrive più file JSON di credenziali. Lo
store canonico delle credenziali è SQLite;
auth-profiles.json,auth.jsonper-agent ecredentials/oauth.jsoncondiviso sono input di migrazione doctor che vengono rimossi dopo l'importazione. - I test di salvataggio/stato dei profili auth ora verificano direttamente tabelle auth SQLite tipizzate e usano nomi file auth-profile legacy solo per input di migrazione doctor.
openclaw secrets applypulisce solo il file config, il file env e lo store auth-profile SQLite. Non porta più logica di compatibilità che modifica il ritiratoauth.jsonper-agent; doctor possiede importazione ed eliminazione di quel file.- I piani e le applicazioni di migrazione segreti Hermes importavano profili API-key direttamente
nello store auth-profile SQLite. Non scrive né verifica più
auth-profiles.jsoncome target intermedio. - La documentazione auth rivolta agli utenti ora descrive
state/openclaw.sqlite#table/auth_profile_stores/<agentDir>invece di dire agli utenti di ispezionare o copiareauth-profiles.json; i nomi JSON legacy OAuth/auth restano documentati solo come input di importazione doctor. - Gli helper dei percorsi di stato core non espongono più il file ritirato
credentials/oauth.json. Il nome file legacy è locale al percorso di importazione auth doctor. - La documentazione di installazione, sicurezza, onboarding, model-auth e SecretRef ora descrive righe auth-profile SQLite e backup/migrazione dell'intero stato invece di file JSON auth-profile per-agent.
- La discovery dei modelli PI ora passa credenziali canoniche nello storage auth
in-memory
pi-coding-agent. Non crea, pulisce né scrive piùauth.jsonper-agent durante la discovery. - Le impostazioni di trigger e routing Voice Wake ora usano tabelle SQLite condivise tipizzate
invece di
settings/voicewake.json,settings/voicewake-routing.jsono righe generiche opache; doctor importa i file JSON legacy e li rimuove dopo una migrazione riuscita. - Lo stato update-check ora usa una riga condivisa tipizzata
update_check_stateinvece diupdate-check.jsono un blob generico opaco; doctor importa il file JSON legacy e lo rimuove dopo una migrazione riuscita. - Lo stato di integrità config ora usa righe condivise tipizzate
config_health_entriesinvece dilogs/config-health.jsono un blob generico opaco; doctor importa il file JSON legacy e lo rimuove dopo una migrazione riuscita. - Le approvazioni dei binding delle conversazioni Plugin ora usano righe tipizzate
plugin_binding_approvalsinvece di stato SQLite condiviso opaco oplugin-binding-approvals.json; il file legacy è un input di migrazione per doctor. - I binding generici della conversazione corrente ora archiviano righe tipizzate
current_conversation_bindingsinvece di riscriverebindings/current-conversations.json; doctor importa il file JSON legacy e lo rimuove dopo una migrazione riuscita. - I registri di sincronizzazione delle fonti importate di Memory Wiki ora archiviano una riga di stato Plugin SQLite
per ogni chiave vault/fonte invece di riscrivere
.openclaw-wiki/source-sync.json; il provider di migrazione importa e rimuove il registro JSON legacy. - I record delle esecuzioni di importazione ChatGPT di Memory Wiki ora archiviano una riga di stato Plugin SQLite
per ogni id vault/esecuzione invece di scrivere
.openclaw-wiki/import-runs/*.json. Gli snapshot di rollback restano file vault espliciti finché l'archiviazione degli snapshot delle esecuzioni di importazione non viene spostata nello storage blob. - I digest compilati di Memory Wiki ora archiviano righe blob Plugin SQLite invece di
scrivere
.openclaw-wiki/cache/agent-digest.jsone.openclaw-wiki/cache/claims.jsonl. Il provider di migrazione importa i vecchi file di cache e rimuove la directory della cache quando diventa vuota. - Il tracciamento dell'installazione delle skill ClawHub ora archivia una riga di stato Plugin SQLite per
workspace/skill invece di scrivere o leggere i file collaterali
.clawhub/lock.jsone.clawhub/origin.jsona runtime. Il codice runtime usa oggetti di stato di installazione tracciata invece di astrazioni lockfile/origin modellate come file. Doctor importa i file collaterali legacy dai workspace agent configurati e li rimuove dopo un'importazione pulita. - L'indice dei Plugin installati ora legge e scrive la riga singleton tipizzata SQLite condivisa
installed_plugin_indexinvece diplugins/installs.json; il file JSON legacy è solo un input di migrazione per doctor e viene rimosso dopo l'importazione. - L'helper del percorso legacy
plugins/installs.jsonora vive nel codice legacy di doctor. I moduli runtime dell'indice Plugin espongono solo opzioni di persistenza basate su SQLite, non un percorso di file JSON. - Il sentinel di riavvio del Gateway, l'intenzione di riavvio e lo stato di passaggio al supervisore ora usano
righe SQLite condivise tipizzate (
gateway_restart_sentinel,gateway_restart_intentegateway_restart_handoff) invece di blob generici opachi. Il codice runtime di riavvio non ha contratti sentinel/intento/passaggio modellati come file. - Cache di sincronizzazione Matrix, metadati di storage, binding dei thread, marcatori di deduplicazione in ingresso,
stato di cooldown della verifica di avvio, snapshot crittografici IndexedDB dell'SDK,
credenziali e chiavi di recupero ora usano tabelle di stato/blob Plugin SQLite
condivise. Le strutture dei percorsi runtime non espongono più un percorso di metadati
storage-meta.json; quel nome file è solo un input di migrazione legacy. Il loro piano di importazione JSON legacy vive nella superficie di configurazione/migrazione doctor del Plugin Matrix. - L'avvio di Matrix non esegue più scansioni, segnalazioni o completamenti dello stato file legacy di Matrix. Il rilevamento dei file Matrix, la creazione di snapshot crittografici legacy, lo stato di migrazione del ripristino delle chiavi delle stanze, l'importazione e la rimozione della fonte sono tutti di proprietà di doctor.
- I barrel di migrazione runtime di Matrix sono stati rimossi. Gli helper di rilevamento e modifica di stato/crittografia legacy sono importati direttamente da Matrix doctor invece di far parte della superficie API runtime.
- I marcatori di riuso degli snapshot di migrazione Matrix ora vivono nello stato Plugin SQLite
invece che in
matrix/migration-snapshot.json; doctor può ancora riusare lo stesso archivio pre-migrazione verificato senza scrivere un file di stato collaterale. - I cursori del bus Nostr e lo stato di pubblicazione profilo ora usano lo stato Plugin SQLite condiviso. Il loro piano di importazione JSON legacy vive nella superficie di configurazione/migrazione doctor del Plugin Nostr.
- Gli interruttori di sessione di Active Memory ora usano lo stato Plugin SQLite condiviso invece di
session-toggles.json; riattivare la memoria elimina la riga invece di riscrivere un oggetto JSON. - Le proposte di Skill Workshop e i contatori di revisione ora usano lo stato Plugin SQLite condiviso
invece degli archivi per workspace
skill-workshop/<workspace>.json. Ogni proposta è una riga separata sottoskill-workshop/proposals, e il contatore di revisione è una riga separata sottoskill-workshop/reviews. - Le esecuzioni dei subagent revisori di Skill Workshop ora usano il resolver delle trascrizioni
di sessione runtime invece di creare percorsi di sessione collaterali
skill-workshop/<sessionId>.json. - I lease di processo ACPX ora usano lo stato Plugin SQLite condiviso sotto
acpx/process-leasesinvece di un registro a file interoprocess-leases.json. Ogni lease è archiviato come riga propria, preservando la pulizia dei processi obsoleti all'avvio senza un percorso runtime di riscrittura JSON. - Gli script wrapper ACPX e la home Codex isolata sono generati nella radice temporanea di OpenClaw. Vengono ricreati quando necessario e non sono input di backup o migrazione.
- La persistenza del registro delle esecuzioni dei subagent usa righe condivise tipizzate
subagent_runs. Il vecchio percorsosubagents/runs.jsonora è solo un input di migrazione per doctor, e i nomi degli helper runtime non descrivono più il livello di stato come basato su disco. I test runtime non creano più fixtureruns.jsonnon valide o vuote per dimostrare il comportamento del registro; inizializzano/leggono direttamente righe SQLite. - Il backup prepara la directory di stato prima dell'archiviazione, copia i file non database,
crea snapshot dei database
*.sqliteconVACUUM INTO, omette i file collaterali WAL/SHM live, registra i metadati degli snapshot nel manifest dell'archivio e registra le esecuzioni di backup completate in SQLite con il manifest dell'archivio.openclaw backup createvalida l'archivio scritto per impostazione predefinita;--no-verifyè il percorso veloce esplicito. openclaw backup restorevalida l'archivio prima dell'estrazione, riusa il manifest normalizzato del verificatore e ripristina gli asset verificati del manifest nei loro percorsi sorgente registrati. Richiede--yesper le scritture e supporta--dry-runper un piano di ripristino.- Il vecchio filtro dei percorsi volatili di backup è eliminato. Il backup non ha più bisogno di una lista di esclusione live-tar per file JSON/JSONL legacy di sessione o Cron perché gli snapshot SQLite vengono preparati prima della creazione dell'archivio.
- La preparazione del workspace in setup e onboarding semplice non crea più
directory
agents/<agentId>/sessions/. Crea solo config/workspace; le righe di sessione SQLite e le righe di trascrizione vengono create su richiesta nel database per agent. - La riparazione dei permessi di sicurezza ora prende di mira i database SQLite
globali e per agent più i file collaterali WAL/SHM invece di
sessions.jsone dei file transcript JSONL. - I nomi runtime del registro sandbox ora descrivono direttamente i tipi di registro SQLite invece di portare la terminologia legacy del registro JSON nello store attivo.
openclaw reset --scope config+creds+sessionsrimuove i databaseopenclaw-agent.sqliteper agent più i file collaterali WAL/SHM, non solo le directorysessions/legacy.- Gli helper di sessione aggregata del Gateway ora usano nomi orientati alle entry:
loadCombinedSessionEntriesForGatewayrestituisce{ databasePath, entries }. La vecchia denominazione combined-store è stata rimossa dai chiamanti runtime. - Il seeding del canale Docker MCP ora scrive la riga di sessione principale e gli eventi di trascrizione
nel database SQLite per agent invece di creare
sessions.jsone una trascrizione JSONL. - L'hook session-memory incluso ora risolve il contesto della sessione precedente da
SQLite tramite
{agentId, sessionId}. Non scansiona, archivia o sintetizza più percorsi di trascrizione o directoryworkspace/sessions. - L'hook command-logger incluso ora scrive righe di audit dei comandi nella tabella SQLite condivisa
command_log_entriesinvece di aggiungerle alogs/commands.log. - Le allowlist di pairing dei canali ora espongono solo helper di lettura/scrittura basati su SQLite a
runtime e nel Plugin SDK. Il vecchio resolver del percorso
*-allowFrom.jsone il lettore di file vivono solo sotto il codice di importazione legacy di doctor. migration_runsregistra le esecuzioni di migrazione dello stato legacy con stato, timestamp e report JSON.migration_sourcesregistra ogni fonte file legacy importata con hash, dimensione, conteggio record, tabella di destinazione, id esecuzione, stato e stato di rimozione della fonte.backup_runsregistra percorsi degli archivi di backup, stato e manifest JSON.- Lo schema globale non conserva una tabella di registro
agentsinutilizzata. La scoperta dei database agent è il registro canonicoagent_databasesfinché il runtime non avrà un vero proprietario dei record agent. - La config generata del catalogo modelli è archiviata in righe SQLite globali tipizzate
agent_model_catalogsindicizzate per directory agent. I chiamanti runtime usanoensureOpenClawModelCatalog; non esiste un'API di compatibilitàmodels.jsonnel codice runtime. L'implementazione scrive SQLite e il registro PI incorporato viene idratato da quel payload archiviato senza creare un filemodels.json. - L'esportazione markdown delle trascrizioni di sessione QMD e la config
memory.qmd.sessionssono state rimosse. Non esiste una raccolta di trascrizioni QMD, nessun percorso runtimeqmd/sessions*e nessun bridge di memoria di sessione basato su file. - Il runtime memory-core importa gli helper di indicizzazione delle trascrizioni SQLite da
openclaw/plugin-sdk/memory-core-host-engine-session-transcripts, non dal sottopercorso SDK QMD. Il sottopercorso QMD mantiene una riesportazione di compatibilità solo per chiamanti esterni finché una pulizia maggiore dell'SDK potrà rimuoverla. - L'
index.sqliteproprio di QMD ora è una materializzazione runtime temporanea basata sulla tabella SQLite principaleplugin_blob_entries. Il runtime non crea più un file collaterale durevole~/.openclaw/agents/<agentId>/qmd. - Il Plugin opzionale
memory-lancedbnon crea più~/.openclaw/memory/lancedbcome store implicito gestito da OpenClaw. È un backend LanceDB esterno e resta disabilitato finché l'operatore non configura undbPathesplicito. check:database-first-legacy-storesfallisce nuovo codice sorgente runtime che abbina nomi di store legacy ad API filesystem in stile scrittura. Fallisce anche codice sorgente runtime che reintroduce i marcatori ritirati del bridge di trascrizionetranscriptLocatorosqlite-transcript://.... Migrazione, doctor, importazione e codice esplicito di esportazione non-sessione restano consentiti. Nomi di contratti legacy più ampi comesessionFile,storePathe le vecchie facciate dell'era fileSessionManagerhanno ancora proprietari correnti e necessitano di lavoro separato sulle guardie di migrazione prima di poter diventare un controllo preflight obbligatorio. La guardia ora copre anche store runtimecache/*.json, file collaterali genericithread-bindings.json, JSON di stato/log di esecuzione Cron, JSON di salute config, file collaterali di riavvio e lock, impostazioni Voice Wake, approvazioni di binding Plugin, JSON dell'indice dei Plugin installati, JSONL di audit File Transfer, log attività di Memory Wiki, il vecchio log di testocommand-loggerincluso e le opzioni diagnostiche JSONL raw-stream pi-mono. Vieta anche i vecchi nomi dei moduli legacy doctor a livello radice, così il codice di compatibilità resta sottosrc/commands/doctor/. Anche gli handler di debug Android usano logcat/output in memoria invece di preparare file cachecamera_debug.logodebug_logs.txt.
Forma dello schema di destinazione
Mantieni gli schemi espliciti. Lo stato di runtime di proprieta dell'host usa tabelle tipizzate. Lo stato opaco di proprieta dei Plugin usa plugin_state_entries / plugin_blob_entries; non esiste una tabella host generica kv.
Database globale:
state_leases(scope, lease_key, owner, expires_at, heartbeat_at, payload_json, created_at, updated_at)exec_approvals_config(config_key, raw_json, socket_path, has_socket_token, default_security, default_ask, default_ask_fallback, auto_allow_skills, agent_count, allowlist_count, updated_at_ms)schema_meta(meta_key, role, schema_version, agent_id, app_version, created_at, updated_at)agent_databases(agent_id, path, schema_version, last_seen_at, size_bytes)task_runs(...)task_delivery_state(...)flow_runs(...)subagent_runs(run_id, child_session_key, requester_session_key, controller_session_key, created_at, ended_at, cleanup_handled, payload_json)current_conversation_bindings(binding_key, binding_id, target_agent_id, target_session_id, target_session_key, channel, account_id, conversation_kind, parent_conversation_id, conversation_id, target_kind, status, bound_at, expires_at, metadata_json, updated_at)plugin_binding_approvals(plugin_root, channel, account_id, plugin_id, plugin_name, approved_at)tui_last_sessions(scope_key, session_key, updated_at)plugin_state_entries(plugin_id, namespace, entry_key, value_json, created_at, expires_at)plugin_blob_entries(plugin_id, namespace, entry_key, metadata_json, blob, created_at, expires_at)media_blobs(subdir, id, content_type, size_bytes, blob, created_at, updated_at)skill_uploads(upload_id, kind, slug, force, size_bytes, sha256, actual_sha256, received_bytes, archive_blob, created_at, expires_at, committed, committed_at, idempotency_key_hash)web_push_subscriptions(endpoint_hash, subscription_id, endpoint, p256dh, auth, created_at_ms, updated_at_ms)web_push_vapid_keys(key_id, public_key, private_key, subject, updated_at_ms)apns_registrations(node_id, transport, token, relay_handle, send_grant, installation_id, topic, environment, distribution, token_debug_suffix, updated_at_ms)node_host_config(config_key, version, node_id, token, display_name, gateway_host, gateway_port, gateway_tls, gateway_tls_fingerprint, updated_at_ms)device_identities(identity_key, device_id, public_key_pem, private_key_pem, created_at_ms, updated_at_ms)device_auth_tokens(device_id, role, token, scopes_json, updated_at_ms)macos_port_guardian_records(pid, port, command, mode, timestamp)workspace_setup_state(workspace_key, workspace_path, version, bootstrap_seeded_at, setup_completed_at, updated_at)native_hook_relay_bridges(relay_id, pid, hostname, port, token, expires_at_ms, updated_at_ms)model_capability_cache(provider_id, model_id, name, input_text, input_image, reasoning, supports_tools, context_window, max_tokens, cost_input, cost_output, cost_cache_read, cost_cache_write, updated_at_ms)agent_model_catalogs(catalog_key, agent_dir, raw_json, updated_at)managed_outgoing_image_records(attachment_id, session_key, message_id, created_at, updated_at, retention_class, alt, original_media_id, original_media_subdir, original_content_type, original_width, original_height, original_size_bytes, original_filename, record_json)gateway_restart_sentinel(sentinel_key, version, kind, status, ts, session_key, thread_id, delivery_channel, delivery_to, delivery_account_id, message, continuation_json, doctor_hint, stats_json, payload_json, updated_at_ms)channel_pairing_requests(channel_key, account_id, request_id, code, created_at, last_seen_at, meta_json)channel_pairing_allow_entries(channel_key, account_id, entry, sort_order, updated_at)voicewake_triggers(config_key, position, trigger, updated_at_ms)voicewake_routing_config(config_key, version, default_target_mode, default_target_agent_id, default_target_session_key, updated_at_ms)voicewake_routing_routes(config_key, position, trigger, target_mode, target_agent_id, target_session_key, updated_at_ms)update_check_state(state_key, last_checked_at, last_notified_version, last_notified_tag, last_available_version, last_available_tag, auto_install_id, auto_first_seen_version, auto_first_seen_tag, auto_first_seen_at, auto_last_attempt_version, auto_last_attempt_at, auto_last_success_version, auto_last_success_at, updated_at_ms)config_health_entries(config_path, last_known_good_json, last_promoted_good_json, last_observed_suspicious_signature, updated_at_ms)sandbox_registry_entries(registry_kind, container_name, session_key, backend_id, runtime_label, image, created_at_ms, last_used_at_ms, config_label_kind, config_hash, cdp_port, no_vnc_port, entry_json, updated_at)cron_run_logs(store_key, job_id, seq, ts, status, error, summary, diagnostics_summary, delivery_status, delivery_error, delivered, session_id, session_key, run_id, run_at_ms, duration_ms, next_run_at_ms, model, provider, total_tokens, entry_json, created_at)cron_jobs(store_key, job_id, name, description, enabled, delete_after_run, created_at_ms, agent_id, session_key, schedule_kind, schedule_expr, schedule_tz, every_ms, anchor_ms, at, stagger_ms, session_target, wake_mode, payload_kind, payload_message, payload_model, payload_fallbacks_json, payload_thinking, payload_timeout_seconds, payload_allow_unsafe_external_content, payload_external_content_source_json, payload_light_context, payload_tools_allow_json, delivery_mode, delivery_channel, delivery_to, delivery_thread_id, delivery_account_id, delivery_best_effort, failure_delivery_mode, failure_delivery_channel, failure_delivery_to, failure_delivery_account_id, failure_alert_disabled, failure_alert_after, failure_alert_channel, failure_alert_to, failure_alert_cooldown_ms, failure_alert_include_skipped, failure_alert_mode, failure_alert_account_id, next_run_at_ms, running_at_ms, last_run_at_ms, last_run_status, last_error, last_duration_ms, consecutive_errors, consecutive_skipped, schedule_error_count, last_delivery_status, last_delivery_error, last_delivered, last_failure_alert_at_ms, job_json, state_json, runtime_updated_at_ms, schedule_identity, sort_order, updated_at)delivery_queue_entries(queue_name, id, status, entry_kind, session_key, channel, target, account_id, retry_count, last_attempt_at, last_error, recovery_state, platform_send_started_at, entry_json, enqueued_at, updated_at, failed_at)commitments(id, agent_id, session_key, channel, account_id, recipient_id, thread_id, sender_id, kind, sensitivity, source, status, reason, suggested_text, dedupe_key, confidence, due_earliest_ms, due_latest_ms, due_timezone, source_message_id, source_run_id, created_at_ms, updated_at_ms, attempts, last_attempt_at_ms, sent_at_ms, dismissed_at_ms, snoozed_until_ms, expired_at_ms, record_json)migration_runs(id, started_at, finished_at, status, report_json)migration_sources(source_key, migration_kind, source_path, target_table, source_sha256, source_size_bytes, source_record_count, last_run_id, status, imported_at, removed_source, report_json)backup_runs(id, created_at, archive_path, status, manifest_json)Database dell'agente:
schema_meta(meta_key, role, schema_version, agent_id, app_version, created_at, updated_at)sessions(session_id, session_key, session_scope, created_at, updated_at, started_at, ended_at, status, chat_type, channel, account_id, primary_conversation_id, model_provider, model, agent_harness_id, parent_session_key, spawned_by, display_name)conversations(conversation_id, channel, account_id, kind, peer_id, parent_conversation_id, thread_id, native_channel_id, native_direct_user_id, label, metadata_json, created_at, updated_at)session_conversations(session_id, conversation_id, role, first_seen_at, last_seen_at)session_routes(session_key, session_id, updated_at)session_entries(session_id, session_key, entry_json, updated_at)transcript_events(session_id, seq, event_json, created_at)transcript_event_identities(session_id, event_id, seq, event_type, has_parent, parent_id, message_idempotency_key, created_at)transcript_snapshots(session_id, snapshot_id, reason, event_count, created_at, metadata_json)vfs_entries(namespace, path, kind, content_blob, metadata_json, updated_at)tool_artifacts(run_id, artifact_id, kind, metadata_json, blob, created_at)run_artifacts(run_id, path, kind, metadata_json, blob, created_at)trajectory_runtime_events(session_id, run_id, seq, event_json, created_at)memory_index_meta(key, value)memory_index_sources(path, source, hash, mtime, size)memory_index_chunks(id, path, source, start_line, end_line, hash, model, text, embedding, updated_at)memory_embedding_cache(provider, model, provider_key, hash, embedding, dims, updated_at)memory_index_state(id, revision)cache_entries(scope, key, value_json, blob, expires_at, updated_at)La ricerca futura puo aggiungere tabelle FTS senza modificare le tabelle canoniche degli eventi:
transcript_events_fts(session_id, seq, text)vfs_entries_fts(namespace, path, text)I valori grandi dovrebbero usare colonne blob, non codifica di stringhe JSON. Mantieni value_json per piccoli dati strutturati che devono restare ispezionabili con normali strumenti SQLite.
agent_databases e il registro canonico per questo ramo. Non aggiungere una tabella agents finche non esiste un vero proprietario dei record degli agenti; la configurazione degli agenti resta in openclaw.json.
Forma della migrazione di doctor
Il comando doctor dovrebbe chiamare un singolo passaggio di migrazione esplicito, rendicontabile e sicuro da rieseguire:
openclaw doctor --fixopenclaw doctor --fix invoca l'implementazione della migrazione dello stato dopo il normale preflight della configurazione e crea un backup verificato prima dell'importazione. L'avvio del runtime e openclaw migrate non devono importare file di stato OpenClaw legacy.
Proprieta della migrazione:
- Un passaggio di migrazione individua tutte le sorgenti di file legacy e produce un piano prima di modificare qualsiasi cosa.
- Doctor crea un archivio di backup pre-migrazione verificato prima di importare i file legacy.
- Le importazioni sono idempotenti e indicizzate per percorso sorgente, mtime, dimensione, hash e tabella di destinazione.
- I file sorgente importati correttamente vengono rimossi o archiviati dopo il commit del database di destinazione.
- Le importazioni non riuscite lasciano la sorgente intatta e registrano un avviso in
migration_runs. - Il codice di runtime legge solo SQLite dopo che la migrazione esiste.
- Non e richiesto alcun percorso di downgrade/esportazione verso file di runtime.
Inventario della migrazione
Sposta questi elementi nel database globale:
- Le scritture runtime del registro delle attività ora usano il database condiviso; l'importatore sidecar non distribuito
tasks/runs.sqliteè stato eliminato. I salvataggi degli snapshot eseguono upsert per id attività ed eliminano solo le righe di attività/consegna mancanti. - Le scritture runtime di Task Flow ora usano il database condiviso; l'importatore sidecar non distribuito
tasks/flows/registry.sqliteè stato eliminato. I salvataggi degli snapshot eseguono upsert per id flusso ed eliminano solo le righe di flusso mancanti. - Le scritture runtime dello stato dei Plugin ora usano il database condiviso; l'importatore sidecar non distribuito
plugin-state/state.sqliteè stato eliminato. - La ricerca di memoria integrata non usa più come impostazione predefinita
memory/<agentId>.sqlite; le sue tabelle indice risiedono nel database dell'agente proprietario, e l'adesione esplicita al sidecarmemorySearch.store.pathè stata ritirata nella migrazione della configurazione di doctor. - La reindicizzazione della memoria integrata reimposta solo le tabelle di proprietà della memoria nel database dell'agente. Non deve sostituire l'intero file SQLite, perché lo stesso database possiede sessioni, trascrizioni, righe VFS, artefatti e cache runtime.
- Registri dei container/browser sandbox da JSON monolitico e frammentato. Le scritture runtime ora usano il database condiviso; l'importazione JSON legacy rimane.
- Le definizioni dei job Cron, lo stato di pianificazione e la cronologia delle esecuzioni ora usano SQLite condiviso;
doctor importa/rimuove i file legacy
jobs.json,jobs-state.jsonecron/runs/*.jsonl - Identità/autenticazione del dispositivo, push, controllo aggiornamenti, commitment, cache dei modelli OpenRouter, indice dei Plugin installati e associazioni dell'app-server
- I record di associazione e bootstrap dispositivo/nodo ora usano tabelle SQLite tipizzate
- I sottoscrittori delle notifiche device-pair e i marcatori delle richieste consegnate ora usano la tabella
plugin-state SQLite condivisa invece di
device-pair-notify.json. - I record delle chiamate vocali ora usano la tabella plugin-state SQLite condivisa nello
spazio dei nomi
voice-call/callsinvece dicalls.jsonl; la CLI del Plugin segue e riassume la cronologia delle chiamate supportata da SQLite. - Le sessioni Gateway QQBot, i record degli utenti noti e la cache delle citazioni ref-index ora usano
lo stato Plugin SQLite negli spazi dei nomi
qqbot(sessions,known-users,ref-index) invece disession-*.json,known-users.jsoneref-index.jsonl; la migrazione doctor/setup di QQBot importa e rimuove i file legacy. - Le preferenze del selettore di modelli Discord, gli hash di distribuzione dei comandi e le associazioni dei thread
ora usano lo stato Plugin SQLite negli spazi dei nomi
discord(model-picker-preferences,command-deploy-hashes,thread-bindings) invece dimodel-picker-preferences.json,command-deploy-cache.jsonethread-bindings.json; la migrazione doctor/setup di Discord importa e rimuove i file legacy. - I cursori di recupero BlueBubbles e i marcatori di deduplicazione in ingresso ora usano lo stato Plugin SQLite
negli spazi dei nomi
bluebubbles(catchup-cursors,inbound-dedupe) invece dibluebubbles/catchup/*.jsonebluebubbles/inbound-dedupe/*.json; la migrazione doctor/setup di BlueBubbles importa e rimuove i file legacy. - Gli offset degli aggiornamenti Telegram, le voci della cache degli sticker, le voci della cache dei messaggi della catena di risposte,
le voci della cache dei messaggi inviati, le voci della cache dei nomi degli argomenti e le
associazioni dei thread ora usano lo stato Plugin SQLite negli spazi dei nomi
telegram(update-offsets,sticker-cache,message-cache,sent-messages,topic-names,thread-bindings) invece diupdate-offset-*.json,sticker-cache.json,*.telegram-messages.json,*.telegram-sent-messages.json,*.telegram-topic-names.jsonethread-bindings-*.json; la migrazione doctor/setup di Telegram importa e rimuove i file legacy. - I cursori di recupero iMessage, le mappature degli short-id delle risposte e le righe di deduplicazione sent-echo
ora usano lo stato Plugin SQLite negli spazi dei nomi
imessage(catchup-cursors,reply-cache,sent-echoes) invece diimessage/catchup/*.json,imessage/reply-cache.jsonleimessage/sent-echoes.jsonl; la migrazione doctor/setup di iMessage importa e rimuove i file legacy. - Le conversazioni, i sondaggi, i token SSO e gli apprendimenti del feedback di Microsoft Teams ora
usano spazi dei nomi dello stato Plugin SQLite (
conversations,polls,sso-tokens,feedback-learnings) invece dimsteams-conversations.json,msteams-polls.json,msteams-sso-tokens.jsone*.learnings.json; la migrazione doctor/setup di Microsoft Teams importa e archivia i file legacy. I caricamenti in sospeso sono una cache SQLite di breve durata e i vecchi file di cache JSON non vengono migrati. - La cache di sincronizzazione Matrix, i metadati di archiviazione, le associazioni dei thread, i marcatori di deduplicazione in ingresso,
lo stato di cooldown della verifica all'avvio, le credenziali, le chiavi di recupero e gli snapshot crittografici
IndexedDB dell'SDK ora usano spazi dei nomi di stato/blob Plugin SQLite sotto
matrix(sync-store,storage-meta,thread-bindings,inbound-dedupe,startup-verification,credentials,recovery-key,idb-snapshots) invece dibot-storage.json,storage-meta.json,thread-bindings.json,inbound-dedupe.json,startup-verification.json,credentials.json,recovery-key.jsonecrypto-idb-snapshot.json; la migrazione doctor/setup di Matrix importa e rimuove quei file legacy dalle radici di archiviazione Matrix con ambito account. - I cursori del bus Nostr e lo stato di pubblicazione del profilo ora usano lo stato Plugin SQLite negli
spazi dei nomi
nostr(bus-state,profile-state) invece dibus-state-*.jsoneprofile-state-*.json; la migrazione doctor/setup di Nostr importa e rimuove i file legacy. - Gli interruttori di sessione di Active Memory ora usano lo stato Plugin SQLite sotto
active-memory/session-togglesinvece disession-toggles.json. - Le code delle proposte e i contatori delle revisioni di Skill Workshop ora usano lo stato Plugin SQLite
sotto
skill-workshop/proposalseskill-workshop/reviewsinvece dei fileskill-workshop/<workspace>.jsonper workspace. - Le code di consegna in uscita e di consegna sessione ora condividono la tabella SQLite globale
delivery_queue_entriessotto nomi di coda separati (outbound-delivery,session-delivery) invece dei file durevolidelivery-queue/*.json,delivery-queue/failed/*.jsonesession-delivery-queue/*.json. Il passaggio doctor legacy-state importa righe in sospeso e non riuscite, rimuove i marcatori consegnati obsoleti ed elimina i vecchi file JSON dopo l'importazione. I campi di instradamento rapido e tentativi sono colonne tipizzate; il payload JSON viene conservato solo per replay/debug. - I lease dei processi ACPX ora usano lo stato Plugin SQLite sotto
acpx/process-leasesinvece diprocess-leases.json. - Metadati delle esecuzioni di backup e migrazione
Sposta questi nei database degli agenti:
- Radici delle sessioni agente e payload session-entry con forma compatibile. Completato per
le scritture runtime: i metadati rapidi delle sessioni sono interrogabili in
sessions, mentre il payload completo legacy-shapedSessionEntryrimane insession_entries. - Eventi di trascrizione degli agenti. Completato per le scritture runtime.
- Checkpoint di Compaction e snapshot delle trascrizioni. Completato per le scritture runtime:
le copie delle trascrizioni dei checkpoint sono righe di trascrizione SQLite e i metadati dei checkpoint
sono registrati in
transcript_snapshots. Gli helper dei checkpoint Gateway ora nominano questi valori come snapshot delle trascrizioni invece che come file sorgente. - Spazi dei nomi scratch/workspace VFS degli agenti. Completato per le scritture VFS runtime.
- Payload degli allegati dei subagenti. Completato per le scritture runtime: sono voci seed VFS SQLite e mai file workspace durevoli.
- Artefatti degli strumenti. Completato per le scritture runtime.
- Artefatti delle esecuzioni. Completato per le scritture runtime dei worker tramite la tabella per agente
run_artifacts. - Cache runtime locali all'agente. Completato per le scritture cache con ambito runtime dei worker tramite
la tabella per agente
cache_entries. Le cache dei modelli a livello Gateway restano nel database globale a meno che non diventino specifiche dell'agente. - Log degli stream padre ACP. Completato per le scritture runtime.
- Sessioni del registro di replay ACP. Completato per le scritture runtime tramite
acp_replay_sessionseacp_replay_events; il file legacyacp/event-ledger.jsonrimane solo come input di doctor. - Metadati delle sessioni ACP. Completato per le scritture runtime tramite
acp_sessions; i blocchi legacyentry.acpinsessions.jsonsono solo input di migrazione doctor. - Sidecar delle traiettorie quando non sono file di esportazione espliciti. Completato per le scritture
runtime: la cattura delle traiettorie scrive righe
trajectory_runtime_eventsnel database dell'agente e replica gli artefatti con ambito esecuzione in SQLite. I sidecar legacy sono solo input di importazione doctor; l'esportazione può materializzare nuovi output JSONL per bundle di supporto ma non legge né migra vecchi sidecar di traiettorie/trascrizioni a runtime. La cattura runtime delle traiettorie espone l'ambito SQLite; gli helper dei percorsi JSONL sono isolati al supporto esportazione/debug e non sono riesportati dal modulo runtime. I metadati delle traiettorie dell'embedded-runner registrano l'identità{agentId, sessionId, sessionKey}invece di rendere persistente un localizzatore di trascrizioni.
Mantieni questi su file per ora:
openclaw.json- file di credenziali del provider o della CLI
- manifest di Plugin/pacchetto
- workspace utente e repository Git quando è selezionata la modalità disco
- log destinati al tailing dell'operatore, a meno che una superficie di log specifica venga spostata
Piano di migrazione
Fase 0: congelare il confine
Rendi esplicito il confine dello stato durevole prima di spostare altre righe:
- Aggiungi una tabella
migration_runsal database globale. Completato per i report di esecuzione della migrazione dello stato legacy. - Aggiungi un unico servizio di migrazione dello stato da file a database di proprietà di doctor.
Completato:
openclaw doctor --fixusa l'implementazione della migrazione legacy-state. - Rendi
planin sola lettura e fai sì cheapplycrei un backup, importi, verifichi e poi elimini o metta in quarantena i vecchi file. Completato: doctor crea un backup pre-migrazione verificato, passa il percorso del backup amigration_runse riutilizza i percorsi di importazione/rimozione. - Aggiungi divieti statici in modo che il nuovo codice runtime non possa scrivere file di stato legacy mentre il codice di migrazione e i test possono ancora seminarli/leggerli. Completato per gli store legacy attualmente migrati; la protezione scansiona anche i test annidati per contratti vietati dei localizzatori di trascrizioni runtime.
Fase 1: completare il piano di controllo globale
Mantieni lo stato di coordinamento condiviso in state/openclaw.sqlite:
- Agenti e registro dei database degli agenti
- Ledger di Task e Task Flow
- Stato dei Plugin
- Registro dei container/browser sandbox
- Cronologia delle esecuzioni di Cron/scheduler
- Associazione, dispositivo, push, controllo aggiornamenti, TUI, cache OpenRouter/modelli e altro piccolo stato runtime con ambito Gateway
- Metadati di backup e migrazione
- Byte degli allegati multimediali Gateway. Completato per le scritture runtime; i percorsi file diretti
sono materializzazioni temporanee per compatibilità con i mittenti dei canali e lo staging sandbox.
Le allowlist runtime accettano percorsi di materializzazione SQLite, non radici media legacy
di stato/configurazione. Doctor importa i file media legacy in
media_blobse rimuove i file sorgente dopo scritture di riga riuscite. - Sessioni, eventi e blob payload di cattura del proxy di debug. Completato: le catture vivono
nel DB di stato condiviso e si aprono tramite bootstrap, schema,
WAL e impostazioni busy-timeout del DB di stato condiviso. I byte dei payload sono compressi con gzip in
capture_blobs.data; non esiste alcun override DB sidecar runtime del proxy di debug, directory blob o target schema/codegen generato solo per proxy-capture. La migrazione doctor/avvio importa le righedebug-proxy/capture.sqlitedistribuite e i blob payload referenziati, inclusi gli override di ambiente legacy DB/blob attivi, poi archivia quelle sorgenti lasciando intatti i certificati CA.
Questa fase elimina anche opener sidecar duplicati, helper dei permessi, configurazione WAL, potatura del filesystem e writer di compatibilità da quei sottosistemi.
Fase 2: introdurre database per agente
Crea un database per ogni agente e registralo dal DB globale:
~/.openclaw/state/openclaw.sqlite~/.openclaw/agents/<agentId>/agent/openclaw-agent.sqliteLa riga globale agent_databases memorizza il percorso, la versione dello schema, il timestamp
dell'ultimo avvistamento e metadati di base su dimensione/integrità. Il codice runtime chiede
al registro il DB dell'agente invece di derivare direttamente i percorsi file.
Il DB dell'agente possiede:
sessionscome radice canonica delle sessioni, consession_entriescome tabella dei payload con forma di compatibilità collegata a quella radice, esession_routescome lookup univoco delsession_keyattivoconversationsesession_conversationscome identità di instradamento del provider normalizzata collegata alle sessionitranscript_events- snapshot delle trascrizioni e checkpoint di Compaction. Fatto per le scritture runtime.
vfs_entriestool_artifactse artefatti di esecuzione- righe runtime/cache locali all'agente. Fatto per le cache con ambito worker.
- eventi dello stream padre ACP
- eventi runtime della traiettoria quando non sono artefatti di esportazione espliciti
Fase 3: Sostituire le API dello store delle sessioni
Fatto per il runtime. La superficie dello store delle sessioni a forma di file non è un contratto runtime attivo:
- Il runtime non chiama più
loadSessionStore(storePath)né trattastorePathcome identità della sessione. - Le operazioni runtime sulle righe sono
getSessionEntry,upsertSessionEntry,patchSessionEntry,deleteSessionEntryelistSessionEntries. - Gli helper di riscrittura dell'intero store, i writer su file, i test delle code, la potatura degli alias e i parametri di eliminazione delle chiavi legacy sono spariti dal runtime.
- Gli export di compatibilità deprecati del pacchetto radice adattano ancora i percorsi
canonici
sessions.jsonsulle API SQLite per righe. - Il parsing di
sessions.jsonrimane solo nel codice di migrazione/import di doctor e nei test di doctor. - Il fallback del ciclo di vita runtime legge gli header delle trascrizioni SQLite, non le prime righe JSONL.
Continuare a eliminare tutto ciò che reintroduce parametri di lock dei file, vocabolario di potatura/troncamento come manutenzione dei file, identità basata sul percorso dello store, o test la cui unica asserzione è la persistenza JSON.
Fase 4: Spostare trascrizioni, stream ACP, traiettorie e VFS
Rendere nativo su database ogni stream di dati dell'agente:
- Le scritture append delle trascrizioni passano attraverso una transazione SQLite che garantisce
l'header della sessione, controlla l'idempotenza dei messaggi, seleziona la coda padre, inserisce
in
transcript_eventse registra metadati di identità interrogabili intranscript_event_identities. Fatto per gli append diretti dei messaggi di trascrizione e per i normali append persistiti diTranscriptSessionManager; le operazioni esplicite sui branch mantengono la loro scelta esplicita del padre e scrivono comunque righe SQLite senza derivare alcun localizzatore di file. - I log dello stream padre ACP diventano righe, non file
.acp-stream.jsonl. Fatto. - La configurazione dello spawn ACP non persiste più i percorsi JSONL delle trascrizioni. Fatto.
- La cattura runtime delle traiettorie scrive direttamente righe/artefatti evento. Il comando esplicito di supporto/esportazione può ancora produrre artefatti JSONL del bundle di supporto come formato di esportazione, ma l'esportazione della sessione non ricrea JSONL di sessione. Fatto.
- Le workspace su disco rimangono su disco quando configurate in modalità disco.
- Lo scratch VFS e la modalità sperimentale workspace solo VFS usano il DB dell'agente.
La migrazione importa i vecchi file JSONL una sola volta, registra conteggi/hash in
migration_runs e rimuove i file importati dopo i controlli di integrità.
Fase 5: Backup, ripristino, Vacuum e verifica
I backup rimangono un unico file archivio:
- Eseguire checkpoint di ogni database globale e agente.
- Creare snapshot di ogni DB con semantica di backup SQLite o
VACUUM INTO. - Archiviare snapshot DB compatti, configurazione, credenziali esterne ed esportazioni workspace richieste.
- Omettere i file live grezzi
*.sqlite-wale*.sqlite-shm. - Verificare aprendo ogni snapshot DB ed eseguendo
PRAGMA integrity_check.openclaw backup createesegue questa verifica dell'archivio per impostazione predefinita;--no-verifysalta solo il passaggio di verifica post-scrittura dell'archivio, non il controllo di integrità della creazione dello snapshot. - Il ripristino copia gli snapshot nei loro percorsi di destinazione. Questo branch reimposta il
layout SQLite non distribuito a
user_version = 1; le future modifiche di schema distribuite potranno aggiungere migrazioni esplicite quando saranno necessarie.
Fase 6: Runtime dei worker
Mantenere sperimentale la modalità worker mentre approda la separazione del database:
- I worker ricevono id agente, id esecuzione, modalità filesystem e identità del registro DB.
- Ogni worker apre la propria connessione SQLite.
- Il padre mantiene autorità su consegna dei canali, approvazioni, configurazione e annullamento.
- Iniziare con un worker per esecuzione attiva; aggiungere pooling solo dopo che ciclo di vita e proprietà delle connessioni DB sono stabili.
Fase 7: Eliminare il vecchio mondo
Fatto per la gestione runtime delle sessioni. Il vecchio mondo è consentito solo come input esplicito di doctor o output di supporto/esportazione:
- Nessuna scrittura runtime di
sessions.json, JSONL delle trascrizioni, JSON del registro sandbox, SQLite sidecar dei task o SQLite sidecar dello stato Plugin. - Nessuna potatura di file JSON/sessione, troncamento di trascrizioni su file, lock dei file di sessione o test di sessione a forma di lock.
- Nessun export di compatibilità runtime il cui scopo sia mantenere aggiornati i vecchi file di sessione.
- Gli export di supporto espliciti rimangono formati di archivio/materializzazione richiesti dall'utente e non devono reintrodurre nomi di file nell'identità runtime.
Backup e ripristino
I backup dovrebbero essere un unico file archivio, ma la cattura del database dovrebbe essere nativa SQLite:
- Fermare l'attività di scrittura a lungo termine o entrare in una breve barriera di backup.
- Per ogni database globale e agente, eseguire un checkpoint.
- Creare uno snapshot di ogni database usando la semantica di backup SQLite o
VACUUM INTOin una directory temporanea di backup. - Archiviare gli snapshot compattati dei database, il file di configurazione, la directory delle credenziali, le workspace selezionate e un manifest.
- Verificare l'archivio aprendo ogni snapshot SQLite incluso ed eseguendo
PRAGMA integrity_check.openclaw backup createlo fa per impostazione predefinita;--no-verifyserve solo a saltare intenzionalmente il passaggio di verifica post-scrittura dell'archivio.
Non affidarsi a copie live grezze di *.sqlite, *.sqlite-wal e *.sqlite-shm come
formato di backup primario. Il manifest dell'archivio dovrebbe registrare ruolo del database,
id agente, versione dello schema, percorso sorgente, percorso dello snapshot, dimensione in byte e stato di
integrità.
Il ripristino dovrebbe ricostruire i file del database globale e dei database agente dagli snapshot dell'archivio. Poiché il layout SQLite non è ancora stato distribuito, questo refactor mantiene solo lo schema versione 1 più l'import da file a database di doctor. Il comando di ripristino valida prima l'archivio, poi sostituisce ogni asset del manifest dal payload estratto verificato.
Piano di refactor del runtime
-
Aggiungere API del registro database.
- Risolvere i percorsi del DB globale e dei DB per agente.
- Mantenere gli schemi non distribuiti a
user_version = 1; non aggiungere codice runner di migrazione dello schema finché uno schema distribuito non lo richiede. - Aggiungere helper di chiusura/checkpoint/integrità usati da test, backup e doctor.
-
Collassare gli store SQLite sidecar.
- Spostare le tabelle di stato Plugin nel database globale. Fatto per le scritture runtime; l'importer del sidecar legacy non distribuito è eliminato.
- Spostare le tabelle del registro task nel database globale. Fatto per le scritture runtime; l'importer del sidecar legacy non distribuito è eliminato.
- Spostare le tabelle Task Flow nel database globale. Fatto per le scritture runtime; l'importer del sidecar legacy non distribuito è eliminato.
- Spostare le tabelle integrate di ricerca memoria in ogni database agente. Fatto; il
memorySearch.store.pathpersonalizzato esplicito ora viene rimosso dalla migrazione della configurazione doctor. La reindicizzazione completa viene eseguita in-place solo sulle tabelle di memoria; il vecchio percorso di swap dell'intero file e l'helper di swap dell'indice sidecar sono eliminati. - Eliminare opener di database duplicati, configurazione WAL, helper dei permessi e percorsi di chiusura da quei sottosistemi.
-
Spostare le tabelle di proprietà dell'agente nei database per agente.
- Creare il DB agente on demand tramite il registro del database globale. Fatto.
- Spostare entry di sessione runtime, eventi di trascrizione, righe VFS e artefatti degli strumenti nei DB agente. Fatto.
- Non migrare entry di sessione, eventi di trascrizione, righe VFS o artefatti degli strumenti branch-local nel DB condiviso; quel layout non è mai stato distribuito. Mantenere solo l'import legacy da file a database in doctor.
-
Sostituire le API dello store delle sessioni.
- Rimuovere
storePathcome identità runtime. Fatto per runtime e protetto dacheck:database-first-legacy-stores: metadati di sessione, aggiornamenti di route, persistenza dei comandi, pulizia sessione CLI, anteprime di ragionamento Feishu, persistenza dello stato della trascrizione, profondità dei subagent, override di sessione del profilo auth, logica parent-fork e ispezione QA-lab ora risolvono il database dalle chiavi canoniche agente/sessione. Le risposte degli elenchi sessione Gateway/TUI/UI/macOS ora espongonodatabasePathinvece delpathlegacy; le superfici debug macOS mostrano il database per agente come stato in sola lettura invece di scrivere la configurazionesession.store./status, esportazione della traiettoria guidata da chat e proxy delle dipendenze CLI non propagano più percorsi store legacy; il fallback di utilizzo della trascrizione legge SQLite per identità agente/sessione. I test runtime e bridge non espongono piùstorePath; gli input doctor/migrazione possiedono quel nome di campo legacy. Il caricamento combinato delle sessioni Gateway non ha più un branch runtime speciale per valorisession.storenon templated; aggrega righe SQLite per agente. La lane doctor del lock sessione legacy e il suo helper di pulizia.jsonl.locksono stati rimossi; SQLite è ora il confine di concorrenza delle sessioni. I call site runtime hot usano nomi di helper orientati alle righe comeresolveSessionRowEntry; il vecchio alias di compatibilitàresolveSessionStoreEntryè stato rimosso dal runtime e dagli export SDK Plugin.
- Rimuovere
- Usare operazioni su righe
{ agentId, sessionKey }. Fatto:getSessionEntry,upsertSessionEntry,deleteSessionEntry,patchSessionEntryelistSessionEntriessono API SQLite-first che non richiedono un percorso dello store sessione. Riepilogo stato, stato agente locale, salute e comando di elencoopenclaw sessionsora leggono direttamente righe per agente e mostrano percorsi dei database SQLite per agente invece di percorsisessions.json. - Sostituire delete/insert dell'intero store con
upsertSessionEntry,deleteSessionEntry,listSessionEntriese query SQL di pulizia. Fatto per il runtime: gli hot path ora usano API su righe e patch di righe ritentate su conflitto; gli helper rimanenti di import/sostituzione dell'intero store sono limitati al codice di importazione migrazione e ai test del backend SQLite.- Eliminare
store-writer.tse i test della coda writer. Fatto. - Eliminare potatura runtime delle chiavi legacy e parametri alias-delete dagli upsert/patch delle righe sessione. Fatto.
- Eliminare
- Eliminare il comportamento runtime del registro JSON.
- Rendere letture e scritture del registro sandbox solo SQLite. Fatto.
- Importare JSON monolitico e sharded solo dal passaggio di migrazione. Fatto.
- Rimuovere lock del registro sharded e scritture JSON. Fatto.
- Mantenere una tabella di registro tipizzata invece di archiviare righe di registro come JSON opaco generico se la forma rimane stato operativo hot-path. Fatto.
-
Eliminare mutazioni di sessione a forma di lock file.
- Fatto per la creazione lock runtime e le API lock runtime.
- La lane autonoma di pulizia doctor legacy
.jsonl.lockè rimossa. session.writeLockè configurazione legacy migrata da doctor, non un'impostazione runtime tipizzata.- L'integrità dello stato non ha più un percorso separato di potatura dei file di trascrizione orfani; la migrazione doctor importa/rimuove le sorgenti JSONL legacy in un unico punto.
- La coordinazione singleton Gateway usa righe SQLite tipizzate
state_leasessottogateway_lockse non espone più una superficie di directory di lock file. - La persistenza dedupe generica dell'SDK Plugin non usa più lock file o file JSON; scrive righe SQLite condivise dello stato Plugin. Fatto.
- La coordinazione embed QMD usa un lease di stato SQLite invece di
qmd/embed.lock. Fatto.
-
Rendere i worker consapevoli del database.
- I worker aprono le proprie connessioni SQLite.
- Il padre possiede consegna, callback dei canali e configurazione.
- Il worker riceve id agente, id esecuzione, modalità filesystem e identità del registro DB, non handle live.
vfs-onlyrimane sperimentale e usa il database agente come radice di storage.- Mantenere prima un worker per esecuzione attiva. Il pooling può aspettare finché la durata delle connessioni DB e il comportamento di annullamento non saranno ordinari.
-
Integrazione del backup.
- Insegna al backup a creare snapshot dei database globali e degli agenti tramite backup SQLite o
VACUUM INTO. Fatto per i file*.sqlitescoperti sotto l'asset di stato. - Aggiungi la verifica del backup per l'integrità SQLite e la versione dello schema. Fatto per la creazione del backup e i controlli di integrità predefiniti per la verifica dell'archivio.
- Registra i metadati dell'esecuzione del backup in SQLite. Fatto tramite la tabella condivisa
backup_runscon percorso dell'archivio, stato e JSON del manifest. - Aggiungi il ripristino da snapshot di archivio verificati. Fatto:
openclaw backup restorevalida prima dell'estrazione, usa il manifest normalizzato del verificatore, supporta--dry-rune richiede--yesprima di sostituire i percorsi sorgente registrati. - Includi l'esportazione VFS/workspace solo quando richiesta; non esportare gli elementi interni della sessione come JSON o JSONL.
- Insegna al backup a creare snapshot dei database globali e degli agenti tramite backup SQLite o
-
Elimina test e codice obsoleti. Fatto per le superfici note delle sessioni runtime.
-
Rimuovi i test che verificano la creazione runtime di
sessions.jsono file transcript JSONL. Fatto per core session store, chat, eventi transcript Gateway, preview, lifecycle, aggiornamenti command session-entry, reset/trace auto-reply e fixture memory-core dreaming, routing target approval, riparazione transcript di sessione, riparazione permessi di sicurezza, esportazione traiettoria ed esportazione sessione. I test transcript Active-memory ora verificano gli scope SQLite e nessuna creazione di file JSONL temporanei o persistiti. La vecchia regressione heartbeat di potatura dei transcript è stata rimossa perché il runtime non tronca più i transcript JSONL. I test dello strumento agent session-list non modellano più i percorsi legacysessions.jsoncome forma della risposta Gateway; i test app/UI/macOS usanodatabasePath. I test di utilizzo transcript/statusora seminano direttamente righe transcript SQLite invece di scrivere file JSONL. I test del lifecycle delle sessioni Gateway ora usano direttamente helper di seeding transcript SQLite; la vecchia forma fixture session-file a riga singola è scomparsa dalla copertura di reset ed eliminazione.sessions.deletenon restituisce più un campo dell'era filearchived: []; l'eliminazione riporta solo il risultato della mutazione della riga. Anche la vecchia opzionedeleteTranscriptè scomparsa: eliminare una sessione rimuove la root canonicasessionse lascia che SQLite propaghi a cascata la rimozione delle righe transcript, snapshot e traiettoria di proprietà della sessione, quindi nessun chiamante può lasciare transcript orfani o dimenticare un ramo di pulizia. I test di cattura traiettoria context-engine ora leggono righetrajectory_runtime_eventsda un database agente isolato invece di leggeresession.trajectory.jsonl. Gli script seed del canale Docker MCP ora seminano direttamente righe SQLite. Le scritture dirette disessions.jsonsono limitate alle fixture doctor. Tool Search Gateway E2E legge le prove tool-call dalle righe transcript SQLite invece di scandire fileagents/<agentId>/sessions/*.jsonl. Gli eventi host memory-core e le righe scratch session-corpus ora vivono nello stato Plugin SQLite condiviso;events.jsonlesession-corpus/*.txtsono solo input di migrazione doctor legacy. Le righe attive usano percorsi virtualimemory/session-ingestion/, non.dreams/session-corpus. Il vecchio modulo di riparazione memory-core dreaming e i suoi test CLI/Gateway sono stati rimossi perché il runtime non possiede più la riparazione dell'archivio file per quel corpus. I test memory-core bridge/public-artifact non espongono più.dreams/events.jsonl; usano il nome dell'artefatto JSON virtuale basato su SQLite. La documentazione di test SDK/Codex pubblica ora parla di stato sessione SQLite invece che di file sessione, e l'esempio channel-turn non espone più un argomentostorePath. Lo stato di sincronizzazione Matrix ora usa direttamente lo store plugin-state SQLite. I contratti attivi client/runtime passano una root di archiviazione account, non un percorsobot-storage.json, e doctor importa ilbot-storage.jsonlegacy in SQLite prima di eliminare la sorgente. Gli scenari QA Matrix di riavvio/distruttivi ora modificano direttamente la riga di sincronizzazione SQLite invece di creare o eliminare filebot-storage.jsonfittizi, e il substrato E2EE passa una root sync-store invece di un percorsosync-store.jsonfittizio. La selezione storage-root di Matrix non assegna più punteggi alle root in base a file JSON legacy di sync/thread; usa metadati root durevoli più stato crypto reale. La suite di test del backend sessioni SQLite runtime non fabbrica più unsessions.json; le fixture sorgente legacy ora vivono nei test doctor che le importano. I test sessione Gateway non espongono più un helpercreateSessionStoreDiro una configurazione inutilizzata di percorso session-store temporaneo; le directory fixture sono esplicite e la configurazione diretta delle righe usa la denominazione session-row SQLite. La copertura del parser session-store JSON5 solo doctor è stata spostata fuori dai test infra e dentro i test di migrazione doctor, quindi le suite di test runtime non possiedono più il parsing legacy dei file sessione. I test runtime SSO/pending-upload di Microsoft Teams non portano più fixture o parser sidecar JSON; il parsing legacy dei token SSO vive solo nel modulo di migrazione del Plugin. I test Telegram non seminano più percorsi store/tmp/*.jsonfittizi; reimpostano direttamente la cache messaggi basata su SQLite. L'helper generico OpenClaw test-state non espone più uno writer legacyauth-profiles.json; i test di migrazione auth doctor possiedono localmente quella fixture. I test runtime per puntatori last-session TUI, approval exec, toggle active-memory, dedupe/verifica startup Matrix, sincronizzazione sorgente Memory Wiki, binding current-conversation, autenticazione onboarding e import secret Hermes non fabbricano più vecchi file sidecar né verificano che vecchi nomi file siano assenti. Dimostrano il comportamento tramite righe SQLite e API di store pubbliche; i test doctor/migration sono l'unico posto a cui appartengono i nomi file sorgente legacy. I test runtime per pairing device/node, channel allowFrom, intent di riavvio, handoff di riavvio, voci della coda di consegna sessione, salute config, cache iMessage, cron job, intestazioni transcript PI, registri subagent e allegati immagine gestiti non creano più nemmeno file JSON/JSONL ritirati solo per dimostrare che sono ignorati o assenti. Il recupero overflow PI non ha più un fallback di riscrittura/troncamento SessionManager: il troncamento tool-result e le riscritture transcript context-engine modificano righe transcript SQLite, poi aggiornano lo stato del prompt attivo dal database. Gli append dei messaggi SessionManager persistiti delegano all'helper atomico di append transcript SQLite per la selezione del parent e l'idempotenza. Anche gli append normali metadata/custom entry selezionano il parent corrente dentro SQLite, quindi le istanze manager obsolete non resuscitano race parent-chain precedenti a SQLite. La pulizia sintetica della coda PI per precheck mid-turn esessions_yieldora ritaglia direttamente lo stato transcript SQLite; il vecchio bridge di rimozione coda SessionManager e i suoi test sono eliminati. Anche la cattura checkpoint Compaction crea snapshot solo da SQLite; i chiamanti non passano più un SessionManager live come sorgente transcript alternativa. -
Mantieni i test che seminano file legacy solo per la migrazione.
-
La prova con file JSON è stata sostituita con prova tramite righe SQL per le superfici runtime attive.
-
Aggiungi divieti statici per scritture runtime verso percorsi JSON legacy di sessione/cache. Fatto per la guardia del repo.
- Rendi verificabile il report di migrazione.
- Registra le esecuzioni di migrazione in SQLite con timestamp di inizio/fine, percorsi
sorgente, hash sorgente, conteggi, avvisi e percorso backup.
Fatto: le esecuzioni di migrazione legacy-state ora persistono un report
migration_runscon inventario percorso/tabella sorgente, SHA-256 del file sorgente, dimensioni, conteggi record, avvisi e percorso backup. Fatto: le esecuzioni di migrazione legacy-state persistono anche righemigration_sourcesper audit a livello sorgente e future decisioni di skip/backfill. - Rendi apply idempotente. La riesecuzione dopo un'importazione parziale dovrebbe saltare una sorgente già importata o fare merge tramite chiave stabile. Fatto: indici sessione, transcript, code di consegna, stato Plugin, registri task e righe SQLite globali di proprietà dell'agente importano tramite chiavi stabili o semantica upsert/replace, quindi le riesecuzioni fanno merge senza duplicare righe durevoli.
- Le importazioni fallite devono lasciare il file sorgente originale al suo posto.
Fatto: le importazioni transcript fallite ora lasciano la sorgente JSONL originale
nel percorso rilevato, e
migration_sourcesregistra la sorgente comewarningconremoved_source=0per la prossima esecuzione doctor.
- Registra le esecuzioni di migrazione in SQLite con timestamp di inizio/fine, percorsi
sorgente, hash sorgente, conteggi, avvisi e percorso backup.
Fatto: le esecuzioni di migrazione legacy-state ora persistono un report
Regole di prestazione
- Una connessione per thread/processo va bene; non condividere handle tra worker.
- Usa WAL,
foreign_keys=ON, un busy timeout di 30s e transazioni di scrittura breviBEGIN IMMEDIATE. - Mantieni sincroni gli helper delle transazioni di scrittura a meno che/fino a quando un'API di transazione async non aggiunge semantiche esplicite di mutex/backpressure.
- Mantieni piccole e transazionali le scritture di consegna parent.
- Evita riscritture dell'intero store; usa upsert/delete a livello di riga.
- Aggiungi indici per percorsi list-by-agent, list-by-session, updated-at, run id ed expiration prima di spostare codice caldo.
- Archivia artefatti grandi, media e vettori come BLOB o righe BLOB chunked, non JSON base64 o array numerici.
- Mantieni piccole e con scope le voci opache plugin-state.
- Aggiungi pulizia SQL per TTL/expiration invece di pruning del filesystem. Fatto per gli store runtime di proprietà del database: media, stato Plugin, blob Plugin, dedupe persistente e cache agente scadono tutti tramite righe SQLite. La pulizia filesystem restante è limitata a materializzazioni temporanee o comandi di rimozione espliciti.
Divieti statici
Aggiungi un controllo repo che fallisca nuove scritture runtime verso percorsi di stato legacy:
sessions.json*.trajectory.jsonleccetto gli output materializzati dei bundle di supporto.acp-stream.jsonlacp/event-ledger.json- file di cache di runtime
cache/*.json agents/<agentId>/agent/auth.jsonagents/<agentId>/agent/models.jsoncredentials/oauth.jsongithub-copilot.token.jsonopenrouter-models.jsonauth-profiles.jsonauth-state.jsonexec-approvals.jsonworkspace-state.jsoncredentials*.jsonerecovery-key.jsondi Matrixcron/runs/*.jsonlcron/jobs.jsonjobs-state.jsondevice-pair-notify.jsondevices/pending.jsondevices/paired.jsondevices/bootstrap.jsonnodes/pending.jsonnodes/paired.jsonidentity/device.jsonidentity/device-auth.jsonpush/web-push-subscriptions.jsonpush/vapid-keys.jsonpush/apns-registrations.jsonprocess-leases.jsongateway-instance-idsession-toggles.json- Memory-core
.dreams/events.jsonl - Memory-core
.dreams/session-corpus/ - Memory-core
.dreams/daily-ingestion.json - Memory-core
.dreams/session-ingestion.json - Memory-core
.dreams/short-term-recall.json - Memory-core
.dreams/phase-signals.json - Memory-core
.dreams/short-term-promotion.lock - Skill Workshop
skill-workshop/<workspace>.json - Skill Workshop
skill-workshop/skill-workshop-review-*.json - Nostr
bus-state-*.json - Nostr
profile-state-*.json calls.jsonlknown-users.jsonref-index.jsonl- QQBot
session-*.json - BlueBubbles
bluebubbles/catchup/*.json - BlueBubbles
bluebubbles/inbound-dedupe/*.json - Telegram
update-offset-*.json - Telegram
sticker-cache.json - Telegram
*.telegram-messages.json - Telegram
*.telegram-sent-messages.json - Telegram
*.telegram-topic-names.json - Telegram
thread-bindings-*.json - iMessage
catchup/*.json - iMessage
reply-cache.jsonl - iMessage
sent-echoes.jsonl - Microsoft Teams
msteams-conversations.json - Microsoft Teams
msteams-polls.json - Microsoft Teams
msteams-sso-tokens.json - Microsoft Teams
*.learnings.json - Matrix
bot-storage.json - Matrix
sync-store.json - Matrix
thread-bindings.json - Matrix
inbound-dedupe.json - Matrix
startup-verification.json - Matrix
storage-meta.json - Matrix
crypto-idb-snapshot.json - Discord
model-picker-preferences.json - Discord
command-deploy-cache.json - file JSON degli shard del registro sandbox
- file JSON del bridge
/tmpdel relay degli hook nativi plugin-state/state.sqlite- sidecar di runtime
openclaw-state.sqlitead hoc tasks/runs.sqlitetasks/flows/registry.sqlitebindings/current-conversations.jsonrestart-sentinel.jsongateway-restart-intent.jsongateway-supervisor-restart-handoff.jsongateway.<hash>.lockqmd/embed.lockcommands.logconfig-health.jsonport-guard.jsonsettings/voicewake.jsonsettings/voicewake-routing.jsonplugin-binding-approvals.jsonplugins/installs.jsonaudit/file-transfer.jsonlaudit/crestodian.jsonlcrestodian/rescue-pending/*.jsonplugins/phone-control/armed.json- Memory Wiki
.openclaw-wiki/log.jsonl - Memory Wiki
.openclaw-wiki/state.json - Memory Wiki
.openclaw-wiki/locks/ - Memory Wiki
.openclaw-wiki/source-sync.json - Memory Wiki
.openclaw-wiki/import-runs/*.json - Memory Wiki
.openclaw-wiki/cache/agent-digest.json - Memory Wiki
.openclaw-wiki/cache/claims.jsonl - ClawHub
.clawhub/lock.json - ClawHub
.clawhub/origin.json - decorazione del profilo browser
.openclaw-profile-decorated - aperture di sessione basate su file
SessionManager.open(...) - facciate di elenco delle trascrizioni
SessionManager.listAll(...)eTranscriptSessionManager.listAll(...) - facciate di fork delle trascrizioni
SessionManager.forkFromSession(...)eTranscriptSessionManager.forkFromSession(...) - facciate di sostituzione delle sessioni mutabili
SessionManager.newSession(...)eTranscriptSessionManager.newSession(...) - facciate per sessioni ramificate
SessionManager.createBranchedSession(...)eTranscriptSessionManager.createBranchedSession(...)
Il divieto deve consentire ai test di creare fixture legacy e al codice di migrazione di leggere/importare/rimuovere sorgenti di file legacy. I sidecar SQLite non rilasciati restano vietati e non ottengono autorizzazioni di importazione tramite doctor.
Criteri di completamento
- Le scritture di dati e cache di runtime vanno nel database SQLite globale o dell'agente.
- Il runtime non scrive più indici di sessione, JSONL di trascrizioni, JSON del registro sandbox, SQLite sidecar dei task o SQLite sidecar dello stato dei Plugin. Gli importer SQLite sidecar non rilasciati dei task e dello stato dei Plugin vengono eliminati.
- L'importazione di file legacy è riservata a doctor.
- Il backup produce un unico archivio con snapshot SQLite compatti e prova di integrità.
- I worker degli agenti possono essere eseguiti con disco, scratch VFS o storage sperimentale solo VFS.
- I file di configurazione e delle credenziali esplicite restano gli unici file di controllo persistenti non basati su database previsti.
- I controlli del repo impediscono di reintrodurre archivi di file di runtime legacy.