| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378 |
- // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
- // SPDX-License-Identifier: Apache-2.0
- package model // import "miniflux.app/v2/internal/model"
- import (
- "os"
- "strconv"
- "testing"
- "time"
- "miniflux.app/v2/internal/config"
- )
- const (
- largeWeeklyCount = 10080
- noRefreshDelay = 0
- )
- func TestFeedCategorySetter(t *testing.T) {
- feed := &Feed{}
- feed.WithCategoryID(int64(123))
- if feed.Category == nil {
- t.Fatal(`The category field should not be null`)
- }
- if feed.Category.ID != int64(123) {
- t.Error(`The category ID must be set`)
- }
- }
- func TestFeedErrorCounter(t *testing.T) {
- feed := &Feed{}
- feed.WithTranslatedErrorMessage("Some Error")
- if feed.ParsingErrorMsg != "Some Error" {
- t.Error(`The error message must be set`)
- }
- if feed.ParsingErrorCount != 1 {
- t.Error(`The error counter must be set to 1`)
- }
- feed.ResetErrorCounter()
- if feed.ParsingErrorMsg != "" {
- t.Error(`The error message must be removed`)
- }
- if feed.ParsingErrorCount != 0 {
- t.Error(`The error counter must be set to 0`)
- }
- }
- func TestFeedCheckedNow(t *testing.T) {
- feed := &Feed{}
- feed.FeedURL = "https://example.org/feed"
- feed.CheckedNow()
- if feed.SiteURL != feed.FeedURL {
- t.Error(`The site URL must not be empty`)
- }
- if feed.CheckedAt.IsZero() {
- t.Error(`The checked date must be set`)
- }
- }
- func checkTargetInterval(t *testing.T, feed *Feed, targetInterval time.Duration, timeBefore time.Time, message string) {
- if feed.NextCheckAt.Before(timeBefore.Add(targetInterval)) {
- t.Errorf(`The next_check_at should be after timeBefore + %s`, message)
- }
- if feed.NextCheckAt.After(time.Now().Add(targetInterval)) {
- t.Errorf(`The next_check_at should be before now + %s`, message)
- }
- }
- func TestFeedScheduleNextCheckRoundRobinDefault(t *testing.T) {
- os.Clearenv()
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- feed.ScheduleNextCheck(0, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := config.Opts.SchedulerRoundRobinMinInterval()
- checkTargetInterval(t, feed, targetInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinDefault")
- }
- func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMinInterval(t *testing.T) {
- os.Clearenv()
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMinInterval()+30)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- expectedInterval := config.Opts.SchedulerRoundRobinMinInterval() + 30
- checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMinInterval")
- }
- func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayBelowMinInterval(t *testing.T) {
- os.Clearenv()
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMinInterval()-30)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- expectedInterval := config.Opts.SchedulerRoundRobinMinInterval()
- checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayBelowMinInterval")
- }
- func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMaxInterval(t *testing.T) {
- os.Clearenv()
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMaxInterval()+30)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- expectedInterval := config.Opts.SchedulerRoundRobinMaxInterval()
- checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMaxInterval")
- }
- func TestFeedScheduleNextCheckRoundRobinMinInterval(t *testing.T) {
- minInterval := 1
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "round_robin")
- os.Setenv("SCHEDULER_ROUND_ROBIN_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- feed.ScheduleNextCheck(0, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- expectedInterval := time.Duration(minInterval) * time.Minute
- checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinMinInterval")
- }
- func TestFeedScheduleNextCheckEntryFrequencyMaxInterval(t *testing.T) {
- maxInterval := 5
- minInterval := 1
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- // Use a very small weekly count to trigger the max interval
- weeklyCount := 1
- feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := time.Duration(maxInterval) * time.Minute
- checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency max interval")
- }
- func TestFeedScheduleNextCheckEntryFrequencyMaxIntervalZeroWeeklyCount(t *testing.T) {
- maxInterval := 5
- minInterval := 1
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- // Use a very small weekly count to trigger the max interval
- weeklyCount := 0
- feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := time.Duration(maxInterval) * time.Minute
- checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency max interval")
- }
- func TestFeedScheduleNextCheckEntryFrequencyMinInterval(t *testing.T) {
- maxInterval := 500
- minInterval := 100
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- // Use a very large weekly count to trigger the min interval
- weeklyCount := largeWeeklyCount
- feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := time.Duration(minInterval) * time.Minute
- checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency min interval")
- }
- func TestFeedScheduleNextCheckEntryFrequencyFactor(t *testing.T) {
- factor := 2
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_FACTOR", strconv.Itoa(factor))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- weeklyCount := 7
- feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := config.Opts.SchedulerEntryFrequencyMaxInterval() / time.Duration(factor)
- checkTargetInterval(t, feed, targetInterval, timeBefore, "factor * count")
- }
- func TestFeedScheduleNextCheckEntryFrequencySmallNewTTL(t *testing.T) {
- // If the feed has a TTL defined, we use it to make sure we don't check it too often.
- maxInterval := 500
- minInterval := 100
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- // Use a very large weekly count to trigger the min interval
- weeklyCount := largeWeeklyCount
- // TTL is smaller than minInterval.
- newTTL := time.Duration(minInterval) * time.Minute / 2
- feed.ScheduleNextCheck(weeklyCount, newTTL)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := time.Duration(minInterval) * time.Minute
- checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency min interval")
- if feed.NextCheckAt.Before(timeBefore.Add(newTTL)) {
- t.Error(`The next_check_at should be after timeBefore + TTL`)
- }
- }
- func TestFeedScheduleNextCheckEntryFrequencyLargeNewTTL(t *testing.T) {
- // If the feed has a TTL defined, we use it to make sure we don't check it too often.
- maxInterval := 500
- minInterval := 100
- os.Clearenv()
- os.Setenv("POLLING_SCHEDULER", "entry_frequency")
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
- os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
- var err error
- parser := config.NewConfigParser()
- config.Opts, err = parser.ParseEnvironmentVariables()
- if err != nil {
- t.Fatalf(`Parsing failure: %v`, err)
- }
- timeBefore := time.Now()
- feed := &Feed{}
- // Use a very large weekly count to trigger the min interval
- weeklyCount := largeWeeklyCount
- // TTL is larger than minInterval.
- newTTL := time.Duration(minInterval) * time.Minute * 2
- feed.ScheduleNextCheck(weeklyCount, newTTL)
- if feed.NextCheckAt.IsZero() {
- t.Error(`The next_check_at must be set`)
- }
- targetInterval := newTTL
- checkTargetInterval(t, feed, targetInterval, timeBefore, "TTL")
- if feed.NextCheckAt.Before(timeBefore.Add(time.Minute * time.Duration(minInterval))) {
- t.Error(`The next_check_at should be after timeBefore + entry frequency min interval`)
- }
- }
|