2023-06-19 23:42:47 +02:00
|
|
|
// SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2018-08-25 20:53:14 +02:00
|
|
|
|
2022-04-14 06:53:06 +02:00
|
|
|
//go:build integration
|
2018-08-25 20:53:14 +02:00
|
|
|
// +build integration
|
|
|
|
|
|
|
|
package tests
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
miniflux "miniflux.app/client"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestGetAllFeedEntries(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
feed, _ := createFeed(t, client)
|
|
|
|
|
|
|
|
allResults, err := client.FeedEntries(feed.ID, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Total == 0 {
|
|
|
|
t.Fatal(`Invalid number of entries`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Entries[0].Title == "" {
|
|
|
|
t.Fatal(`Invalid entry title`)
|
|
|
|
}
|
|
|
|
|
|
|
|
filteredResults, err := client.FeedEntries(feed.ID, &miniflux.Filter{Limit: 1, Offset: 5})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Total != filteredResults.Total {
|
|
|
|
t.Fatal(`Total should always contains the total number of items regardless of filters`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Entries[0].ID == filteredResults.Entries[0].ID {
|
|
|
|
t.Fatal(`Filtered entries should be different than previous results`)
|
|
|
|
}
|
|
|
|
|
2023-03-01 17:58:01 +01:00
|
|
|
filteredResultsByEntryID, err := client.FeedEntries(feed.ID, &miniflux.Filter{AfterEntryID: allResults.Entries[0].ID})
|
2018-08-25 20:53:14 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if filteredResultsByEntryID.Entries[0].ID == allResults.Entries[0].ID {
|
2021-01-19 04:44:02 +01:00
|
|
|
t.Fatal(`The first entry should be filtered out`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetAllCategoryEntries(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
_, category := createFeed(t, client)
|
|
|
|
|
|
|
|
allResults, err := client.CategoryEntries(category.ID, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Total == 0 {
|
|
|
|
t.Fatal(`Invalid number of entries`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Entries[0].Title == "" {
|
|
|
|
t.Fatal(`Invalid entry title`)
|
|
|
|
}
|
|
|
|
|
|
|
|
filteredResults, err := client.CategoryEntries(category.ID, &miniflux.Filter{Limit: 1, Offset: 5})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Total != filteredResults.Total {
|
|
|
|
t.Fatal(`Total should always contains the total number of items regardless of filters`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if allResults.Entries[0].ID == filteredResults.Entries[0].ID {
|
|
|
|
t.Fatal(`Filtered entries should be different than previous results`)
|
|
|
|
}
|
|
|
|
|
2023-03-01 17:58:01 +01:00
|
|
|
filteredResultsByEntryID, err := client.CategoryEntries(category.ID, &miniflux.Filter{AfterEntryID: allResults.Entries[0].ID})
|
2021-01-19 04:44:02 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if filteredResultsByEntryID.Entries[0].ID == allResults.Entries[0].ID {
|
|
|
|
t.Fatal(`The first entry should be filtered out`)
|
2018-08-25 20:53:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetAllEntries(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
resultWithoutSorting, err := client.Entries(nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if resultWithoutSorting.Total == 0 {
|
|
|
|
t.Fatal(`Invalid number of entries`)
|
|
|
|
}
|
|
|
|
|
|
|
|
resultWithStatusFilter, err := client.Entries(&miniflux.Filter{Status: miniflux.EntryStatusRead})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if resultWithStatusFilter.Total != 0 {
|
|
|
|
t.Fatal(`We should have 0 read entries`)
|
|
|
|
}
|
|
|
|
|
|
|
|
resultWithDifferentSorting, err := client.Entries(&miniflux.Filter{Order: "published_at", Direction: "desc"})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if resultWithDifferentSorting.Entries[0].Title == resultWithoutSorting.Entries[0].Title {
|
|
|
|
t.Fatalf(`The items should be sorted differently "%v" vs "%v"`, resultWithDifferentSorting.Entries[0].Title, resultWithoutSorting.Entries[0].Title)
|
|
|
|
}
|
|
|
|
|
2022-04-14 06:53:06 +02:00
|
|
|
resultWithStarredEntries, err := client.Entries(&miniflux.Filter{Starred: miniflux.FilterOnlyStarred})
|
2018-08-25 20:53:14 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if resultWithStarredEntries.Total != 0 {
|
|
|
|
t.Fatalf(`We are not supposed to have starred entries yet`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-18 07:53:11 +01:00
|
|
|
func TestFilterEntriesByCategory(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
category, err := client.CreateCategory("Test Filter by Category")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-03 01:33:41 +01:00
|
|
|
feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
|
|
|
|
FeedURL: testFeedURL,
|
|
|
|
CategoryID: category.ID,
|
|
|
|
})
|
2019-11-18 07:53:11 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if feedID == 0 {
|
|
|
|
t.Fatalf(`Invalid feed ID, got %q`, feedID)
|
|
|
|
}
|
|
|
|
|
|
|
|
results, err := client.Entries(&miniflux.Filter{CategoryID: category.ID})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Total == 0 {
|
|
|
|
t.Fatalf(`We should have more than one entry`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[0].Feed.Category == nil {
|
|
|
|
t.Fatalf(`The entry feed category should not be nil`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[0].Feed.Category.ID != category.ID {
|
|
|
|
t.Errorf(`Entries should be filtered by category_id=%d`, category.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
func TestFilterEntriesByFeed(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
category, err := client.CreateCategory("Test Filter by Feed")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
|
|
|
|
FeedURL: testFeedURL,
|
|
|
|
CategoryID: category.ID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if feedID == 0 {
|
|
|
|
t.Fatalf(`Invalid feed ID, got %q`, feedID)
|
|
|
|
}
|
|
|
|
|
|
|
|
results, err := client.Entries(&miniflux.Filter{FeedID: feedID})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Total == 0 {
|
|
|
|
t.Fatalf(`We should have more than one entry`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[0].Feed.Category == nil {
|
|
|
|
t.Fatalf(`The entry feed category should not be nil`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[0].Feed.Category.ID != category.ID {
|
|
|
|
t.Errorf(`Entries should be filtered by category_id=%d`, category.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-15 05:03:24 +02:00
|
|
|
func TestFilterEntriesByStatuses(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
category, err := client.CreateCategory("Test Filter by statuses")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-03 01:33:41 +01:00
|
|
|
feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
|
|
|
|
FeedURL: testFeedURL,
|
|
|
|
CategoryID: category.ID,
|
|
|
|
})
|
2020-09-15 05:03:24 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if feedID == 0 {
|
|
|
|
t.Fatalf(`Invalid feed ID, got %q`, feedID)
|
|
|
|
}
|
|
|
|
|
|
|
|
results, err := client.Entries(&miniflux.Filter{FeedID: feedID})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-05 00:32:32 +01:00
|
|
|
if err := client.UpdateEntries([]int64{results.Entries[0].ID}, miniflux.EntryStatusRead); err != nil {
|
2020-09-15 05:03:24 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-05 00:32:32 +01:00
|
|
|
if err := client.UpdateEntries([]int64{results.Entries[1].ID}, miniflux.EntryStatusRemoved); err != nil {
|
2020-09-15 05:03:24 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-05 00:32:32 +01:00
|
|
|
results, err = client.Entries(&miniflux.Filter{Statuses: []string{miniflux.EntryStatusRead, miniflux.EntryStatusRemoved}})
|
2020-09-15 05:03:24 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Total != 2 {
|
|
|
|
t.Fatalf(`We should have 2 entries`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[0].Status != "read" {
|
|
|
|
t.Errorf(`The first entry has the wrong status: %s`, results.Entries[0].Status)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Entries[1].Status != "removed" {
|
|
|
|
t.Errorf(`The 2nd entry has the wrong status: %s`, results.Entries[1].Status)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-25 20:53:14 +02:00
|
|
|
func TestSearchEntries(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
categories, err := client.Categories()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-03 01:33:41 +01:00
|
|
|
feedID, err := client.CreateFeed(&miniflux.FeedCreationRequest{
|
|
|
|
FeedURL: testFeedURL,
|
|
|
|
CategoryID: categories[0].ID,
|
|
|
|
})
|
2018-08-25 20:53:14 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if feedID == 0 {
|
|
|
|
t.Fatalf(`Invalid feed ID, got %q`, feedID)
|
|
|
|
}
|
|
|
|
|
|
|
|
results, err := client.Entries(&miniflux.Filter{Search: "2.0.8"})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if results.Total != 1 {
|
|
|
|
t.Fatalf(`We should have only one entry instead of %d`, results.Total)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInvalidFilters(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
_, err := client.Entries(&miniflux.Filter{Status: "invalid"})
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal(`Using invalid status should raise an error`)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Entries(&miniflux.Filter{Direction: "invalid"})
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal(`Using invalid direction should raise an error`)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Entries(&miniflux.Filter{Order: "invalid"})
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal(`Using invalid order should raise an error`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
func TestGetFeedEntry(t *testing.T) {
|
2018-08-25 20:53:14 +02:00
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
// Test get entry by entry id and feed id
|
2018-08-25 20:53:14 +02:00
|
|
|
entry, err := client.FeedEntry(result.Entries[0].FeedID, result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-01-19 04:44:02 +01:00
|
|
|
if entry.ID != result.Entries[0].ID {
|
|
|
|
t.Fatal("Wrong entry returned")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetCategoryEntry(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
_, category := createFeed(t, client)
|
|
|
|
|
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-25 20:53:14 +02:00
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
// Test get entry by entry id and category id
|
|
|
|
entry, err := client.CategoryEntry(category.ID, result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-25 20:53:14 +02:00
|
|
|
if entry.ID != result.Entries[0].ID {
|
|
|
|
t.Fatal("Wrong entry returned")
|
|
|
|
}
|
2021-01-19 04:44:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetEntry(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
2018-08-25 20:53:14 +02:00
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 1})
|
2018-08-25 20:53:14 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:44:02 +01:00
|
|
|
// Test get entry by entry id only
|
|
|
|
entry, err := client.Entry(result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-25 20:53:14 +02:00
|
|
|
if entry.ID != result.Entries[0].ID {
|
|
|
|
t.Fatal("Wrong entry returned")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUpdateStatus(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = client.UpdateEntries([]int64{result.Entries[0].ID}, miniflux.EntryStatusRead)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
entry, err := client.Entry(result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if entry.Status != miniflux.EntryStatusRead {
|
|
|
|
t.Fatal("The entry status should be updated")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = client.UpdateEntries([]int64{result.Entries[0].ID}, "invalid")
|
|
|
|
if err == nil {
|
2021-01-05 00:32:32 +01:00
|
|
|
t.Fatal(`Invalid entry status should not be accepted`)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = client.UpdateEntries([]int64{}, miniflux.EntryStatusRead)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal(`An empty list of entry should not be accepted`)
|
2018-08-25 20:53:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestToggleBookmark(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if result.Entries[0].Starred {
|
|
|
|
t.Fatal("The entry should not be starred")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = client.ToggleBookmark(result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
entry, err := client.Entry(result.Entries[0].ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !entry.Starred {
|
|
|
|
t.Fatal("The entry should be starred")
|
|
|
|
}
|
|
|
|
}
|
2020-02-11 05:20:03 +01:00
|
|
|
|
|
|
|
func TestHistoryOrder(t *testing.T) {
|
|
|
|
client := createClient(t)
|
|
|
|
createFeed(t, client)
|
|
|
|
|
|
|
|
result, err := client.Entries(&miniflux.Filter{Limit: 3})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
selectedEntry := result.Entries[2].ID
|
|
|
|
|
|
|
|
err = client.UpdateEntries([]int64{selectedEntry}, miniflux.EntryStatusRead)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
history, err := client.Entries(&miniflux.Filter{Order: "changed_at", Direction: "desc", Limit: 1})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if history.Entries[0].ID != selectedEntry {
|
|
|
|
t.Fatal("The entry that we just read should be at the top of the history")
|
|
|
|
}
|
|
|
|
}
|