2025-01-19 10:56:03 +01:00
|
|
|
package schedule
|
2024-09-30 07:34:40 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2025-01-05 12:20:35 +01:00
|
|
|
"slices"
|
2024-10-29 07:22:04 +01:00
|
|
|
"strings"
|
2024-09-30 07:34:40 +02:00
|
|
|
|
2024-10-01 07:36:31 +02:00
|
|
|
"github.com/google/uuid"
|
2024-09-30 07:34:40 +02:00
|
|
|
"go-mod.ewintr.nl/planner/item"
|
2025-01-19 10:56:03 +01:00
|
|
|
"go-mod.ewintr.nl/planner/plan/cli/arg"
|
|
|
|
"go-mod.ewintr.nl/planner/plan/command"
|
2025-01-19 09:57:52 +01:00
|
|
|
"go-mod.ewintr.nl/planner/plan/format"
|
2025-01-13 09:13:48 +01:00
|
|
|
"go-mod.ewintr.nl/planner/sync/client"
|
2024-09-30 07:34:40 +02:00
|
|
|
)
|
|
|
|
|
2024-12-27 11:20:32 +01:00
|
|
|
type AddArgs struct {
|
|
|
|
fieldTPL map[string][]string
|
2025-01-19 10:56:03 +01:00
|
|
|
Schedule item.Schedule
|
2024-09-30 07:34:40 +02:00
|
|
|
}
|
|
|
|
|
2024-12-27 11:20:32 +01:00
|
|
|
func NewAddArgs() AddArgs {
|
|
|
|
return AddArgs{
|
|
|
|
fieldTPL: map[string][]string{
|
2024-12-29 09:32:49 +01:00
|
|
|
"date": {"d", "date", "on"},
|
|
|
|
"recurrer": {"rec", "recurrer"},
|
2024-10-29 07:22:04 +01:00
|
|
|
},
|
2024-10-01 07:36:31 +02:00
|
|
|
}
|
2024-09-30 07:34:40 +02:00
|
|
|
}
|
|
|
|
|
2025-01-19 10:56:03 +01:00
|
|
|
func (aa AddArgs) Parse(main []string, fields map[string]string) (command.Command, error) {
|
|
|
|
if len(main) == 0 || !slices.Contains([]string{"s", "sched", "schedule"}, main[0]) {
|
|
|
|
return nil, command.ErrWrongCommand
|
|
|
|
}
|
|
|
|
main = main[1:]
|
2025-01-05 12:20:35 +01:00
|
|
|
if len(main) == 0 || !slices.Contains([]string{"add", "a", "new", "n"}, main[0]) {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, command.ErrWrongCommand
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
2025-01-05 12:20:35 +01:00
|
|
|
|
2024-12-27 11:20:32 +01:00
|
|
|
main = main[1:]
|
|
|
|
if len(main) == 0 {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, fmt.Errorf("%w: title is required for add", command.ErrInvalidArg)
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
2025-01-19 10:56:03 +01:00
|
|
|
fields, err := arg.ResolveFields(fields, aa.fieldTPL)
|
2024-12-27 11:20:32 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
2024-12-27 11:20:32 +01:00
|
|
|
|
2025-01-19 10:56:03 +01:00
|
|
|
sched := item.Schedule{
|
2024-12-27 11:20:32 +01:00
|
|
|
ID: uuid.New().String(),
|
2025-01-19 10:56:03 +01:00
|
|
|
ScheduleBody: item.ScheduleBody{
|
2024-12-29 09:32:49 +01:00
|
|
|
Title: strings.Join(main, " "),
|
2024-12-27 11:20:32 +01:00
|
|
|
},
|
2024-10-01 07:36:31 +02:00
|
|
|
}
|
2024-12-27 11:20:32 +01:00
|
|
|
|
|
|
|
if val, ok := fields["date"]; ok {
|
|
|
|
d := item.NewDateFromString(val)
|
|
|
|
if d.IsZero() {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, fmt.Errorf("%w: could not parse date", command.ErrInvalidArg)
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
2025-01-19 10:56:03 +01:00
|
|
|
sched.Date = d
|
2024-10-01 07:36:31 +02:00
|
|
|
}
|
2024-12-27 11:20:32 +01:00
|
|
|
if val, ok := fields["recurrer"]; ok {
|
|
|
|
rec := item.NewRecurrer(val)
|
|
|
|
if rec == nil {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, fmt.Errorf("%w: could not parse recurrer", command.ErrInvalidArg)
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
2025-01-19 10:56:03 +01:00
|
|
|
sched.Recurrer = rec
|
|
|
|
sched.RecurNext = sched.Recurrer.First()
|
2024-10-01 07:36:31 +02:00
|
|
|
}
|
2024-10-29 07:22:04 +01:00
|
|
|
|
2024-12-27 11:20:32 +01:00
|
|
|
return &Add{
|
2024-12-29 09:32:49 +01:00
|
|
|
Args: AddArgs{
|
2025-01-19 10:56:03 +01:00
|
|
|
Schedule: sched,
|
2024-12-27 11:20:32 +01:00
|
|
|
},
|
|
|
|
}, nil
|
2024-10-29 07:22:04 +01:00
|
|
|
}
|
|
|
|
|
2024-12-27 11:20:32 +01:00
|
|
|
type Add struct {
|
2024-12-29 09:32:49 +01:00
|
|
|
Args AddArgs
|
2024-12-27 11:20:32 +01:00
|
|
|
}
|
2024-12-01 10:22:47 +01:00
|
|
|
|
2025-01-19 10:56:03 +01:00
|
|
|
func (a Add) Do(repos command.Repositories, _ client.Client) (command.CommandResult, error) {
|
2025-01-13 09:13:48 +01:00
|
|
|
tx, err := repos.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not start transaction: %v", err)
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
2025-01-19 10:56:03 +01:00
|
|
|
if err := repos.Schedule(tx).Store(a.Args.Schedule); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not store schedule: %v", err)
|
2024-10-01 07:36:31 +02:00
|
|
|
}
|
|
|
|
|
2025-01-13 09:13:48 +01:00
|
|
|
localID, err := repos.LocalID(tx).Next()
|
2024-10-03 07:32:48 +02:00
|
|
|
if err != nil {
|
2024-12-29 09:32:49 +01:00
|
|
|
return nil, fmt.Errorf("could not create next local id: %v", err)
|
2024-10-03 07:32:48 +02:00
|
|
|
}
|
2025-01-19 10:56:03 +01:00
|
|
|
if err := repos.LocalID(tx).Store(a.Args.Schedule.ID, localID); err != nil {
|
2024-12-29 09:32:49 +01:00
|
|
|
return nil, fmt.Errorf("could not store local id: %v", err)
|
2024-10-03 07:32:48 +02:00
|
|
|
}
|
|
|
|
|
2025-01-19 10:56:03 +01:00
|
|
|
it, err := a.Args.Schedule.Item()
|
2024-10-07 11:11:18 +02:00
|
|
|
if err != nil {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, fmt.Errorf("could not convert schedule to sync item: %v", err)
|
2024-10-07 11:11:18 +02:00
|
|
|
}
|
2025-01-13 09:13:48 +01:00
|
|
|
if err := repos.Sync(tx).Store(it); err != nil {
|
2024-12-29 09:32:49 +01:00
|
|
|
return nil, fmt.Errorf("could not store sync item: %v", err)
|
2024-10-07 11:11:18 +02:00
|
|
|
}
|
|
|
|
|
2025-01-13 09:13:48 +01:00
|
|
|
if err := tx.Commit(); err != nil {
|
2025-01-19 10:56:03 +01:00
|
|
|
return nil, fmt.Errorf("could not add schedule: %v", err)
|
2025-01-13 09:13:48 +01:00
|
|
|
}
|
|
|
|
|
2025-01-19 09:57:52 +01:00
|
|
|
return AddResult{
|
|
|
|
LocalID: localID,
|
|
|
|
}, nil
|
2024-12-29 09:32:49 +01:00
|
|
|
}
|
|
|
|
|
2025-01-19 09:57:52 +01:00
|
|
|
type AddResult struct {
|
|
|
|
LocalID int
|
2024-12-29 09:32:49 +01:00
|
|
|
}
|
|
|
|
|
2025-01-19 09:57:52 +01:00
|
|
|
func (ar AddResult) Render() string {
|
2025-01-19 10:56:03 +01:00
|
|
|
return fmt.Sprintf("stored schedule %s", format.Bold(fmt.Sprintf("%d", ar.LocalID)))
|
2024-09-30 07:34:40 +02:00
|
|
|
}
|