2152baeb4f
- 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
287 lines
6.0 KiB
Go
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))
|
|
}
|
|
}
|