STATUS —
basectl historyand the local history index are implemented as the first slice.basectl explain last-error,basectl report, and history cleanup integration remain future work.
Issue: #396
Base currently exposes raw runtime logs through basectl logs and structured
local command metadata through basectl history. This document defines the
local observability layer: shipped command history, future last-error
explanation, and future report generation.
- Preserve
basectl logsas the raw evidence surface. - Add a local structured history index that makes recent command outcomes easy to scan without opening individual log files.
- Allow a future
basectl explain last-errorcommand to summarize the latest failed Base run from local evidence. - Allow a future
basectl reportcommand to create a redacted local diagnostic bundle for bug reports or support handoff. - Keep all data local by default, with no telemetry and no automatic upload.
- Do not add hosted telemetry.
- Do not upload logs, reports, history, or diagnostics automatically.
- Do not use network services or AI providers to explain failures.
- Do not make command history writes a reason for the primary command to fail.
- Do not replace raw log files with summaries.
basectl logs scans the Base cache root, finds recent Python-layer runtime log
files, and prints a table with command, run id, status, and path. It also
supports opening, tailing, or printing the newest matching log path.
This is useful when the user already knows they need raw logs. It is less useful when they want to answer questions such as:
- What did I run recently?
- Which project did the failure belong to?
- Which command failed most recently?
- What should I inspect first?
- What local evidence can I attach to a bug report?
Those questions need a structured index in addition to raw log discovery.
Base writes a structured command history index under the Base cache root:
<base-cache-root>/history/runs.jsonl
Each line is one JSON object. JSON Lines keeps writes append-only, easy to inspect with ordinary tools, and easy to recover when one line is malformed.
History writes are best-effort:
- write a final completion record when a Python-backed command with a persistent log exits
- never fail the user command because the history file cannot be written
- ignore malformed history lines while warning in debug output
The first implementation emits one finished record per recorded run. Shell-only
commands and no-durable-write modes such as ctx.dry_run or
App(log_to_file=False) do not create history records in this slice.
A history record should include:
{
"schema_version": 1,
"run_id": "20260610T101500_ab12cd",
"event": "finished",
"command": "setup",
"raw_command": "base_setup",
"argv": ["basectl", "setup", "base"],
"project": "base",
"project_root": "~/work/base",
"manifest": "~/work/base/base_manifest.yaml",
"workspace_root": "~/work",
"started_at": "2026-06-10T10:15:00Z",
"ended_at": "2026-06-10T10:15:12Z",
"duration_ms": 12000,
"exit_code": 0,
"status": "ok",
"log_path": "~/Library/Caches/base/cli/base_setup/logs/20260610T101500_ab12cd.log",
"base_version": "0.4.0",
"os": "macos",
"shell": "bash",
"profiles": ["dev"]
}Fields should be omitted when unknown instead of guessed.
The first implementation should not store:
- full environment variables
- raw command output
- secret values
- unbounded absolute paths
- package index URLs with credentials
- arbitrary shell history outside the Base command being run
Raw output remains in the log file. History stores metadata and a pointer to the evidence.
History and reports must be local by default. Users opt in to sharing by copying, attaching, or otherwise sending a generated report themselves.
Before writing history or reports, Base should redact:
- URL credentials such as
https://user:secret@example.invalid/path - key/value fragments whose key looks like
token,password,secret,api-key,api_key, orauthorization - shell arguments passed to options whose name looks secret-bearing
- home-directory paths by compacting them to
~/...
The live terminal stream is not redacted by Base. Redaction protects persistent metadata and generated reports.
History belongs to the same local cache lifecycle as runtime logs. basectl clean should eventually understand history records and remove or compact
records whose log files have been pruned.
The retention contract should be:
basectl clean --older-than <age>removes history records older than the age when their corresponding logs are also eligible for removal.basectl clean --keep-last <count>keeps history records for the retained log files and prunes older records for that command family.- Orphaned history records may remain temporarily when cleanup cannot match the
log path safely, but
basectl historyshould mark missing logs clearly. - Durable user state under
~/.base.dis never cleaned by history retention.
basectl history reads the structured index and prints a compact table of
recent Base command runs:
TIME COMMAND PROJECT STATUS EXIT LOG
2026-06-10 10:15:12 setup base ok 0 ~/Library/Caches/base/...
2026-06-10 10:10:03 check demo error 1 ~/Library/Caches/base/...
Expected options:
--project <name>filters by Base project name.--command <name>filters by command.--status <ok|warn|error>filters by status.--limit <count>limits the number of rows.--format jsonprints structured records for scripts.
basectl logs should remain the command for opening or tailing raw log files.
basectl history should point to logs, not replace them.
basectl explain last-error should find the latest failed history record,
inspect its linked log file if present, and print a deterministic local
summary:
- command, project, exit code, and time
- likely failing subsystem when detectable
- the most relevant log tail
- suggested next commands such as
basectl logs --pathorbasectl doctor
The explanation should be rule-based. It should not call external services.
If the history record is missing its log file, the command should still report the metadata and explain that raw evidence has been cleaned or moved.
basectl report should generate a local diagnostic artifact, preferably a
Markdown report by default with an optional JSON format for automation.
The report should include:
- Base version and runtime environment summary
- recent command history summary
- selected doctor/check findings when requested
- paths to relevant logs
- redacted log excerpts when explicitly requested
- project manifest metadata, limited to non-secret fields
The command should write to an explicit output path or print to stdout. It should never upload the report.
The commands should ship in separate, reviewable slices:
- Add history recording and
basectl history. Shipped. - Add
basectl explain last-errorafter history records exist. - Add
basectl reportafter history and explanation have stable local data. - Extend
basectl cleanto compact or prune history records once the history format is stable.
This order keeps the data model and privacy boundary reviewable before Base starts producing summaries or shareable diagnostic artifacts.
- The history writer lives in
base_cli.App, so Python-backed commands share the same local metadata lifecycle as persistent logs. - Shell-only commands are deferred until Base has a shell-side writer with the same redaction and best-effort guarantees.
basectl historymarks missing log files directly in text output and exposeslog_existsin JSON output.- Future report generation should require an explicit option before embedding raw log excerpts.