proper add

This commit is contained in:
Erik Winter 2024-10-01 07:36:31 +02:00
parent de7f5b8c74
commit cd784f999d
2 changed files with 192 additions and 27 deletions

View File

@ -1,14 +1,20 @@
package command package command
import ( import (
"errors"
"fmt" "fmt"
"time" "time"
"github.com/google/uuid"
"github.com/urfave/cli/v2" "github.com/urfave/cli/v2"
"go-mod.ewintr.nl/planner/item" "go-mod.ewintr.nl/planner/item"
"go-mod.ewintr.nl/planner/plan/storage" "go-mod.ewintr.nl/planner/plan/storage"
) )
var (
ErrInvalidArg = errors.New("invalid argument")
)
var AddCmd = &cli.Command{ var AddCmd = &cli.Command{
Name: "add", Name: "add",
Usage: "Add a new event", Usage: "Add a new event",
@ -17,11 +23,13 @@ var AddCmd = &cli.Command{
Name: "name", Name: "name",
Aliases: []string{"n"}, Aliases: []string{"n"},
Usage: "The event that will happen", Usage: "The event that will happen",
Required: true,
}, },
&cli.StringFlag{ &cli.StringFlag{
Name: "on", Name: "on",
Aliases: []string{"o"}, Aliases: []string{"o"},
Usage: "The date, in YYYY-MM-DD format", Usage: "The date, in YYYY-MM-DD format",
Required: true,
}, },
&cli.StringFlag{ &cli.StringFlag{
Name: "at", Name: "at",
@ -37,29 +45,55 @@ var AddCmd = &cli.Command{
} }
func NewAddCmd(repo storage.EventRepo) *cli.Command { func NewAddCmd(repo storage.EventRepo) *cli.Command {
AddCmd.Action = NewAddAction(repo) AddCmd.Action = func(cCtx *cli.Context) error {
return Add(cCtx.String("name"), cCtx.String("on"), cCtx.String("at"), cCtx.String("for"), repo)
}
return AddCmd return AddCmd
} }
func NewAddAction(repo storage.EventRepo) func(*cli.Context) error { func Add(nameStr, onStr, atStr, frStr string, repo storage.EventRepo) error {
return func(cCtx *cli.Context) error { if nameStr == "" {
desc := cCtx.String("name") return fmt.Errorf("%w: name is required", ErrInvalidArg)
date, err := time.Parse("2006-01-02", cCtx.String("date")) }
if err != nil { if onStr == "" {
return fmt.Errorf("could not parse date: %v", err) return fmt.Errorf("%w: date is required", ErrInvalidArg)
}
if atStr == "" && frStr != "" {
return fmt.Errorf("%w: can not have duration without start time", ErrInvalidArg)
}
if atStr == "" && frStr == "" {
frStr = "24h"
} }
one := item.Event{ startFormat := "2006-01-02"
ID: "a", startStr := onStr
if atStr != "" {
startFormat = fmt.Sprintf("%s 15:04", startFormat)
startStr = fmt.Sprintf("%s %s", startStr, atStr)
}
start, err := time.Parse(startFormat, startStr)
if err != nil {
return fmt.Errorf("%w: could not parse start time and date: %v", ErrInvalidArg, err)
}
e := item.Event{
ID: uuid.New().String(),
EventBody: item.EventBody{ EventBody: item.EventBody{
Title: desc, Title: nameStr,
Start: date, Start: start,
}, },
} }
if err := repo.Store(one); err != nil {
if frStr != "" {
fr, err := time.ParseDuration(frStr)
if err != nil {
return fmt.Errorf("%w: could not parse time: %s", ErrInvalidArg, err)
}
e.Duration = fr
}
if err := repo.Store(e); err != nil {
return fmt.Errorf("could not store event: %v", err) return fmt.Errorf("could not store event: %v", err)
} }
return nil return nil
}
} }

131
plan/command/add_test.go Normal file
View File

@ -0,0 +1,131 @@
package command_test
import (
"testing"
"time"
"github.com/google/go-cmp/cmp"
"go-mod.ewintr.nl/planner/item"
"go-mod.ewintr.nl/planner/plan/command"
"go-mod.ewintr.nl/planner/plan/storage"
)
func TestAdd(t *testing.T) {
t.Parallel()
oneHour, err := time.ParseDuration("1h")
if err != nil {
t.Errorf("exp nil, got %v", err)
}
oneDay, err := time.ParseDuration("24h")
if err != nil {
t.Errorf("exp nil, got %v", err)
}
for _, tc := range []struct {
name string
args map[string]string
expEvent item.Event
expErr bool
}{
{
name: "no name",
args: map[string]string{
"on": "2024-10-01",
"at": "9:00",
"for": "1h",
},
expErr: true,
},
{
name: "no date",
args: map[string]string{
"name": "event",
"at": "9:00",
"for": "1h",
},
expErr: true,
},
{
name: "duration, but no time",
args: map[string]string{
"name": "event",
"on": "2024-10-01",
"for": "1h",
},
expErr: true,
},
{
name: "time, but no duration",
args: map[string]string{
"name": "event",
"on": "2024-10-01",
"at": "9:00",
},
expEvent: item.Event{
ID: "a",
EventBody: item.EventBody{
Title: "event",
Start: time.Date(2024, 10, 1, 9, 0, 0, 0, time.UTC),
},
},
},
{
name: "no time, no duration",
args: map[string]string{
"name": "event",
"on": "2024-10-01",
},
expEvent: item.Event{
ID: "a",
EventBody: item.EventBody{
Title: "event",
Start: time.Date(2024, 10, 1, 0, 0, 0, 0, time.UTC),
Duration: oneDay,
},
},
},
{
name: "full",
args: map[string]string{
"name": "event",
"on": "2024-10-01",
"at": "9:00",
"for": "1h",
},
expEvent: item.Event{
ID: "a",
EventBody: item.EventBody{
Title: "event",
Start: time.Date(2024, 10, 1, 9, 0, 0, 0, time.UTC),
Duration: oneHour,
},
},
},
} {
t.Run(tc.name, func(t *testing.T) {
mem := storage.NewMemory()
actErr := command.Add(tc.args["name"], tc.args["on"], tc.args["at"], tc.args["for"], mem) != nil
if tc.expErr != actErr {
t.Errorf("exp %v, got %v", tc.expErr, actErr)
}
if tc.expErr {
return
}
actEvents, err := mem.FindAll()
if err != nil {
t.Errorf("exp nil, got %v", err)
}
if len(actEvents) != 1 {
t.Errorf("exp 1, got %d", len(actEvents))
}
if actEvents[0].ID == "" {
t.Errorf("exp string not te be empty")
}
tc.expEvent.ID = actEvents[0].ID
if diff := cmp.Diff(tc.expEvent, actEvents[0]); diff != "" {
t.Errorf("(exp +, got -)\n%s", diff)
}
})
}
}