aboutsummaryrefslogtreecommitdiff
path: root/ui/entry.go
diff options
context:
space:
mode:
authorGravatar Frédéric Guillot <fred@miniflux.net> 2018-01-02 22:04:48 -0800
committerGravatar Frédéric Guillot <fred@miniflux.net> 2018-01-02 22:04:48 -0800
commit320d1b016747ba4501da9417d9ce5f99368a5768 (patch)
tree1054d96afde6022951b76cc4a09b78e1e3f05058 /ui/entry.go
parentc39f2e1a8d2de6d412bcc673d29eb0f7a2d1f5f7 (diff)
downloadv2-320d1b016747ba4501da9417d9ce5f99368a5768.tar.gz
v2-320d1b016747ba4501da9417d9ce5f99368a5768.tar.zst
v2-320d1b016747ba4501da9417d9ce5f99368a5768.zip
Refactor packages to have more idiomatic code base
Diffstat (limited to 'ui/entry.go')
-rw-r--r--ui/entry.go493
1 files changed, 493 insertions, 0 deletions
diff --git a/ui/entry.go b/ui/entry.go
new file mode 100644
index 00000000..a67fa683
--- /dev/null
+++ b/ui/entry.go
@@ -0,0 +1,493 @@
+// Copyright 2017 Frédéric Guillot. All rights reserved.
+// Use of this source code is governed by the Apache 2.0
+// license that can be found in the LICENSE file.
+
+package ui
+
+import (
+ "errors"
+
+ "github.com/miniflux/miniflux/http/handler"
+ "github.com/miniflux/miniflux/integration"
+ "github.com/miniflux/miniflux/logger"
+ "github.com/miniflux/miniflux/model"
+ "github.com/miniflux/miniflux/reader/sanitizer"
+ "github.com/miniflux/miniflux/reader/scraper"
+ "github.com/miniflux/miniflux/storage"
+)
+
+// FetchContent downloads the original HTML page and returns relevant contents.
+func (c *Controller) FetchContent(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ user := ctx.LoggedUser()
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.JSON().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.JSON().NotFound(errors.New("Entry not found"))
+ return
+ }
+
+ content, err := scraper.Fetch(entry.URL, entry.Feed.ScraperRules)
+ if err != nil {
+ response.JSON().ServerError(err)
+ return
+ }
+
+ entry.Content = sanitizer.Sanitize(entry.URL, content)
+ c.store.UpdateEntryContent(entry)
+
+ response.JSON().Created(map[string]string{"content": entry.Content})
+}
+
+// SaveEntry send the link to external services.
+func (c *Controller) SaveEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ user := ctx.LoggedUser()
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.JSON().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.JSON().NotFound(errors.New("Entry not found"))
+ return
+ }
+
+ settings, err := c.store.Integration(user.ID)
+ if err != nil {
+ response.JSON().ServerError(err)
+ return
+ }
+
+ go func() {
+ integration.SendEntry(entry, settings)
+ }()
+
+ response.JSON().Created(map[string]string{"message": "saved"})
+}
+
+// ShowFeedEntry shows a single feed entry in "feed" mode.
+func (c *Controller) ShowFeedEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ feedID, err := request.IntegerParam("feedID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithFeedID(feedID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.HTML().NotFound()
+ return
+ }
+
+ if entry.Status == model.EntryStatusUnread {
+ err = c.store.SetEntriesStatus(user.ID, []int64{entry.ID}, model.EntryStatusRead)
+ if err != nil {
+ logger.Error("[Controller:ShowFeedEntry] %v", err)
+ response.HTML().ServerError(nil)
+ return
+ }
+ }
+
+ args, err := c.getCommonTemplateArgs(ctx)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ builder = c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithFeedID(feedID)
+
+ prevEntry, nextEntry, err := c.getEntryPrevNext(user, builder, entry.ID)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ nextEntryRoute := ""
+ if nextEntry != nil {
+ nextEntryRoute = ctx.Route("feedEntry", "feedID", feedID, "entryID", nextEntry.ID)
+ }
+
+ prevEntryRoute := ""
+ if prevEntry != nil {
+ prevEntryRoute = ctx.Route("feedEntry", "feedID", feedID, "entryID", prevEntry.ID)
+ }
+
+ response.HTML().Render("entry", args.Merge(tplParams{
+ "entry": entry,
+ "prevEntry": prevEntry,
+ "nextEntry": nextEntry,
+ "nextEntryRoute": nextEntryRoute,
+ "prevEntryRoute": prevEntryRoute,
+ "menu": "feeds",
+ }))
+}
+
+// ShowCategoryEntry shows a single feed entry in "category" mode.
+func (c *Controller) ShowCategoryEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ categoryID, err := request.IntegerParam("categoryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithCategoryID(categoryID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.HTML().NotFound()
+ return
+ }
+
+ if entry.Status == model.EntryStatusUnread {
+ err = c.store.SetEntriesStatus(user.ID, []int64{entry.ID}, model.EntryStatusRead)
+ if err != nil {
+ logger.Error("[Controller:ShowCategoryEntry] %v", err)
+ response.HTML().ServerError(nil)
+ return
+ }
+ }
+
+ args, err := c.getCommonTemplateArgs(ctx)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ builder = c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithCategoryID(categoryID)
+
+ prevEntry, nextEntry, err := c.getEntryPrevNext(user, builder, entry.ID)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ nextEntryRoute := ""
+ if nextEntry != nil {
+ nextEntryRoute = ctx.Route("categoryEntry", "categoryID", categoryID, "entryID", nextEntry.ID)
+ }
+
+ prevEntryRoute := ""
+ if prevEntry != nil {
+ prevEntryRoute = ctx.Route("categoryEntry", "categoryID", categoryID, "entryID", prevEntry.ID)
+ }
+
+ response.HTML().Render("entry", args.Merge(tplParams{
+ "entry": entry,
+ "prevEntry": prevEntry,
+ "nextEntry": nextEntry,
+ "nextEntryRoute": nextEntryRoute,
+ "prevEntryRoute": prevEntryRoute,
+ "menu": "categories",
+ }))
+}
+
+// ShowUnreadEntry shows a single feed entry in "unread" mode.
+func (c *Controller) ShowUnreadEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.HTML().NotFound()
+ return
+ }
+
+ args, err := c.getCommonTemplateArgs(ctx)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ builder = c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithStatus(model.EntryStatusUnread)
+
+ prevEntry, nextEntry, err := c.getEntryPrevNext(user, builder, entry.ID)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ nextEntryRoute := ""
+ if nextEntry != nil {
+ nextEntryRoute = ctx.Route("unreadEntry", "entryID", nextEntry.ID)
+ }
+
+ prevEntryRoute := ""
+ if prevEntry != nil {
+ prevEntryRoute = ctx.Route("unreadEntry", "entryID", prevEntry.ID)
+ }
+
+ // We change the status here, otherwise we cannot get the pagination for unread items.
+ if entry.Status == model.EntryStatusUnread {
+ err = c.store.SetEntriesStatus(user.ID, []int64{entry.ID}, model.EntryStatusRead)
+ if err != nil {
+ logger.Error("[Controller:ShowUnreadEntry] %v", err)
+ response.HTML().ServerError(nil)
+ return
+ }
+ }
+
+ response.HTML().Render("entry", args.Merge(tplParams{
+ "entry": entry,
+ "prevEntry": prevEntry,
+ "nextEntry": nextEntry,
+ "nextEntryRoute": nextEntryRoute,
+ "prevEntryRoute": prevEntryRoute,
+ "menu": "unread",
+ }))
+}
+
+// ShowReadEntry shows a single feed entry in "history" mode.
+func (c *Controller) ShowReadEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.HTML().NotFound()
+ return
+ }
+
+ args, err := c.getCommonTemplateArgs(ctx)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ builder = c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithStatus(model.EntryStatusRead)
+
+ prevEntry, nextEntry, err := c.getEntryPrevNext(user, builder, entry.ID)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ nextEntryRoute := ""
+ if nextEntry != nil {
+ nextEntryRoute = ctx.Route("readEntry", "entryID", nextEntry.ID)
+ }
+
+ prevEntryRoute := ""
+ if prevEntry != nil {
+ prevEntryRoute = ctx.Route("readEntry", "entryID", prevEntry.ID)
+ }
+
+ response.HTML().Render("entry", args.Merge(tplParams{
+ "entry": entry,
+ "prevEntry": prevEntry,
+ "nextEntry": nextEntry,
+ "nextEntryRoute": nextEntryRoute,
+ "prevEntryRoute": prevEntryRoute,
+ "menu": "history",
+ }))
+}
+
+// ShowStarredEntry shows a single feed entry in "starred" mode.
+func (c *Controller) ShowStarredEntry(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ entryID, err := request.IntegerParam("entryID")
+ if err != nil {
+ response.HTML().BadRequest(err)
+ return
+ }
+
+ builder := c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithEntryID(entryID)
+ builder.WithoutStatus(model.EntryStatusRemoved)
+
+ entry, err := builder.GetEntry()
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ if entry == nil {
+ response.HTML().NotFound()
+ return
+ }
+
+ if entry.Status == model.EntryStatusUnread {
+ err = c.store.SetEntriesStatus(user.ID, []int64{entry.ID}, model.EntryStatusRead)
+ if err != nil {
+ logger.Error("[Controller:ShowReadEntry] %v", err)
+ response.HTML().ServerError(nil)
+ return
+ }
+ }
+
+ args, err := c.getCommonTemplateArgs(ctx)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ builder = c.store.NewEntryQueryBuilder(user.ID)
+ builder.WithStarred()
+
+ prevEntry, nextEntry, err := c.getEntryPrevNext(user, builder, entry.ID)
+ if err != nil {
+ response.HTML().ServerError(err)
+ return
+ }
+
+ nextEntryRoute := ""
+ if nextEntry != nil {
+ nextEntryRoute = ctx.Route("starredEntry", "entryID", nextEntry.ID)
+ }
+
+ prevEntryRoute := ""
+ if prevEntry != nil {
+ prevEntryRoute = ctx.Route("starredEntry", "entryID", prevEntry.ID)
+ }
+
+ response.HTML().Render("entry", args.Merge(tplParams{
+ "entry": entry,
+ "prevEntry": prevEntry,
+ "nextEntry": nextEntry,
+ "nextEntryRoute": nextEntryRoute,
+ "prevEntryRoute": prevEntryRoute,
+ "menu": "starred",
+ }))
+}
+
+// UpdateEntriesStatus handles Ajax request to update the status for a list of entries.
+func (c *Controller) UpdateEntriesStatus(ctx *handler.Context, request *handler.Request, response *handler.Response) {
+ user := ctx.LoggedUser()
+
+ entryIDs, status, err := decodeEntryStatusPayload(request.Body())
+ if err != nil {
+ logger.Error("[Controller:UpdateEntryStatus] %v", err)
+ response.JSON().BadRequest(nil)
+ return
+ }
+
+ if len(entryIDs) == 0 {
+ response.JSON().BadRequest(errors.New("The list of entryID is empty"))
+ return
+ }
+
+ err = c.store.SetEntriesStatus(user.ID, entryIDs, status)
+ if err != nil {
+ logger.Error("[Controller:UpdateEntryStatus] %v", err)
+ response.JSON().ServerError(nil)
+ return
+ }
+
+ response.JSON().Standard("OK")
+}
+
+func (c *Controller) getEntryPrevNext(user *model.User, builder *storage.EntryQueryBuilder, entryID int64) (prev *model.Entry, next *model.Entry, err error) {
+ builder.WithoutStatus(model.EntryStatusRemoved)
+ builder.WithOrder(model.DefaultSortingOrder)
+ builder.WithDirection(user.EntryDirection)
+
+ entries, err := builder.GetEntries()
+ if err != nil {
+ return nil, nil, err
+ }
+
+ n := len(entries)
+ for i := 0; i < n; i++ {
+ if entries[i].ID == entryID {
+ if i-1 >= 0 {
+ prev = entries[i-1]
+ }
+
+ if i+1 < n {
+ next = entries[i+1]
+ }
+ }
+ }
+
+ return prev, next, nil
+}