mirror of
https://github.com/lukaszraczylo/claude-mnemonic.git
synced 2026-06-05 23:03:55 +00:00
201 lines
5.9 KiB
Go
201 lines
5.9 KiB
Go
// Package sqlite provides SQLite database operations for claude-mnemonic.
|
|
package sqlite
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"time"
|
|
|
|
"github.com/lukaszraczylo/claude-mnemonic/pkg/models"
|
|
)
|
|
|
|
// SummaryStore provides summary-related database operations.
|
|
type SummaryStore struct {
|
|
store *Store
|
|
}
|
|
|
|
// NewSummaryStore creates a new summary store.
|
|
func NewSummaryStore(store *Store) *SummaryStore {
|
|
return &SummaryStore{store: store}
|
|
}
|
|
|
|
// StoreSummary stores a new session summary.
|
|
func (s *SummaryStore) StoreSummary(ctx context.Context, sdkSessionID, project string, summary *models.ParsedSummary, promptNumber int, discoveryTokens int64) (int64, int64, error) {
|
|
now := time.Now()
|
|
nowEpoch := now.UnixMilli()
|
|
|
|
// Ensure session exists (auto-create if missing)
|
|
if err := s.ensureSessionExists(ctx, sdkSessionID, project); err != nil {
|
|
return 0, 0, err
|
|
}
|
|
|
|
const query = `
|
|
INSERT INTO session_summaries
|
|
(sdk_session_id, project, request, investigated, learned, completed,
|
|
next_steps, notes, prompt_number, discovery_tokens, created_at, created_at_epoch)
|
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
`
|
|
|
|
result, err := s.store.ExecContext(ctx, query,
|
|
sdkSessionID, project,
|
|
nullString(summary.Request), nullString(summary.Investigated),
|
|
nullString(summary.Learned), nullString(summary.Completed),
|
|
nullString(summary.NextSteps), nullString(summary.Notes),
|
|
nullInt(promptNumber), discoveryTokens,
|
|
now.Format(time.RFC3339), nowEpoch,
|
|
)
|
|
if err != nil {
|
|
return 0, 0, err
|
|
}
|
|
|
|
id, _ := result.LastInsertId()
|
|
return id, nowEpoch, nil
|
|
}
|
|
|
|
// ensureSessionExists creates a session if it doesn't exist.
|
|
func (s *SummaryStore) ensureSessionExists(ctx context.Context, sdkSessionID, project string) error {
|
|
const checkQuery = `SELECT id FROM sdk_sessions WHERE sdk_session_id = ?`
|
|
var id int64
|
|
err := s.store.QueryRowContext(ctx, checkQuery, sdkSessionID).Scan(&id)
|
|
if err == nil {
|
|
return nil // Session exists
|
|
}
|
|
if err != sql.ErrNoRows {
|
|
return err
|
|
}
|
|
|
|
// Auto-create session
|
|
now := time.Now()
|
|
const insertQuery = `
|
|
INSERT INTO sdk_sessions
|
|
(claude_session_id, sdk_session_id, project, started_at, started_at_epoch, status)
|
|
VALUES (?, ?, ?, ?, ?, 'active')
|
|
`
|
|
_, err = s.store.ExecContext(ctx, insertQuery,
|
|
sdkSessionID, sdkSessionID, project,
|
|
now.Format(time.RFC3339), now.UnixMilli(),
|
|
)
|
|
return err
|
|
}
|
|
|
|
// GetSummariesByIDs retrieves summaries by a list of IDs.
|
|
func (s *SummaryStore) GetSummariesByIDs(ctx context.Context, ids []int64, orderBy string, limit int) ([]*models.SessionSummary, error) {
|
|
if len(ids) == 0 {
|
|
return nil, nil
|
|
}
|
|
|
|
// Build query with placeholders
|
|
// #nosec G202 -- query uses parameterized placeholders, not user input
|
|
query := `
|
|
SELECT id, sdk_session_id, project, request, investigated, learned, completed,
|
|
next_steps, notes, prompt_number, discovery_tokens, created_at, created_at_epoch
|
|
FROM session_summaries
|
|
WHERE id IN (?` + repeatPlaceholders(len(ids)-1) + `)
|
|
ORDER BY created_at_epoch `
|
|
|
|
if orderBy == "date_asc" {
|
|
query += "ASC"
|
|
} else {
|
|
query += "DESC"
|
|
}
|
|
|
|
if limit > 0 {
|
|
query += " LIMIT ?"
|
|
}
|
|
|
|
// Convert []int64 to []interface{}
|
|
args := make([]interface{}, len(ids))
|
|
for i, id := range ids {
|
|
args[i] = id
|
|
}
|
|
if limit > 0 {
|
|
args = append(args, limit)
|
|
}
|
|
|
|
rows, err := s.store.db.QueryContext(ctx, query, args...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer rows.Close()
|
|
|
|
var summaries []*models.SessionSummary
|
|
for rows.Next() {
|
|
var summary models.SessionSummary
|
|
if err := rows.Scan(
|
|
&summary.ID, &summary.SDKSessionID, &summary.Project,
|
|
&summary.Request, &summary.Investigated, &summary.Learned, &summary.Completed,
|
|
&summary.NextSteps, &summary.Notes, &summary.PromptNumber, &summary.DiscoveryTokens,
|
|
&summary.CreatedAt, &summary.CreatedAtEpoch,
|
|
); err != nil {
|
|
return nil, err
|
|
}
|
|
summaries = append(summaries, &summary)
|
|
}
|
|
return summaries, rows.Err()
|
|
}
|
|
|
|
// GetRecentSummaries retrieves recent summaries for a project.
|
|
func (s *SummaryStore) GetRecentSummaries(ctx context.Context, project string, limit int) ([]*models.SessionSummary, error) {
|
|
const query = `
|
|
SELECT id, sdk_session_id, project, request, investigated, learned, completed,
|
|
next_steps, notes, prompt_number, discovery_tokens, created_at, created_at_epoch
|
|
FROM session_summaries
|
|
WHERE project = ?
|
|
ORDER BY created_at_epoch DESC
|
|
LIMIT ?
|
|
`
|
|
|
|
rows, err := s.store.QueryContext(ctx, query, project, limit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer rows.Close()
|
|
|
|
var summaries []*models.SessionSummary
|
|
for rows.Next() {
|
|
var summary models.SessionSummary
|
|
if err := rows.Scan(
|
|
&summary.ID, &summary.SDKSessionID, &summary.Project,
|
|
&summary.Request, &summary.Investigated, &summary.Learned, &summary.Completed,
|
|
&summary.NextSteps, &summary.Notes, &summary.PromptNumber, &summary.DiscoveryTokens,
|
|
&summary.CreatedAt, &summary.CreatedAtEpoch,
|
|
); err != nil {
|
|
return nil, err
|
|
}
|
|
summaries = append(summaries, &summary)
|
|
}
|
|
return summaries, rows.Err()
|
|
}
|
|
|
|
// GetAllRecentSummaries retrieves recent summaries across all projects.
|
|
func (s *SummaryStore) GetAllRecentSummaries(ctx context.Context, limit int) ([]*models.SessionSummary, error) {
|
|
const query = `
|
|
SELECT id, sdk_session_id, project, request, investigated, learned, completed,
|
|
next_steps, notes, prompt_number, discovery_tokens, created_at, created_at_epoch
|
|
FROM session_summaries
|
|
ORDER BY created_at_epoch DESC
|
|
LIMIT ?
|
|
`
|
|
|
|
rows, err := s.store.QueryContext(ctx, query, limit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer rows.Close()
|
|
|
|
var summaries []*models.SessionSummary
|
|
for rows.Next() {
|
|
var summary models.SessionSummary
|
|
if err := rows.Scan(
|
|
&summary.ID, &summary.SDKSessionID, &summary.Project,
|
|
&summary.Request, &summary.Investigated, &summary.Learned, &summary.Completed,
|
|
&summary.NextSteps, &summary.Notes, &summary.PromptNumber, &summary.DiscoveryTokens,
|
|
&summary.CreatedAt, &summary.CreatedAtEpoch,
|
|
); err != nil {
|
|
return nil, err
|
|
}
|
|
summaries = append(summaries, &summary)
|
|
}
|
|
return summaries, rows.Err()
|
|
}
|