entry_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. //go:build integration
  4. // +build integration
  5. package tests
  6. import (
  7. "testing"
  8. miniflux "miniflux.app/v2/client"
  9. )
  10. func TestGetAllFeedEntries(t *testing.T) {
  11. client := createClient(t)
  12. feed, _ := createFeed(t, client)
  13. allResults, err := client.FeedEntries(feed.ID, nil)
  14. if err != nil {
  15. t.Fatal(err)
  16. }
  17. if allResults.Total == 0 {
  18. t.Fatal(`Invalid number of entries`)
  19. }
  20. if allResults.Entries[0].Title == "" {
  21. t.Fatal(`Invalid entry title`)
  22. }
  23. filteredResults, err := client.FeedEntries(feed.ID, &miniflux.Filter{Limit: 1, Offset: 5})
  24. if err != nil {
  25. t.Fatal(err)
  26. }
  27. if allResults.Total != filteredResults.Total {
  28. t.Fatal(`Total should always contains the total number of items regardless of filters`)
  29. }
  30. if allResults.Entries[0].ID == filteredResults.Entries[0].ID {
  31. t.Fatal(`Filtered entries should be different than previous results`)
  32. }
  33. filteredResultsByEntryID, err := client.FeedEntries(feed.ID, &miniflux.Filter{AfterEntryID: allResults.Entries[0].ID})
  34. if err != nil {
  35. t.Fatal(err)
  36. }
  37. if filteredResultsByEntryID.Entries[0].ID == allResults.Entries[0].ID {
  38. t.Fatal(`The first entry should be filtered out`)
  39. }
  40. }
  41. func TestGetAllCategoryEntries(t *testing.T) {
  42. client := createClient(t)
  43. _, category := createFeed(t, client)
  44. allResults, err := client.CategoryEntries(category.ID, nil)
  45. if err != nil {
  46. t.Fatal(err)
  47. }
  48. if allResults.Total == 0 {
  49. t.Fatal(`Invalid number of entries`)
  50. }
  51. if allResults.Entries[0].Title == "" {
  52. t.Fatal(`Invalid entry title`)
  53. }
  54. filteredResults, err := client.CategoryEntries(category.ID, &miniflux.Filter{Limit: 1, Offset: 5})
  55. if err != nil {
  56. t.Fatal(err)
  57. }
  58. if allResults.Total != filteredResults.Total {
  59. t.Fatal(`Total should always contains the total number of items regardless of filters`)
  60. }
  61. if allResults.Entries[0].ID == filteredResults.Entries[0].ID {
  62. t.Fatal(`Filtered entries should be different than previous results`)
  63. }
  64. filteredResultsByEntryID, err := client.CategoryEntries(category.ID, &miniflux.Filter{AfterEntryID: allResults.Entries[0].ID})
  65. if err != nil {
  66. t.Fatal(err)
  67. }
  68. if filteredResultsByEntryID.Entries[0].ID == allResults.Entries[0].ID {
  69. t.Fatal(`The first entry should be filtered out`)
  70. }
  71. }
  72. func TestGetAllEntries(t *testing.T) {
  73. client := createClient(t)
  74. createFeed(t, client)
  75. resultWithoutSorting, err := client.Entries(nil)
  76. if err != nil {
  77. t.Fatal(err)
  78. }
  79. if resultWithoutSorting.Total == 0 {
  80. t.Fatal(`Invalid number of entries`)
  81. }
  82. resultWithStatusFilter, err := client.Entries(&miniflux.Filter{Status: miniflux.EntryStatusRead})
  83. if err != nil {
  84. t.Fatal(err)
  85. }
  86. if resultWithStatusFilter.Total != 0 {
  87. t.Fatal(`We should have 0 read entries`)
  88. }
  89. resultWithDifferentSorting, err := client.Entries(&miniflux.Filter{Order: "published_at", Direction: "desc"})
  90. if err != nil {
  91. t.Fatal(err)
  92. }
  93. if resultWithDifferentSorting.Entries[0].Title == resultWithoutSorting.Entries[0].Title {
  94. t.Fatalf(`The items should be sorted differently "%v" vs "%v"`, resultWithDifferentSorting.Entries[0].Title, resultWithoutSorting.Entries[0].Title)
  95. }
  96. resultWithStarredEntries, err := client.Entries(&miniflux.Filter{Starred: miniflux.FilterOnlyStarred})
  97. if err != nil {
  98. t.Fatal(err)
  99. }
  100. if resultWithStarredEntries.Total != 0 {
  101. t.Fatalf(`We are not supposed to have starred entries yet`)
  102. }
  103. }
  104. func TestFilterEntriesByCategory(t *testing.T) {
  105. client := createClient(t)
  106. category, err := client.CreateCategory("Test Filter by Category")
  107. if err != nil {
  108. t.Fatal(err)
  109. }
  110. feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
  111. FeedURL: testFeedURL,
  112. CategoryID: category.ID,
  113. })
  114. if err != nil {
  115. t.Fatal(err)
  116. }
  117. if feedID == 0 {
  118. t.Fatalf(`Invalid feed ID, got %q`, feedID)
  119. }
  120. results, err := client.Entries(&miniflux.Filter{CategoryID: category.ID})
  121. if err != nil {
  122. t.Fatal(err)
  123. }
  124. if results.Total == 0 {
  125. t.Fatalf(`We should have more than one entry`)
  126. }
  127. if results.Entries[0].Feed.Category == nil {
  128. t.Fatalf(`The entry feed category should not be nil`)
  129. }
  130. if results.Entries[0].Feed.Category.ID != category.ID {
  131. t.Errorf(`Entries should be filtered by category_id=%d`, category.ID)
  132. }
  133. }
  134. func TestFilterEntriesByFeed(t *testing.T) {
  135. client := createClient(t)
  136. category, err := client.CreateCategory("Test Filter by Feed")
  137. if err != nil {
  138. t.Fatal(err)
  139. }
  140. feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
  141. FeedURL: testFeedURL,
  142. CategoryID: category.ID,
  143. })
  144. if err != nil {
  145. t.Fatal(err)
  146. }
  147. if feedID == 0 {
  148. t.Fatalf(`Invalid feed ID, got %q`, feedID)
  149. }
  150. results, err := client.Entries(&miniflux.Filter{FeedID: feedID})
  151. if err != nil {
  152. t.Fatal(err)
  153. }
  154. if results.Total == 0 {
  155. t.Fatalf(`We should have more than one entry`)
  156. }
  157. if results.Entries[0].Feed.Category == nil {
  158. t.Fatalf(`The entry feed category should not be nil`)
  159. }
  160. if results.Entries[0].Feed.Category.ID != category.ID {
  161. t.Errorf(`Entries should be filtered by category_id=%d`, category.ID)
  162. }
  163. }
  164. func TestFilterEntriesByStatuses(t *testing.T) {
  165. client := createClient(t)
  166. category, err := client.CreateCategory("Test Filter by statuses")
  167. if err != nil {
  168. t.Fatal(err)
  169. }
  170. feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
  171. FeedURL: testFeedURL,
  172. CategoryID: category.ID,
  173. })
  174. if err != nil {
  175. t.Fatal(err)
  176. }
  177. if feedID == 0 {
  178. t.Fatalf(`Invalid feed ID, got %q`, feedID)
  179. }
  180. results, err := client.Entries(&miniflux.Filter{FeedID: feedID})
  181. if err != nil {
  182. t.Fatal(err)
  183. }
  184. if err := client.UpdateEntries([]int64{results.Entries[0].ID}, miniflux.EntryStatusRead); err != nil {
  185. t.Fatal(err)
  186. }
  187. if err := client.UpdateEntries([]int64{results.Entries[1].ID}, miniflux.EntryStatusRemoved); err != nil {
  188. t.Fatal(err)
  189. }
  190. results, err = client.Entries(&miniflux.Filter{Statuses: []string{miniflux.EntryStatusRead, miniflux.EntryStatusRemoved}})
  191. if err != nil {
  192. t.Fatal(err)
  193. }
  194. if results.Total != 2 {
  195. t.Fatalf(`We should have 2 entries`)
  196. }
  197. if results.Entries[0].Status != "read" {
  198. t.Errorf(`The first entry has the wrong status: %s`, results.Entries[0].Status)
  199. }
  200. if results.Entries[1].Status != "removed" {
  201. t.Errorf(`The 2nd entry has the wrong status: %s`, results.Entries[1].Status)
  202. }
  203. }
  204. func TestSearchEntries(t *testing.T) {
  205. client := createClient(t)
  206. categories, err := client.Categories()
  207. if err != nil {
  208. t.Fatal(err)
  209. }
  210. feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
  211. FeedURL: testFeedURL,
  212. CategoryID: categories[0].ID,
  213. })
  214. if err != nil {
  215. t.Fatal(err)
  216. }
  217. if feedID == 0 {
  218. t.Fatalf(`Invalid feed ID, got %q`, feedID)
  219. }
  220. results, err := client.Entries(&miniflux.Filter{Search: "2.0.8"})
  221. if err != nil {
  222. t.Fatal(err)
  223. }
  224. if results.Total != 1 {
  225. t.Fatalf(`We should have only one entry instead of %d`, results.Total)
  226. }
  227. }
  228. func TestInvalidFilters(t *testing.T) {
  229. client := createClient(t)
  230. createFeed(t, client)
  231. _, err := client.Entries(&miniflux.Filter{Status: "invalid"})
  232. if err == nil {
  233. t.Fatal(`Using invalid status should raise an error`)
  234. }
  235. _, err = client.Entries(&miniflux.Filter{Direction: "invalid"})
  236. if err == nil {
  237. t.Fatal(`Using invalid direction should raise an error`)
  238. }
  239. _, err = client.Entries(&miniflux.Filter{Order: "invalid"})
  240. if err == nil {
  241. t.Fatal(`Using invalid order should raise an error`)
  242. }
  243. }
  244. func TestGetFeedEntry(t *testing.T) {
  245. client := createClient(t)
  246. createFeed(t, client)
  247. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  248. if err != nil {
  249. t.Fatal(err)
  250. }
  251. // Test get entry by entry id and feed id
  252. entry, err := client.FeedEntry(result.Entries[0].FeedID, result.Entries[0].ID)
  253. if err != nil {
  254. t.Fatal(err)
  255. }
  256. if entry.ID != result.Entries[0].ID {
  257. t.Fatal("Wrong entry returned")
  258. }
  259. }
  260. func TestGetCategoryEntry(t *testing.T) {
  261. client := createClient(t)
  262. _, category := createFeed(t, client)
  263. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  264. if err != nil {
  265. t.Fatal(err)
  266. }
  267. // Test get entry by entry id and category id
  268. entry, err := client.CategoryEntry(category.ID, result.Entries[0].ID)
  269. if err != nil {
  270. t.Fatal(err)
  271. }
  272. if entry.ID != result.Entries[0].ID {
  273. t.Fatal("Wrong entry returned")
  274. }
  275. }
  276. func TestGetEntry(t *testing.T) {
  277. client := createClient(t)
  278. createFeed(t, client)
  279. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  280. if err != nil {
  281. t.Fatal(err)
  282. }
  283. // Test get entry by entry id only
  284. entry, err := client.Entry(result.Entries[0].ID)
  285. if err != nil {
  286. t.Fatal(err)
  287. }
  288. if entry.ID != result.Entries[0].ID {
  289. t.Fatal("Wrong entry returned")
  290. }
  291. }
  292. func TestUpdateStatus(t *testing.T) {
  293. client := createClient(t)
  294. createFeed(t, client)
  295. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  296. if err != nil {
  297. t.Fatal(err)
  298. }
  299. err = client.UpdateEntries([]int64{result.Entries[0].ID}, miniflux.EntryStatusRead)
  300. if err != nil {
  301. t.Fatal(err)
  302. }
  303. entry, err := client.Entry(result.Entries[0].ID)
  304. if err != nil {
  305. t.Fatal(err)
  306. }
  307. if entry.Status != miniflux.EntryStatusRead {
  308. t.Fatal("The entry status should be updated")
  309. }
  310. err = client.UpdateEntries([]int64{result.Entries[0].ID}, "invalid")
  311. if err == nil {
  312. t.Fatal(`Invalid entry status should not be accepted`)
  313. }
  314. err = client.UpdateEntries([]int64{}, miniflux.EntryStatusRead)
  315. if err == nil {
  316. t.Fatal(`An empty list of entry should not be accepted`)
  317. }
  318. }
  319. func TestUpdateEntry(t *testing.T) {
  320. client := createClient(t)
  321. createFeed(t, client)
  322. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  323. if err != nil {
  324. t.Fatal(err)
  325. }
  326. title := "New title"
  327. content := "New content"
  328. _, err = client.UpdateEntry(result.Entries[0].ID, &miniflux.EntryModificationRequest{
  329. Title: &title,
  330. Content: &content,
  331. })
  332. if err != nil {
  333. t.Fatal(err)
  334. }
  335. entry, err := client.Entry(result.Entries[0].ID)
  336. if err != nil {
  337. t.Fatal(err)
  338. }
  339. if entry.Title != title {
  340. t.Fatal("The entry title should be updated")
  341. }
  342. if entry.Content != content {
  343. t.Fatal("The entry content should be updated")
  344. }
  345. }
  346. func TestToggleBookmark(t *testing.T) {
  347. client := createClient(t)
  348. createFeed(t, client)
  349. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  350. if err != nil {
  351. t.Fatal(err)
  352. }
  353. if result.Entries[0].Starred {
  354. t.Fatal("The entry should not be starred")
  355. }
  356. err = client.ToggleBookmark(result.Entries[0].ID)
  357. if err != nil {
  358. t.Fatal(err)
  359. }
  360. entry, err := client.Entry(result.Entries[0].ID)
  361. if err != nil {
  362. t.Fatal(err)
  363. }
  364. if !entry.Starred {
  365. t.Fatal("The entry should be starred")
  366. }
  367. }
  368. func TestHistoryOrder(t *testing.T) {
  369. client := createClient(t)
  370. createFeed(t, client)
  371. result, err := client.Entries(&miniflux.Filter{Limit: 3})
  372. if err != nil {
  373. t.Fatal(err)
  374. }
  375. selectedEntryID := result.Entries[2].ID
  376. err = client.UpdateEntries([]int64{selectedEntryID}, miniflux.EntryStatusRead)
  377. if err != nil {
  378. t.Fatal(err)
  379. }
  380. history, err := client.Entries(&miniflux.Filter{Order: "changed_at", Direction: "desc", Limit: 1})
  381. if err != nil {
  382. t.Fatal(err)
  383. }
  384. if history.Entries[0].ID != selectedEntryID {
  385. t.Fatal("The entry that we just read should be at the top of the history")
  386. }
  387. }
  388. func TestFlushHistory(t *testing.T) {
  389. client := createClient(t)
  390. createFeed(t, client)
  391. result, err := client.Entries(&miniflux.Filter{Limit: 1})
  392. if err != nil {
  393. t.Fatal(err)
  394. }
  395. selectedEntryID := result.Entries[0].ID
  396. err = client.UpdateEntries([]int64{selectedEntryID}, miniflux.EntryStatusRead)
  397. if err != nil {
  398. t.Fatal(err)
  399. }
  400. err = client.FlushHistory()
  401. if err != nil {
  402. t.Fatal(err)
  403. }
  404. history, err := client.Entries(&miniflux.Filter{Status: miniflux.EntryStatusRemoved})
  405. if err != nil {
  406. t.Fatal(err)
  407. }
  408. if history.Entries[0].ID != selectedEntryID {
  409. t.Fatal("The entry that we just read should have the removed status")
  410. }
  411. }