diff --git a/internal/model/comment.go b/internal/model/comment.go index 489d411..fc79bb9 100644 --- a/internal/model/comment.go +++ b/internal/model/comment.go @@ -1,7 +1,6 @@ package model import ( - "html/template" "strings" "github.com/goyalmunish/reminder/pkg/utils" @@ -21,19 +20,19 @@ type Comment struct { } // String provides basic string representation of a commment. -func (comment *Comment) String() (string, error) { - var escapeString bool = false +func (comment *Comment) String() string { + // var escapeString bool = false // way 1 - if escapeString { - reportTemplate := `[{{.CreatedAt | mediumTimeStr}}] {{.Text}}` - funcMap := template.FuncMap{ - "mediumTimeStr": utils.UnixTimestampToMediumTimeStr, - } - return utils.TemplateResult(reportTemplate, funcMap, comment) - } + // if escapeString { + // reportTemplate := `[{{.CreatedAt | mediumTimeStr}}] {{.Text}}` + // funcMap := template.FuncMap{ + // "mediumTimeStr": utils.UnixTimestampToMediumTimeStr, + // } + // return utils.TemplateResult(reportTemplate, funcMap, comment) + // } // way 2 parts := []string{utils.UnixTimestampToMediumTimeStr(comment.CreatedAt), comment.Text} - return strings.Join(parts, " | "), nil + return strings.Join(parts, " | ") } diff --git a/internal/model/comment_test.go b/internal/model/comment_test.go new file mode 100644 index 0000000..447399b --- /dev/null +++ b/internal/model/comment_test.go @@ -0,0 +1,15 @@ +package model_test + +import ( + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestCommentString(t *testing.T) { + utils.Location = utils.UTCLocation() + c := model.Comment{Text: "c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters", BaseStruct: model.BaseStruct{CreatedAt: 1600000004, UpdatedAt: 1600001004}} + want := "13-Sep-20 12:26:44 | c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters" + utils.AssertEqual(t, c.String(), want) +} diff --git a/internal/model/comments.go b/internal/model/comments.go index 1d818cc..c586af0 100644 --- a/internal/model/comments.go +++ b/internal/model/comments.go @@ -12,15 +12,12 @@ func (c Comments) Swap(i, j int) { c[i], c[j] = c[j], c[i] } func (c Comments) Less(i, j int) bool { return c[i].CreatedAt > c[j].CreatedAt } // Strings provides representation of Commments in terms of slice of strings. -func (comments Comments) Strings() ([]string, error) { +func (comments Comments) Strings() []string { // assuming each note will have 10 comments on average strs := make([]string, 0, 10) for _, comment := range comments { - s, err := comment.String() - if err != nil { - return nil, err - } + s := comment.String() strs = append(strs, s) } - return strs, nil + return strs } diff --git a/internal/model/comments_test.go b/internal/model/comments_test.go new file mode 100644 index 0000000..55d9724 --- /dev/null +++ b/internal/model/comments_test.go @@ -0,0 +1,26 @@ +package model_test + +import ( + "sort" + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestCommentsStrings(t *testing.T) { + utils.Location = utils.UTCLocation() + comments := model.Comments{&model.Comment{Text: "c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters"}, &model.Comment{Text: "c2"}, &model.Comment{Text: "c3"}} + want := []string{"nil | c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters", "nil | c2", "nil | c3"} + utils.AssertEqual(t, comments.Strings(), want) +} + +func TestCommentsSort(t *testing.T) { + utils.Location = utils.UTCLocation() + c1 := &model.Comment{Text: "c1", BaseStruct: model.BaseStruct{CreatedAt: 1600000004}} + c2 := &model.Comment{Text: "c2", BaseStruct: model.BaseStruct{CreatedAt: 1600000002}} + c3 := &model.Comment{Text: "c3", BaseStruct: model.BaseStruct{CreatedAt: 1600000003}} + comments := model.Comments{c1, c2, c3} + sort.Sort(comments) + utils.AssertEqual(t, comments, model.Comments{c1, c3, c2}) +} diff --git a/internal/model/functions_test.go b/internal/model/functions_test.go new file mode 100644 index 0000000..36208b5 --- /dev/null +++ b/internal/model/functions_test.go @@ -0,0 +1,84 @@ +package model_test + +import ( + "errors" + "io/fs" + "os" + "path" + "testing" + "time" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestNewNote(t *testing.T) { + tagIDs := []int{1, 3, 5} + dummyText := "a random note text" + note, _ := model.NewNote(tagIDs, dummyText) + want := &model.Note{ + Text: dummyText, + TagIds: tagIDs, + Status: note.Status, + BaseStruct: model.BaseStruct{UpdatedAt: note.UpdatedAt, CreatedAt: note.CreatedAt}, + } + utils.AssertEqual(t, note, want) +} + +func TestBasicTags(t *testing.T) { + basicTags := model.BasicTags() + slugs := basicTags.Slugs() + want := "[current priority-urgent priority-medium priority-low repeat-annually repeat-monthly tips]" + utils.AssertEqual(t, slugs, want) +} + +func TestNewTag(t *testing.T) { + dummySlug := "test_tag_slug" + dummyGroup := "test_tag_group" + tag, _ := model.NewTag(10, dummySlug, dummyGroup) + want := &model.Tag{ + Id: 10, + Slug: dummySlug, + Group: dummyGroup, + } + utils.AssertEqual(t, tag, want) +} + +func TestMakeSureFileExists(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + + // make sure file doesn't exists already + _, err := os.Stat(dataFilePath) + utils.AssertEqual(t, err != nil, true) + utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), true) + // attempt to create the file and required dirs, when the file doesn't exist already + _ = model.MakeSureFileExists(dataFilePath, false) + // prove that the file was created + stats, err := os.Stat(dataFilePath) + utils.AssertEqual(t, err != nil, false) + utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), false) + + // make sure that the existing file is not replaced + modificationTime := stats.ModTime() + // attempt to create the file and required dirs, when the file does exist already + time.Sleep(10 * time.Millisecond) + _ = model.MakeSureFileExists(dataFilePath, false) + utils.AssertEqual(t, err != nil, false) + utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), false) + stats, _ = os.Stat(dataFilePath) + newModificationTime := stats.ModTime() + utils.AssertEqual(t, newModificationTime == modificationTime, true) +} + +func TestReadDataFile(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + // attempt to read file and parse it + reminderData, _ := model.ReadDataFile(dataFilePath, false) + utils.AssertEqual(t, reminderData.UpdatedAt > 0, true) +} diff --git a/internal/model/model_test.go b/internal/model/model_test.go deleted file mode 100644 index f1cf8e3..0000000 --- a/internal/model/model_test.go +++ /dev/null @@ -1,1068 +0,0 @@ -package model_test - -import ( - "errors" - "fmt" - - // "fmt" - "io/fs" - "os" - "path" - "sort" - "strings" - "testing" - "time" - - // "github.com/golang/mock/gomock" - - model "github.com/goyalmunish/reminder/internal/model" - "github.com/goyalmunish/reminder/internal/settings" - utils "github.com/goyalmunish/reminder/pkg/utils" - gc "google.golang.org/api/calendar/v3" -) - -// mocks - -type MockPromptTagSlug struct { -} -type MockPromptTagGroup struct { -} -type MockPromptNoteText struct { -} -type TestTagger struct{} - -func (tagger TestTagger) TagsFromIds(tagIDs []int) []string { - slugs := []string{} - for i, id := range tagIDs { - slug := fmt.Sprintf("%d-%d", i, id) - slugs = append(slugs, slug) - } - return slugs -} - -func (prompt *MockPromptTagSlug) Run() (string, error) { - return "test_tag_slug", nil -} - -func (prompt *MockPromptTagGroup) Run() (string, error) { - return "test_tag_group", nil -} - -func (prompt *MockPromptNoteText) Run() (string, error) { - return "a random note text", nil -} - -// test examples - -func TestDataFile(t *testing.T) { - defaultDataFilePath := settings.DefaultSettings().AppInfo.DataFile - utils.AssertEqual(t, strings.HasPrefix(defaultDataFilePath, "/") || strings.HasPrefix(defaultDataFilePath, "~/"), true) - utils.AssertEqual(t, strings.HasSuffix(defaultDataFilePath, ".json"), true) -} - -func TestUser(t *testing.T) { - got := model.User{Name: "Test User", EmailId: "user@test.com"} - want := "{Name: Test User, EmailId: user@test.com}" - utils.AssertEqual(t, got, want) -} - -func TestTag(t *testing.T) { - // case 1: general case - got := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - want := "tag_group1#a#1" - utils.AssertEqual(t, got, want) - // case 2: blank group - got = model.Tag{Id: 1, Slug: "a", Group: ""} - want = "#a#1" - utils.AssertEqual(t, got, want) - // case 3: omitted group - got = model.Tag{Id: 1, Slug: "a"} - want = "#a#1" - utils.AssertEqual(t, got, want) -} - -func TestTags(t *testing.T) { - var tags model.Tags - tags = append(tags, &model.Tag{Id: 1, Slug: "a", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 2, Slug: "z", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 3, Slug: "c", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 4, Slug: "b", Group: "tag_group2"}) - sort.Sort(model.Tags(tags)) - var got []int - for _, value := range tags { - got = append(got, value.Id) - } - want := []int{1, 4, 3, 2} - utils.AssertEqual(t, got, want) -} - -func TestSlugs(t *testing.T) { - var tags model.Tags - utils.AssertEqual(t, tags, "[]") - // case 1 (no tags) - utils.AssertEqual(t, tags.Slugs(), "[]") - // case 2 (non-empty tags) - tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group"}) - tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group"}) - tags = append(tags, &model.Tag{Id: 3, Slug: "tag_3", Group: "tag_group"}) - got := tags.Slugs() - want := "[tag_1 tag_2 tag_3]" - utils.AssertEqual(t, got, want) -} - -func TestFromSlug(t *testing.T) { - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - // case 1 (passing non-existing slug) - utils.AssertEqual(t, tags.FromSlug("no_such_slug"), nil) - // case 2 (passing tag which is part of another tag as well) - utils.AssertEqual(t, tags.FromSlug("a"), &tag1) - // case 3 - utils.AssertEqual(t, tags.FromSlug("a1"), &tag2) -} - -func TestFromIds(t *testing.T) { - var tags model.Tags - // creating tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "z", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "c", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - // case 1 (passing blank tagIDs) - tagIDs := []int{} - gotSlugs := tags.FromIds(tagIDs) - wantSlugs := model.Tags{} - utils.AssertEqual(t, gotSlugs, wantSlugs) - // case 2 (no matching tagIDs) - tagIDs = []int{100, 101} - gotSlugs = tags.FromIds(tagIDs) - wantSlugs = model.Tags{} - utils.AssertEqual(t, gotSlugs, wantSlugs) - // case 3 (two matching tagIDs) - tagIDs = []int{1, 3} - gotSlugs = tags.FromIds(tagIDs) - wantSlugs = model.Tags{&tag1, &tag3} - utils.AssertEqual(t, gotSlugs, wantSlugs) - // case 4 - tagIDs = []int{1, 4, 2, 3} - gotSlugs = tags.FromIds(tagIDs) - wantSlugs = model.Tags{&tag1, &tag4, &tag2, &tag3} - utils.AssertEqual(t, gotSlugs, wantSlugs) -} - -func TestIdsForGroup(t *testing.T) { - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - // case 1 (group with no such name) - utils.AssertEqual(t, tags.IdsForGroup("tag_group_NO"), []int{}) - // case 1 (group with multiple tags) - utils.AssertEqual(t, tags.IdsForGroup("tag_group1"), []int{1, 2, 3}) - // case 2 (group with single tag) - utils.AssertEqual(t, tags.IdsForGroup("tag_group2"), []int{4}) -} - -func TestBasicTags(t *testing.T) { - basicTags := model.BasicTags() - slugs := basicTags.Slugs() - want := "[current priority-urgent priority-medium priority-low repeat-annually repeat-monthly tips]" - utils.AssertEqual(t, slugs, want) -} - -func TestNotesByDueDate(t *testing.T) { - var notes []*model.Note - notes = append(notes, &model.Note{Text: "1", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: 1800000003}) - notes = append(notes, &model.Note{Text: "2", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}, CompleteBy: 1800000004}) - notes = append(notes, &model.Note{Text: "3", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}, CompleteBy: 1800000002}) - notes = append(notes, &model.Note{Text: "4", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}, CompleteBy: 1800000001}) - sort.Sort(model.NotesByDueDate(notes)) - var gotTexts []string - for _, value := range notes { - gotTexts = append(gotTexts, value.Text) - } - wantTexts := []string{"4", "3", "1", "2"} - utils.AssertEqual(t, gotTexts, wantTexts) -} - -func TestNoteStrings(t *testing.T) { - utils.Location = utils.UTCLocation() - comments := model.Comments{&model.Comment{Text: "c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters"}, &model.Comment{Text: "c2"}, &model.Comment{Text: "c3"}} - note := &model.Note{Text: "dummy text with \" and < characters", Comments: comments, Status: model.NoteStatus_Pending, Summary: "summary heading:\n- line 1\n- line 2", TagIds: []int{1, 2}, CompleteBy: 1609669235} - want := `[ | Text: dummy text with " and < characters - | Comments: - | : nil | c1: - | - line 1 - | - line 2 - | - line 3 with " and < characters - | : nil | c2 - | : nil | c3 - | Summary: summary heading: - | - line 1 - | - line 2 - | Status: pending - | Tags: [1 2] - | IsMain: false - | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC - | CreatedAt: nil - | UpdatedAt: nil -]` - text, _ := note.Strings() - utils.AssertEqual(t, text, want) -} - -func TestExternalText(t *testing.T) { - utils.Location = utils.UTCLocation() - comments := model.Comments{&model.Comment{Text: "c < 1"}, &model.Comment{Text: "c > 2"}, &model.Comment{Text: "c & \" 3"}} - note := &model.Note{Text: "dummy < > \" text", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669235} - var tags model.Tags - tags = append(tags, &model.Tag{Id: 0, Slug: "tag_0", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group2"}) - reminderData := &model.ReminderData{Tags: tags} - want := `Note Details: ------------------------------------------------- - | Text: dummy < > " text - | Comments: - | : nil | c < 1 - | : nil | c > 2 - | : nil | c & " 3 - | Summary: - | Status: pending - | Tags: - | : tag_1 - | : tag_2 - | IsMain: false - | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC - | CreatedAt: nil - | UpdatedAt: nil -` - text, _ := note.ExternalText(reminderData) - utils.AssertEqual(t, text, want) -} - -func TestSafeExtText(t *testing.T) { - utils.Location = utils.UTCLocation() - comments := model.Comments{&model.Comment{Text: "c < 1"}, &model.Comment{Text: "c > 2"}, &model.Comment{Text: "c & \" 3"}} - note := &model.Note{Text: "dummy < > \" text", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669235} - var tags model.Tags - tags = append(tags, &model.Tag{Id: 0, Slug: "tag_0", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group2"}) - reminderData := &model.ReminderData{Tags: tags} - want := `Note Details: ------------------------------------------------- - | Text: dummy < > " text - | Summary: - | Status: pending - | Tags: - | : tag_1 - | : tag_2 - | IsMain: false - | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC - | CreatedAt: nil - | UpdatedAt: nil -` - text, _ := note.SafeExtText(reminderData) - utils.AssertEqual(t, text, want) -} - -func TestSearchableText(t *testing.T) { - // case 1 - comments := model.Comments{&model.Comment{Text: "c1"}} - note := model.Note{Text: "a beautiful cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} - got, _ := note.SearchableText() - utils.AssertEqual(t, got, "| incidental | pending | ├ a beautiful cat ┤ [nil | c1]") - // case 2 - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} - note = model.Note{Text: "a cute dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 2}, CompleteBy: 1609669232} - got, _ = note.SearchableText() - utils.AssertEqual(t, got, "| incidental | done | ├ a cute dog ┤ [nil | c1, nil | foo bar, nil | c3]") - // case 3 - comments = model.Comments{} - note = model.Note{Text: "a cute dog", Comments: comments} - got, _ = note.SearchableText() - utils.AssertEqual(t, got, "| incidental | | ├ a cute dog ┤ [no-comments]") - // case 4 - comments = model.Comments{} - note = model.Note{Text: "first line\nsecondline\nthird line", Comments: comments} - got, _ = note.SearchableText() - utils.AssertEqual(t, got, "| incidental | | ├ first line NWL secondline NWL third line ┤ [no-comments]") - // case 5 - comments = model.Comments{&model.Comment{Text: "c1"}} - note = model.Note{Text: "a beautiful cat", Comments: comments, Status: model.NoteStatus_Suspended, TagIds: []int{1, 2}, CompleteBy: 1609669231} - got, _ = note.SearchableText() - utils.AssertEqual(t, got, "| incidental | suspended | ├ a beautiful cat ┤ [nil | c1]") -} - -func TestExternalTexts(t *testing.T) { - var notes model.Notes - // case 1 (no notes) - utils.AssertEqual(t, "[]", "[]") - // add notes - comments := model.Comments{&model.Comment{Text: "c1"}} - notes = append(notes, &model.Note{Text: "beautiful little cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231}) - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}, &model.Comment{Text: "baz"}} - notes = append(notes, &model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 2}, CompleteBy: 1609669232}) - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "f b"}, &model.Comment{Text: "c4"}, &model.Comment{Text: "b"}} - notes = append(notes, &model.Note{Text: "cbd", Comments: comments, Status: model.NoteStatus_Suspended, TagIds: []int{1}, CompleteBy: 1609669235}) - // case 2 - got := notes.ExternalTexts(0, 0, 0) - want := "[beautiful little cat {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" - utils.AssertEqual(t, got, want) - // case 3 - got = notes.ExternalTexts(5, 0, 0) - want = "[be... {R: -, C:01, S:P, D:03-Jan-21} cu... {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" - utils.AssertEqual(t, got, want) - // case 4 - got = notes.ExternalTexts(15, 0, 0) - want = "[beautiful li... {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" - utils.AssertEqual(t, got, want) - // case 5 - got = notes.ExternalTexts(25, 0, 0) - want = "[beautiful little cat {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" - utils.AssertEqual(t, got, want) -} - -func TestWithStatus(t *testing.T) { - var notes model.Notes - // case 1 (no notes) - utils.AssertEqual(t, notes.WithStatus(model.NoteStatus_Pending), model.Notes{}) - // add some notes - comments := model.Comments{&model.Comment{Text: "c1"}} - note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} - notes = append(notes, ¬e1) - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} - note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, CompleteBy: 1609669232} - notes = append(notes, ¬e2) - comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} - note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}, CompleteBy: 1609669233} - notes = append(notes, ¬e3) - // case 2 (with an invalid status) - utils.AssertEqual(t, notes.WithStatus("no-such-status"), model.Notes{}) - // case 3 (with valid status "pending") - got := notes.WithStatus(model.NoteStatus_Pending) - want := model.Notes{¬e1, ¬e3} - utils.AssertEqual(t, got, want) - // case 4 (with valid status "done") - got = notes.WithStatus(model.NoteStatus_Done) - want = model.Notes{¬e2} - utils.AssertEqual(t, got, want) -} - -func TestWithCompleteBy(t *testing.T) { - var notes model.Notes - // case 1 (no notes) - utils.AssertEqual(t, notes.WithCompleteBy(), model.Notes{}) - // add some notes - comments := model.Comments{&model.Comment{Text: "c1"}} - note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} - notes = append(notes, ¬e1) - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} - note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, CompleteBy: 1609669232} - notes = append(notes, ¬e2) - comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} - note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}} - notes = append(notes, ¬e3) - // case 3 (with only few notes to be filtered in) - got := notes.WithCompleteBy() - want := model.Notes{¬e1, ¬e2} - utils.AssertEqual(t, got, want) -} - -func TestOnlyMain(t *testing.T) { - var notes model.Notes - // case 1 (no notes) - utils.AssertEqual(t, notes.OnlyMain(), model.Notes{}) - // add some notes - comments := model.Comments{&model.Comment{Text: "c1"}} - note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} - notes = append(notes, ¬e1) - comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} - note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, IsMain: true, CompleteBy: 1609669232} - notes = append(notes, ¬e2) - comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} - note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}} - notes = append(notes, ¬e3) - // case 3 (with only few notes to be filtered in) - got := notes.OnlyMain() - want := model.Notes{¬e2} - utils.AssertEqual(t, got, want) -} - -func TestWithTagIdAndStatus(t *testing.T) { - // var tags model.Tags - var notes model.Notes - // case 1 (no notes) - utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Pending), model.Notes{}) - // creating tags - // tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - // tags = append(tags, &tag1) - // tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - // tags = append(tags, &tag2) - // tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - // tags = append(tags, &tag3) - // tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - // tags = append(tags, &tag4) - // create notes - note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - notes = append(notes, ¬e1) - note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} - notes = append(notes, ¬e2) - note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} - notes = append(notes, ¬e3) - note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} - notes = append(notes, ¬e4) - note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} - notes = append(notes, ¬e5) - note6 := model.Note{Text: "6", Status: model.NoteStatus_Suspended, TagIds: []int{1}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000006}} - notes = append(notes, ¬e6) - // case 2 - utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Pending), []*model.Note{¬e2}) - // case 3 - utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Done), []*model.Note{¬e3}) - // case 4 - utils.AssertEqual(t, notes.WithTagIdAndStatus(4, model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) - // case 5 - utils.AssertEqual(t, notes.WithTagIdAndStatus(1, model.NoteStatus_Done), []*model.Note{}) - // case 6 - utils.AssertEqual(t, notes.WithTagIdAndStatus(1, model.NoteStatus_Suspended), []*model.Note{¬e6}) -} - -func TestAddComment(t *testing.T) { - // create notes - note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // add comments - // case 1 - err := note1.AddComment("test comment 1") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, len(note1.Comments), 1) - utils.AssertEqual(t, strings.Contains(note1.Comments[0].Text, "test comment 1"), true) - // case 2 - err = note1.AddComment("test comment 2") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, len(note1.Comments), 2) - utils.AssertEqual(t, strings.Contains(note1.Comments[1].Text, "test comment 2"), true) - // case 3 - err = note1.AddComment("") - utils.AssertEqual(t, strings.Contains(err.Error(), "Note's comment text is empty"), true) - utils.AssertEqual(t, len(note1.Comments), 2) - utils.AssertEqual(t, strings.Contains(note1.Comments[1].Text, "test comment 2"), true) -} - -func TestUpdateText(t *testing.T) { - // create notes - note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // update text - // case 1 - err := note1.UpdateText("updated text 1") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.Text, "updated text 1") - // case 2 - err = note1.UpdateText("") - utils.AssertEqual(t, strings.Contains(err.Error(), "Note's text is empty"), true) - utils.AssertEqual(t, note1.Text, "updated text 1") -} - -func TestUpdateSummary(t *testing.T) { - // create notes - note1 := model.Note{Summary: "original summary", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // update summary - // case 1 - err := note1.UpdateSummary("updated summary 1") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.Summary, "updated summary 1") - // case 2 - err = note1.UpdateSummary("") - utils.AssertEqual(t, strings.Contains(err.Error(), "Note's summary is empty"), true) - utils.AssertEqual(t, note1.Summary, "updated summary 1") -} - -func TestUpdateCompleteBy(t *testing.T) { - // create notes - note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - utils.AssertEqual(t, note1.CompleteBy, 0) - // update complete_by - // case 1 - err := note1.UpdateCompleteBy("15-12-2021") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.CompleteBy, 1639526400) // Wed Dec 15 2021 00:00:00 GMT+0000 - // case 2 - err = note1.UpdateCompleteBy("31-12-2022") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.CompleteBy, 1672444800) // Sat Dec 31 2022 00:00:00 GMT+0000 - // case 3 - err = note1.UpdateCompleteBy("nil") - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.CompleteBy, 0) -} - -func TestUpdateTags(t *testing.T) { - // create notes - note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // update TagIds - // case 1 - tagIds := []int{2, 5} - err := note1.UpdateTags(tagIds) - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.TagIds, tagIds) - // case 2 - tagIds = []int{} - err = note1.UpdateTags(tagIds) - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.TagIds, tagIds) -} - -func TestUpdateStatus(t *testing.T) { - // create notes - note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // update TagIds - // case 1 - err := note1.UpdateStatus(model.NoteStatus_Done, []int{1, 2, 3}) - utils.AssertEqual(t, err, errors.New("Note is part of a \"repeat\" group")) - utils.AssertEqual(t, note1.Status, model.NoteStatus_Pending) - // case 2 - err = note1.UpdateStatus(model.NoteStatus_Done, []int{5, 6, 7}) - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.Status, model.NoteStatus_Done) - // case 3 - err = note1.UpdateStatus(model.NoteStatus_Pending, []int{5, 6, 7}) - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, note1.Status, model.NoteStatus_Pending) -} - -func TestRepeatType(t *testing.T) { - repeatAnnuallyTagId := 3 - repeatMonthlyTagId := 4 - // create notes - note1 := model.Note{Text: "original text1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - note2 := model.Note{Text: "original text2", Status: model.NoteStatus_Done, TagIds: []int{3, 5}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - note3 := model.Note{Text: "original text3", Status: model.NoteStatus_Done, TagIds: []int{2, 6}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // assert repeat type - utils.AssertEqual(t, note1.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "M") - utils.AssertEqual(t, note2.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "A") - utils.AssertEqual(t, note3.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "-") - utils.AssertEqual(t, note3.RepeatType(0, 0), "-") -} - -func TestToggleMainFlag(t *testing.T) { - // create notes - note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - // update TagIds - // case 1 - originalPriority := note1.IsMain - err := note1.ToggleMainFlag() - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, originalPriority != note1.IsMain, true) - // case 2 - originalPriority = note1.IsMain - err = note1.ToggleMainFlag() - utils.AssertEqual(t, err, nil) - utils.AssertEqual(t, originalPriority != note1.IsMain, true) -} - -func TestGoogleCalendarEvent(t *testing.T) { - tagger := TestTagger{} - var tests = []struct { - name string // has to be string - note model.Note - inputRATID int - inputRMTID int - inputTimezone string - inputTagger model.Tagger - want *gc.Event - wantErr error - wantedErr bool - }{ - { - name: "general case 1", - note: model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}}, - inputRATID: 1, - inputRMTID: 3, - inputTimezone: "Australia/Melbourne", - inputTagger: tagger, - want: &gc.Event{ - Summary: "[reminder] original text", - }, - wantedErr: false, - }, - } - for position, subtest := range tests { - t.Run(subtest.name, func(t *testing.T) { - got, err := subtest.note.GoogleCalendarEvent(subtest.inputRATID, subtest.inputRMTID, subtest.inputTimezone, tagger) - if (err != nil) != subtest.wantedErr { - t.Fatalf("GoogleCalendarEvent case %q (position=%d) with input <%+v> returns error <%v>; wantError <%v>", subtest.name, position, subtest.note, err, subtest.wantErr) - } - if got.Summary != subtest.want.Summary { - t.Errorf("GoogleCalendarEvent case %q (position=%d) with input <%+v> returns <%+v>; want <%+v>", subtest.name, position, subtest.note, got, subtest.want) - } - }) - } -} - -func TestSortedTagsSlug(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - // case 1 (no tags) - utils.AssertEqual(t, reminderData.SortedTagSlugs(), []string{}) - // case 2 (has couple of tags) - tags = append(tags, &model.Tag{Id: 1, Slug: "a", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 2, Slug: "z", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 3, Slug: "c", Group: "tag_group1"}) - tags = append(tags, &model.Tag{Id: 4, Slug: "b", Group: "tag_group2"}) - reminderData.Tags = tags - gotSlugs := reminderData.SortedTagSlugs() - wantSlugs := []string{"a", "b", "c", "z"} - utils.AssertEqual(t, gotSlugs, wantSlugs) -} - -func TestTagsFromIds(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "z", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "c", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - // case 1 - tagIDs := []int{1, 3} - gotSlugs := reminderData.TagsFromIds(tagIDs) - wantSlugs := model.Tags{&tag1, &tag3}.Slugs() - utils.AssertEqual(t, gotSlugs, wantSlugs) - // case 2 - tagIDs = []int{} - gotSlugs = reminderData.TagsFromIds(tagIDs) - wantSlugs = model.Tags{}.Slugs() - utils.AssertEqual(t, gotSlugs, wantSlugs) - // case 3 - tagIDs = []int{1, 4, 2, 3} - gotSlugs = reminderData.TagsFromIds(tagIDs) - wantSlugs = model.Tags{&tag1, &tag4, &tag2, &tag3}.Slugs() - utils.AssertEqual(t, gotSlugs, wantSlugs) -} - -func TestTagFromSlug(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - // case 1 - utils.AssertEqual(t, reminderData.TagFromSlug("a"), &tag1) - // case 2 - utils.AssertEqual(t, reminderData.TagFromSlug("a1"), &tag2) - // case 3 - utils.AssertEqual(t, reminderData.TagFromSlug("no_slug"), nil) -} - -func TestTagIdsForGroup(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - // case 1 - utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group1"), []int{1, 2, 3}) - // case 2 - utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group2"), []int{4}) - // case 3 - utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group_NO"), []int{}) -} - -/* -func TestNextPossibleTagId(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - utils.AssertEqual(t, reminderData.nextPossibleTagId(), 4) -} -*/ - -func TestFindNotesByTagId(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - // create notes - var notes model.Notes - note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - notes = append(notes, ¬e1) - note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} - notes = append(notes, ¬e2) - note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} - notes = append(notes, ¬e3) - note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} - notes = append(notes, ¬e4) - note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} - notes = append(notes, ¬e5) - reminderData.Notes = notes - // searching notes - // case 1 - utils.AssertEqual(t, reminderData.FindNotesByTagId(2, model.NoteStatus_Pending), []*model.Note{¬e2}) - // case 2 - utils.AssertEqual(t, reminderData.FindNotesByTagId(2, model.NoteStatus_Done), []*model.Note{¬e3}) - // case 3 - utils.AssertEqual(t, reminderData.FindNotesByTagId(4, model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) - // case 4 - utils.AssertEqual(t, reminderData.FindNotesByTagId(1, model.NoteStatus_Done), []*model.Note{}) -} - -func TestFindNotesByTagSlug(t *testing.T) { - reminderData := model.ReminderData{ - User: &model.User{Name: "Test User", EmailId: "user@test.com"}, - Notes: []*model.Note{}, - Tags: model.Tags{}, - } - // creating tags - var tags model.Tags - tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} - tags = append(tags, &tag1) - tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} - tags = append(tags, &tag2) - tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} - tags = append(tags, &tag3) - tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} - tags = append(tags, &tag4) - reminderData.Tags = tags - // create notes - var notes model.Notes - note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} - notes = append(notes, ¬e1) - note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} - notes = append(notes, ¬e2) - note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} - notes = append(notes, ¬e3) - note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} - notes = append(notes, ¬e4) - note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} - notes = append(notes, ¬e5) - reminderData.Notes = notes - // searching notes - // case 1 - utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a1", model.NoteStatus_Pending), []*model.Note{¬e2}) - // case 2 - utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a1", model.NoteStatus_Done), []*model.Note{¬e3}) - // case 3 - utils.AssertEqual(t, reminderData.FindNotesByTagSlug("b", model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) - // case 4 - utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a", model.NoteStatus_Done), []*model.Note{}) -} - -func TestNewTagRegistration(t *testing.T) { - dataFilePath := path.Join("..", "..", "test", "test_data_file.json") - reminderData, _ := model.ReadDataFile(dataFilePath, false) - utils.AssertEqual(t, len(reminderData.Tags), 5) -} - -func TestNewTag(t *testing.T) { - dummySlug := "test_tag_slug" - dummyGroup := "test_tag_group" - tag, _ := model.NewTag(10, dummySlug, dummyGroup) - want := &model.Tag{ - Id: 10, - Slug: dummySlug, - Group: dummyGroup, - } - utils.AssertEqual(t, tag, want) -} -func TestNewNote(t *testing.T) { - tagIDs := []int{1, 3, 5} - dummyText := "a random note text" - note, _ := model.NewNote(tagIDs, dummyText) - want := &model.Note{ - Text: dummyText, - TagIds: tagIDs, - Status: note.Status, - BaseStruct: model.BaseStruct{UpdatedAt: note.UpdatedAt, CreatedAt: note.CreatedAt}, - } - utils.AssertEqual(t, note, want) -} - -func TestNotes(t *testing.T) { - var notes []*model.Note - notes = append(notes, &model.Note{Text: "1", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}}) - notes = append(notes, &model.Note{Text: "2", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}}) - notes = append(notes, &model.Note{Text: "3", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}}) - notes = append(notes, &model.Note{Text: "4", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}}) - sort.Sort(model.Notes(notes)) - var gotTexts []string - for _, value := range notes { - gotTexts = append(gotTexts, value.Text) - } - wantTexts := []string{"2", "3", "4", "1"} - utils.AssertEqual(t, gotTexts, wantTexts) -} - -func TestMakeSureFileExists(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - - // make sure file doesn't exists already - _, err := os.Stat(dataFilePath) - utils.AssertEqual(t, err != nil, true) - utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), true) - // attempt to create the file and required dirs, when the file doesn't exist already - _ = model.MakeSureFileExists(dataFilePath, false) - // prove that the file was created - stats, err := os.Stat(dataFilePath) - utils.AssertEqual(t, err != nil, false) - utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), false) - - // make sure that the existing file is not replaced - modificationTime := stats.ModTime() - // attempt to create the file and required dirs, when the file does exist already - time.Sleep(10 * time.Millisecond) - _ = model.MakeSureFileExists(dataFilePath, false) - utils.AssertEqual(t, err != nil, false) - utils.AssertEqual(t, errors.Is(err, fs.ErrNotExist), false) - stats, _ = os.Stat(dataFilePath) - newModificationTime := stats.ModTime() - utils.AssertEqual(t, newModificationTime == modificationTime, true) -} - -func TestReadDataFile(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - // attempt to read file and parse it - reminderData, _ := model.ReadDataFile(dataFilePath, false) - utils.AssertEqual(t, reminderData.UpdatedAt > 0, true) -} -func TestCreateDataFile(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - reminderData, _ := model.ReadDataFile(dataFilePath, false) - // old_updated_at := reminderData.UpdatedAt - testUser := model.User{Name: "Test User", EmailId: "user@test.com"} - reminderData.User = &testUser - _ = reminderData.CreateDataFile("") - remiderDataRe, _ := model.ReadDataFile(dataFilePath, false) - // utils.AssertEqual(t, remiderDataRe.UpdatedAt > old_updated_at, true) - utils.AssertEqual(t, remiderDataRe.User.EmailId == testUser.EmailId, true) -} - -func TestUpdateDataFile(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - reminderData, _ := model.ReadDataFile(dataFilePath, false) - // old_updated_at := reminderData.UpdatedAt - testUser := model.User{Name: "Test User", EmailId: "user@test.com"} - reminderData.User = &testUser - _ = reminderData.UpdateDataFile("") - remiderDataRe, _ := model.ReadDataFile(dataFilePath, false) - // utils.AssertEqual(t, remiderDataRe.UpdatedAt > old_updated_at, true) - utils.AssertEqual(t, remiderDataRe.User.EmailId == testUser.EmailId, true) -} - -func TestRegisterBasicTags(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - reminderData, _ := model.ReadDataFile(dataFilePath, false) - // register basic tags - _ = reminderData.RegisterBasicTags() - utils.AssertEqual(t, len(reminderData.Tags), 7) -} - -func TestNotesApproachingDueDate(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - reminderData, _ := model.ReadDataFile(dataFilePath, false) - // register basic tags - _ = reminderData.RegisterBasicTags() - // get current time - currentTime := utils.CurrentUnixTimestamp() - // register notes - // for reference: here is the list of basic tags - // {"slug": "current", "group": ""}, - // {"slug": "priority-urgent", "group": "priority"}, - // {"slug": "priority-medium", "group": "priority"}, - // {"slug": "priority-low", "group": "priority"}, - // {"slug": "repeat-annually", "group": "repeat"}, - // {"slug": "repeat-monthly", "group": "repeat"}, - // {"slug": "tips", "group": "tips"}} - currentTagId := reminderData.TagFromSlug("current").Id - repeatAnnuallyTagId := reminderData.TagFromSlug("repeat-annually").Id - repeatMonthlyTagId := reminderData.TagFromSlug("repeat-monthly").Id - var notes model.Notes - // non-repeat done notes - notes = append(notes, &model.Note{Text: "NRD01a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) - notes = append(notes, &model.Note{Text: "NRD02a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) - notes = append(notes, &model.Note{Text: "NRD03a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) - // repeat-annually done notes - notes = append(notes, &model.Note{Text: "RAD01a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) - notes = append(notes, &model.Note{Text: "RAD02a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) - notes = append(notes, &model.Note{Text: "RAD03a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) - // repeat-monthally done notes - notes = append(notes, &model.Note{Text: "RMD01a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) - notes = append(notes, &model.Note{Text: "RMD02a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) - notes = append(notes, &model.Note{Text: "RMD03a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) - // non-repeat pending notes - notes = append(notes, &model.Note{Text: "NRP01a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) // expected - notes = append(notes, &model.Note{Text: "NRP02a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*24*3600 - 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP02b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*24*3600 + 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP03a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected - notes = append(notes, &model.Note{Text: "NRP04a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 - 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP04b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 + 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP05a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600 - 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP05b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600 + 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP06a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600 - 3600}) // expected - notes = append(notes, &model.Note{Text: "NRP06b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600 + 3600}) - notes = append(notes, &model.Note{Text: "NRP07a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) - // repeat-annually pending notes - notes = append(notes, &model.Note{Text: "RAP01", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) - notes = append(notes, &model.Note{Text: "RAP02", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RAP03", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime}) // expected - notes = append(notes, &model.Note{Text: "RAP04", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RAP05", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RAP06", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) - notes = append(notes, &model.Note{Text: "RAP07", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600 - 2*365}) - notes = append(notes, &model.Note{Text: "RAP08", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600 - 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP09", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP10", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600 - 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP11", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 - 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP12", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600 - 2*365}) - notes = append(notes, &model.Note{Text: "RAP13", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600 + 2*365}) - notes = append(notes, &model.Note{Text: "RAP14", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600 + 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP15", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP16", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600 + 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP17", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 + 2*365}) // expected - notes = append(notes, &model.Note{Text: "RAP18", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600 + 2*365}) - // repeat-monthly pending notes - notes = append(notes, &model.Note{Text: "RMP01", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) - notes = append(notes, &model.Note{Text: "RMP02", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RMP03", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime}) // expected - notes = append(notes, &model.Note{Text: "RMP04", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RMP05", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600}) // expected - notes = append(notes, &model.Note{Text: "RMP06", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600}) - notes = append(notes, &model.Note{Text: "RMP07", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) - reminderData.Notes = notes - // get urgent notes - urgentNotes := reminderData.NotesApprachingDueDate("default") - var urgentNotesText []string - for _, note := range urgentNotes { - urgentNotesText = append(urgentNotesText, note.Text) - } - expectNotesText := []string{ - "NRP01a", "NRP02a", "NRP02b", "NRP03a", "NRP04a", "NRP04b", "NRP05a", "NRP05b", "NRP06a", - "RAP02", "RAP03", "RAP04", "RAP05", "RAP08", "RAP09", "RAP10", "RAP11", "RAP14", "RAP15", "RAP16", "RAP17", - "RMP02", "RMP03", - } - t.Logf("Received Texts: %v", urgentNotesText) - t.Logf("Expected Texts: %v", expectNotesText) - utils.SkipCI(t) - utils.AssertEqual(t, urgentNotesText, expectNotesText) - // [NRP01a NRP02a NRP02b NRP03a NRP04a NRP04b NRP05a NRP05b NRP06a RAP02 RAP03 RAP04 RAP05 RAP08 RAP09 RAP10 RAP11 RAP14 RAP15 RAP16 RAP17 RMP03] - // [NRP01a NRP02a NRP02b NRP03a NRP04a NRP04b NRP05a NRP05b NRP06a RAP02 RAP03 RAP04 RAP05 RAP08 RAP09 RAP10 RAP11 RAP14 RAP15 RAP16 RAP17 RMP02 RMP03]} -} - -func TestPrintStats(t *testing.T) { - var dataFilePath = "temp_test_dir/mydata.json" - // make sure temporary files and dirs are removed at the end of the test - defer os.RemoveAll(path.Dir(dataFilePath)) - // create the file and required dirs - _ = model.MakeSureFileExists(dataFilePath, false) - reminderData, _ := model.ReadDataFile(dataFilePath, false) - // register basic tags - _ = reminderData.RegisterBasicTags() - got, _ := reminderData.Stats() - want := ` -Stats of "temp_test_dir/mydata.json": - - Number of Tags: 7 - - Pending Notes: 0/0 - - Suspended Notes: 0 - - Done Notes: 0 -` - utils.AssertEqual(t, got, want) -} diff --git a/internal/model/note.go b/internal/model/note.go index 79319e6..b127617 100644 --- a/internal/model/note.go +++ b/internal/model/note.go @@ -58,10 +58,7 @@ func (note *Note) Strings() ([]string, error) { // allocating 10 members before hand, considering there will be around 10 status fields strs := make([]string, 0, 10) strs = append(strs, printNoteField("Text", note.Text)) - comments, err := note.Comments.Strings() - if err != nil { - return nil, err - } + comments := note.Comments.Strings() strs = append(strs, printNoteField("Comments", comments)) strs = append(strs, printNoteField("Summary", note.Summary)) strs = append(strs, printNoteField("Status", note.Status)) @@ -121,10 +118,7 @@ func (note *Note) SearchableText() (string, error) { if len(note.Comments) == 0 { commentsText = append(commentsText, "no-comments") } else { - text, err := note.Comments.Strings() - if err != nil { - return "", nil - } + text := note.Comments.Strings() commentsText = append(commentsText, strings.Join(text, ", ")) } commentsText = append(commentsText, "]") diff --git a/internal/model/note_test.go b/internal/model/note_test.go new file mode 100644 index 0000000..383c462 --- /dev/null +++ b/internal/model/note_test.go @@ -0,0 +1,290 @@ +package model_test + +import ( + "errors" + "strings" + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" + gc "google.golang.org/api/calendar/v3" +) + +func TestNoteStrings(t *testing.T) { + utils.Location = utils.UTCLocation() + comments := model.Comments{&model.Comment{Text: "c1:\n- line 1\n\n- line 2\n- line 3 with \" and < characters"}, &model.Comment{Text: "c2"}, &model.Comment{Text: "c3"}} + note := &model.Note{Text: "dummy text with \" and < characters", Comments: comments, Status: model.NoteStatus_Pending, Summary: "summary heading:\n- line 1\n- line 2", TagIds: []int{1, 2}, CompleteBy: 1609669235} + want := `[ | Text: dummy text with " and < characters + | Comments: + | : nil | c1: + | - line 1 + | - line 2 + | - line 3 with " and < characters + | : nil | c2 + | : nil | c3 + | Summary: summary heading: + | - line 1 + | - line 2 + | Status: pending + | Tags: [1 2] + | IsMain: false + | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC + | CreatedAt: nil + | UpdatedAt: nil +]` + text, _ := note.Strings() + utils.AssertEqual(t, text, want) +} + +func TestNoteExternalText(t *testing.T) { + utils.Location = utils.UTCLocation() + comments := model.Comments{&model.Comment{Text: "c < 1"}, &model.Comment{Text: "c > 2"}, &model.Comment{Text: "c & \" 3"}} + note := &model.Note{Text: "dummy < > \" text", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669235} + var tags model.Tags + tags = append(tags, &model.Tag{Id: 0, Slug: "tag_0", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group2"}) + reminderData := &model.ReminderData{Tags: tags} + want := `Note Details: ------------------------------------------------- + | Text: dummy < > " text + | Comments: + | : nil | c < 1 + | : nil | c > 2 + | : nil | c & " 3 + | Summary: + | Status: pending + | Tags: + | : tag_1 + | : tag_2 + | IsMain: false + | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC + | CreatedAt: nil + | UpdatedAt: nil +` + text, _ := note.ExternalText(reminderData) + utils.AssertEqual(t, text, want) +} + +func TestNoteSafeExtText(t *testing.T) { + utils.Location = utils.UTCLocation() + comments := model.Comments{&model.Comment{Text: "c < 1"}, &model.Comment{Text: "c > 2"}, &model.Comment{Text: "c & \" 3"}} + note := &model.Note{Text: "dummy < > \" text", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669235} + var tags model.Tags + tags = append(tags, &model.Tag{Id: 0, Slug: "tag_0", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group2"}) + reminderData := &model.ReminderData{Tags: tags} + want := `Note Details: ------------------------------------------------- + | Text: dummy < > " text + | Summary: + | Status: pending + | Tags: + | : tag_1 + | : tag_2 + | IsMain: false + | CompleteBy: Sunday, 03-Jan-21 10:20:35 UTC + | CreatedAt: nil + | UpdatedAt: nil +` + text, _ := note.SafeExtText(reminderData) + utils.AssertEqual(t, text, want) +} + +func TestNoteSearchableText(t *testing.T) { + // case 1 + comments := model.Comments{&model.Comment{Text: "c1"}} + note := model.Note{Text: "a beautiful cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} + got, _ := note.SearchableText() + utils.AssertEqual(t, got, "| incidental | pending | ├ a beautiful cat ┤ [nil | c1]") + // case 2 + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} + note = model.Note{Text: "a cute dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 2}, CompleteBy: 1609669232} + got, _ = note.SearchableText() + utils.AssertEqual(t, got, "| incidental | done | ├ a cute dog ┤ [nil | c1, nil | foo bar, nil | c3]") + // case 3 + comments = model.Comments{} + note = model.Note{Text: "a cute dog", Comments: comments} + got, _ = note.SearchableText() + utils.AssertEqual(t, got, "| incidental | | ├ a cute dog ┤ [no-comments]") + // case 4 + comments = model.Comments{} + note = model.Note{Text: "first line\nsecondline\nthird line", Comments: comments} + got, _ = note.SearchableText() + utils.AssertEqual(t, got, "| incidental | | ├ first line NWL secondline NWL third line ┤ [no-comments]") + // case 5 + comments = model.Comments{&model.Comment{Text: "c1"}} + note = model.Note{Text: "a beautiful cat", Comments: comments, Status: model.NoteStatus_Suspended, TagIds: []int{1, 2}, CompleteBy: 1609669231} + got, _ = note.SearchableText() + utils.AssertEqual(t, got, "| incidental | suspended | ├ a beautiful cat ┤ [nil | c1]") +} + +func TestNoteAddComment(t *testing.T) { + // create notes + note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // add comments + // case 1 + err := note1.AddComment("test comment 1") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, len(note1.Comments), 1) + utils.AssertEqual(t, strings.Contains(note1.Comments[0].Text, "test comment 1"), true) + // case 2 + err = note1.AddComment("test comment 2") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, len(note1.Comments), 2) + utils.AssertEqual(t, strings.Contains(note1.Comments[1].Text, "test comment 2"), true) + // case 3 + err = note1.AddComment("") + utils.AssertEqual(t, strings.Contains(err.Error(), "Note's comment text is empty"), true) + utils.AssertEqual(t, len(note1.Comments), 2) + utils.AssertEqual(t, strings.Contains(note1.Comments[1].Text, "test comment 2"), true) +} + +func TestNoteUpdateTags(t *testing.T) { + // create notes + note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // update TagIds + // case 1 + tagIds := []int{2, 5} + err := note1.UpdateTags(tagIds) + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.TagIds, tagIds) + // case 2 + tagIds = []int{} + err = note1.UpdateTags(tagIds) + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.TagIds, tagIds) +} + +func TestNoteUpdateStatus(t *testing.T) { + // create notes + note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // update TagIds + // case 1 + err := note1.UpdateStatus(model.NoteStatus_Done, []int{1, 2, 3}) + utils.AssertEqual(t, err, errors.New("Note is part of a \"repeat\" group")) + utils.AssertEqual(t, note1.Status, model.NoteStatus_Pending) + // case 2 + err = note1.UpdateStatus(model.NoteStatus_Done, []int{5, 6, 7}) + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.Status, model.NoteStatus_Done) + // case 3 + err = note1.UpdateStatus(model.NoteStatus_Pending, []int{5, 6, 7}) + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.Status, model.NoteStatus_Pending) +} + +func TestNoteUpdateText(t *testing.T) { + // create notes + note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // update text + // case 1 + err := note1.UpdateText("updated text 1") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.Text, "updated text 1") + // case 2 + err = note1.UpdateText("") + utils.AssertEqual(t, strings.Contains(err.Error(), "Note's text is empty"), true) + utils.AssertEqual(t, note1.Text, "updated text 1") +} + +func TestNoteUpdateSummary(t *testing.T) { + // create notes + note1 := model.Note{Summary: "original summary", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // update summary + // case 1 + err := note1.UpdateSummary("updated summary 1") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.Summary, "updated summary 1") + // case 2 + err = note1.UpdateSummary("") + utils.AssertEqual(t, strings.Contains(err.Error(), "Note's summary is empty"), true) + utils.AssertEqual(t, note1.Summary, "updated summary 1") +} + +func TestNoteUpdateCompleteBy(t *testing.T) { + // create notes + note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + utils.AssertEqual(t, note1.CompleteBy, 0) + // update complete_by + // case 1 + err := note1.UpdateCompleteBy("15-12-2021") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.CompleteBy, 1639526400) // Wed Dec 15 2021 00:00:00 GMT+0000 + // case 2 + err = note1.UpdateCompleteBy("31-12-2022") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.CompleteBy, 1672444800) // Sat Dec 31 2022 00:00:00 GMT+0000 + // case 3 + err = note1.UpdateCompleteBy("nil") + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, note1.CompleteBy, 0) +} + +func TestNoteRepeatType(t *testing.T) { + repeatAnnuallyTagId := 3 + repeatMonthlyTagId := 4 + // create notes + note1 := model.Note{Text: "original text1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + note2 := model.Note{Text: "original text2", Status: model.NoteStatus_Done, TagIds: []int{3, 5}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + note3 := model.Note{Text: "original text3", Status: model.NoteStatus_Done, TagIds: []int{2, 6}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // assert repeat type + utils.AssertEqual(t, note1.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "M") + utils.AssertEqual(t, note2.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "A") + utils.AssertEqual(t, note3.RepeatType(repeatAnnuallyTagId, repeatMonthlyTagId), "-") + utils.AssertEqual(t, note3.RepeatType(0, 0), "-") +} + +func TestNoteToggleMainFlag(t *testing.T) { + // create notes + note1 := model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + // update TagIds + // case 1 + originalPriority := note1.IsMain + err := note1.ToggleMainFlag() + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, originalPriority != note1.IsMain, true) + // case 2 + originalPriority = note1.IsMain + err = note1.ToggleMainFlag() + utils.AssertEqual(t, err, nil) + utils.AssertEqual(t, originalPriority != note1.IsMain, true) +} + +func TestGoogleCalendarEvent(t *testing.T) { + tagger := TestTagger{} + var tests = []struct { + name string // has to be string + note model.Note + inputRATID int + inputRMTID int + inputTimezone string + inputTagger model.Tagger + want *gc.Event + wantErr error + wantedErr bool + }{ + { + name: "general case 1", + note: model.Note{Text: "original text", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}}, + inputRATID: 1, + inputRMTID: 3, + inputTimezone: "Australia/Melbourne", + inputTagger: tagger, + want: &gc.Event{ + Summary: "[reminder] original text", + }, + wantedErr: false, + }, + } + for position, subtest := range tests { + t.Run(subtest.name, func(t *testing.T) { + got, err := subtest.note.GoogleCalendarEvent(subtest.inputRATID, subtest.inputRMTID, subtest.inputTimezone, tagger) + if (err != nil) != subtest.wantedErr { + t.Fatalf("GoogleCalendarEvent case %q (position=%d) with input <%+v> returns error <%v>; wantError <%v>", subtest.name, position, subtest.note, err, subtest.wantErr) + } + if got.Summary != subtest.want.Summary { + t.Errorf("GoogleCalendarEvent case %q (position=%d) with input <%+v> returns <%+v>; want <%+v>", subtest.name, position, subtest.note, got, subtest.want) + } + }) + } +} diff --git a/internal/model/notes_by_due_date_test.go b/internal/model/notes_by_due_date_test.go new file mode 100644 index 0000000..26a201f --- /dev/null +++ b/internal/model/notes_by_due_date_test.go @@ -0,0 +1,24 @@ +package model_test + +import ( + "sort" + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestNotesByDueDate(t *testing.T) { + var notes []*model.Note + notes = append(notes, &model.Note{Text: "1", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: 1800000003}) + notes = append(notes, &model.Note{Text: "2", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}, CompleteBy: 1800000004}) + notes = append(notes, &model.Note{Text: "3", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}, CompleteBy: 1800000002}) + notes = append(notes, &model.Note{Text: "4", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}, CompleteBy: 1800000001}) + sort.Sort(model.NotesByDueDate(notes)) + var gotTexts []string + for _, value := range notes { + gotTexts = append(gotTexts, value.Text) + } + wantTexts := []string{"4", "3", "1", "2"} + utils.AssertEqual(t, gotTexts, wantTexts) +} diff --git a/internal/model/notes_test.go b/internal/model/notes_test.go new file mode 100644 index 0000000..48bed2a --- /dev/null +++ b/internal/model/notes_test.go @@ -0,0 +1,142 @@ +package model_test + +import ( + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestNotesExternalTexts(t *testing.T) { + var notes model.Notes + // case 1 (no notes) + utils.AssertEqual(t, "[]", "[]") + // add notes + comments := model.Comments{&model.Comment{Text: "c1"}} + notes = append(notes, &model.Note{Text: "beautiful little cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231}) + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}, &model.Comment{Text: "baz"}} + notes = append(notes, &model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 2}, CompleteBy: 1609669232}) + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "f b"}, &model.Comment{Text: "c4"}, &model.Comment{Text: "b"}} + notes = append(notes, &model.Note{Text: "cbd", Comments: comments, Status: model.NoteStatus_Suspended, TagIds: []int{1}, CompleteBy: 1609669235}) + // case 2 + got := notes.ExternalTexts(0, 0, 0) + want := "[beautiful little cat {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" + utils.AssertEqual(t, got, want) + // case 3 + got = notes.ExternalTexts(5, 0, 0) + want = "[be... {R: -, C:01, S:P, D:03-Jan-21} cu... {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" + utils.AssertEqual(t, got, want) + // case 4 + got = notes.ExternalTexts(15, 0, 0) + want = "[beautiful li... {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" + utils.AssertEqual(t, got, want) + // case 5 + got = notes.ExternalTexts(25, 0, 0) + want = "[beautiful little cat {R: -, C:01, S:P, D:03-Jan-21} cute brown dog {R: -, C:04, S:D, D:03-Jan-21} cbd {R: -, C:04, S:S, D:03-Jan-21}]" + utils.AssertEqual(t, got, want) +} + +func TestNotesWithStatus(t *testing.T) { + var notes model.Notes + // case 1 (no notes) + utils.AssertEqual(t, notes.WithStatus(model.NoteStatus_Pending), model.Notes{}) + // add some notes + comments := model.Comments{&model.Comment{Text: "c1"}} + note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} + notes = append(notes, ¬e1) + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} + note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, CompleteBy: 1609669232} + notes = append(notes, ¬e2) + comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} + note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}, CompleteBy: 1609669233} + notes = append(notes, ¬e3) + // case 2 (with an invalid status) + utils.AssertEqual(t, notes.WithStatus("no-such-status"), model.Notes{}) + // case 3 (with valid status "pending") + got := notes.WithStatus(model.NoteStatus_Pending) + want := model.Notes{¬e1, ¬e3} + utils.AssertEqual(t, got, want) + // case 4 (with valid status "done") + got = notes.WithStatus(model.NoteStatus_Done) + want = model.Notes{¬e2} + utils.AssertEqual(t, got, want) +} + +func TestNotesWithCompleteBy(t *testing.T) { + var notes model.Notes + // case 1 (no notes) + utils.AssertEqual(t, notes.WithCompleteBy(), model.Notes{}) + // add some notes + comments := model.Comments{&model.Comment{Text: "c1"}} + note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} + notes = append(notes, ¬e1) + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} + note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, CompleteBy: 1609669232} + notes = append(notes, ¬e2) + comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} + note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}} + notes = append(notes, ¬e3) + // case 3 (with only few notes to be filtered in) + got := notes.WithCompleteBy() + want := model.Notes{¬e1, ¬e2} + utils.AssertEqual(t, got, want) +} + +func TestNotesOnlyMain(t *testing.T) { + var notes model.Notes + // case 1 (no notes) + utils.AssertEqual(t, notes.OnlyMain(), model.Notes{}) + // add some notes + comments := model.Comments{&model.Comment{Text: "c1"}} + note1 := model.Note{Text: "big fat cat", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1, 2}, CompleteBy: 1609669231} + notes = append(notes, ¬e1) + comments = model.Comments{&model.Comment{Text: "c1"}, &model.Comment{Text: "foo bar"}} + note2 := model.Note{Text: "cute brown dog", Comments: comments, Status: model.NoteStatus_Done, TagIds: []int{1, 3}, IsMain: true, CompleteBy: 1609669232} + notes = append(notes, ¬e2) + comments = model.Comments{&model.Comment{Text: "foo bar"}, &model.Comment{Text: "c3"}} + note3 := model.Note{Text: "little hamster", Comments: comments, Status: model.NoteStatus_Pending, TagIds: []int{1}} + notes = append(notes, ¬e3) + // case 3 (with only few notes to be filtered in) + got := notes.OnlyMain() + want := model.Notes{¬e2} + utils.AssertEqual(t, got, want) +} + +func TestNotesWithTagIdAndStatus(t *testing.T) { + // var tags model.Tags + var notes model.Notes + // case 1 (no notes) + utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Pending), model.Notes{}) + // creating tags + // tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + // tags = append(tags, &tag1) + // tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + // tags = append(tags, &tag2) + // tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + // tags = append(tags, &tag3) + // tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + // tags = append(tags, &tag4) + // create notes + note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + notes = append(notes, ¬e1) + note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} + notes = append(notes, ¬e2) + note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} + notes = append(notes, ¬e3) + note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} + notes = append(notes, ¬e4) + note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} + notes = append(notes, ¬e5) + note6 := model.Note{Text: "6", Status: model.NoteStatus_Suspended, TagIds: []int{1}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000006}} + notes = append(notes, ¬e6) + // case 2 + utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Pending), []*model.Note{¬e2}) + // case 3 + utils.AssertEqual(t, notes.WithTagIdAndStatus(2, model.NoteStatus_Done), []*model.Note{¬e3}) + // case 4 + utils.AssertEqual(t, notes.WithTagIdAndStatus(4, model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) + // case 5 + utils.AssertEqual(t, notes.WithTagIdAndStatus(1, model.NoteStatus_Done), []*model.Note{}) + // case 6 + utils.AssertEqual(t, notes.WithTagIdAndStatus(1, model.NoteStatus_Suspended), []*model.Note{¬e6}) +} diff --git a/internal/model/reminder_data.go b/internal/model/reminder_data.go index 0a84474..4298cdc 100644 --- a/internal/model/reminder_data.go +++ b/internal/model/reminder_data.go @@ -94,7 +94,7 @@ func (rd *ReminderData) SyncCalendar(calOptions *calendar.Options) error { // Add events to Cloud Calendar logger.Info("Fetching events to be Synced.") - newEvents, err := rd.GoogleCalendarEvents(timeZone, rd) + newEvents, err := rd.GoogleCalendarEvents(timeZone) if err != nil { return err } @@ -112,7 +112,7 @@ func (rd *ReminderData) SyncCalendar(calOptions *calendar.Options) error { } // GoogleCalendarEvents returns list of Google Calendar Events. -func (rd *ReminderData) GoogleCalendarEvents(timezoneIANA string, reminderData *ReminderData) ([]*gc.Event, error) { +func (rd *ReminderData) GoogleCalendarEvents(timezoneIANA string) ([]*gc.Event, error) { logger.Info("Start: GoogleCalendarEvents") defer logger.Info("End: GoogleCalendarEvents") // get all pending notes @@ -123,7 +123,7 @@ func (rd *ReminderData) GoogleCalendarEvents(timezoneIANA string, reminderData * repeatMonthlyTag := rd.TagFromSlug("repeat-monthly") var events []*gc.Event for _, note := range relevantNotes { - event, err := note.GoogleCalendarEvent(repeatAnnuallyTag.Id, repeatMonthlyTag.Id, timezoneIANA, reminderData) + event, err := note.GoogleCalendarEvent(repeatAnnuallyTag.Id, repeatMonthlyTag.Id, timezoneIANA, rd) if err != nil { return nil, err } diff --git a/internal/model/reminder_data_test.go b/internal/model/reminder_data_test.go new file mode 100644 index 0000000..eed7900 --- /dev/null +++ b/internal/model/reminder_data_test.go @@ -0,0 +1,447 @@ +package model_test + +import ( + "fmt" + + // "fmt" + + "os" + "path" + "sort" + "strings" + "testing" + + // "github.com/golang/mock/gomock" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/internal/settings" + utils "github.com/goyalmunish/reminder/pkg/utils" +) + +// mocks + +type MockPromptTagSlug struct { +} +type MockPromptTagGroup struct { +} +type MockPromptNoteText struct { +} +type TestTagger struct{} + +func (tagger TestTagger) TagsFromIds(tagIDs []int) []string { + slugs := []string{} + for i, id := range tagIDs { + slug := fmt.Sprintf("%d-%d", i, id) + slugs = append(slugs, slug) + } + return slugs +} + +func (prompt *MockPromptTagSlug) Run() (string, error) { + return "test_tag_slug", nil +} + +func (prompt *MockPromptTagGroup) Run() (string, error) { + return "test_tag_group", nil +} + +func (prompt *MockPromptNoteText) Run() (string, error) { + return "a random note text", nil +} + +// test examples + +func TestDataFile(t *testing.T) { + defaultDataFilePath := settings.DefaultSettings().AppInfo.DataFile + utils.AssertEqual(t, strings.HasPrefix(defaultDataFilePath, "/") || strings.HasPrefix(defaultDataFilePath, "~/"), true) + utils.AssertEqual(t, strings.HasSuffix(defaultDataFilePath, ".json"), true) +} + +func TestReminderDataTagFromSlug(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + // case 1 + utils.AssertEqual(t, reminderData.TagFromSlug("a"), &tag1) + // case 2 + utils.AssertEqual(t, reminderData.TagFromSlug("a1"), &tag2) + // case 3 + utils.AssertEqual(t, reminderData.TagFromSlug("no_slug"), nil) +} + +func TestReminderDataTagIdsForGroup(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + // case 1 + utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group1"), []int{1, 2, 3}) + // case 2 + utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group2"), []int{4}) + // case 3 + utils.AssertEqual(t, reminderData.TagIdsForGroup("tag_group_NO"), []int{}) +} + +func TestSortedTagsSlug(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + // case 1 (no tags) + utils.AssertEqual(t, reminderData.SortedTagSlugs(), []string{}) + // case 2 (has couple of tags) + tags = append(tags, &model.Tag{Id: 1, Slug: "a", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 2, Slug: "z", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 3, Slug: "c", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 4, Slug: "b", Group: "tag_group2"}) + reminderData.Tags = tags + gotSlugs := reminderData.SortedTagSlugs() + wantSlugs := []string{"a", "b", "c", "z"} + utils.AssertEqual(t, gotSlugs, wantSlugs) +} + +func TestReminderDataTagsFromIds(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "z", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "c", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + // case 1 + tagIDs := []int{1, 3} + gotSlugs := reminderData.TagsFromIds(tagIDs) + wantSlugs := model.Tags{&tag1, &tag3}.Slugs() + utils.AssertEqual(t, gotSlugs, wantSlugs) + // case 2 + tagIDs = []int{} + gotSlugs = reminderData.TagsFromIds(tagIDs) + wantSlugs = model.Tags{}.Slugs() + utils.AssertEqual(t, gotSlugs, wantSlugs) + // case 3 + tagIDs = []int{1, 4, 2, 3} + gotSlugs = reminderData.TagsFromIds(tagIDs) + wantSlugs = model.Tags{&tag1, &tag4, &tag2, &tag3}.Slugs() + utils.AssertEqual(t, gotSlugs, wantSlugs) +} + +/* +func TestNextPossibleTagId(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + utils.AssertEqual(t, reminderData.nextPossibleTagId(), 4) +} +*/ + +func TestFindNotesByTagId(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + // create notes + var notes model.Notes + note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + notes = append(notes, ¬e1) + note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} + notes = append(notes, ¬e2) + note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} + notes = append(notes, ¬e3) + note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} + notes = append(notes, ¬e4) + note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} + notes = append(notes, ¬e5) + reminderData.Notes = notes + // searching notes + // case 1 + utils.AssertEqual(t, reminderData.FindNotesByTagId(2, model.NoteStatus_Pending), []*model.Note{¬e2}) + // case 2 + utils.AssertEqual(t, reminderData.FindNotesByTagId(2, model.NoteStatus_Done), []*model.Note{¬e3}) + // case 3 + utils.AssertEqual(t, reminderData.FindNotesByTagId(4, model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) + // case 4 + utils.AssertEqual(t, reminderData.FindNotesByTagId(1, model.NoteStatus_Done), []*model.Note{}) +} + +func TestFindNotesByTagSlug(t *testing.T) { + reminderData := model.ReminderData{ + User: &model.User{Name: "Test User", EmailId: "user@test.com"}, + Notes: []*model.Note{}, + Tags: model.Tags{}, + } + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + reminderData.Tags = tags + // create notes + var notes model.Notes + note1 := model.Note{Text: "1", Status: model.NoteStatus_Pending, TagIds: []int{1, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}} + notes = append(notes, ¬e1) + note2 := model.Note{Text: "2", Status: model.NoteStatus_Pending, TagIds: []int{2, 4}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}} + notes = append(notes, ¬e2) + note3 := model.Note{Text: "3", Status: model.NoteStatus_Done, TagIds: []int{2}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}} + notes = append(notes, ¬e3) + note4 := model.Note{Text: "4", Status: model.NoteStatus_Done, TagIds: []int{}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}} + notes = append(notes, ¬e4) + note5 := model.Note{Text: "5", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000005}} + notes = append(notes, ¬e5) + reminderData.Notes = notes + // searching notes + // case 1 + utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a1", model.NoteStatus_Pending), []*model.Note{¬e2}) + // case 2 + utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a1", model.NoteStatus_Done), []*model.Note{¬e3}) + // case 3 + utils.AssertEqual(t, reminderData.FindNotesByTagSlug("b", model.NoteStatus_Pending), []*model.Note{¬e1, ¬e2}) + // case 4 + utils.AssertEqual(t, reminderData.FindNotesByTagSlug("a", model.NoteStatus_Done), []*model.Note{}) +} + +func TestNewTagRegistration(t *testing.T) { + dataFilePath := path.Join("..", "..", "test", "test_data_file.json") + reminderData, _ := model.ReadDataFile(dataFilePath, false) + utils.AssertEqual(t, len(reminderData.Tags), 5) +} + +func TestNotes(t *testing.T) { + var notes []*model.Note + notes = append(notes, &model.Note{Text: "1", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}}) + notes = append(notes, &model.Note{Text: "2", Status: model.NoteStatus_Pending, BaseStruct: model.BaseStruct{UpdatedAt: 1600000004}}) + notes = append(notes, &model.Note{Text: "3", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000003}}) + notes = append(notes, &model.Note{Text: "4", Status: model.NoteStatus_Done, BaseStruct: model.BaseStruct{UpdatedAt: 1600000002}}) + sort.Sort(model.Notes(notes)) + var gotTexts []string + for _, value := range notes { + gotTexts = append(gotTexts, value.Text) + } + wantTexts := []string{"2", "3", "4", "1"} + utils.AssertEqual(t, gotTexts, wantTexts) +} + +func TestCreateDataFile(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + reminderData, _ := model.ReadDataFile(dataFilePath, false) + // old_updated_at := reminderData.UpdatedAt + testUser := model.User{Name: "Test User", EmailId: "user@test.com"} + reminderData.User = &testUser + _ = reminderData.CreateDataFile("") + remiderDataRe, _ := model.ReadDataFile(dataFilePath, false) + // utils.AssertEqual(t, remiderDataRe.UpdatedAt > old_updated_at, true) + utils.AssertEqual(t, remiderDataRe.User.EmailId == testUser.EmailId, true) +} + +func TestUpdateDataFile(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + reminderData, _ := model.ReadDataFile(dataFilePath, false) + // old_updated_at := reminderData.UpdatedAt + testUser := model.User{Name: "Test User", EmailId: "user@test.com"} + reminderData.User = &testUser + _ = reminderData.UpdateDataFile("") + remiderDataRe, _ := model.ReadDataFile(dataFilePath, false) + // utils.AssertEqual(t, remiderDataRe.UpdatedAt > old_updated_at, true) + utils.AssertEqual(t, remiderDataRe.User.EmailId == testUser.EmailId, true) +} + +func TestRegisterBasicTags(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + reminderData, _ := model.ReadDataFile(dataFilePath, false) + // register basic tags + _ = reminderData.RegisterBasicTags() + utils.AssertEqual(t, len(reminderData.Tags), 7) +} + +func TestNotesApproachingDueDate(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + reminderData, _ := model.ReadDataFile(dataFilePath, false) + // register basic tags + _ = reminderData.RegisterBasicTags() + // get current time + currentTime := utils.CurrentUnixTimestamp() + // register notes + // for reference: here is the list of basic tags + // {"slug": "current", "group": ""}, + // {"slug": "priority-urgent", "group": "priority"}, + // {"slug": "priority-medium", "group": "priority"}, + // {"slug": "priority-low", "group": "priority"}, + // {"slug": "repeat-annually", "group": "repeat"}, + // {"slug": "repeat-monthly", "group": "repeat"}, + // {"slug": "tips", "group": "tips"}} + currentTagId := reminderData.TagFromSlug("current").Id + repeatAnnuallyTagId := reminderData.TagFromSlug("repeat-annually").Id + repeatMonthlyTagId := reminderData.TagFromSlug("repeat-monthly").Id + var notes model.Notes + // non-repeat done notes + notes = append(notes, &model.Note{Text: "NRD01a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) + notes = append(notes, &model.Note{Text: "NRD02a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) + notes = append(notes, &model.Note{Text: "NRD03a", Status: model.NoteStatus_Done, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) + // repeat-annually done notes + notes = append(notes, &model.Note{Text: "RAD01a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) + notes = append(notes, &model.Note{Text: "RAD02a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) + notes = append(notes, &model.Note{Text: "RAD03a", Status: model.NoteStatus_Done, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) + // repeat-monthally done notes + notes = append(notes, &model.Note{Text: "RMD01a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 8*24*3600}) + notes = append(notes, &model.Note{Text: "RMD02a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) + notes = append(notes, &model.Note{Text: "RMD03a", Status: model.NoteStatus_Done, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) + // non-repeat pending notes + notes = append(notes, &model.Note{Text: "NRP01a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) // expected + notes = append(notes, &model.Note{Text: "NRP02a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*24*3600 - 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP02b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*24*3600 + 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP03a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected + notes = append(notes, &model.Note{Text: "NRP04a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 - 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP04b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 + 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP05a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600 - 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP05b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600 + 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP06a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600 - 3600}) // expected + notes = append(notes, &model.Note{Text: "NRP06b", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600 + 3600}) + notes = append(notes, &model.Note{Text: "NRP07a", Status: model.NoteStatus_Pending, TagIds: []int{currentTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) + // repeat-annually pending notes + notes = append(notes, &model.Note{Text: "RAP01", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) + notes = append(notes, &model.Note{Text: "RAP02", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RAP03", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime}) // expected + notes = append(notes, &model.Note{Text: "RAP04", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RAP05", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RAP06", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) + notes = append(notes, &model.Note{Text: "RAP07", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600 - 2*365}) + notes = append(notes, &model.Note{Text: "RAP08", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600 - 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP09", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP10", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600 - 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP11", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 - 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP12", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600 - 2*365}) + notes = append(notes, &model.Note{Text: "RAP13", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600 + 2*365}) + notes = append(notes, &model.Note{Text: "RAP14", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600 + 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP15", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP16", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 1*24*3600 + 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP17", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600 + 2*365}) // expected + notes = append(notes, &model.Note{Text: "RAP18", Status: model.NoteStatus_Pending, TagIds: []int{repeatAnnuallyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600 + 2*365}) + // repeat-monthly pending notes + notes = append(notes, &model.Note{Text: "RMP01", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 9*24*3600}) + notes = append(notes, &model.Note{Text: "RMP02", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime - 1*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RMP03", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime}) // expected + notes = append(notes, &model.Note{Text: "RMP04", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 2*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RMP05", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 3*24*3600}) // expected + notes = append(notes, &model.Note{Text: "RMP06", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 7*24*3600}) + notes = append(notes, &model.Note{Text: "RMP07", Status: model.NoteStatus_Pending, TagIds: []int{repeatMonthlyTagId}, BaseStruct: model.BaseStruct{UpdatedAt: 1600000001}, CompleteBy: currentTime + 9*24*3600}) + reminderData.Notes = notes + // get urgent notes + urgentNotes := reminderData.NotesApprachingDueDate("default") + var urgentNotesText []string + for _, note := range urgentNotes { + urgentNotesText = append(urgentNotesText, note.Text) + } + expectNotesText := []string{ + "NRP01a", "NRP02a", "NRP02b", "NRP03a", "NRP04a", "NRP04b", "NRP05a", "NRP05b", "NRP06a", + "RAP02", "RAP03", "RAP04", "RAP05", "RAP08", "RAP09", "RAP10", "RAP11", "RAP14", "RAP15", "RAP16", "RAP17", + "RMP02", "RMP03", + } + t.Logf("Received Texts: %v", urgentNotesText) + t.Logf("Expected Texts: %v", expectNotesText) + utils.SkipCI(t) + utils.AssertEqual(t, urgentNotesText, expectNotesText) + // [NRP01a NRP02a NRP02b NRP03a NRP04a NRP04b NRP05a NRP05b NRP06a RAP02 RAP03 RAP04 RAP05 RAP08 RAP09 RAP10 RAP11 RAP14 RAP15 RAP16 RAP17 RMP03] + // [NRP01a NRP02a NRP02b NRP03a NRP04a NRP04b NRP05a NRP05b NRP06a RAP02 RAP03 RAP04 RAP05 RAP08 RAP09 RAP10 RAP11 RAP14 RAP15 RAP16 RAP17 RMP02 RMP03]} +} + +func TestPrintStats(t *testing.T) { + var dataFilePath = "temp_test_dir/mydata.json" + // make sure temporary files and dirs are removed at the end of the test + defer os.RemoveAll(path.Dir(dataFilePath)) + // create the file and required dirs + _ = model.MakeSureFileExists(dataFilePath, false) + reminderData, _ := model.ReadDataFile(dataFilePath, false) + // register basic tags + _ = reminderData.RegisterBasicTags() + got, _ := reminderData.Stats() + want := ` +Stats of "temp_test_dir/mydata.json": + - Number of Tags: 7 + - Pending Notes: 0/0 + - Suspended Notes: 0 + - Done Notes: 0 +` + utils.AssertEqual(t, got, want) +} diff --git a/internal/model/tag.go b/internal/model/tag.go index 0e60572..51f0c1e 100644 --- a/internal/model/tag.go +++ b/internal/model/tag.go @@ -1,7 +1,6 @@ package model import ( - "context" "fmt" ) @@ -11,10 +10,9 @@ A Tag represents classification of a note. A note can have multiple tags, and a tag can be associated with multiple notes. */ type Tag struct { - context context.Context - Id int `json:"id"` // internal int-based id of the tag - Slug string `json:"slug"` // client-facing string-based id for tag - Group string `json:"group"` // a note can be part of only one tag within a group + Id int `json:"id"` // internal int-based id of the tag + Slug string `json:"slug"` // client-facing string-based id for tag + Group string `json:"group"` // a note can be part of only one tag within a group BaseStruct } @@ -22,8 +20,3 @@ type Tag struct { func (t Tag) String() string { return fmt.Sprintf("%v#%v#%v", t.Group, t.Slug, t.Id) } - -// SetContext sets given context to the receiver. -func (tag *Tag) SetContext(ctx context.Context) { - tag.context = ctx -} diff --git a/internal/model/tag_test.go b/internal/model/tag_test.go new file mode 100644 index 0000000..4603a7a --- /dev/null +++ b/internal/model/tag_test.go @@ -0,0 +1,23 @@ +package model_test + +import ( + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestTagString(t *testing.T) { + // case 1: general case + got := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + want := "tag_group1#a#1" + utils.AssertEqual(t, got, want) + // case 2: blank group + got = model.Tag{Id: 1, Slug: "a", Group: ""} + want = "#a#1" + utils.AssertEqual(t, got, want) + // case 3: omitted group + got = model.Tag{Id: 1, Slug: "a"} + want = "#a#1" + utils.AssertEqual(t, got, want) +} diff --git a/internal/model/tags_test.go b/internal/model/tags_test.go new file mode 100644 index 0000000..38a47f4 --- /dev/null +++ b/internal/model/tags_test.go @@ -0,0 +1,109 @@ +package model_test + +import ( + "sort" + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestTagsSort(t *testing.T) { + var tags model.Tags + tags = append(tags, &model.Tag{Id: 1, Slug: "a", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 2, Slug: "z", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 3, Slug: "c", Group: "tag_group1"}) + tags = append(tags, &model.Tag{Id: 4, Slug: "b", Group: "tag_group2"}) + sort.Sort(model.Tags(tags)) + var got []int + for _, value := range tags { + got = append(got, value.Id) + } + want := []int{1, 4, 3, 2} + utils.AssertEqual(t, got, want) +} + +func TestTagsSlugs(t *testing.T) { + var tags model.Tags + utils.AssertEqual(t, tags, "[]") + // case 1 (no tags) + utils.AssertEqual(t, tags.Slugs(), "[]") + // case 2 (non-empty tags) + tags = append(tags, &model.Tag{Id: 1, Slug: "tag_1", Group: "tag_group"}) + tags = append(tags, &model.Tag{Id: 2, Slug: "tag_2", Group: "tag_group"}) + tags = append(tags, &model.Tag{Id: 3, Slug: "tag_3", Group: "tag_group"}) + got := tags.Slugs() + want := "[tag_1 tag_2 tag_3]" + utils.AssertEqual(t, got, want) +} + +func TestTagsFromSlug(t *testing.T) { + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + // case 1 (passing non-existing slug) + utils.AssertEqual(t, tags.FromSlug("no_such_slug"), nil) + // case 2 (passing tag which is part of another tag as well) + utils.AssertEqual(t, tags.FromSlug("a"), &tag1) + // case 3 + utils.AssertEqual(t, tags.FromSlug("a1"), &tag2) +} + +func TestTagsFromIds(t *testing.T) { + var tags model.Tags + // creating tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "z", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "c", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + // case 1 (passing blank tagIDs) + tagIDs := []int{} + gotSlugs := tags.FromIds(tagIDs) + wantSlugs := model.Tags{} + utils.AssertEqual(t, gotSlugs, wantSlugs) + // case 2 (no matching tagIDs) + tagIDs = []int{100, 101} + gotSlugs = tags.FromIds(tagIDs) + wantSlugs = model.Tags{} + utils.AssertEqual(t, gotSlugs, wantSlugs) + // case 3 (two matching tagIDs) + tagIDs = []int{1, 3} + gotSlugs = tags.FromIds(tagIDs) + wantSlugs = model.Tags{&tag1, &tag3} + utils.AssertEqual(t, gotSlugs, wantSlugs) + // case 4 + tagIDs = []int{1, 4, 2, 3} + gotSlugs = tags.FromIds(tagIDs) + wantSlugs = model.Tags{&tag1, &tag4, &tag2, &tag3} + utils.AssertEqual(t, gotSlugs, wantSlugs) +} + +func TestTagsIdsForGroup(t *testing.T) { + // creating tags + var tags model.Tags + tag1 := model.Tag{Id: 1, Slug: "a", Group: "tag_group1"} + tags = append(tags, &tag1) + tag2 := model.Tag{Id: 2, Slug: "a1", Group: "tag_group1"} + tags = append(tags, &tag2) + tag3 := model.Tag{Id: 3, Slug: "a2", Group: "tag_group1"} + tags = append(tags, &tag3) + tag4 := model.Tag{Id: 4, Slug: "b", Group: "tag_group2"} + tags = append(tags, &tag4) + // case 1 (group with no such name) + utils.AssertEqual(t, tags.IdsForGroup("tag_group_NO"), []int{}) + // case 1 (group with multiple tags) + utils.AssertEqual(t, tags.IdsForGroup("tag_group1"), []int{1, 2, 3}) + // case 2 (group with single tag) + utils.AssertEqual(t, tags.IdsForGroup("tag_group2"), []int{4}) +} diff --git a/internal/model/user.go b/internal/model/user.go index 0e1091b..fcb8a88 100644 --- a/internal/model/user.go +++ b/internal/model/user.go @@ -1,7 +1,6 @@ package model import ( - "context" "fmt" ) @@ -12,7 +11,6 @@ The app doesn't have multi-tenant setting, but it is supposed to be used only locally and by the user who is owner of the OS account. */ type User struct { - context context.Context Name string `json:"name"` EmailId string `json:"email_id"` } @@ -21,8 +19,3 @@ type User struct { func (u User) String() string { return fmt.Sprintf("{Name: %v, EmailId: %v}", u.Name, u.EmailId) } - -// SetContext sets given context to the receiver. -func (u *User) SetContext(ctx context.Context) { - u.context = ctx -} diff --git a/internal/model/user_test.go b/internal/model/user_test.go new file mode 100644 index 0000000..fb001b2 --- /dev/null +++ b/internal/model/user_test.go @@ -0,0 +1,14 @@ +package model_test + +import ( + "testing" + + model "github.com/goyalmunish/reminder/internal/model" + "github.com/goyalmunish/reminder/pkg/utils" +) + +func TestUserString(t *testing.T) { + user := model.User{Name: "Foo Bar", EmailId: "user@test.com"} + want := "{Name: Foo Bar, EmailId: user@test.com}" + utils.AssertEqual(t, user.String(), want) +}