From f72610f2f9e1d977394f9046cd1ad96e77709f08 Mon Sep 17 00:00:00 2001 From: masterElmar <18119527+masterElmar@users.noreply.github.com> Date: Tue, 21 Nov 2023 01:48:09 +0100 Subject: [PATCH] feat:#6 added tests with db for unused feed delete --- backend/model/feedModel.go | 6 +- backend/service/addSchedule.go | 37 ++------- backend/service/course/courseFunctions.go | 20 +++++ backend/service/db/dbFeeds.go | 8 +- backend/service/feed/feedFunctions.go | 35 +++++++++ backend/service/feed/feedFunctions_test.go | 83 ++++++++++++++++++++ backend/service/feed/mockData/data.db | Bin 0 -> 143360 bytes backend/service/feed/mockData/logs.db | Bin 0 -> 49152 bytes backend/service/functions/time/mockClock.go | 10 +++ backend/service/functions/time/realClock.go | 8 ++ backend/service/functions/time/time.go | 8 ++ 11 files changed, 178 insertions(+), 37 deletions(-) create mode 100644 backend/service/course/courseFunctions.go create mode 100644 backend/service/feed/feedFunctions.go create mode 100644 backend/service/feed/feedFunctions_test.go create mode 100644 backend/service/feed/mockData/data.db create mode 100644 backend/service/feed/mockData/logs.db create mode 100644 backend/service/functions/time/mockClock.go create mode 100644 backend/service/functions/time/realClock.go create mode 100644 backend/service/functions/time/time.go diff --git a/backend/model/feedModel.go b/backend/model/feedModel.go index 516ae8a..cc2e413 100644 --- a/backend/model/feedModel.go +++ b/backend/model/feedModel.go @@ -2,12 +2,12 @@ package model import ( "github.com/pocketbase/pocketbase/models" - "time" + "github.com/pocketbase/pocketbase/tools/types" ) type Feed struct { - Modules string `db:"modules" json:"modules"` - Retrieved time.Time `db:"retrieved" json:"retrieved"` + Modules string `db:"modules" json:"modules"` + Retrieved types.DateTime `db:"retrieved" json:"retrieved"` models.BaseModel } diff --git a/backend/service/addSchedule.go b/backend/service/addSchedule.go index 280cb32..bd0cd96 100644 --- a/backend/service/addSchedule.go +++ b/backend/service/addSchedule.go @@ -4,10 +4,9 @@ import ( "github.com/pocketbase/pocketbase" "github.com/pocketbase/pocketbase/core" "github.com/pocketbase/pocketbase/tools/cron" - "htwkalender/service/db" - "htwkalender/service/events" - "log" - "time" + "htwkalender/service/course" + "htwkalender/service/feed" + "htwkalender/service/functions/time" ) func AddSchedules(app *pocketbase.PocketBase) { @@ -19,37 +18,13 @@ func AddSchedules(app *pocketbase.PocketBase) { // Every three hours update all courses (5 segments - minute, hour, day, month, weekday) "0 */3 * * *" // Every 10 minutes update all courses (5 segments - minute, hour, day, month, weekday) "*/10 * * * *" scheduler.MustAdd("updateCourse", "0 */3 * * *", func() { - courses := events.GetAllCourses(app) - for _, course := range courses { - err := events.UpdateModulesForCourse(app, course) - if err != nil { - log.Println("Update Course: " + course + " failed") - log.Println(err) - } else { - log.Println("Update Course: " + course + " successful") - } - } + course.UpdateCourse(app) }) // Every sunday at 3am clean all courses (5 segments - minute, hour, day, month, weekday) "0 3 * * 0" scheduler.MustAdd("cleanFeeds", "0 3 * * 0", func() { - feeds, err := db.GetAllFeeds(app) - if err != nil { - log.Println("CleanFeeds: get all feeds failed") - return - } - for _, feed := range feeds { - // if retrieved time is older than a half year delete feed - if feed.GetTime("retrieved").Before(time.Now().AddDate(0, -6, 0)) { - err = app.Dao().DeleteRecord(feed) - if err != nil { - log.Println("CleanFeeds: delete feed " + feed.GetId() + " failed") - log.Println(err) - } else { - log.Println("CleanFeeds: delete feed " + feed.GetId() + " successful") - } - } - } + // clean feeds older than 6 months + feed.ClearFeeds(app.Dao(), 6, time.RealClock{}) }) scheduler.Start() return nil diff --git a/backend/service/course/courseFunctions.go b/backend/service/course/courseFunctions.go new file mode 100644 index 0000000..0d24700 --- /dev/null +++ b/backend/service/course/courseFunctions.go @@ -0,0 +1,20 @@ +package course + +import ( + "github.com/pocketbase/pocketbase" + "htwkalender/service/events" + "log" +) + +func UpdateCourse(app *pocketbase.PocketBase) { + courses := events.GetAllCourses(app) + for _, course := range courses { + err := events.UpdateModulesForCourse(app, course) + if err != nil { + log.Println("Update Course: " + course + " failed") + log.Println(err) + } else { + log.Println("Update Course: " + course + " successful") + } + } +} diff --git a/backend/service/db/dbFeeds.go b/backend/service/db/dbFeeds.go index f2f0804..408d763 100644 --- a/backend/service/db/dbFeeds.go +++ b/backend/service/db/dbFeeds.go @@ -2,6 +2,7 @@ package db import ( "github.com/pocketbase/pocketbase" + "github.com/pocketbase/pocketbase/daos" "github.com/pocketbase/pocketbase/models" "htwkalender/model" "time" @@ -28,7 +29,7 @@ func FindFeedByToken(token string, app *pocketbase.PocketBase) (*model.Feed, err var feed model.Feed feed.Modules = record.GetString("modules") - feed.Retrieved = record.GetTime("retrieved") + feed.Retrieved = record.GetDateTime("retrieved") //update retrieved time record.Set("retrieved", time.Now()) @@ -38,8 +39,9 @@ func FindFeedByToken(token string, app *pocketbase.PocketBase) (*model.Feed, err return &feed, err } -func GetAllFeeds(app *pocketbase.PocketBase) ([]*models.Record, error) { - feeds, err := app.Dao().FindRecordsByFilter("feeds", "", "", 0, 0) +func GetAllFeeds(db *daos.Dao) ([]model.Feed, error) { + var feeds []model.Feed + err := db.DB().Select("*").From("feeds").All(&feeds) if err != nil { return nil, err } diff --git a/backend/service/feed/feedFunctions.go b/backend/service/feed/feedFunctions.go new file mode 100644 index 0000000..853a144 --- /dev/null +++ b/backend/service/feed/feedFunctions.go @@ -0,0 +1,35 @@ +package feed + +import ( + "github.com/pocketbase/dbx" + "github.com/pocketbase/pocketbase/daos" + database "htwkalender/service/db" + localTime "htwkalender/service/functions/time" + "log" +) + +func ClearFeeds(db *daos.Dao, months int, clock localTime.Clock) { + feeds, err := database.GetAllFeeds(db) + if err != nil { + log.Println("CleanFeeds: get all feeds failed") + return + } + for _, feed := range feeds { + // if retrieved time is older than a half year delete feed + now := clock.Now() + feedRetrievedTime := feed.Retrieved.Time() + timeShift := now.AddDate(0, -months, 0) + + if feedRetrievedTime.Before(timeShift) { + // delete feed + sqlResult, err := db.DB().Delete("feeds", dbx.NewExp("id = {:id}", dbx.Params{"id": feed.GetId()})).Execute() + if err != nil { + log.Println("CleanFeeds: delete feed " + feed.GetId() + " failed") + log.Println(err) + log.Println(sqlResult) + } else { + log.Println("CleanFeeds: delete feed " + feed.GetId() + " successful") + } + } + } +} diff --git a/backend/service/feed/feedFunctions_test.go b/backend/service/feed/feedFunctions_test.go new file mode 100644 index 0000000..5eb1182 --- /dev/null +++ b/backend/service/feed/feedFunctions_test.go @@ -0,0 +1,83 @@ +package feed + +import ( + "github.com/pocketbase/pocketbase/daos" + "github.com/pocketbase/pocketbase/tests" + "htwkalender/model" + mockTime "htwkalender/service/functions/time" + "testing" + "time" +) + +const testDataDir = "./mockData" + +func TestClearFeeds(t *testing.T) { + + setupTestApp := func(t *testing.T) *daos.Dao { + testApp, err := tests.NewTestApp(testDataDir) + if err != nil { + t.Fatal(err) + } + dao := daos.New(testApp.Dao().DB()) + return dao + } + + type args struct { + db *daos.Dao + months int + mockClock mockTime.MockClock + } + testCases := []struct { + name string + args args + want int + }{ + { + name: "TestClearFeeds", + args: args{ + db: setupTestApp(t), + months: 6, + mockClock: mockTime.MockClock{ + NowTime: time.Date(2023, 12, 1, 0, 0, 0, 0, time.UTC), + }, + }, + want: 1, + }, + { + name: "TestClearAllFeeds", + args: args{ + db: setupTestApp(t), + months: 1, + mockClock: mockTime.MockClock{ + NowTime: time.Date(2024, 1, 1, 0, 0, 0, 0, time.UTC), + }, + }, + want: 0, + }, + { + name: "TestClearFeedsClearBeforeRetrievedTime", + args: args{ + db: setupTestApp(t), + months: 1, + mockClock: mockTime.MockClock{ + NowTime: time.Date(2010, 1, 1, 0, 0, 0, 0, time.UTC), + }, + }, + want: 3, + }, + } + for _, tt := range testCases { + t.Run(tt.name, func(t *testing.T) { + ClearFeeds(tt.args.db, tt.args.months, tt.args.mockClock) + // count all feeds in db + var feeds []*model.Feed + err := tt.args.db.DB().Select("id").From("feeds").All(&feeds) + if err != nil { + t.Fatal(err) + } + if got := len(feeds); got != tt.want { + t.Errorf("ClearFeeds() = %v, want %v", got, tt.want) + } + }) + } +} diff --git a/backend/service/feed/mockData/data.db b/backend/service/feed/mockData/data.db new file mode 100644 index 0000000000000000000000000000000000000000..acca65d300f16e4a1b6ca5af4e8a927b572656cb GIT binary patch literal 143360 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
  • 1Wc{rvdq>wYM17E=RaNN2 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