2026-03-29 23:56:43 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
2026-04-02 01:58:48 +02:00
|
|
|
"fmt"
|
2026-03-29 23:56:43 +02:00
|
|
|
"os"
|
|
|
|
|
"os/exec"
|
|
|
|
|
"path/filepath"
|
2026-03-31 22:36:49 +02:00
|
|
|
"slices"
|
2026-03-29 23:56:43 +02:00
|
|
|
"strings"
|
|
|
|
|
"testing"
|
|
|
|
|
)
|
|
|
|
|
|
2026-03-31 22:36:49 +02:00
|
|
|
// ── JSON response types ───────────────────────────────────────────────────────
|
|
|
|
|
|
2026-03-29 23:56:43 +02:00
|
|
|
type NodeResponse struct {
|
|
|
|
|
ID string `json:"id"`
|
|
|
|
|
Title string `json:"title"`
|
|
|
|
|
Content string `json:"content"`
|
2026-03-31 22:36:49 +02:00
|
|
|
DueDate string `json:"due_date"`
|
|
|
|
|
CreatedAt string `json:"created_at"`
|
|
|
|
|
UpdatedAt string `json:"updated_at"`
|
2026-03-29 23:56:43 +02:00
|
|
|
Tags []string `json:"tags"`
|
|
|
|
|
Relations map[string][]string `json:"relations"`
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-02 01:58:48 +02:00
|
|
|
func (n NodeResponse) HasTag(tag string) bool { return slices.Contains(n.Tags, tag) }
|
2026-03-31 22:36:49 +02:00
|
|
|
func (n NodeResponse) Property(key string) string {
|
|
|
|
|
prefix := "_" + key + "::"
|
|
|
|
|
for _, t := range n.Tags {
|
|
|
|
|
if strings.HasPrefix(t, prefix) {
|
|
|
|
|
return strings.TrimPrefix(t, prefix)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
func (n NodeResponse) HasRelation(relType, targetID string) bool {
|
|
|
|
|
return slices.Contains(n.Relations[relType], targetID)
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-02 01:58:48 +02:00
|
|
|
// ── Test binary ───────────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
var axBinary string
|
|
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
|
if err := exec.Command("go", "build", "-o", "ax", ".").Run(); err != nil {
|
|
|
|
|
fmt.Fprintln(os.Stderr, "build failed:", err)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
abs, err := filepath.Abs("./ax")
|
|
|
|
|
if err != nil {
|
|
|
|
|
fmt.Fprintln(os.Stderr, "resolve binary path:", err)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
axBinary = abs
|
|
|
|
|
code := m.Run()
|
|
|
|
|
os.Remove("./ax")
|
|
|
|
|
os.Exit(code)
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-31 22:36:49 +02:00
|
|
|
// ── Test environment ──────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
type testEnv struct {
|
2026-03-31 23:52:24 +02:00
|
|
|
t *testing.T
|
|
|
|
|
dir string
|
2026-04-02 01:58:48 +02:00
|
|
|
user string
|
2026-03-31 23:52:24 +02:00
|
|
|
}
|
|
|
|
|
|
2026-04-02 01:58:48 +02:00
|
|
|
func newTestEnv(t *testing.T, user string) *testEnv {
|
|
|
|
|
t.Helper()
|
|
|
|
|
env := &testEnv{t: t, dir: t.TempDir(), user: user}
|
|
|
|
|
env.mustAx("init")
|
|
|
|
|
return env
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// withUser returns a new testEnv pointing at the same directory but with a different user.
|
|
|
|
|
func (e *testEnv) withUser(user string) *testEnv {
|
|
|
|
|
return &testEnv{t: e.t, dir: e.dir, user: user}
|
2026-03-31 22:36:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (e *testEnv) ax(args ...string) (string, error) {
|
2026-04-02 01:58:48 +02:00
|
|
|
cmd := exec.Command(axBinary, args...)
|
2026-03-31 22:36:49 +02:00
|
|
|
cmd.Dir = e.dir
|
2026-04-02 01:58:48 +02:00
|
|
|
var env []string
|
|
|
|
|
for _, v := range os.Environ() {
|
|
|
|
|
if !strings.HasPrefix(v, "AX_USER=") {
|
|
|
|
|
env = append(env, v)
|
|
|
|
|
}
|
2026-03-31 23:52:24 +02:00
|
|
|
}
|
2026-04-02 01:58:48 +02:00
|
|
|
env = append(env, "AX_USER="+e.user)
|
|
|
|
|
cmd.Env = env
|
2026-03-29 23:56:43 +02:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
return string(out), err
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-31 22:36:49 +02:00
|
|
|
func (e *testEnv) mustAx(args ...string) string {
|
|
|
|
|
e.t.Helper()
|
|
|
|
|
out, err := e.ax(args...)
|
2026-03-29 23:56:43 +02:00
|
|
|
if err != nil {
|
2026-03-31 22:36:49 +02:00
|
|
|
e.t.Fatalf("ax %v failed: %v\n%s", args, err, out)
|
2026-03-29 23:56:43 +02:00
|
|
|
}
|
2026-03-31 22:36:49 +02:00
|
|
|
return out
|
2026-03-29 23:56:43 +02:00
|
|
|
}
|
|
|
|
|
|
2026-03-31 22:36:49 +02:00
|
|
|
func (e *testEnv) parseNode(out string) NodeResponse {
|
|
|
|
|
e.t.Helper()
|
|
|
|
|
var n NodeResponse
|
|
|
|
|
if err := json.Unmarshal([]byte(out), &n); err != nil {
|
2026-04-02 01:58:48 +02:00
|
|
|
e.t.Fatalf("parse node JSON: %v\noutput: %s", err, out)
|
2026-03-31 22:36:49 +02:00
|
|
|
}
|
|
|
|
|
return n
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (e *testEnv) parseNodes(out string) []NodeResponse {
|
|
|
|
|
e.t.Helper()
|
|
|
|
|
if strings.TrimSpace(out) == "null" {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
var nodes []NodeResponse
|
|
|
|
|
if err := json.Unmarshal([]byte(out), &nodes); err != nil {
|
2026-04-02 01:58:48 +02:00
|
|
|
e.t.Fatalf("parse node list JSON: %v\noutput: %s", err, out)
|
2026-03-31 22:36:49 +02:00
|
|
|
}
|
|
|
|
|
return nodes
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (e *testEnv) findInList(nodes []NodeResponse, id string) (NodeResponse, bool) {
|
|
|
|
|
for _, n := range nodes {
|
|
|
|
|
if n.ID == id {
|
|
|
|
|
return n, true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return NodeResponse{}, false
|
|
|
|
|
}
|