mirror of
https://github.com/miniflux/v2.git
synced 2025-06-27 16:36:00 +00:00
feat: combine feed icon handlers to use only externalIconID
This commit is contained in:
parent
c531be8780
commit
c75e0ceea1
16 changed files with 33 additions and 52 deletions
|
@ -210,7 +210,6 @@ func (r RequestModifiers) String() string {
|
||||||
func Serve(router *mux.Router, store *storage.Storage) {
|
func Serve(router *mux.Router, store *storage.Storage) {
|
||||||
handler := &handler{store, router}
|
handler := &handler{store, router}
|
||||||
router.HandleFunc("/accounts/ClientLogin", handler.clientLoginHandler).Methods(http.MethodPost).Name("ClientLogin")
|
router.HandleFunc("/accounts/ClientLogin", handler.clientLoginHandler).Methods(http.MethodPost).Name("ClientLogin")
|
||||||
router.HandleFunc("/reader/api/0/icons/{externalIconID}", handler.iconHandler).Methods(http.MethodGet).Name("Icons")
|
|
||||||
|
|
||||||
middleware := newMiddleware(store)
|
middleware := newMiddleware(store)
|
||||||
sr := router.PathPrefix("/reader/api/0").Subrouter()
|
sr := router.PathPrefix("/reader/api/0").Subrouter()
|
||||||
|
@ -728,39 +727,6 @@ func (h *handler) quickAddHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *handler) iconHandler(w http.ResponseWriter, r *http.Request) {
|
|
||||||
clientIP := request.ClientIP(r)
|
|
||||||
externalIconID := request.RouteStringParam(r, "externalIconID")
|
|
||||||
|
|
||||||
slog.Debug("[GoogleReader] Handle /icons/{externalIconID}",
|
|
||||||
slog.String("handler", "iconHandler"),
|
|
||||||
slog.String("client_ip", clientIP),
|
|
||||||
slog.String("user_agent", r.UserAgent()),
|
|
||||||
slog.String("external_icon_id", externalIconID),
|
|
||||||
)
|
|
||||||
|
|
||||||
icon, err := h.store.IconByExternalID(externalIconID)
|
|
||||||
if err != nil {
|
|
||||||
json.ServerError(w, r, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if icon == nil {
|
|
||||||
json.NotFound(w, r)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
response.New(w, r).WithCaching(icon.Hash, 72*time.Hour, func(b *response.Builder) {
|
|
||||||
b.WithHeader("Content-Security-Policy", `sandbox`)
|
|
||||||
b.WithHeader("Content-Type", icon.MimeType)
|
|
||||||
b.WithBody(icon.Content)
|
|
||||||
if icon.MimeType != "image/svg+xml" {
|
|
||||||
b.WithoutCompression()
|
|
||||||
}
|
|
||||||
b.Write()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func getFeed(stream Stream, store *storage.Storage, userID int64) (*model.Feed, error) {
|
func getFeed(stream Stream, store *storage.Storage, userID int64) (*model.Feed, error) {
|
||||||
feedID, err := strconv.ParseInt(stream.ID, 10, 64)
|
feedID, err := strconv.ParseInt(stream.ID, 10, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -863,7 +829,7 @@ func move(stream Stream, destination Stream, store *storage.Storage, userID int6
|
||||||
|
|
||||||
func (h *handler) feedIconURL(f *model.Feed) string {
|
func (h *handler) feedIconURL(f *model.Feed) string {
|
||||||
if f.Icon != nil && f.Icon.ExternalIconID != "" {
|
if f.Icon != nil && f.Icon.ExternalIconID != "" {
|
||||||
return config.Opts.RootURL() + route.Path(h.router, "Icons", "externalIconID", f.Icon.ExternalIconID)
|
return config.Opts.RootURL() + route.Path(h.router, "feedIcon", "externalIconID", f.Icon.ExternalIconID)
|
||||||
} else {
|
} else {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
|
@ -293,6 +293,7 @@ func (e *EntryQueryBuilder) GetEntries() (model.Entries, error) {
|
||||||
f.hide_globally,
|
f.hide_globally,
|
||||||
f.no_media_player,
|
f.no_media_player,
|
||||||
fi.icon_id,
|
fi.icon_id,
|
||||||
|
i.external_id AS icon_external_id,
|
||||||
u.timezone
|
u.timezone
|
||||||
FROM
|
FROM
|
||||||
entries e
|
entries e
|
||||||
|
@ -302,6 +303,8 @@ func (e *EntryQueryBuilder) GetEntries() (model.Entries, error) {
|
||||||
categories c ON c.id=f.category_id
|
categories c ON c.id=f.category_id
|
||||||
LEFT JOIN
|
LEFT JOIN
|
||||||
feed_icons fi ON fi.feed_id=f.id
|
feed_icons fi ON fi.feed_id=f.id
|
||||||
|
LEFT JOIN
|
||||||
|
icons i ON i.id=fi.icon_id
|
||||||
LEFT JOIN
|
LEFT JOIN
|
||||||
users u ON u.id=e.user_id
|
users u ON u.id=e.user_id
|
||||||
WHERE %s %s
|
WHERE %s %s
|
||||||
|
@ -323,6 +326,7 @@ func (e *EntryQueryBuilder) GetEntries() (model.Entries, error) {
|
||||||
|
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var iconID sql.NullInt64
|
var iconID sql.NullInt64
|
||||||
|
var externalIconID sql.NullString
|
||||||
var tz string
|
var tz string
|
||||||
|
|
||||||
entry := model.NewEntry()
|
entry := model.NewEntry()
|
||||||
|
@ -361,6 +365,7 @@ func (e *EntryQueryBuilder) GetEntries() (model.Entries, error) {
|
||||||
&entry.Feed.HideGlobally,
|
&entry.Feed.HideGlobally,
|
||||||
&entry.Feed.NoMediaPlayer,
|
&entry.Feed.NoMediaPlayer,
|
||||||
&iconID,
|
&iconID,
|
||||||
|
&externalIconID,
|
||||||
&tz,
|
&tz,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -368,8 +373,10 @@ func (e *EntryQueryBuilder) GetEntries() (model.Entries, error) {
|
||||||
return nil, fmt.Errorf("store: unable to fetch entry row: %v", err)
|
return nil, fmt.Errorf("store: unable to fetch entry row: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if iconID.Valid {
|
if iconID.Valid && externalIconID.Valid && externalIconID.String != "" {
|
||||||
|
entry.Feed.Icon.FeedID = entry.FeedID
|
||||||
entry.Feed.Icon.IconID = iconID.Int64
|
entry.Feed.Icon.IconID = iconID.Int64
|
||||||
|
entry.Feed.Icon.ExternalIconID = externalIconID.String
|
||||||
} else {
|
} else {
|
||||||
entry.Feed.Icon.IconID = 0
|
entry.Feed.Icon.IconID = 0
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,7 +59,7 @@ func (s *Storage) IconByExternalID(externalIconID string) (*model.Icon, error) {
|
||||||
if err == sql.ErrNoRows {
|
if err == sql.ErrNoRows {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
return nil, fmt.Errorf("store: unable to fetch icon #%s: %w", externalIconID, err)
|
return nil, fmt.Errorf("store: unable to fetch icon %s: %w", externalIconID, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return &icon, nil
|
return &icon, nil
|
||||||
|
|
|
@ -10,7 +10,7 @@
|
||||||
<h2 id="feed-title-{{ .ID }}" class="item-title">
|
<h2 id="feed-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "feedEntries" "feedID" .ID }}">
|
<a href="{{ route "feedEntries" "feedID" .ID }}">
|
||||||
{{ if and (.Icon) (gt .Icon.IconID 0) }}
|
{{ if and (.Icon) (gt .Icon.IconID 0) }}
|
||||||
<img src="{{ route "icon" "iconID" .Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ if .Disabled }} 🚫 {{ end }}
|
{{ if .Disabled }} 🚫 {{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "starredEntry" "entryID" .ID }}">
|
<a href="{{ route "starredEntry" "entryID" .ID }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -102,7 +102,7 @@
|
||||||
{{ end }}
|
{{ end }}
|
||||||
>
|
>
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -108,8 +108,8 @@
|
||||||
{{ end }}
|
{{ end }}
|
||||||
<div class="entry-meta" dir="auto">
|
<div class="entry-meta" dir="auto">
|
||||||
<span class="entry-website">
|
<span class="entry-website">
|
||||||
{{ if and .user (ne .entry.Feed.Icon.IconID 0) }}
|
{{ if ne .entry.Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .entry.Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .entry.Feed.Title }}">
|
<img src="{{ route "feedIcon" "externalIconID" .entry.Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="{{ .entry.Feed.Title }}">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ if .user }}
|
{{ if .user }}
|
||||||
<a href="{{ route "feedEntries" "feedID" .entry.Feed.ID }}">{{ .entry.Feed.Title }}</a>
|
<a href="{{ route "feedEntries" "feedID" .entry.Feed.ID }}">{{ .entry.Feed.Title }}</a>
|
||||||
|
|
|
@ -113,7 +113,7 @@
|
||||||
{{ end }}
|
{{ end }}
|
||||||
>
|
>
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -48,7 +48,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "readEntry" "entryID" .ID }}">
|
<a href="{{ route "readEntry" "entryID" .ID }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -32,7 +32,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "searchEntry" "entryID" .ID }}?q={{ $.searchQuery }}">
|
<a href="{{ route "searchEntry" "entryID" .ID }}?q={{ $.searchQuery }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
|
||||||
{{ else }}
|
{{ else }}
|
||||||
<span class="sr-only">{{ .Feed.Title }}</span>
|
<span class="sr-only">{{ .Feed.Title }}</span>
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
|
|
@ -48,7 +48,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "readEntry" "entryID" .ID }}">
|
<a href="{{ route "readEntry" "entryID" .ID }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "tagEntry" "entryID" .ID "tagName" (urlEncode $.tagName) }}">
|
<a href="{{ route "tagEntry" "entryID" .ID "tagName" (urlEncode $.tagName) }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -56,7 +56,7 @@
|
||||||
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
<h2 id="entry-title-{{ .ID }}" class="item-title">
|
||||||
<a href="{{ route "unreadEntry" "entryID" .ID }}">
|
<a href="{{ route "unreadEntry" "entryID" .ID }}">
|
||||||
{{ if ne .Feed.Icon.IconID 0 }}
|
{{ if ne .Feed.Icon.IconID 0 }}
|
||||||
<img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="">
|
<img src="{{ route "feedIcon" "externalIconID" .Feed.Icon.ExternalIconID }}" width="16" height="16" loading="lazy" alt="">
|
||||||
{{ end }}
|
{{ end }}
|
||||||
{{ .Title }}
|
{{ .Title }}
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -12,9 +12,9 @@ import (
|
||||||
"miniflux.app/v2/internal/http/response/html"
|
"miniflux.app/v2/internal/http/response/html"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (h *handler) showIcon(w http.ResponseWriter, r *http.Request) {
|
func (h *handler) showFeedIcon(w http.ResponseWriter, r *http.Request) {
|
||||||
iconID := request.RouteInt64Param(r, "iconID")
|
externalIconID := request.RouteStringParam(r, "externalIconID")
|
||||||
icon, err := h.store.IconByID(iconID)
|
icon, err := h.store.IconByExternalID(externalIconID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
html.ServerError(w, r, err)
|
html.ServerError(w, r, err)
|
||||||
return
|
return
|
||||||
|
|
|
@ -63,6 +63,13 @@ func (m *middleware) handleUserSession(next http.Handler) http.Handler {
|
||||||
|
|
||||||
func (m *middleware) handleAppSession(next http.Handler) http.Handler {
|
func (m *middleware) handleAppSession(next http.Handler) http.Handler {
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if mux.CurrentRoute(r).GetName() == "feedIcon" {
|
||||||
|
// Skip app session handling for the feed icon route to avoid unnecessary session creation
|
||||||
|
// when fetching feed icons.
|
||||||
|
next.ServeHTTP(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
session := m.getAppSessionValueFromCookie(r)
|
session := m.getAppSessionValueFromCookie(r)
|
||||||
|
|
||||||
|
@ -154,6 +161,7 @@ func (m *middleware) isPublicRoute(r *http.Request) bool {
|
||||||
"oauth2Redirect",
|
"oauth2Redirect",
|
||||||
"oauth2Callback",
|
"oauth2Callback",
|
||||||
"appIcon",
|
"appIcon",
|
||||||
|
"feedIcon",
|
||||||
"favicon",
|
"favicon",
|
||||||
"webManifest",
|
"webManifest",
|
||||||
"robots",
|
"robots",
|
||||||
|
|
|
@ -74,7 +74,7 @@ func Serve(router *mux.Router, store *storage.Storage, pool *worker.Pool) {
|
||||||
uiRouter.HandleFunc("/feed/{feedID}/entries/all", handler.showFeedEntriesAllPage).Name("feedEntriesAll").Methods(http.MethodGet)
|
uiRouter.HandleFunc("/feed/{feedID}/entries/all", handler.showFeedEntriesAllPage).Name("feedEntriesAll").Methods(http.MethodGet)
|
||||||
uiRouter.HandleFunc("/feed/{feedID}/entry/{entryID}", handler.showFeedEntryPage).Name("feedEntry").Methods(http.MethodGet)
|
uiRouter.HandleFunc("/feed/{feedID}/entry/{entryID}", handler.showFeedEntryPage).Name("feedEntry").Methods(http.MethodGet)
|
||||||
uiRouter.HandleFunc("/unread/feed/{feedID}/entry/{entryID}", handler.showUnreadFeedEntryPage).Name("unreadFeedEntry").Methods(http.MethodGet)
|
uiRouter.HandleFunc("/unread/feed/{feedID}/entry/{entryID}", handler.showUnreadFeedEntryPage).Name("unreadFeedEntry").Methods(http.MethodGet)
|
||||||
uiRouter.HandleFunc("/feed/icon/{iconID}", handler.showIcon).Name("icon").Methods(http.MethodGet)
|
uiRouter.HandleFunc("/feed/icon/{externalIconID}", handler.showFeedIcon).Name("feedIcon").Methods(http.MethodGet)
|
||||||
uiRouter.HandleFunc("/feed/{feedID}/mark-all-as-read", handler.markFeedAsRead).Name("markFeedAsRead").Methods(http.MethodPost)
|
uiRouter.HandleFunc("/feed/{feedID}/mark-all-as-read", handler.markFeedAsRead).Name("markFeedAsRead").Methods(http.MethodPost)
|
||||||
|
|
||||||
// Category pages.
|
// Category pages.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue