enclosure.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package storage // import "miniflux.app/v2/internal/storage"
  4. import (
  5. "database/sql"
  6. "errors"
  7. "fmt"
  8. "strings"
  9. "miniflux.app/v2/internal/model"
  10. "github.com/lib/pq"
  11. )
  12. // EnclosuresByEntryID returns all enclosures for the given entry.
  13. func (s *Storage) EnclosuresByEntryID(entryID int64) (model.EnclosureList, error) {
  14. query := `
  15. SELECT
  16. id,
  17. user_id,
  18. entry_id,
  19. url,
  20. size,
  21. mime_type,
  22. media_progression
  23. FROM
  24. enclosures
  25. WHERE
  26. entry_id = $1
  27. ORDER BY id ASC
  28. `
  29. rows, err := s.db.Query(query, entryID)
  30. if err != nil {
  31. return nil, fmt.Errorf(`store: unable to fetch enclosures: %v`, err)
  32. }
  33. defer rows.Close()
  34. enclosures := make(model.EnclosureList, 0)
  35. for rows.Next() {
  36. var enclosure model.Enclosure
  37. err := rows.Scan(
  38. &enclosure.ID,
  39. &enclosure.UserID,
  40. &enclosure.EntryID,
  41. &enclosure.URL,
  42. &enclosure.Size,
  43. &enclosure.MimeType,
  44. &enclosure.MediaProgression,
  45. )
  46. if err != nil {
  47. return nil, fmt.Errorf(`store: unable to fetch enclosure row: %v`, err)
  48. }
  49. enclosures = append(enclosures, &enclosure)
  50. }
  51. return enclosures, nil
  52. }
  53. // EnclosuresByEntryIDs returns enclosures for the given entries, grouped by entry ID.
  54. func (s *Storage) EnclosuresByEntryIDs(entryIDs []int64) (map[int64]model.EnclosureList, error) {
  55. query := `
  56. SELECT
  57. id,
  58. user_id,
  59. entry_id,
  60. url,
  61. size,
  62. mime_type,
  63. media_progression
  64. FROM
  65. enclosures
  66. WHERE
  67. entry_id = ANY($1)
  68. ORDER BY id ASC
  69. `
  70. rows, err := s.db.Query(query, pq.Array(entryIDs))
  71. if err != nil {
  72. return nil, fmt.Errorf("store: unable to fetch enclosures: %w", err)
  73. }
  74. defer rows.Close()
  75. enclosuresMap := make(map[int64]model.EnclosureList)
  76. for rows.Next() {
  77. var enclosure model.Enclosure
  78. err := rows.Scan(
  79. &enclosure.ID,
  80. &enclosure.UserID,
  81. &enclosure.EntryID,
  82. &enclosure.URL,
  83. &enclosure.Size,
  84. &enclosure.MimeType,
  85. &enclosure.MediaProgression,
  86. )
  87. if err != nil {
  88. return nil, fmt.Errorf("store: unable to scan enclosure row: %w", err)
  89. }
  90. enclosuresMap[enclosure.EntryID] = append(enclosuresMap[enclosure.EntryID], &enclosure)
  91. }
  92. return enclosuresMap, nil
  93. }
  94. // EnclosureByID returns the enclosure for the given user and enclosure ID.
  95. func (s *Storage) EnclosureByID(userID, enclosureID int64) (*model.Enclosure, error) {
  96. query := `
  97. SELECT
  98. id,
  99. user_id,
  100. entry_id,
  101. url,
  102. size,
  103. mime_type,
  104. media_progression
  105. FROM
  106. enclosures
  107. WHERE
  108. id = $1 AND user_id = $2
  109. `
  110. row := s.db.QueryRow(query, enclosureID, userID)
  111. var enclosure model.Enclosure
  112. err := row.Scan(
  113. &enclosure.ID,
  114. &enclosure.UserID,
  115. &enclosure.EntryID,
  116. &enclosure.URL,
  117. &enclosure.Size,
  118. &enclosure.MimeType,
  119. &enclosure.MediaProgression,
  120. )
  121. if errors.Is(err, sql.ErrNoRows) {
  122. return nil, nil
  123. } else if err != nil {
  124. return nil, fmt.Errorf(`store: unable to fetch enclosure row: %v`, err)
  125. }
  126. return &enclosure, nil
  127. }
  128. func (s *Storage) createEnclosure(tx *sql.Tx, enclosure *model.Enclosure) error {
  129. enclosureURL := strings.TrimSpace(enclosure.URL)
  130. if enclosureURL == "" {
  131. return nil
  132. }
  133. query := `
  134. INSERT INTO enclosures
  135. (url, size, mime_type, entry_id, user_id, media_progression)
  136. VALUES
  137. ($1, $2, $3, $4, $5, $6)
  138. ON CONFLICT (user_id, entry_id, encode(sha256(url::bytea), 'hex')) DO NOTHING
  139. RETURNING
  140. id
  141. `
  142. if err := tx.QueryRow(
  143. query,
  144. enclosureURL,
  145. enclosure.Size,
  146. enclosure.MimeType,
  147. enclosure.EntryID,
  148. enclosure.UserID,
  149. enclosure.MediaProgression,
  150. ).Scan(&enclosure.ID); err != nil && err != sql.ErrNoRows {
  151. return fmt.Errorf(`store: unable to create enclosure: %w`, err)
  152. }
  153. return nil
  154. }
  155. func (s *Storage) updateEnclosures(tx *sql.Tx, entry *model.Entry) error {
  156. if len(entry.Enclosures) == 0 {
  157. // Do not keep any old enclosures if there is none in the updated entry.
  158. query := `
  159. DELETE FROM
  160. enclosures
  161. WHERE
  162. user_id=$1 AND entry_id=$2
  163. `
  164. _, err := tx.Exec(query, entry.UserID, entry.ID)
  165. if err != nil {
  166. return fmt.Errorf(`store: unable to delete old enclosures: %v`, err)
  167. }
  168. return nil
  169. }
  170. sqlValues := make([]string, 0, len(entry.Enclosures))
  171. for _, enclosure := range entry.Enclosures {
  172. sqlValues = append(sqlValues, strings.TrimSpace(enclosure.URL))
  173. if err := s.createEnclosure(tx, enclosure); err != nil {
  174. return err
  175. }
  176. }
  177. query := `
  178. DELETE FROM
  179. enclosures
  180. WHERE
  181. user_id=$1 AND entry_id=$2 AND url <> ALL($3)
  182. `
  183. _, err := tx.Exec(query, entry.UserID, entry.ID, pq.Array(sqlValues))
  184. if err != nil {
  185. return fmt.Errorf(`store: unable to delete old enclosures: %v`, err)
  186. }
  187. return nil
  188. }
  189. // UpdateEnclosure persists changes to the given enclosure.
  190. func (s *Storage) UpdateEnclosure(enclosure *model.Enclosure) error {
  191. query := `
  192. UPDATE
  193. enclosures
  194. SET
  195. url=$1,
  196. size=$2,
  197. mime_type=$3,
  198. entry_id=$4,
  199. user_id=$5,
  200. media_progression=$6
  201. WHERE
  202. id=$7
  203. `
  204. _, err := s.db.Exec(query,
  205. enclosure.URL,
  206. enclosure.Size,
  207. enclosure.MimeType,
  208. enclosure.EntryID,
  209. enclosure.UserID,
  210. enclosure.MediaProgression,
  211. enclosure.ID,
  212. )
  213. if err != nil {
  214. return fmt.Errorf(`store: unable to update enclosure #%d : %v`, enclosure.ID, err)
  215. }
  216. return nil
  217. }