Files
nib-v1/cmd/cmd_test.go
lerko 2152baeb4f feat: add export and backup commands
- nib export: dump all entities to JSON (stdout or --output file)
- nib backup: atomic SQLite backup via VACUUM INTO (WAL-safe)
- Store.Backup() method on db layer
- Tests for both commands
2026-05-20 20:54:44 -04:00

287 lines
6.0 KiB
Go

package cmd
import (
"bytes"
"context"
"encoding/json"
"os"
"path/filepath"
"strings"
"testing"
"github.com/lerko/nib/internal/db"
"github.com/spf13/cobra"
)
func testStore(t *testing.T) *db.Store {
t.Helper()
dbPath := filepath.Join(t.TempDir(), "test.db")
t.Setenv("NIB_DB", dbPath)
store, err := db.Open(dbPath)
if err != nil {
t.Fatal(err)
}
t.Cleanup(func() { store.Close() })
return store
}
func newCmd() *cobra.Command {
c := &cobra.Command{}
c.SetContext(context.Background())
return c
}
func captureOutput(t *testing.T, fn func()) string {
t.Helper()
var buf bytes.Buffer
rootCmd.SetOut(&buf)
defer rootCmd.SetOut(nil)
fn()
return buf.String()
}
func seedEntity(t *testing.T, store *db.Store, body string, glyph db.Glyph) *db.Entity {
t.Helper()
e := &db.Entity{Body: body, Glyph: glyph, Tags: []string{}}
if err := store.Create(context.Background(), e); err != nil {
t.Fatal(err)
}
return e
}
func TestRunAdd(t *testing.T) {
testStore(t)
err := runAdd(newCmd(), []string{"hello", "world"})
if err != nil {
t.Fatalf("runAdd: %v", err)
}
}
func TestRunAddWithGlyph(t *testing.T) {
testStore(t)
err := runAdd(newCmd(), []string{"-", "buy", "milk", "#errands"})
if err != nil {
t.Fatalf("runAdd todo: %v", err)
}
}
func TestRunAddWithTimeAnchor(t *testing.T) {
testStore(t)
err := runAdd(newCmd(), []string{"@", "dentist", "@14:00"})
if err != nil {
t.Fatalf("runAdd event: %v", err)
}
}
func TestRunDelete(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "to delete", db.GlyphNote)
store.Close()
err := runDelete(newCmd(), []string{e.ID})
if err != nil {
t.Fatalf("runDelete soft: %v", err)
}
err = runDelete(newCmd(), []string{e.ID})
if err != nil {
t.Fatalf("runDelete hard: %v", err)
}
}
func TestRunDeleteNotFound(t *testing.T) {
testStore(t)
err := runDelete(newCmd(), []string{"nonexistent"})
if err == nil {
t.Fatal("expected error for nonexistent id")
}
if !strings.Contains(err.Error(), "not_found") {
t.Fatalf("expected not_found error, got: %v", err)
}
}
func TestRunPromote(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "reusable snippet", db.GlyphNote)
store.Close()
err := runPromote(newCmd(), []string{e.ID, "snippet"})
if err != nil {
t.Fatalf("runPromote: %v", err)
}
}
func TestRunPromoteAlreadyPromoted(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "already a card", db.GlyphNote)
store.Promote(context.Background(), e.ID, db.CardSnippet, nil)
store.Close()
err := runPromote(newCmd(), []string{e.ID, "snippet"})
if err == nil {
t.Fatal("expected error for already promoted")
}
}
func TestRunDemote(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "demote me", db.GlyphNote)
store.Promote(context.Background(), e.ID, db.CardSnippet, nil)
store.Close()
err := runDemote(newCmd(), []string{e.ID})
if err != nil {
t.Fatalf("runDemote: %v", err)
}
}
func TestRunDemoteAlreadyFluid(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "already fluid", db.GlyphNote)
store.Close()
err := runDemote(newCmd(), []string{e.ID})
if err == nil {
t.Fatal("expected error for already fluid")
}
}
func TestRunAbsorb(t *testing.T) {
store := testStore(t)
target := seedEntity(t, store, "target body", db.GlyphNote)
source := seedEntity(t, store, "source body", db.GlyphNote)
store.Close()
err := runAbsorb(newCmd(), []string{target.ID, source.ID})
if err != nil {
t.Fatalf("runAbsorb: %v", err)
}
}
func TestRunAbsorbSameEntity(t *testing.T) {
store := testStore(t)
e := seedEntity(t, store, "same entity", db.GlyphNote)
store.Close()
err := runAbsorb(newCmd(), []string{e.ID, e.ID})
if err == nil {
t.Fatal("expected error for same entity absorb")
}
}
func TestRunAbsorbCrystallizedTarget(t *testing.T) {
store := testStore(t)
target := seedEntity(t, store, "crystallized", db.GlyphNote)
source := seedEntity(t, store, "source", db.GlyphNote)
store.Promote(context.Background(), target.ID, db.CardSnippet, nil)
store.Close()
err := runAbsorb(newCmd(), []string{target.ID, source.ID})
if err == nil {
t.Fatal("expected error for crystallized target")
}
}
func TestRunLs(t *testing.T) {
store := testStore(t)
seedEntity(t, store, "recent note", db.GlyphNote)
store.Close()
lsTag = ""
lsDate = ""
lsMonth = ""
lsFrom = ""
lsTo = ""
lsLimit = 0
lsAll = false
err := runLs(newCmd(), nil)
if err != nil {
t.Fatalf("runLs: %v", err)
}
}
func TestRunLsEmpty(t *testing.T) {
testStore(t)
lsTag = ""
lsDate = ""
lsMonth = ""
lsFrom = ""
lsTo = ""
lsLimit = 0
lsAll = false
err := runLs(newCmd(), nil)
if err != nil {
t.Fatalf("runLs empty: %v", err)
}
}
func TestRunExport(t *testing.T) {
store := testStore(t)
seedEntity(t, store, "export me", db.GlyphNote)
seedEntity(t, store, "export me too", db.GlyphTodo)
store.Close()
outFile := filepath.Join(t.TempDir(), "export.json")
exportOutput = outFile
err := runExport(newCmd(), nil)
if err != nil {
t.Fatalf("runExport: %v", err)
}
data, err := os.ReadFile(outFile)
if err != nil {
t.Fatalf("read export: %v", err)
}
var entities []exportEntity
if err := json.Unmarshal(data, &entities); err != nil {
t.Fatalf("unmarshal export: %v", err)
}
if len(entities) != 2 {
t.Fatalf("expected 2 entities, got %d", len(entities))
}
}
func TestRunBackup(t *testing.T) {
store := testStore(t)
seedEntity(t, store, "backup me", db.GlyphNote)
store.Close()
dst := filepath.Join(t.TempDir(), "backup.db")
err := runBackup(newCmd(), []string{dst})
if err != nil {
t.Fatalf("runBackup: %v", err)
}
info, err := os.Stat(dst)
if err != nil {
t.Fatalf("backup file missing: %v", err)
}
if info.Size() == 0 {
t.Fatal("backup file is empty")
}
backed, err := db.Open(dst)
if err != nil {
t.Fatalf("open backup: %v", err)
}
defer backed.Close()
entities, err := backed.List(context.Background(), db.DefaultListParams())
if err != nil {
t.Fatalf("list backup: %v", err)
}
if len(entities) != 1 {
t.Fatalf("expected 1 entity in backup, got %d", len(entities))
}
}