enclosure.go 3.7 KB

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