mirror of
https://github.com/lukaszraczylo/claude-mnemonic.git
synced 2026-06-11 00:09:28 +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
199 lines
5.4 KiB
Go
199 lines
5.4 KiB
Go
// Package gorm provides GORM-based database operations for claude-mnemonic.
|
|
package gorm
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"time"
|
|
|
|
"gorm.io/gorm"
|
|
"gorm.io/gorm/clause"
|
|
|
|
"github.com/lukaszraczylo/claude-mnemonic/pkg/models"
|
|
)
|
|
|
|
// SessionStore provides session-related database operations using GORM.
|
|
type SessionStore struct {
|
|
db *gorm.DB
|
|
}
|
|
|
|
// NewSessionStore creates a new session store.
|
|
func NewSessionStore(store *Store) *SessionStore {
|
|
return &SessionStore{db: store.DB}
|
|
}
|
|
|
|
// CreateSDKSession creates a new SDK session (idempotent - returns existing ID if exists).
|
|
// This is the KEY to how claude-mnemonic stays unified across hooks.
|
|
func (s *SessionStore) CreateSDKSession(ctx context.Context, claudeSessionID, project, userPrompt string) (int64, error) {
|
|
now := time.Now()
|
|
|
|
session := &SDKSession{
|
|
ClaudeSessionID: claudeSessionID,
|
|
SDKSessionID: func() sql.NullString {
|
|
return sql.NullString{String: claudeSessionID, Valid: true}
|
|
}(),
|
|
Project: project,
|
|
UserPrompt: func() sql.NullString {
|
|
if userPrompt != "" {
|
|
return sql.NullString{String: userPrompt, Valid: true}
|
|
}
|
|
return sql.NullString{Valid: false}
|
|
}(),
|
|
Status: "active",
|
|
StartedAt: now.Format(time.RFC3339),
|
|
StartedAtEpoch: now.UnixMilli(),
|
|
}
|
|
|
|
// CRITICAL: INSERT OR IGNORE makes this idempotent
|
|
// Use OnConflict with DoNothing to achieve INSERT OR IGNORE behavior
|
|
result := s.db.WithContext(ctx).
|
|
Clauses(clause.OnConflict{
|
|
Columns: []clause.Column{{Name: "claude_session_id"}},
|
|
DoNothing: true,
|
|
}).
|
|
Create(session)
|
|
|
|
if result.Error != nil {
|
|
return 0, result.Error
|
|
}
|
|
|
|
// Check if insert happened
|
|
if result.RowsAffected == 0 {
|
|
// Session exists - UPDATE project and user_prompt if we have non-empty values
|
|
if project != "" {
|
|
updates := map[string]interface{}{
|
|
"project": project,
|
|
}
|
|
if userPrompt != "" {
|
|
updates["user_prompt"] = userPrompt
|
|
}
|
|
s.db.WithContext(ctx).
|
|
Model(&SDKSession{}).
|
|
Where("claude_session_id = ?", claudeSessionID).
|
|
Updates(updates)
|
|
}
|
|
|
|
// Fetch existing session
|
|
var existing SDKSession
|
|
err := s.db.WithContext(ctx).
|
|
Where("claude_session_id = ?", claudeSessionID).
|
|
First(&existing).Error
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return existing.ID, nil
|
|
}
|
|
|
|
return session.ID, nil
|
|
}
|
|
|
|
// GetSessionByID retrieves a session by its database ID.
|
|
func (s *SessionStore) GetSessionByID(ctx context.Context, id int64) (*models.SDKSession, error) {
|
|
var sess SDKSession
|
|
err := s.db.WithContext(ctx).First(&sess, id).Error
|
|
if err == gorm.ErrRecordNotFound {
|
|
return nil, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return toModelSDKSession(&sess), nil
|
|
}
|
|
|
|
// FindAnySDKSession finds any session by Claude session ID (any status).
|
|
func (s *SessionStore) FindAnySDKSession(ctx context.Context, claudeSessionID string) (*models.SDKSession, error) {
|
|
var sess SDKSession
|
|
err := s.db.WithContext(ctx).
|
|
Where("claude_session_id = ?", claudeSessionID).
|
|
First(&sess).Error
|
|
if err == gorm.ErrRecordNotFound {
|
|
return nil, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return toModelSDKSession(&sess), nil
|
|
}
|
|
|
|
// IncrementPromptCounter increments the prompt counter and returns the new value.
|
|
func (s *SessionStore) IncrementPromptCounter(ctx context.Context, id int64) (int, error) {
|
|
// Atomic increment using GORM expression
|
|
err := s.db.WithContext(ctx).
|
|
Model(&SDKSession{}).
|
|
Where("id = ?", id).
|
|
Update("prompt_counter", gorm.Expr("COALESCE(prompt_counter, 0) + 1")).Error
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Fetch updated value
|
|
var sess SDKSession
|
|
err = s.db.WithContext(ctx).
|
|
Select("prompt_counter").
|
|
First(&sess, id).Error
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return sess.PromptCounter, nil
|
|
}
|
|
|
|
// GetPromptCounter returns the current prompt counter for a session.
|
|
func (s *SessionStore) GetPromptCounter(ctx context.Context, id int64) (int, error) {
|
|
var sess SDKSession
|
|
err := s.db.WithContext(ctx).
|
|
Select("prompt_counter").
|
|
First(&sess, id).Error
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return sess.PromptCounter, nil
|
|
}
|
|
|
|
// GetSessionsToday returns the count of sessions started today.
|
|
func (s *SessionStore) GetSessionsToday(ctx context.Context) (int, error) {
|
|
// Get start of today in milliseconds
|
|
now := time.Now()
|
|
startOfDay := time.Date(now.Year(), now.Month(), now.Day(), 0, 0, 0, 0, now.Location())
|
|
startEpoch := startOfDay.UnixMilli()
|
|
|
|
var count int64
|
|
err := s.db.WithContext(ctx).
|
|
Model(&SDKSession{}).
|
|
Where("started_at_epoch >= ?", startEpoch).
|
|
Count(&count).Error
|
|
|
|
return int(count), err
|
|
}
|
|
|
|
// GetAllProjects returns all unique project names.
|
|
func (s *SessionStore) GetAllProjects(ctx context.Context) ([]string, error) {
|
|
var projects []string
|
|
err := s.db.WithContext(ctx).
|
|
Model(&SDKSession{}).
|
|
Distinct("project").
|
|
Where("project IS NOT NULL AND project != ''").
|
|
Order("project ASC").
|
|
Pluck("project", &projects).Error
|
|
|
|
return projects, err
|
|
}
|
|
|
|
// toModelSDKSession converts a GORM SDKSession to pkg/models.SDKSession.
|
|
func toModelSDKSession(sess *SDKSession) *models.SDKSession {
|
|
return &models.SDKSession{
|
|
ID: sess.ID,
|
|
ClaudeSessionID: sess.ClaudeSessionID,
|
|
SDKSessionID: sess.SDKSessionID,
|
|
Project: sess.Project,
|
|
UserPrompt: sess.UserPrompt,
|
|
WorkerPort: sess.WorkerPort,
|
|
PromptCounter: int64(sess.PromptCounter),
|
|
Status: models.SessionStatus(sess.Status),
|
|
StartedAt: sess.StartedAt,
|
|
StartedAtEpoch: sess.StartedAtEpoch,
|
|
CompletedAt: sess.CompletedAt,
|
|
CompletedAtEpoch: sess.CompletedAtEpoch,
|
|
}
|
|
}
|