mirror of
https://github.com/lukaszraczylo/claude-mnemonic.git
synced 2026-06-05 23:03:55 +00:00
7a061c85eb
* refactor(hooks): simplify hook execution with shared context - [x] Extract BaseInput struct to eliminate duplicate fields across hooks - [x] Create RunHook handler pattern for session-start and user-prompt - [x] Create RunStatuslineHook for fast statusline rendering without worker startup - [x] Add HookContext struct to pass port, project, CWD, SessionID to handlers - [x] Add db/interface.go with ObservationReader/Writer interfaces - [x] Add comprehensive conflict management tests in sqlite/conflict_test.go - [x] Add vector client tests for Count, ModelVersion, NeedsRebuild, GetStaleVectors - [x] Add FilterByThreshold helper tests for query result filtering - [x] Make handlers_test more robust for network-dependent update checks - [x] Update package versions in UI * Move to GORM + general cleanup * feat(mcp): add observation relations discovery and scoring integration - [x] Add find_related_observations MCP tool for discovering related observations by confidence - [x] Integrate scoring calculator and recalculator into MCP server initialization - [x] Add pattern, relation, and session stores to MCP server dependencies - [x] Register MCP server in Claude Code settings during plugin installation - [x] Update install scripts (bash, PowerShell) to configure MCP server settings - [x] Switch plugin manifest files to template-based versioning (plugin.json.tpl, marketplace.json.tpl) - [x] Update all MCP server tests to pass new dependency parameters
130 lines
3.4 KiB
Go
130 lines
3.4 KiB
Go
// Package main provides the session-start hook entry point.
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/lukaszraczylo/claude-mnemonic/pkg/hooks"
|
|
)
|
|
|
|
// Input is the hook input from Claude Code.
|
|
type Input struct {
|
|
hooks.BaseInput
|
|
Source string `json:"source"` // "startup", "resume", "clear", "compact"
|
|
}
|
|
|
|
// Observation represents an observation from the API.
|
|
type Observation struct {
|
|
ID int64 `json:"id"`
|
|
Type string `json:"type"`
|
|
Title string `json:"title"`
|
|
Subtitle string `json:"subtitle"`
|
|
Narrative string `json:"narrative"`
|
|
Facts []string `json:"facts"`
|
|
}
|
|
|
|
func main() {
|
|
hooks.RunHook("SessionStart", handleSessionStart)
|
|
}
|
|
|
|
func handleSessionStart(ctx *hooks.HookContext, input *Input) (string, error) {
|
|
// Fetch observations for context injection
|
|
endpoint := fmt.Sprintf("/api/context/inject?project=%s&cwd=%s",
|
|
url.QueryEscape(ctx.Project),
|
|
url.QueryEscape(ctx.CWD))
|
|
|
|
result, err := hooks.GET(ctx.Port, endpoint)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "[claude-mnemonic] Warning: context fetch failed: %v\n", err)
|
|
return "", nil
|
|
}
|
|
|
|
// Parse observations from response
|
|
obsData, ok := result["observations"].([]interface{})
|
|
if !ok || len(obsData) == 0 {
|
|
// No observations - just continue normally
|
|
return "", nil
|
|
}
|
|
|
|
// Get full_count from response (how many observations get full detail)
|
|
fullCount := 25 // default
|
|
if fc, ok := result["full_count"].(float64); ok && fc > 0 {
|
|
fullCount = int(fc)
|
|
}
|
|
|
|
// Show count to user via stderr
|
|
fmt.Fprintf(os.Stderr, "[claude-mnemonic] Injecting %d observations from project memory (%d detailed, %d condensed)\n",
|
|
len(obsData), min(fullCount, len(obsData)), max(0, len(obsData)-fullCount))
|
|
|
|
// Build context string
|
|
contextBuilder := "<claude-mnemonic-context>\n"
|
|
contextBuilder += fmt.Sprintf("# Project Memory (%d observations)\n", len(obsData))
|
|
contextBuilder += "Use this knowledge to answer questions without re-exploring the codebase.\n\n"
|
|
|
|
for i, o := range obsData {
|
|
obs, ok := o.(map[string]interface{})
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
title := getString(obs, "title")
|
|
obsType := getString(obs, "type")
|
|
|
|
// First `fullCount` observations get full detail, rest are condensed
|
|
if i < fullCount {
|
|
// Full detail: include narrative and facts
|
|
narrative := getString(obs, "narrative")
|
|
|
|
contextBuilder += fmt.Sprintf("## %d. [%s] %s\n", i+1, strings.ToUpper(obsType), title)
|
|
if narrative != "" {
|
|
contextBuilder += narrative + "\n"
|
|
}
|
|
|
|
if facts, ok := obs["facts"].([]interface{}); ok && len(facts) > 0 {
|
|
contextBuilder += "Key facts:\n"
|
|
for _, f := range facts {
|
|
if fact, ok := f.(string); ok && fact != "" {
|
|
contextBuilder += fmt.Sprintf("- %s\n", fact)
|
|
}
|
|
}
|
|
}
|
|
contextBuilder += "\n"
|
|
} else {
|
|
// Condensed: just title and subtitle (one line)
|
|
subtitle := getString(obs, "subtitle")
|
|
if subtitle != "" {
|
|
contextBuilder += fmt.Sprintf("- [%s] %s: %s\n", strings.ToUpper(obsType), title, subtitle)
|
|
} else {
|
|
contextBuilder += fmt.Sprintf("- [%s] %s\n", strings.ToUpper(obsType), title)
|
|
}
|
|
}
|
|
}
|
|
|
|
contextBuilder += "</claude-mnemonic-context>\n"
|
|
return contextBuilder, nil
|
|
}
|
|
|
|
func getString(m map[string]interface{}, key string) string {
|
|
if v, ok := m[key].(string); ok {
|
|
return v
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func min(a, b int) int {
|
|
if a < b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func max(a, b int) int {
|
|
if a > b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|