feed.go 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  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. "fmt"
  6. "io"
  7. "math"
  8. "time"
  9. "miniflux.app/v2/internal/config"
  10. )
  11. // List of supported schedulers.
  12. const (
  13. SchedulerRoundRobin = "round_robin"
  14. SchedulerEntryFrequency = "entry_frequency"
  15. // Default settings for the feed query builder
  16. DefaultFeedSorting = "parsing_error_count"
  17. DefaultFeedSortingDirection = "desc"
  18. )
  19. // Feed represents a feed in the application.
  20. type Feed struct {
  21. ID int64 `json:"id"`
  22. UserID int64 `json:"user_id"`
  23. FeedURL string `json:"feed_url"`
  24. SiteURL string `json:"site_url"`
  25. Title string `json:"title"`
  26. Description string `json:"description"`
  27. CheckedAt time.Time `json:"checked_at"`
  28. NextCheckAt time.Time `json:"next_check_at"`
  29. EtagHeader string `json:"etag_header"`
  30. LastModifiedHeader string `json:"last_modified_header"`
  31. ParsingErrorMsg string `json:"parsing_error_message"`
  32. ParsingErrorCount int `json:"parsing_error_count"`
  33. ScraperRules string `json:"scraper_rules"`
  34. RewriteRules string `json:"rewrite_rules"`
  35. Crawler bool `json:"crawler"`
  36. BlocklistRules string `json:"blocklist_rules"`
  37. KeeplistRules string `json:"keeplist_rules"`
  38. UrlRewriteRules string `json:"urlrewrite_rules"`
  39. UserAgent string `json:"user_agent"`
  40. Cookie string `json:"cookie"`
  41. Username string `json:"username"`
  42. Password string `json:"password"`
  43. Disabled bool `json:"disabled"`
  44. NoMediaPlayer bool `json:"no_media_player"`
  45. IgnoreHTTPCache bool `json:"ignore_http_cache"`
  46. AllowSelfSignedCertificates bool `json:"allow_self_signed_certificates"`
  47. FetchViaProxy bool `json:"fetch_via_proxy"`
  48. HideGlobally bool `json:"hide_globally"`
  49. DisableHTTP2 bool `json:"disable_http2"`
  50. AppriseServiceURLs string `json:"apprise_service_urls"`
  51. WebhookURL string `json:"webhook_url"`
  52. NtfyEnabled bool `json:"ntfy_enabled"`
  53. NtfyPriority int `json:"ntfy_priority"`
  54. NtfyTopic string `json:"ntfy_topic"`
  55. PushoverEnabled bool `json:"pushover_enabled,omitempty"`
  56. PushoverPriority int `json:"pushover_priority,omitempty"`
  57. // Non-persisted attributes
  58. Category *Category `json:"category,omitempty"`
  59. Icon *FeedIcon `json:"icon"`
  60. Entries Entries `json:"entries,omitempty"`
  61. // Internal attributes (not exposed in the API and not persisted in the database)
  62. TTL int `json:"-"`
  63. IconURL string `json:"-"`
  64. UnreadCount int `json:"-"`
  65. ReadCount int `json:"-"`
  66. NumberOfVisibleEntries int `json:"-"`
  67. }
  68. type FeedCounters struct {
  69. ReadCounters map[int64]int `json:"reads"`
  70. UnreadCounters map[int64]int `json:"unreads"`
  71. }
  72. func (f *Feed) String() string {
  73. return fmt.Sprintf("ID=%d, UserID=%d, FeedURL=%s, SiteURL=%s, Title=%s, Category={%s}",
  74. f.ID,
  75. f.UserID,
  76. f.FeedURL,
  77. f.SiteURL,
  78. f.Title,
  79. f.Category,
  80. )
  81. }
  82. // WithCategoryID initializes the category attribute of the feed.
  83. func (f *Feed) WithCategoryID(categoryID int64) {
  84. f.Category = &Category{ID: categoryID}
  85. }
  86. // WithTranslatedErrorMessage adds a new error message and increment the error counter.
  87. func (f *Feed) WithTranslatedErrorMessage(message string) {
  88. f.ParsingErrorCount++
  89. f.ParsingErrorMsg = message
  90. }
  91. // ResetErrorCounter removes all previous errors.
  92. func (f *Feed) ResetErrorCounter() {
  93. f.ParsingErrorCount = 0
  94. f.ParsingErrorMsg = ""
  95. }
  96. // CheckedNow set attribute values when the feed is refreshed.
  97. func (f *Feed) CheckedNow() {
  98. f.CheckedAt = time.Now()
  99. if f.SiteURL == "" {
  100. f.SiteURL = f.FeedURL
  101. }
  102. }
  103. // ScheduleNextCheck set "next_check_at" of a feed based on the scheduler selected from the configuration.
  104. func (f *Feed) ScheduleNextCheck(weeklyCount int, refreshDelayInMinutes int) {
  105. f.TTL = refreshDelayInMinutes
  106. // Default to the global config Polling Frequency.
  107. intervalMinutes := config.Opts.SchedulerRoundRobinMinInterval()
  108. if config.Opts.PollingScheduler() == SchedulerEntryFrequency {
  109. if weeklyCount <= 0 {
  110. intervalMinutes = config.Opts.SchedulerEntryFrequencyMaxInterval()
  111. } else {
  112. intervalMinutes = int(math.Round(float64(7*24*60) / float64(weeklyCount*config.Opts.SchedulerEntryFrequencyFactor())))
  113. intervalMinutes = min(intervalMinutes, config.Opts.SchedulerEntryFrequencyMaxInterval())
  114. intervalMinutes = max(intervalMinutes, config.Opts.SchedulerEntryFrequencyMinInterval())
  115. }
  116. }
  117. // If the feed has a TTL or a Retry-After defined, we use it to make sure we don't check it too often.
  118. if refreshDelayInMinutes > 0 && refreshDelayInMinutes > intervalMinutes {
  119. intervalMinutes = refreshDelayInMinutes
  120. }
  121. f.NextCheckAt = time.Now().Add(time.Minute * time.Duration(intervalMinutes))
  122. }
  123. // FeedCreationRequest represents the request to create a feed.
  124. type FeedCreationRequest struct {
  125. FeedURL string `json:"feed_url"`
  126. CategoryID int64 `json:"category_id"`
  127. UserAgent string `json:"user_agent"`
  128. Cookie string `json:"cookie"`
  129. Username string `json:"username"`
  130. Password string `json:"password"`
  131. Crawler bool `json:"crawler"`
  132. Disabled bool `json:"disabled"`
  133. NoMediaPlayer bool `json:"no_media_player"`
  134. IgnoreHTTPCache bool `json:"ignore_http_cache"`
  135. AllowSelfSignedCertificates bool `json:"allow_self_signed_certificates"`
  136. FetchViaProxy bool `json:"fetch_via_proxy"`
  137. ScraperRules string `json:"scraper_rules"`
  138. RewriteRules string `json:"rewrite_rules"`
  139. BlocklistRules string `json:"blocklist_rules"`
  140. KeeplistRules string `json:"keeplist_rules"`
  141. HideGlobally bool `json:"hide_globally"`
  142. UrlRewriteRules string `json:"urlrewrite_rules"`
  143. DisableHTTP2 bool `json:"disable_http2"`
  144. }
  145. type FeedCreationRequestFromSubscriptionDiscovery struct {
  146. Content io.ReadSeeker
  147. ETag string
  148. LastModified string
  149. FeedCreationRequest
  150. }
  151. // FeedModificationRequest represents the request to update a feed.
  152. type FeedModificationRequest struct {
  153. FeedURL *string `json:"feed_url"`
  154. SiteURL *string `json:"site_url"`
  155. Title *string `json:"title"`
  156. Description *string `json:"description"`
  157. ScraperRules *string `json:"scraper_rules"`
  158. RewriteRules *string `json:"rewrite_rules"`
  159. BlocklistRules *string `json:"blocklist_rules"`
  160. KeeplistRules *string `json:"keeplist_rules"`
  161. UrlRewriteRules *string `json:"urlrewrite_rules"`
  162. Crawler *bool `json:"crawler"`
  163. UserAgent *string `json:"user_agent"`
  164. Cookie *string `json:"cookie"`
  165. Username *string `json:"username"`
  166. Password *string `json:"password"`
  167. CategoryID *int64 `json:"category_id"`
  168. Disabled *bool `json:"disabled"`
  169. NoMediaPlayer *bool `json:"no_media_player"`
  170. IgnoreHTTPCache *bool `json:"ignore_http_cache"`
  171. AllowSelfSignedCertificates *bool `json:"allow_self_signed_certificates"`
  172. FetchViaProxy *bool `json:"fetch_via_proxy"`
  173. HideGlobally *bool `json:"hide_globally"`
  174. DisableHTTP2 *bool `json:"disable_http2"`
  175. }
  176. // Patch updates a feed with modified values.
  177. func (f *FeedModificationRequest) Patch(feed *Feed) {
  178. if f.FeedURL != nil && *f.FeedURL != "" {
  179. feed.FeedURL = *f.FeedURL
  180. }
  181. if f.SiteURL != nil && *f.SiteURL != "" {
  182. feed.SiteURL = *f.SiteURL
  183. }
  184. if f.Title != nil && *f.Title != "" {
  185. feed.Title = *f.Title
  186. }
  187. if f.Description != nil && *f.Description != "" {
  188. feed.Description = *f.Description
  189. }
  190. if f.ScraperRules != nil {
  191. feed.ScraperRules = *f.ScraperRules
  192. }
  193. if f.RewriteRules != nil {
  194. feed.RewriteRules = *f.RewriteRules
  195. }
  196. if f.KeeplistRules != nil {
  197. feed.KeeplistRules = *f.KeeplistRules
  198. }
  199. if f.UrlRewriteRules != nil {
  200. feed.UrlRewriteRules = *f.UrlRewriteRules
  201. }
  202. if f.BlocklistRules != nil {
  203. feed.BlocklistRules = *f.BlocklistRules
  204. }
  205. if f.Crawler != nil {
  206. feed.Crawler = *f.Crawler
  207. }
  208. if f.UserAgent != nil {
  209. feed.UserAgent = *f.UserAgent
  210. }
  211. if f.Cookie != nil {
  212. feed.Cookie = *f.Cookie
  213. }
  214. if f.Username != nil {
  215. feed.Username = *f.Username
  216. }
  217. if f.Password != nil {
  218. feed.Password = *f.Password
  219. }
  220. if f.CategoryID != nil && *f.CategoryID > 0 {
  221. feed.Category.ID = *f.CategoryID
  222. }
  223. if f.Disabled != nil {
  224. feed.Disabled = *f.Disabled
  225. }
  226. if f.NoMediaPlayer != nil {
  227. feed.NoMediaPlayer = *f.NoMediaPlayer
  228. }
  229. if f.IgnoreHTTPCache != nil {
  230. feed.IgnoreHTTPCache = *f.IgnoreHTTPCache
  231. }
  232. if f.AllowSelfSignedCertificates != nil {
  233. feed.AllowSelfSignedCertificates = *f.AllowSelfSignedCertificates
  234. }
  235. if f.FetchViaProxy != nil {
  236. feed.FetchViaProxy = *f.FetchViaProxy
  237. }
  238. if f.HideGlobally != nil {
  239. feed.HideGlobally = *f.HideGlobally
  240. }
  241. if f.DisableHTTP2 != nil {
  242. feed.DisableHTTP2 = *f.DisableHTTP2
  243. }
  244. }
  245. // Feeds is a list of feed
  246. type Feeds []*Feed