feed_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package model // import "miniflux.app/v2/internal/model"
  4. import (
  5. "os"
  6. "strconv"
  7. "testing"
  8. "time"
  9. "miniflux.app/v2/internal/config"
  10. )
  11. const (
  12. largeWeeklyCount = 10080
  13. noRefreshDelay = 0
  14. )
  15. func TestFeedCategorySetter(t *testing.T) {
  16. feed := &Feed{}
  17. feed.WithCategoryID(int64(123))
  18. if feed.Category == nil {
  19. t.Fatal(`The category field should not be null`)
  20. }
  21. if feed.Category.ID != int64(123) {
  22. t.Error(`The category ID must be set`)
  23. }
  24. }
  25. func TestFeedErrorCounter(t *testing.T) {
  26. feed := &Feed{}
  27. feed.WithTranslatedErrorMessage("Some Error")
  28. if feed.ParsingErrorMsg != "Some Error" {
  29. t.Error(`The error message must be set`)
  30. }
  31. if feed.ParsingErrorCount != 1 {
  32. t.Error(`The error counter must be set to 1`)
  33. }
  34. feed.ResetErrorCounter()
  35. if feed.ParsingErrorMsg != "" {
  36. t.Error(`The error message must be removed`)
  37. }
  38. if feed.ParsingErrorCount != 0 {
  39. t.Error(`The error counter must be set to 0`)
  40. }
  41. }
  42. func TestFeedCheckedNow(t *testing.T) {
  43. feed := &Feed{}
  44. feed.FeedURL = "https://example.org/feed"
  45. feed.CheckedNow()
  46. if feed.SiteURL != feed.FeedURL {
  47. t.Error(`The site URL must not be empty`)
  48. }
  49. if feed.CheckedAt.IsZero() {
  50. t.Error(`The checked date must be set`)
  51. }
  52. }
  53. func checkTargetInterval(t *testing.T, feed *Feed, targetInterval time.Duration, timeBefore time.Time, message string) {
  54. if feed.NextCheckAt.Before(timeBefore.Add(targetInterval)) {
  55. t.Errorf(`The next_check_at should be after timeBefore + %s`, message)
  56. }
  57. if feed.NextCheckAt.After(time.Now().Add(targetInterval)) {
  58. t.Errorf(`The next_check_at should be before now + %s`, message)
  59. }
  60. }
  61. func TestFeedScheduleNextCheckRoundRobinDefault(t *testing.T) {
  62. os.Clearenv()
  63. var err error
  64. parser := config.NewConfigParser()
  65. config.Opts, err = parser.ParseEnvironmentVariables()
  66. if err != nil {
  67. t.Fatalf(`Parsing failure: %v`, err)
  68. }
  69. timeBefore := time.Now()
  70. feed := &Feed{}
  71. feed.ScheduleNextCheck(0, noRefreshDelay)
  72. if feed.NextCheckAt.IsZero() {
  73. t.Error(`The next_check_at must be set`)
  74. }
  75. targetInterval := config.Opts.SchedulerRoundRobinMinInterval()
  76. checkTargetInterval(t, feed, targetInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinDefault")
  77. }
  78. func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMinInterval(t *testing.T) {
  79. os.Clearenv()
  80. var err error
  81. parser := config.NewConfigParser()
  82. config.Opts, err = parser.ParseEnvironmentVariables()
  83. if err != nil {
  84. t.Fatalf(`Parsing failure: %v`, err)
  85. }
  86. timeBefore := time.Now()
  87. feed := &Feed{}
  88. feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMinInterval()+30)
  89. if feed.NextCheckAt.IsZero() {
  90. t.Error(`The next_check_at must be set`)
  91. }
  92. expectedInterval := config.Opts.SchedulerRoundRobinMinInterval() + 30
  93. checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMinInterval")
  94. }
  95. func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayBelowMinInterval(t *testing.T) {
  96. os.Clearenv()
  97. var err error
  98. parser := config.NewConfigParser()
  99. config.Opts, err = parser.ParseEnvironmentVariables()
  100. if err != nil {
  101. t.Fatalf(`Parsing failure: %v`, err)
  102. }
  103. timeBefore := time.Now()
  104. feed := &Feed{}
  105. feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMinInterval()-30)
  106. if feed.NextCheckAt.IsZero() {
  107. t.Error(`The next_check_at must be set`)
  108. }
  109. expectedInterval := config.Opts.SchedulerRoundRobinMinInterval()
  110. checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayBelowMinInterval")
  111. }
  112. func TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMaxInterval(t *testing.T) {
  113. os.Clearenv()
  114. var err error
  115. parser := config.NewConfigParser()
  116. config.Opts, err = parser.ParseEnvironmentVariables()
  117. if err != nil {
  118. t.Fatalf(`Parsing failure: %v`, err)
  119. }
  120. timeBefore := time.Now()
  121. feed := &Feed{}
  122. feed.ScheduleNextCheck(0, config.Opts.SchedulerRoundRobinMaxInterval()+30)
  123. if feed.NextCheckAt.IsZero() {
  124. t.Error(`The next_check_at must be set`)
  125. }
  126. expectedInterval := config.Opts.SchedulerRoundRobinMaxInterval()
  127. checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinWithRefreshDelayAboveMaxInterval")
  128. }
  129. func TestFeedScheduleNextCheckRoundRobinMinInterval(t *testing.T) {
  130. minInterval := 1
  131. os.Clearenv()
  132. os.Setenv("POLLING_SCHEDULER", "round_robin")
  133. os.Setenv("SCHEDULER_ROUND_ROBIN_MIN_INTERVAL", strconv.Itoa(minInterval))
  134. var err error
  135. parser := config.NewConfigParser()
  136. config.Opts, err = parser.ParseEnvironmentVariables()
  137. if err != nil {
  138. t.Fatalf(`Parsing failure: %v`, err)
  139. }
  140. timeBefore := time.Now()
  141. feed := &Feed{}
  142. feed.ScheduleNextCheck(0, noRefreshDelay)
  143. if feed.NextCheckAt.IsZero() {
  144. t.Error(`The next_check_at must be set`)
  145. }
  146. expectedInterval := time.Duration(minInterval) * time.Minute
  147. checkTargetInterval(t, feed, expectedInterval, timeBefore, "TestFeedScheduleNextCheckRoundRobinMinInterval")
  148. }
  149. func TestFeedScheduleNextCheckEntryFrequencyMaxInterval(t *testing.T) {
  150. maxInterval := 5
  151. minInterval := 1
  152. os.Clearenv()
  153. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  154. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
  155. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
  156. var err error
  157. parser := config.NewConfigParser()
  158. config.Opts, err = parser.ParseEnvironmentVariables()
  159. if err != nil {
  160. t.Fatalf(`Parsing failure: %v`, err)
  161. }
  162. timeBefore := time.Now()
  163. feed := &Feed{}
  164. // Use a very small weekly count to trigger the max interval
  165. weeklyCount := 1
  166. feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
  167. if feed.NextCheckAt.IsZero() {
  168. t.Error(`The next_check_at must be set`)
  169. }
  170. targetInterval := time.Duration(maxInterval) * time.Minute
  171. checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency max interval")
  172. }
  173. func TestFeedScheduleNextCheckEntryFrequencyMaxIntervalZeroWeeklyCount(t *testing.T) {
  174. maxInterval := 5
  175. minInterval := 1
  176. os.Clearenv()
  177. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  178. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
  179. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
  180. var err error
  181. parser := config.NewConfigParser()
  182. config.Opts, err = parser.ParseEnvironmentVariables()
  183. if err != nil {
  184. t.Fatalf(`Parsing failure: %v`, err)
  185. }
  186. timeBefore := time.Now()
  187. feed := &Feed{}
  188. // Use a very small weekly count to trigger the max interval
  189. weeklyCount := 0
  190. feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
  191. if feed.NextCheckAt.IsZero() {
  192. t.Error(`The next_check_at must be set`)
  193. }
  194. targetInterval := time.Duration(maxInterval) * time.Minute
  195. checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency max interval")
  196. }
  197. func TestFeedScheduleNextCheckEntryFrequencyMinInterval(t *testing.T) {
  198. maxInterval := 500
  199. minInterval := 100
  200. os.Clearenv()
  201. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  202. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
  203. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
  204. var err error
  205. parser := config.NewConfigParser()
  206. config.Opts, err = parser.ParseEnvironmentVariables()
  207. if err != nil {
  208. t.Fatalf(`Parsing failure: %v`, err)
  209. }
  210. timeBefore := time.Now()
  211. feed := &Feed{}
  212. // Use a very large weekly count to trigger the min interval
  213. weeklyCount := largeWeeklyCount
  214. feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
  215. if feed.NextCheckAt.IsZero() {
  216. t.Error(`The next_check_at must be set`)
  217. }
  218. targetInterval := time.Duration(minInterval) * time.Minute
  219. checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency min interval")
  220. }
  221. func TestFeedScheduleNextCheckEntryFrequencyFactor(t *testing.T) {
  222. factor := 2
  223. os.Clearenv()
  224. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  225. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_FACTOR", strconv.Itoa(factor))
  226. var err error
  227. parser := config.NewConfigParser()
  228. config.Opts, err = parser.ParseEnvironmentVariables()
  229. if err != nil {
  230. t.Fatalf(`Parsing failure: %v`, err)
  231. }
  232. timeBefore := time.Now()
  233. feed := &Feed{}
  234. weeklyCount := 7
  235. feed.ScheduleNextCheck(weeklyCount, noRefreshDelay)
  236. if feed.NextCheckAt.IsZero() {
  237. t.Error(`The next_check_at must be set`)
  238. }
  239. targetInterval := config.Opts.SchedulerEntryFrequencyMaxInterval() / time.Duration(factor)
  240. checkTargetInterval(t, feed, targetInterval, timeBefore, "factor * count")
  241. }
  242. func TestFeedScheduleNextCheckEntryFrequencySmallNewTTL(t *testing.T) {
  243. // If the feed has a TTL defined, we use it to make sure we don't check it too often.
  244. maxInterval := 500
  245. minInterval := 100
  246. os.Clearenv()
  247. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  248. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
  249. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
  250. var err error
  251. parser := config.NewConfigParser()
  252. config.Opts, err = parser.ParseEnvironmentVariables()
  253. if err != nil {
  254. t.Fatalf(`Parsing failure: %v`, err)
  255. }
  256. timeBefore := time.Now()
  257. feed := &Feed{}
  258. // Use a very large weekly count to trigger the min interval
  259. weeklyCount := largeWeeklyCount
  260. // TTL is smaller than minInterval.
  261. newTTL := time.Duration(minInterval) * time.Minute / 2
  262. feed.ScheduleNextCheck(weeklyCount, newTTL)
  263. if feed.NextCheckAt.IsZero() {
  264. t.Error(`The next_check_at must be set`)
  265. }
  266. targetInterval := time.Duration(minInterval) * time.Minute
  267. checkTargetInterval(t, feed, targetInterval, timeBefore, "entry frequency min interval")
  268. if feed.NextCheckAt.Before(timeBefore.Add(newTTL)) {
  269. t.Error(`The next_check_at should be after timeBefore + TTL`)
  270. }
  271. }
  272. func TestFeedScheduleNextCheckEntryFrequencyLargeNewTTL(t *testing.T) {
  273. // If the feed has a TTL defined, we use it to make sure we don't check it too often.
  274. maxInterval := 500
  275. minInterval := 100
  276. os.Clearenv()
  277. os.Setenv("POLLING_SCHEDULER", "entry_frequency")
  278. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL", strconv.Itoa(maxInterval))
  279. os.Setenv("SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL", strconv.Itoa(minInterval))
  280. var err error
  281. parser := config.NewConfigParser()
  282. config.Opts, err = parser.ParseEnvironmentVariables()
  283. if err != nil {
  284. t.Fatalf(`Parsing failure: %v`, err)
  285. }
  286. timeBefore := time.Now()
  287. feed := &Feed{}
  288. // Use a very large weekly count to trigger the min interval
  289. weeklyCount := largeWeeklyCount
  290. // TTL is larger than minInterval.
  291. newTTL := time.Duration(minInterval) * time.Minute * 2
  292. feed.ScheduleNextCheck(weeklyCount, newTTL)
  293. if feed.NextCheckAt.IsZero() {
  294. t.Error(`The next_check_at must be set`)
  295. }
  296. targetInterval := newTTL
  297. checkTargetInterval(t, feed, targetInterval, timeBefore, "TTL")
  298. if feed.NextCheckAt.Before(timeBefore.Add(time.Minute * time.Duration(minInterval))) {
  299. t.Error(`The next_check_at should be after timeBefore + entry frequency min interval`)
  300. }
  301. }