Files
claude-mnemonic/internal/db/gorm/conflict_store.go
T
lukaszraczylo 7a061c85eb general improvements (#17)
* 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
2026-01-07 00:26:20 +00:00

282 lines
8.1 KiB
Go

// Package gorm provides GORM-based database operations for claude-mnemonic.
package gorm
import (
"context"
"database/sql"
"time"
"gorm.io/gorm"
"github.com/lukaszraczylo/claude-mnemonic/pkg/models"
)
// SupersededRetentionDays is the number of days to keep superseded observations before deletion.
const SupersededRetentionDays = 3
// ConflictStore provides conflict-related database operations using GORM.
type ConflictStore struct {
db *gorm.DB
}
// NewConflictStore creates a new conflict store.
func NewConflictStore(store *Store) *ConflictStore {
return &ConflictStore{
db: store.DB,
}
}
// StoreConflict stores a new observation conflict.
func (s *ConflictStore) StoreConflict(ctx context.Context, conflict *models.ObservationConflict) (int64, error) {
dbConflict := &ObservationConflict{
NewerObsID: conflict.NewerObsID,
OlderObsID: conflict.OlderObsID,
ConflictType: conflict.ConflictType,
Resolution: conflict.Resolution,
DetectedAt: conflict.DetectedAt,
DetectedAtEpoch: conflict.DetectedAtEpoch,
Resolved: 0,
}
// Convert bool to int
if conflict.Resolved {
dbConflict.Resolved = 1
}
// Handle nullable fields
if conflict.Reason != "" {
dbConflict.Reason = sql.NullString{String: conflict.Reason, Valid: true}
}
if conflict.ResolvedAt != nil && *conflict.ResolvedAt != "" {
dbConflict.ResolvedAt = sql.NullString{String: *conflict.ResolvedAt, Valid: true}
}
result := s.db.WithContext(ctx).Create(dbConflict)
if result.Error != nil {
return 0, result.Error
}
return dbConflict.ID, nil
}
// MarkObservationSuperseded marks an observation as superseded.
func (s *ConflictStore) MarkObservationSuperseded(ctx context.Context, obsID int64) error {
result := s.db.WithContext(ctx).
Model(&Observation{}).
Where("id = ?", obsID).
Update("is_superseded", 1)
return result.Error
}
// MarkObservationsSuperseded marks multiple observations as superseded.
func (s *ConflictStore) MarkObservationsSuperseded(ctx context.Context, obsIDs []int64) error {
if len(obsIDs) == 0 {
return nil
}
result := s.db.WithContext(ctx).
Model(&Observation{}).
Where("id IN ?", obsIDs).
Update("is_superseded", 1)
return result.Error
}
// GetConflictsByObservationID retrieves all conflicts involving an observation.
func (s *ConflictStore) GetConflictsByObservationID(ctx context.Context, obsID int64) ([]*models.ObservationConflict, error) {
var conflicts []ObservationConflict
err := s.db.WithContext(ctx).
Where("newer_obs_id = ? OR older_obs_id = ?", obsID, obsID).
Order("detected_at_epoch DESC").
Find(&conflicts).Error
if err != nil {
return nil, err
}
return toModelConflicts(conflicts), nil
}
// GetUnresolvedConflicts retrieves all unresolved conflicts.
func (s *ConflictStore) GetUnresolvedConflicts(ctx context.Context, limit int) ([]*models.ObservationConflict, error) {
var conflicts []ObservationConflict
err := s.db.WithContext(ctx).
Where("resolved = 0").
Order("detected_at_epoch DESC").
Limit(limit).
Find(&conflicts).Error
if err != nil {
return nil, err
}
return toModelConflicts(conflicts), nil
}
// GetSupersededObservationIDs returns IDs of all observations that have been superseded.
func (s *ConflictStore) GetSupersededObservationIDs(ctx context.Context, project string) ([]int64, error) {
var ids []int64
err := s.db.WithContext(ctx).
Table("observation_conflicts oc").
Select("DISTINCT oc.older_obs_id").
Joins("JOIN observations o ON o.id = oc.older_obs_id").
Where("oc.resolution = ?", models.ResolutionPreferNewer).
Where("o.project = ? OR o.scope = 'global'", project).
Pluck("oc.older_obs_id", &ids).Error
return ids, err
}
// ResolveConflict marks a conflict as resolved.
func (s *ConflictStore) ResolveConflict(ctx context.Context, conflictID int64, resolution models.ConflictResolution) error {
now := time.Now().Format(time.RFC3339)
result := s.db.WithContext(ctx).
Model(&ObservationConflict{}).
Where("id = ?", conflictID).
Updates(map[string]interface{}{
"resolved": 1,
"resolved_at": now,
"resolution": resolution,
})
return result.Error
}
// DeleteConflictsByObservationID deletes all conflicts involving an observation.
// Called when an observation is deleted.
func (s *ConflictStore) DeleteConflictsByObservationID(ctx context.Context, obsID int64) error {
result := s.db.WithContext(ctx).
Where("newer_obs_id = ? OR older_obs_id = ?", obsID, obsID).
Delete(&ObservationConflict{})
return result.Error
}
// ConflictWithDetails contains a conflict with its observation details.
type ConflictWithDetails struct {
Conflict *models.ObservationConflict
NewerObsTitle string
OlderObsTitle string
}
// CleanupSupersededObservations deletes observations that have been superseded for longer than
// SupersededRetentionDays. Returns the IDs of deleted observations for downstream cleanup (e.g., vector DB).
func (s *ConflictStore) CleanupSupersededObservations(ctx context.Context, project string) ([]int64, error) {
// Calculate cutoff time (3 days ago in milliseconds)
cutoffEpoch := time.Now().AddDate(0, 0, -SupersededRetentionDays).UnixMilli()
var toDelete []int64
// Use a transaction to prevent TOCTOU race condition
err := s.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
// Find IDs to delete
err := tx.Table("observations o").
Select("DISTINCT o.id").
Joins("JOIN observation_conflicts oc ON o.id = oc.older_obs_id").
Where("o.is_superseded = 1").
Where("o.project = ?", project).
Where("oc.detected_at_epoch < ?", cutoffEpoch).
Pluck("o.id", &toDelete).Error
if err != nil {
return err
}
if len(toDelete) == 0 {
return nil
}
// Delete the conflict records first (due to foreign key constraints)
for _, obsID := range toDelete {
err := tx.Where("newer_obs_id = ? OR older_obs_id = ?", obsID, obsID).
Delete(&ObservationConflict{}).Error
if err != nil {
return err
}
}
// Delete the observations
return tx.Delete(&Observation{}, toDelete).Error
})
if err != nil {
return nil, err
}
return toDelete, nil
}
// GetConflictsWithDetails retrieves all conflicts with observation titles for display.
func (s *ConflictStore) GetConflictsWithDetails(ctx context.Context, project string, limit int) ([]*ConflictWithDetails, error) {
var results []struct {
ObservationConflict
NewerTitle sql.NullString `gorm:"column:newer_title"`
OlderTitle sql.NullString `gorm:"column:older_title"`
}
err := s.db.WithContext(ctx).
Table("observation_conflicts oc").
Select("oc.*, "+
"COALESCE(newer.title, '') as newer_title, "+
"COALESCE(older.title, '') as older_title").
Joins("JOIN observations newer ON newer.id = oc.newer_obs_id").
Joins("JOIN observations older ON older.id = oc.older_obs_id").
Where("newer.project = ? OR older.project = ?", project, project).
Order("oc.detected_at_epoch DESC").
Limit(limit).
Scan(&results).Error
if err != nil {
return nil, err
}
conflicts := make([]*ConflictWithDetails, len(results))
for i, r := range results {
conflicts[i] = &ConflictWithDetails{
Conflict: toModelConflict(&r.ObservationConflict),
NewerObsTitle: r.NewerTitle.String,
OlderObsTitle: r.OlderTitle.String,
}
}
return conflicts, nil
}
// toModelConflict converts a GORM ObservationConflict to a pkg/models ObservationConflict.
func toModelConflict(c *ObservationConflict) *models.ObservationConflict {
conflict := &models.ObservationConflict{
ID: c.ID,
NewerObsID: c.NewerObsID,
OlderObsID: c.OlderObsID,
ConflictType: c.ConflictType,
Resolution: c.Resolution,
DetectedAt: c.DetectedAt,
DetectedAtEpoch: c.DetectedAtEpoch,
Resolved: c.Resolved == 1,
}
if c.Reason.Valid {
conflict.Reason = c.Reason.String
}
if c.ResolvedAt.Valid {
s := c.ResolvedAt.String
conflict.ResolvedAt = &s
}
return conflict
}
// toModelConflicts converts a slice of GORM ObservationConflicts to pkg/models ObservationConflicts.
func toModelConflicts(conflicts []ObservationConflict) []*models.ObservationConflict {
result := make([]*models.ObservationConflict, len(conflicts))
for i, c := range conflicts {
result[i] = toModelConflict(&c)
}
return result
}