2026-03-29 16:11:44 +02:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
|
|
|
|
|
|
package handler
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"errors"
|
2026-03-29 16:19:44 +02:00
|
|
|
"html"
|
2026-03-29 16:11:44 +02:00
|
|
|
"log/slog"
|
|
|
|
|
"net/http"
|
|
|
|
|
"strconv"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"github.com/gorilla/feeds"
|
|
|
|
|
|
|
|
|
|
"kode.naiv.no/olemd/favoritter/internal/model"
|
|
|
|
|
"kode.naiv.no/olemd/favoritter/internal/store"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const feedPageSize = 50
|
|
|
|
|
|
|
|
|
|
// handleFeedGlobal generates an Atom feed of all public faves.
|
|
|
|
|
func (h *Handler) handleFeedGlobal(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
baseURL := h.deps.Config.BaseURL(r.Host)
|
|
|
|
|
|
|
|
|
|
faves, _, err := h.deps.Faves.ListPublic(feedPageSize, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error("feed: list public error", "error", err)
|
|
|
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := h.deps.Faves.LoadTags(faves); err != nil {
|
|
|
|
|
slog.Error("feed: load tags error", "error", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
settings, _ := h.deps.Settings.Get()
|
|
|
|
|
siteName := "Favoritter"
|
|
|
|
|
if settings != nil {
|
|
|
|
|
siteName = settings.SiteName
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed := &feeds.Feed{
|
|
|
|
|
Title: siteName + " — Siste favoritter",
|
|
|
|
|
Link: &feeds.Link{Href: baseURL},
|
|
|
|
|
Description: "Siste offentlige favoritter",
|
2026-03-29 16:19:44 +02:00
|
|
|
Updated: feedUpdatedTime(faves),
|
2026-03-29 16:11:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed.Items = favesToFeedItems(faves, baseURL)
|
|
|
|
|
h.writeAtom(w, feed)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// handleFeedUser generates an Atom feed for a user's public faves.
|
|
|
|
|
func (h *Handler) handleFeedUser(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
username := r.PathValue("username")
|
|
|
|
|
baseURL := h.deps.Config.BaseURL(r.Host)
|
|
|
|
|
|
|
|
|
|
user, err := h.deps.Users.GetByUsername(username)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if errors.Is(err, store.ErrUserNotFound) {
|
|
|
|
|
http.NotFound(w, r)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
slog.Error("feed: get user error", "error", err)
|
|
|
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if user.Disabled || user.ProfileVisibility == "limited" {
|
|
|
|
|
http.NotFound(w, r)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
faves, _, err := h.deps.Faves.ListPublicByUser(user.ID, feedPageSize, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error("feed: list user faves error", "error", err)
|
|
|
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := h.deps.Faves.LoadTags(faves); err != nil {
|
|
|
|
|
slog.Error("feed: load tags error", "error", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed := &feeds.Feed{
|
|
|
|
|
Title: user.DisplayNameOrUsername() + " sine favoritter",
|
|
|
|
|
Link: &feeds.Link{Href: baseURL + "/u/" + user.Username},
|
2026-03-29 16:19:44 +02:00
|
|
|
Updated: feedUpdatedTime(faves),
|
2026-03-29 16:11:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed.Items = favesToFeedItems(faves, baseURL)
|
|
|
|
|
h.writeAtom(w, feed)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// handleFeedTag generates an Atom feed for a tag's public faves.
|
|
|
|
|
func (h *Handler) handleFeedTag(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
tagName := r.PathValue("name")
|
|
|
|
|
baseURL := h.deps.Config.BaseURL(r.Host)
|
|
|
|
|
|
|
|
|
|
faves, _, err := h.deps.Faves.ListByTag(tagName, feedPageSize, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error("feed: list by tag error", "error", err)
|
|
|
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := h.deps.Faves.LoadTags(faves); err != nil {
|
|
|
|
|
slog.Error("feed: load tags error", "error", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed := &feeds.Feed{
|
|
|
|
|
Title: "Favoritter med merkelapp: " + tagName,
|
|
|
|
|
Link: &feeds.Link{Href: baseURL + "/tags/" + tagName},
|
2026-03-29 16:19:44 +02:00
|
|
|
Updated: feedUpdatedTime(faves),
|
2026-03-29 16:11:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
feed.Items = favesToFeedItems(faves, baseURL)
|
|
|
|
|
h.writeAtom(w, feed)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func favesToFeedItems(faves []*model.Fave, baseURL string) []*feeds.Item {
|
|
|
|
|
items := make([]*feeds.Item, 0, len(faves))
|
|
|
|
|
for _, f := range faves {
|
|
|
|
|
item := &feeds.Item{
|
|
|
|
|
Title: f.Description,
|
|
|
|
|
Link: &feeds.Link{Href: baseURL + "/faves/" + itoa(f.ID)},
|
|
|
|
|
Author: &feeds.Author{Name: f.DisplayName},
|
|
|
|
|
Created: f.CreatedAt,
|
|
|
|
|
Updated: f.UpdatedAt,
|
|
|
|
|
}
|
|
|
|
|
|
feat: add notes field to favorites and enhance OG meta tags
Add an optional long-form "notes" text field to each favorite for
reviews, thoughts, or extended descriptions. The field is stored in
SQLite via a new migration (002_add_fave_notes.sql) and propagated
through the entire stack:
- Model: Notes field on Fave struct
- Store: All SQL queries (Create, GetByID, Update, list methods,
scanFaves) updated with notes column
- Web handlers: Read/write notes in create, edit, update forms
- API handlers: Notes in create, update, get, import request/response
- Export: Notes included in both JSON and CSV exports
- Import: Notes parsed from both JSON and CSV imports
- Feed: Notes used as Atom feed item summary when present
- Form template: New textarea between URL and image fields
- Detail template: Display notes, enhanced og:description with
cascade: notes (truncated) → URL → generic fallback text
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 00:40:08 +02:00
|
|
|
if f.Notes != "" {
|
|
|
|
|
item.Description = f.Notes
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-29 16:11:44 +02:00
|
|
|
if f.URL != "" {
|
2026-03-29 16:19:44 +02:00
|
|
|
escaped := html.EscapeString(f.URL)
|
|
|
|
|
item.Content = `<p><a href="` + escaped + `">` + escaped + `</a></p>`
|
2026-03-29 16:11:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if f.ImagePath != "" {
|
|
|
|
|
item.Enclosure = &feeds.Enclosure{
|
|
|
|
|
Url: baseURL + "/uploads/" + f.ImagePath,
|
|
|
|
|
Type: "image/jpeg",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
items = append(items, item)
|
|
|
|
|
}
|
|
|
|
|
return items
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *Handler) writeAtom(w http.ResponseWriter, feed *feeds.Feed) {
|
|
|
|
|
atom, err := feed.ToAtom()
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error("feed: generate atom error", "error", err)
|
|
|
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
w.Header().Set("Content-Type", "application/atom+xml; charset=utf-8")
|
|
|
|
|
w.Write([]byte(atom))
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-29 16:19:44 +02:00
|
|
|
// feedUpdatedTime returns the most recent UpdatedAt from the faves,
|
|
|
|
|
// falling back to now if the list is empty.
|
|
|
|
|
func feedUpdatedTime(faves []*model.Fave) time.Time {
|
|
|
|
|
if len(faves) > 0 {
|
|
|
|
|
return faves[0].UpdatedAt
|
|
|
|
|
}
|
|
|
|
|
return time.Now()
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-29 16:11:44 +02:00
|
|
|
func itoa(n int64) string {
|
|
|
|
|
return strconv.FormatInt(n, 10)
|
|
|
|
|
}
|