Pulse/internal/ai/knowledge/store_extended_test.go
rcourtman e194e17159 Update AI core services and adapters
AI module improvements:

Patrol System:
- Better trigger handling
- Improved history persistence
- Enhanced coverage testing

Knowledge Store:
- Extended functionality
- Better test coverage

Adapters:
- Discovery adapter updates
- Investigation adapter improvements

Unified Bridge:
- Setup improvements
- Better test coverage

Alert handling and service updates.
2026-01-28 16:51:53 +00:00

416 lines
11 KiB
Go

package knowledge
import (
"os"
"strings"
"testing"
)
// Additional tests to improve coverage
func TestNewStore_InvalidDir(t *testing.T) {
// Test with a file as directory (should work, creates subdir)
store, err := NewStore("/nonexistent/path/that/should/fail")
if store == nil && err != nil {
// Expected - can't create in nonexistent path
t.Log("Store creation failed as expected for nonexistent path")
}
}
func TestStore_GetKnowledge_Empty(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Get knowledge for non-existent guest
knowledge, err := store.GetKnowledge("nonexistent")
if err != nil {
t.Fatalf("GetKnowledge should not error for non-existent guest: %v", err)
}
// Should return empty knowledge
if knowledge != nil && len(knowledge.Notes) > 0 {
t.Error("Expected empty knowledge for non-existent guest")
}
}
func TestStore_SaveNote_Basic(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save a note
err = store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
if err != nil {
t.Fatalf("Failed to save note: %v", err)
}
// Verify it's retrievable
knowledge, err := store.GetKnowledge("vm-100")
if err != nil {
t.Fatalf("Failed to get knowledge: %v", err)
}
if knowledge == nil {
t.Fatal("Expected knowledge to be non-nil")
}
if len(knowledge.Notes) != 1 {
t.Fatalf("Expected 1 note, got %d", len(knowledge.Notes))
}
note := knowledge.Notes[0]
if note.Category != "config" {
t.Errorf("Expected category 'config', got '%s'", note.Category)
}
if note.Title != "Database" {
t.Errorf("Expected title 'Database', got '%s'", note.Title)
}
if note.Content != "PostgreSQL 15" {
t.Errorf("Expected content 'PostgreSQL 15', got '%s'", note.Content)
}
}
func TestStore_SaveNote_UpdateExisting(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save initial note
err = store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 14")
if err != nil {
t.Fatalf("Failed to save note: %v", err)
}
// Save another note with same title/category - should update
err = store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
if err != nil {
t.Fatalf("Failed to update note: %v", err)
}
// Should still have only 1 note
knowledge, err := store.GetKnowledge("vm-100")
if err != nil {
t.Fatalf("Failed to get knowledge: %v", err)
}
if len(knowledge.Notes) != 1 {
t.Errorf("Expected 1 note after update, got %d", len(knowledge.Notes))
}
if knowledge.Notes[0].Content != "PostgreSQL 15" {
t.Errorf("Expected updated content 'PostgreSQL 15', got '%s'", knowledge.Notes[0].Content)
}
}
func TestStore_FormatForContextForResources_Scoped(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Seed knowledge for multiple resources
if err := store.SaveNote("vm-100", "web-server", "vm", "config", "DB", "Postgres"); err != nil {
t.Fatalf("Failed to save note: %v", err)
}
if err := store.SaveNote("qemu/200", "app", "vm", "service", "API", "FastAPI"); err != nil {
t.Fatalf("Failed to save note: %v", err)
}
if err := store.SaveNote("host:alpha", "alpha", "host", "service", "Agent", "v1"); err != nil {
t.Fatalf("Failed to save note: %v", err)
}
if err := store.SaveNote("docker:host1/container1", "container1", "docker", "service", "Web", "Nginx"); err != nil {
t.Fatalf("Failed to save note: %v", err)
}
// Scope to a single VM by qemu resource ID
scoped := store.FormatForContextForResources([]string{"qemu/100"})
if scoped == "" {
t.Fatalf("Expected scoped context, got empty")
}
if !strings.Contains(scoped, "web-server") {
t.Fatalf("Expected vm-100 notes in scoped context")
}
if strings.Contains(scoped, "app") {
t.Fatalf("Did not expect qemu/200 notes in scoped context")
}
// Scope to docker resource ID
scoped = store.FormatForContextForResources([]string{"docker:host1/container1"})
if !strings.Contains(scoped, "container1") {
t.Fatalf("Expected docker container notes in scoped context")
}
if strings.Contains(scoped, "web-server") {
t.Fatalf("Did not expect vm-100 notes in docker scoped context")
}
// Scope to host resource ID
scoped = store.FormatForContextForResources([]string{"host:alpha"})
if !strings.Contains(scoped, "alpha") {
t.Fatalf("Expected host notes in scoped context")
}
if strings.Contains(scoped, "container1") {
t.Fatalf("Did not expect docker notes in host scoped context")
}
}
func TestStore_DeleteNote(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save a note
err = store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
if err != nil {
t.Fatalf("Failed to save note: %v", err)
}
// Get the note ID
knowledge, _ := store.GetKnowledge("vm-100")
noteID := knowledge.Notes[0].ID
// Delete the note
err = store.DeleteNote("vm-100", noteID)
if err != nil {
t.Fatalf("Failed to delete note: %v", err)
}
// Verify it's deleted
knowledge, _ = store.GetKnowledge("vm-100")
if len(knowledge.Notes) != 0 {
t.Errorf("Expected 0 notes after delete, got %d", len(knowledge.Notes))
}
}
func TestStore_DeleteNote_NonExistent(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Delete non-existent note - may return error for non-existent guest
err = store.DeleteNote("nonexistent", "nonexistent-note")
// It's ok either way - just checking it doesn't panic
_ = err
}
func TestStore_GetNotesByCategory(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save notes in different categories
store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
store.SaveNote("vm-100", "web-server", "vm", "service", "Web Server", "nginx")
store.SaveNote("vm-100", "web-server", "vm", "config", "Cache", "Redis")
// Get config notes only
notes, err := store.GetNotesByCategory("vm-100", "config")
if err != nil {
t.Fatalf("Failed to get notes by category: %v", err)
}
if len(notes) != 2 {
t.Errorf("Expected 2 config notes, got %d", len(notes))
}
for _, note := range notes {
if note.Category != "config" {
t.Errorf("Expected category 'config', got '%s'", note.Category)
}
}
}
func TestStore_FormatForContext_Empty(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Format for non-existent guest
result := store.FormatForContext("nonexistent")
if result != "" {
t.Errorf("Expected empty result for non-existent guest, got: %s", result)
}
}
func TestStore_FormatForContext_WithData(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save some notes
store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
store.SaveNote("vm-100", "web-server", "vm", "service", "Web Server", "nginx on port 80")
// Format for context
result := store.FormatForContext("vm-100")
if result == "" {
t.Error("Expected non-empty result")
}
if !contains(result, "PostgreSQL") {
t.Error("Expected result to contain PostgreSQL")
}
if !contains(result, "nginx") {
t.Error("Expected result to contain nginx")
}
}
func TestStore_ListGuests(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Initially empty
guests, err := store.ListGuests()
if err != nil {
t.Fatalf("Failed to list guests: %v", err)
}
if len(guests) != 0 {
t.Errorf("Expected 0 guests initially, got %d", len(guests))
}
// Add some guests
store.SaveNote("vm-100", "web-server", "vm", "config", "DB", "PostgreSQL")
store.SaveNote("vm-200", "db-server", "vm", "config", "DB", "MySQL")
// List again
guests, err = store.ListGuests()
if err != nil {
t.Fatalf("Failed to list guests: %v", err)
}
if len(guests) != 2 {
t.Errorf("Expected 2 guests, got %d", len(guests))
}
}
func TestStore_Persistence(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
// Create store and save data
store1, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
store1.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
// Create new store in same dir - should load data
store2, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create second store: %v", err)
}
knowledge, err := store2.GetKnowledge("vm-100")
if err != nil {
t.Fatalf("Failed to get knowledge from second store: %v", err)
}
if knowledge == nil || len(knowledge.Notes) == 0 {
t.Error("Expected persisted note to be loaded in second store")
}
}
func TestStore_MultipleNotes(t *testing.T) {
tmpDir, err := os.MkdirTemp("", "knowledge-test")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tmpDir)
store, err := NewStore(tmpDir)
if err != nil {
t.Fatalf("Failed to create store: %v", err)
}
// Save multiple notes with different titles
store.SaveNote("vm-100", "web-server", "vm", "config", "Database", "PostgreSQL 15")
store.SaveNote("vm-100", "web-server", "vm", "config", "Cache", "Redis 7")
store.SaveNote("vm-100", "web-server", "vm", "service", "Web", "nginx")
knowledge, err := store.GetKnowledge("vm-100")
if err != nil {
t.Fatalf("Failed to get knowledge: %v", err)
}
if len(knowledge.Notes) != 3 {
t.Errorf("Expected 3 notes, got %d", len(knowledge.Notes))
}
}