Files
ax/cmd/update.go

132 lines
3.0 KiB
Go
Raw Normal View History

2026-03-26 12:48:47 +00:00
package cmd
import (
"axolotl/models"
"axolotl/output"
2026-03-29 18:58:34 +02:00
"axolotl/service"
2026-03-26 12:48:47 +00:00
"fmt"
"os"
"slices"
2026-03-26 12:48:47 +00:00
"github.com/spf13/cobra"
)
var (
uTitle, uContent, uDue string
uClearDue bool
uAddTags, uRmTags, uAddRels, uRmRels []string
)
2026-03-26 12:48:47 +00:00
var updateCmd = &cobra.Command{
Use: "update <id>", Short: "Update a node", Args: cobra.ExactArgs(1),
2026-03-26 12:48:47 +00:00
Run: func(cmd *cobra.Command, args []string) {
svc, err := service.GetNodeService(cfg)
2026-03-26 12:48:47 +00:00
if err != nil {
fmt.Fprintln(os.Stderr, err)
return
}
2026-03-29 18:58:34 +02:00
node, err := svc.GetByID(args[0])
if err != nil {
fmt.Fprintln(os.Stderr, "node not found:", args[0])
return
}
// parse relations
2026-03-29 18:58:34 +02:00
addRels, rmRels := make(map[models.RelType][]string), make(map[models.RelType][]string)
parseRel := func(src []string, dst map[models.RelType][]string) bool {
for _, r := range src {
rel, err := parseRelFlag(svc, r)
if err != nil {
fmt.Fprintln(os.Stderr, err)
return false
}
dst[rel.Type] = append(dst[rel.Type], rel.Target)
2026-03-26 12:48:47 +00:00
}
return true
2026-03-26 12:48:47 +00:00
}
if !parseRel(uAddRels, addRels) || !parseRel(uRmRels, rmRels) {
return
2026-03-26 12:48:47 +00:00
}
// enforce blocking of tasks
//TODO: mabye part of the backend?
if slices.Contains(uAddTags, "_status::done") {
2026-03-29 18:58:34 +02:00
ok, blockers, err := svc.CanClose(args[0])
2026-03-26 12:48:47 +00:00
if err != nil {
fmt.Fprintln(os.Stderr, "failed to check blockers:", err)
return
}
if !ok {
fmt.Fprintf(os.Stderr, "cannot close: blocked by %v\n", blockers)
return
}
}
// update main fields
if cmd.Flags().Changed("title") {
2026-03-29 18:58:34 +02:00
node.Title = uTitle
}
if cmd.Flags().Changed("content") {
2026-03-29 18:58:34 +02:00
node.Content = uContent
}
if cmd.Flags().Changed("due") {
2026-03-29 18:58:34 +02:00
node.DueDate = uDue
}
if uClearDue {
node.DueDate = ""
}
// udpate tags
2026-03-29 18:58:34 +02:00
for _, t := range uRmTags {
if err := node.RemoveTag(t); err != nil {
fmt.Fprintln(os.Stderr, "failed to remove tag:", err)
return
}
}
2026-03-29 18:58:34 +02:00
for _, t := range uAddTags {
node.AddTag(t)
2026-03-29 18:58:34 +02:00
}
// update relations
2026-03-29 18:58:34 +02:00
for rt, tgts := range rmRels {
for _, tgt := range tgts {
if err := node.RemoveRelation(rt, tgt); err != nil {
fmt.Fprintln(os.Stderr, "failed to remove relation:", err)
return
}
2026-03-29 18:58:34 +02:00
}
}
for rt, tgts := range addRels {
for _, tgt := range tgts {
node.AddRelation(rt, tgt)
2026-03-29 18:58:34 +02:00
}
}
// persist update
2026-03-29 18:58:34 +02:00
if err := svc.Update(node); err != nil {
2026-03-26 12:48:47 +00:00
fmt.Fprintln(os.Stderr, "failed to update:", err)
return
}
2026-03-29 18:58:34 +02:00
if n, err := svc.GetByID(args[0]); err == nil {
output.PrintNode(cmd.OutOrStdout(), svc, n, jsonFlag)
} else {
2026-03-26 12:48:47 +00:00
fmt.Fprintln(os.Stderr, "failed to fetch node:", err)
}
},
}
func init() {
rootCmd.AddCommand(updateCmd)
addPropertyFlags(updateCmd)
f := updateCmd.Flags()
f.StringVar(&uTitle, "title", "", "")
f.StringVar(&uContent, "content", "", "")
f.StringVar(&uDue, "due", "", "")
f.BoolVar(&uClearDue, "clear-due", false, "")
f.StringArrayVar(&uAddTags, "tag", nil, "")
f.StringArrayVar(&uRmTags, "tag-remove", nil, "")
f.StringArrayVar(&uAddRels, "rel", nil, "")
f.StringArrayVar(&uRmRels, "rel-remove", nil, "")
2026-03-26 12:48:47 +00:00
}