enclosure.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  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. "fmt"
  7. "strings"
  8. "miniflux.app/v2/internal/model"
  9. "github.com/lib/pq"
  10. )
  11. // GetEnclosures returns all attachments for the given entry.
  12. func (s *Storage) GetEnclosures(entryID int64) (model.EnclosureList, error) {
  13. query := `
  14. SELECT
  15. id,
  16. user_id,
  17. entry_id,
  18. url,
  19. size,
  20. mime_type,
  21. media_progression
  22. FROM
  23. enclosures
  24. WHERE
  25. entry_id = $1
  26. ORDER BY id ASC
  27. `
  28. rows, err := s.db.Query(query, entryID)
  29. if err != nil {
  30. return nil, fmt.Errorf(`store: unable to fetch enclosures: %v`, err)
  31. }
  32. defer rows.Close()
  33. enclosures := make(model.EnclosureList, 0)
  34. for rows.Next() {
  35. var enclosure model.Enclosure
  36. err := rows.Scan(
  37. &enclosure.ID,
  38. &enclosure.UserID,
  39. &enclosure.EntryID,
  40. &enclosure.URL,
  41. &enclosure.Size,
  42. &enclosure.MimeType,
  43. &enclosure.MediaProgression,
  44. )
  45. if err != nil {
  46. return nil, fmt.Errorf(`store: unable to fetch enclosure row: %v`, err)
  47. }
  48. enclosures = append(enclosures, &enclosure)
  49. }
  50. return enclosures, nil
  51. }
  52. func (s *Storage) GetEnclosure(enclosureID int64) (*model.Enclosure, error) {
  53. query := `
  54. SELECT
  55. id,
  56. user_id,
  57. entry_id,
  58. url,
  59. size,
  60. mime_type,
  61. media_progression
  62. FROM
  63. enclosures
  64. WHERE
  65. id = $1
  66. ORDER BY id ASC
  67. `
  68. row := s.db.QueryRow(query, enclosureID)
  69. var enclosure model.Enclosure
  70. err := row.Scan(
  71. &enclosure.ID,
  72. &enclosure.UserID,
  73. &enclosure.EntryID,
  74. &enclosure.URL,
  75. &enclosure.Size,
  76. &enclosure.MimeType,
  77. &enclosure.MediaProgression,
  78. )
  79. if err != nil {
  80. return nil, fmt.Errorf(`store: unable to fetch enclosure row: %v`, err)
  81. }
  82. return &enclosure, nil
  83. }
  84. func (s *Storage) createEnclosure(tx *sql.Tx, enclosure *model.Enclosure) error {
  85. enclosureURL := strings.TrimSpace(enclosure.URL)
  86. if enclosureURL == "" {
  87. return nil
  88. }
  89. query := `
  90. INSERT INTO enclosures
  91. (url, size, mime_type, entry_id, user_id, media_progression)
  92. VALUES
  93. ($1, $2, $3, $4, $5, $6)
  94. ON CONFLICT (user_id, entry_id, md5(url)) DO NOTHING
  95. RETURNING
  96. id
  97. `
  98. if err := tx.QueryRow(
  99. query,
  100. enclosureURL,
  101. enclosure.Size,
  102. enclosure.MimeType,
  103. enclosure.EntryID,
  104. enclosure.UserID,
  105. enclosure.MediaProgression,
  106. ).Scan(&enclosure.ID); err != nil && err != sql.ErrNoRows {
  107. return fmt.Errorf(`store: unable to create enclosure: %w`, err)
  108. }
  109. return nil
  110. }
  111. func (s *Storage) updateEnclosures(tx *sql.Tx, entry *model.Entry) error {
  112. if len(entry.Enclosures) == 0 {
  113. return nil
  114. }
  115. sqlValues := make([]string, 0, len(entry.Enclosures))
  116. for _, enclosure := range entry.Enclosures {
  117. sqlValues = append(sqlValues, strings.TrimSpace(enclosure.URL))
  118. if err := s.createEnclosure(tx, enclosure); err != nil {
  119. return err
  120. }
  121. }
  122. query := `
  123. DELETE FROM
  124. enclosures
  125. WHERE
  126. user_id=$1 AND entry_id=$2 AND url <> ALL($3)
  127. `
  128. _, err := tx.Exec(query, entry.UserID, entry.ID, pq.Array(sqlValues))
  129. if err != nil {
  130. return fmt.Errorf(`store: unable to delete old enclosures: %v`, err)
  131. }
  132. return nil
  133. }
  134. func (s *Storage) UpdateEnclosure(enclosure *model.Enclosure) error {
  135. query := `
  136. UPDATE
  137. enclosures
  138. SET
  139. url=$1,
  140. size=$2,
  141. mime_type=$3,
  142. entry_id=$4,
  143. user_id=$5,
  144. media_progression=$6
  145. WHERE
  146. id=$7
  147. `
  148. _, err := s.db.Exec(query,
  149. enclosure.URL,
  150. enclosure.Size,
  151. enclosure.MimeType,
  152. enclosure.EntryID,
  153. enclosure.UserID,
  154. enclosure.MediaProgression,
  155. enclosure.ID,
  156. )
  157. if err != nil {
  158. return fmt.Errorf(`store: unable to update enclosure #%d : %v`, enclosure.ID, err)
  159. }
  160. return nil
  161. }