Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions extensions/copilot/assets/prompts/chronicle-reindex.prompt.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
name: chronicle:reindex
description: Rebuild the local session index and sync to cloud
---
Reindex my session store to pick up any missing sessions. Add 'force' to re-process already indexed sessions.
5 changes: 5 additions & 0 deletions extensions/copilot/assets/prompts/chronicle-standup.prompt.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
name: chronicle:standup
description: Generate a standup report from recent chat sessions
---
Generate a standup report from my recent coding sessions.
5 changes: 5 additions & 0 deletions extensions/copilot/assets/prompts/chronicle-tips.prompt.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
name: chronicle:tips
description: Get personalized tips based on your chat session usage patterns
---
Analyze my recent chat session history and give me personalized tips to improve my workflow.
153 changes: 153 additions & 0 deletions extensions/copilot/assets/prompts/skills/chronicle/SKILL.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
---
name: chronicle
description: Analyze Copilot session history for standup reports, usage tips, and session reindexing. Use when the user asks for a standup, daily summary, usage tips, workflow recommendations, wants to reindex their session store, or asks about deleting session data.
---

# Chronicle

Analyze the user's Copilot session history using the `copilot_sessionStoreSql` tool. This skill handles standup reports, usage analysis, and session store maintenance.

Sessions may be stored locally (SQLite) and optionally synced to the cloud for cross-device access. Cloud sync is controlled by the `chat.sessionSync.enabled` setting.

**Prerequisite:** Chronicle requires the `github.copilot.chat.localIndex.enabled` setting to be `true`. If the `copilot_sessionStoreSql` tool is not available, tell the user to enable this setting in VS Code Settings.
Comment thread
vijayupadya marked this conversation as resolved.

## Available Tool Actions

The `copilot_sessionStoreSql` tool supports three actions:

| Action | Purpose | `query` param |
|--------|---------|---------------|
| `standup` | Pre-fetch last 24h sessions, turns, files, refs | Not needed |
| `query` | Execute a read-only SQL query | Required |
| `reindex` | Rebuild local session index + cloud sync | Not needed |

## Workflows

### Standup

When the user asks for a standup, daily summary, or "what did I do":

1. Call `copilot_sessionStoreSql` with `action: "standup"` and `description: "Generate standup"`.
2. The tool returns pre-fetched session data (sessions, turns, files, refs from the last 24 hours).
3. For any PR references in the data, check their current status (open, merged, draft) if possible.
4. Format the returned data as a standup report grouped by work stream (branch/feature):

```
Standup for <date>:

**✅ Done**

**Feature name** (`branch-name` branch, `repo-name`)
- 3-7 words describing the status
- Key files: 2-3 most important files changed
- Merged: [#123](link)
- Session: `session-id`

**🚧 In Progress**

**Feature name** (`branch-name` branch, `repo-name`)
- 3-7 words describing the current state of work
- Key files: 2-3 most important files being worked on
- Draft: [#789](link)
- Session: `session-id`
```

Rules:
- Keep it concise and succinct — the user can always ask follow-up questions
- Use turn data (user messages AND assistant responses) to understand WHAT was done
- Use file paths to identify which components/areas were affected
- Group related sessions on the same branch into one entry
- For sessions, only show the most recent session per feature/branch
- Link PRs and issues using markdown link syntax
- Classify as Done if work appears complete, In Progress otherwise

### Tips

When the user asks for tips, workflow recommendations, or how to improve:

**Step 1: Investigate how the user works**

Use `copilot_sessionStoreSql` with `action: "query"` to explore their recent sessions. The goal is to understand their patterns — how they prompt, what tools they use, and where they spend time.

Queries to run (do not explain what you will do first — start querying immediately):
- Sessions from the last 7 days: counts, durations, repositories
- Turn data: read actual user messages to understand prompting patterns
- session_files: which files and tools are used most frequently
- session_refs: PR/issue/commit activity patterns

**Step 2: Consider available features**

If the current workspace has a `.github/` folder, check for `.github/copilot-instructions.md`, `.github/skills/`, and `.github/agents/` to see what custom configuration exists. Do NOT look outside the workspace. Look for gaps between what's available and what the user actually uses.

**Step 3: Provide tips**

Based on what you learned, provide 3-5 specific, actionable tips. Each tip should:
- Be grounded in actual usage data — reference specific patterns you observed
- Be non-obvious — skip basic features that any regular user would already know
- Focus on gaps where a feature, workflow change, or different approach would meaningfully improve their experience

Analysis dimensions to explore:
- **Prompting patterns**: Are user messages vague or specific? Do they provide context? Do they correct or redirect the agent frequently?
- **Tool usage**: Which tools are used most? Are there underutilized tools that could help?
- **Session patterns**: How long are sessions? Are there many short abandoned sessions?
- **File patterns**: Which areas of the codebase get the most attention? Any repeated edits to the same files?
- **Workflow**: Is the user leveraging agent mode, custom instructions, prompt files, skills?

If the session store has little data, acknowledge that and suggest features to try based on what configuration you found in the workspace.

### Reindex

When the user asks to reindex, rebuild, or refresh their session store:

1. Call `copilot_sessionStoreSql` with `action: "reindex"` and `description: "Reindex sessions"`.
2. The tool rebuilds the local session store from debug logs and, if cloud sync is enabled, uploads new sessions to the cloud.
3. Present the before/after stats and cloud sync results to the user.

If the user says "force reindex" or wants to re-process already-indexed sessions, add `force: true` to the call. By default, already-indexed sessions are skipped for speed.

### Delete Sessions

When the user asks to delete session data or clear their history:

- Guide them to run the **Delete Session Sync Data** command from the Command Palette (`github.copilot.sessionSync.deleteSessions`).
- This command lets them choose which sessions to delete from both local storage and the cloud.
- The tool itself does NOT support deletion — this is intentional to prevent accidental data loss.

## Query Guidelines

When using `action: "query"`:
- Only one query per call — do not combine multiple statements with semicolons
- Always use LIMIT (max 100) and prefer aggregations (COUNT, GROUP BY) over raw row dumps
- Query the **turns** table for conversation content — it gives the richest insight into what happened
- Query **session_files** for file paths and tool usage patterns
- Query **session_refs** for PR/issue/commit links
- Join tables using session_id for complete analysis
- Always filter on **updated_at** (not created_at) for time ranges
- Always JOIN sessions with turns to get session content — do not rely on sessions.summary alone

### Query routing

The tool automatically routes queries based on the user's cloud sync settings:
- **Cloud enabled**: Queries go to the cloud DuckDB backend which contains ALL sessions across devices and agents (VS Code, CLI, Copilot Coding Agent, PR reviews). The tool description will show DuckDB SQL syntax — follow it.
- **Cloud disabled**: Queries go to local SQLite which only contains sessions from this device. The tool description will show SQLite syntax.

The tool's description dynamically changes based on the active backend. **Always follow the SQL syntax shown in the tool description** — it matches the active backend.

## Database Schema

### Tables (both local and cloud unless noted)

- **sessions**: id, cwd (workspace folder path — always NULL in cloud), repository, branch, host_type, summary, agent_name, agent_description, created_at, updated_at
- **turns**: session_id, turn_index, user_message, assistant_response (first ~1000 chars, may be truncated), timestamp
- **checkpoints**: session_id, checkpoint_number, title, overview, history, work_done, technical_details, important_files, next_steps, created_at — compaction checkpoints storing summarized state. Note: cloud has fewer columns (no history/work_done/technical_details).
- **session_files**: session_id, file_path, tool_name, turn_index, first_seen_at
- **session_refs**: session_id, ref_type (commit/pr/issue), ref_value, turn_index, created_at
- **search_index**: FTS5 table (local only). Use `WHERE search_index MATCH 'query'` for full-text search

### Cloud-only tables

- **events**: Raw event table (~90 columns). Key columns: session_id, timestamp, type, user_content, assistant_content, tool_start_name, tool_complete_success, tool_complete_result_content, usage_model, usage_input_tokens, usage_output_tokens
- **tool_requests**: session_id, tool_call_id, name, arguments_json

Date math (SQLite): `datetime('now', '-1 day')`, `datetime('now', '-7 days')`
Date math (Cloud/DuckDB): `now() - INTERVAL '1 day'`, `now() - INTERVAL '7 days'`. Use `ILIKE` for text search (no FTS5/MATCH), `date_diff('minute', start, end)` for durations.
65 changes: 41 additions & 24 deletions extensions/copilot/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -1268,24 +1268,33 @@
"name": "copilot_sessionStoreSql",
"displayName": "Session Store SQL",
"toolReferenceName": "sessionStoreSql",
"when": "github.copilot.sessionSearch.enabled",
"userDescription": "Query your Copilot session history using SQL",
"modelDescription": "Execute read-only SQL queries against the global session store containing history from ALL past coding sessions. Use this proactively when the user asks about:\n- What they've worked on recently or in the past\n- Prior approaches to similar problems\n- Project history and file changes\n- Sessions linked to PRs, issues, or commits\n- Temporal queries ('what was I doing yesterday?')\n\nSupports SQLite SQL including JOINs, FTS5 MATCH queries, aggregations, and subqueries.\n\n**Only one query per call — do not combine multiple statements with semicolons.**\n\nSchema:\n- sessions — id, cwd, repository, branch, summary, created_at, updated_at\n- turns — session_id, turn_index, user_message, assistant_response, timestamp\n- session_files — session_id, file_path, tool_name (edit/create), turn_index\n- session_refs — session_id, ref_type (commit/pr/issue), ref_value, turn_index\n- search_index — FTS5 virtual table (content, session_id, source_type). Use WHERE search_index MATCH 'query' for full-text search.",
"modelDescription": "Query the local session store containing history from past coding sessions. Uses SQLite SQL syntax.\n\nActions: 'query' (execute SQL — supports JOINs, FTS5 MATCH, aggregations), 'standup' (pre-fetch last 24h data), 'reindex' (rebuild index from debug logs).",
"tags": [],
"canBeReferencedInPrompt": false,
"inputSchema": {
"type": "object",
"properties": {
"action": {
"type": "string",
"enum": ["query", "standup", "reindex"],
"description": "The action to perform. 'query' (default) executes a SQL query. 'standup' pre-fetches last 24h session data for standup reports. 'reindex' rebuilds the local session index and syncs to cloud if enabled."
},
"query": {
"type": "string",
"description": "A single read-only SQL query to execute. Supports SELECT, WITH, JOINs, aggregations, and FTS5 MATCH. Only one statement per call — do not combine multiple queries with semicolons."
"description": "A single read-only SQL query to execute. Required when action is 'query'. Supports SELECT, WITH, JOINs, aggregations, and FTS5 MATCH. Only one statement per call — do not combine multiple queries with semicolons."
},
"force": {
"type": "boolean",
"description": "When true with action 'reindex', re-processes all sessions including already-indexed ones. Default false (skips already-indexed sessions)."
},
"description": {
"type": "string",
"description": "A 2-5 word summary of what this query does (e.g. 'Recent sessions overview', 'Find PR sessions')."
"description": "A 2-5 word summary of what this call does (e.g. 'Recent sessions overview', 'Generate standup', 'Reindex sessions')."
}
},
"required": [
"query",
"description"
]
}
Expand Down Expand Up @@ -1534,26 +1543,6 @@
"name": "compact",
"description": "%copilot.agent.compact.description%"
},
{
"name": "chronicle",
"description": "%copilot.chronicle.description%",
"when": "github.copilot.sessionSearch.enabled"
},
{
"name": "chronicle:standup",
"description": "%copilot.chronicle.standup.description%",
"when": "github.copilot.sessionSearch.enabled"
},
{
"name": "chronicle:tips",
"description": "%copilot.chronicle.tips.description%",
"when": "github.copilot.sessionSearch.enabled"
},
{
"name": "chronicle:reindex",
"description": "%copilot.chronicle.reindex.description%",
"when": "github.copilot.sessionSearch.enabled"
},
{
"name": "explain",
"description": "%copilot.workspace.explain.description%"
Expand Down Expand Up @@ -6370,6 +6359,27 @@
"sessionTypes": [
"local"
]
},
{
"path": "./assets/prompts/chronicle-standup.prompt.md",
"when": "github.copilot.sessionSearch.enabled",
"sessionTypes": [
"local"
]
},
{
"path": "./assets/prompts/chronicle-tips.prompt.md",
"when": "github.copilot.sessionSearch.enabled",
"sessionTypes": [
"local"
]
},
{
"path": "./assets/prompts/chronicle-reindex.prompt.md",
"when": "github.copilot.sessionSearch.enabled",
"sessionTypes": [
"local"
]
}
],
"chatSkills": [
Expand Down Expand Up @@ -6450,6 +6460,13 @@
"sessionTypes": [
"local"
]
},
{
"path": "./assets/prompts/skills/chronicle/SKILL.md",
"when": "github.copilot.sessionSearch.enabled",
"sessionTypes": [
"local"
]
}
],
"terminal": {
Expand Down
4 changes: 0 additions & 4 deletions extensions/copilot/package.nls.json
Original file line number Diff line number Diff line change
Expand Up @@ -168,12 +168,8 @@
"copilot.edits.description": "Edit files in your workspace",
"copilot.agent.description": "Edit files in your workspace in agent mode",
"copilot.agent.compact.description": "Free up context by compacting the conversation history. Optionally include extra instructions for compaction.",
"copilot.chronicle.description": "Session history tools and insights",
"copilot.chronicle.standup.description": "Generate a standup report from recent chat sessions",
"copilot.chronicle.tips.description": "Get personalized tips based on your chat session usage patterns",
"github.copilot.command.sessionSync.deleteSessions": "Delete Session Sync Data",
"github.copilot.command.chronicle.reindex": "Reindex Sessions",
"copilot.chronicle.reindex.description": "Rebuild the local session index from stored session logs. Add 'force' to re-process already indexed sessions.",
"github.copilot.config.sessionSearch.enabled": "Enable session search and /chronicle commands. This is a team-internal setting.",
"github.copilot.config.sessionSearch.localIndex.enabled": "Enable local session tracking. When enabled, Copilot tracks session data locally for /chronicle commands.",
"github.copilot.config.localIndex.enabled": "Enable local session tracking. When enabled, session data is tracked locally for /chronicle commands.",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,9 +56,10 @@ export class CloudSessionStoreClient {

/**
* Execute a SQL query against the cloud session store (user-scoped).
* Returns an array of row objects on success, or undefined on failure.
* Returns rows on success, an error string on query errors (4xx bad SQL),
* or undefined on auth/network/infrastructure failures (401, 403, network errors).
*/
async executeQuery(sql: string): Promise<{ rows: Record<string, unknown>[]; truncated: boolean } | undefined> {
async executeQuery(sql: string): Promise<{ rows: Record<string, unknown>[]; truncated: boolean } | { error: string } | undefined> {
try {
const copilotToken = await this._tokenManager.getCopilotToken();
const baseUrl = copilotToken.endpoints?.api;
Expand All @@ -85,7 +86,19 @@ export class CloudSessionStoreClient {
});

if (!res.ok) {
return undefined;
// Auth/permission failures → return undefined so callers fall back to local
if (res.status === 401 || res.status === 403) {
return undefined;
}

// Query errors (bad SQL, etc.) → surface error to model
try {
const body = await res.json() as { error?: string; message?: string };
const msg = body?.error ?? body?.message ?? `HTTP ${res.status}`;
return { error: msg };
} catch {
return { error: `HTTP ${res.status}` };
}
}
Comment thread
vijayupadya marked this conversation as resolved.

const data = await res.json() as CloudQueryResponse;
Expand Down
Loading
Loading