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"
|
2026-03-27 02:11:46 +01:00
|
|
|
"slices"
|
2026-03-26 12:48:47 +00:00
|
|
|
|
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
|
)
|
|
|
|
|
|
2026-03-27 02:11:46 +01:00
|
|
|
var (
|
2026-03-29 21:24:09 +02:00
|
|
|
uTitle, uContent, uDue string
|
2026-03-27 02:11:46 +01:00
|
|
|
uClearDue bool
|
|
|
|
|
uAddTags, uRmTags, uAddRels, uRmRels []string
|
|
|
|
|
)
|
2026-03-26 12:48:47 +00:00
|
|
|
|
|
|
|
|
var updateCmd = &cobra.Command{
|
2026-03-27 02:11:46 +01:00
|
|
|
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) {
|
2026-03-29 21:24:09 +02:00
|
|
|
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
|
|
|
|
|
}
|
2026-03-27 02:11:46 +01:00
|
|
|
|
2026-03-29 23:16:44 +02:00
|
|
|
// parse relations
|
2026-03-29 18:58:34 +02:00
|
|
|
addRels, rmRels := make(map[models.RelType][]string), make(map[models.RelType][]string)
|
2026-03-27 02:11:46 +01:00
|
|
|
parseRel := func(src []string, dst map[models.RelType][]string) bool {
|
|
|
|
|
for _, r := range src {
|
2026-03-29 23:16:44 +02:00
|
|
|
rel, err := parseRelFlag(svc, r)
|
2026-03-27 02:11:46 +01:00
|
|
|
if err != nil {
|
|
|
|
|
fmt.Fprintln(os.Stderr, err)
|
|
|
|
|
return false
|
|
|
|
|
}
|
2026-03-29 23:16:44 +02:00
|
|
|
dst[rel.Type] = append(dst[rel.Type], rel.Target)
|
2026-03-26 12:48:47 +00:00
|
|
|
}
|
2026-03-27 02:11:46 +01:00
|
|
|
return true
|
2026-03-26 12:48:47 +00:00
|
|
|
}
|
2026-03-27 02:11:46 +01:00
|
|
|
if !parseRel(uAddRels, addRels) || !parseRel(uRmRels, rmRels) {
|
|
|
|
|
return
|
2026-03-26 12:48:47 +00:00
|
|
|
}
|
2026-03-27 02:11:46 +01:00
|
|
|
|
2026-03-29 23:16:44 +02:00
|
|
|
// enforce blocking of tasks
|
|
|
|
|
//TODO: mabye part of the backend?
|
2026-03-27 02:11:46 +01:00
|
|
|
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
|
|
|
|
|
}
|
2026-03-27 02:11:46 +01:00
|
|
|
}
|
|
|
|
|
|
2026-03-29 23:16:44 +02:00
|
|
|
// update main fields
|
2026-03-27 02:41:27 +01:00
|
|
|
if cmd.Flags().Changed("title") {
|
2026-03-29 18:58:34 +02:00
|
|
|
node.Title = uTitle
|
2026-03-27 02:41:27 +01:00
|
|
|
}
|
|
|
|
|
if cmd.Flags().Changed("content") {
|
2026-03-29 18:58:34 +02:00
|
|
|
node.Content = uContent
|
2026-03-27 02:41:27 +01:00
|
|
|
}
|
|
|
|
|
if cmd.Flags().Changed("due") {
|
2026-03-29 18:58:34 +02:00
|
|
|
node.DueDate = uDue
|
|
|
|
|
}
|
|
|
|
|
if uClearDue {
|
|
|
|
|
node.DueDate = ""
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-29 23:16:44 +02:00
|
|
|
// udpate tags
|
2026-03-29 18:58:34 +02:00
|
|
|
for _, t := range uRmTags {
|
2026-03-29 23:16:44 +02:00
|
|
|
if err := node.RemoveTag(t); err != nil {
|
|
|
|
|
fmt.Fprintln(os.Stderr, "failed to remove tag:", err)
|
|
|
|
|
return
|
|
|
|
|
}
|
2026-03-27 02:41:27 +01:00
|
|
|
}
|
2026-03-29 18:58:34 +02:00
|
|
|
for _, t := range uAddTags {
|
2026-03-29 23:16:44 +02:00
|
|
|
node.AddTag(t)
|
2026-03-29 18:58:34 +02:00
|
|
|
}
|
|
|
|
|
|
2026-03-29 23:16:44 +02:00
|
|
|
// update relations
|
2026-03-29 18:58:34 +02:00
|
|
|
for rt, tgts := range rmRels {
|
|
|
|
|
for _, tgt := range tgts {
|
2026-03-29 23:16:44 +02:00
|
|
|
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 {
|
2026-03-29 23:16:44 +02:00
|
|
|
node.AddRelation(rt, tgt)
|
2026-03-29 18:58:34 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-29 23:16:44 +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 {
|
2026-03-29 21:24:09 +02:00
|
|
|
output.PrintNode(cmd.OutOrStdout(), svc, n, jsonFlag)
|
2026-03-27 02:11:46 +01:00
|
|
|
} else {
|
2026-03-26 12:48:47 +00:00
|
|
|
fmt.Fprintln(os.Stderr, "failed to fetch node:", err)
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
|
rootCmd.AddCommand(updateCmd)
|
2026-03-29 19:22:44 +02:00
|
|
|
addPropertyFlags(updateCmd)
|
2026-03-27 02:11:46 +01:00
|
|
|
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
|
|
|
}
|