summaryrefslogtreecommitdiff
path: root/internal/generator/atom/atom.go
diff options
context:
space:
mode:
authorPaul Buetow <paul@buetow.org>2026-04-10 09:49:58 +0300
committerPaul Buetow <paul@buetow.org>2026-04-10 09:49:58 +0300
commit65e2c4ad6b7f8d9d1685d26e5c976dd846453252 (patch)
tree224c8e14e3a8be9c5e19740c53168324ce813012 /internal/generator/atom/atom.go
parent4c10490e0488b03de70a6e0d7d7432347dcce00a (diff)
generator: isolate Atom feed in atom subpackage, document boundaries
Move Atom 1.0 XML generation to internal/generator/atom so it depends only on config and post, not on themes or the HTML template pipeline. Add generator/doc.go mapping files (orchestration, registry, themes, shared nav templates) and dependency direction. Made-with: Cursor
Diffstat (limited to 'internal/generator/atom/atom.go')
-rw-r--r--internal/generator/atom/atom.go106
1 files changed, 106 insertions, 0 deletions
diff --git a/internal/generator/atom/atom.go b/internal/generator/atom/atom.go
new file mode 100644
index 0000000..f57ad0d
--- /dev/null
+++ b/internal/generator/atom/atom.go
@@ -0,0 +1,106 @@
+// Package atom serializes the site Atom 1.0 feed (atom.xml). It depends only on
+// internal/config and internal/post. It does not import HTML themes, shared nav
+// templates, or the html/template page pipeline — those live in the parent
+// generator package.
+package atom
+
+import (
+ "encoding/xml"
+ "fmt"
+ "os"
+ "path/filepath"
+ "time"
+
+ "codeberg.org/snonux/snonux/internal/config"
+ "codeberg.org/snonux/snonux/internal/post"
+)
+
+// feed is the root element of an Atom 1.0 feed document.
+type feed struct {
+ XMLName xml.Name `xml:"feed"`
+ XMLNS string `xml:"xmlns,attr"`
+ Title string `xml:"title"`
+ Link link `xml:"link"`
+ Updated string `xml:"updated"`
+ ID string `xml:"id"`
+ Entries []entry `xml:"entry"`
+}
+
+type link struct {
+ Href string `xml:"href,attr"`
+ Rel string `xml:"rel,attr,omitempty"`
+}
+
+type entry struct {
+ Title string `xml:"title"`
+ Link link `xml:"link"`
+ ID string `xml:"id"`
+ Updated string `xml:"updated"`
+ Content content `xml:"content"`
+}
+
+type content struct {
+ Type string `xml:"type,attr"`
+ Value string `xml:",chardata"`
+}
+
+// Generate writes atom.xml to cfg.OutputDir containing the most recent
+// min(len(posts), config.PostsPerPage) entries. Posts must already be sorted
+// newest-first (as produced by generator.Run).
+func Generate(posts []*post.Post, cfg *config.Config) error {
+ limit := config.PostsPerPage
+ if len(posts) < limit {
+ limit = len(posts)
+ }
+
+ recent := posts[:limit]
+ entries := buildEntries(recent, cfg.BaseURL)
+
+ updated := time.Now().UTC().Format(time.RFC3339)
+ if len(recent) > 0 {
+ updated = recent[0].Timestamp.UTC().Format(time.RFC3339)
+ }
+
+ f := feed{
+ XMLNS: "http://www.w3.org/2005/Atom",
+ Title: "snonux.foo",
+ Link: link{Href: cfg.BaseURL + "/"},
+ Updated: updated,
+ ID: cfg.BaseURL + "/",
+ Entries: entries,
+ }
+
+ return writeFile(f, filepath.Join(cfg.OutputDir, "atom.xml"))
+}
+
+func buildEntries(posts []*post.Post, baseURL string) []entry {
+ entries := make([]entry, 0, len(posts))
+
+ for _, p := range posts {
+ entryURL := fmt.Sprintf("%s/posts/%s/", baseURL, p.ID)
+ entries = append(entries, entry{
+ Title: fmt.Sprintf("Post %s", p.ID),
+ Link: link{Href: entryURL, Rel: "alternate"},
+ ID: entryURL,
+ Updated: p.Timestamp.UTC().Format(time.RFC3339),
+ Content: content{Type: "html", Value: p.Content},
+ })
+ }
+
+ return entries
+}
+
+func writeFile(f feed, path string) error {
+ data, err := xml.MarshalIndent(f, "", " ")
+ if err != nil {
+ return fmt.Errorf("marshal atom feed: %w", err)
+ }
+
+ content := append([]byte(xml.Header), data...)
+
+ if err := os.WriteFile(path, content, 0o644); err != nil {
+ return fmt.Errorf("write atom.xml: %w", err)
+ }
+
+ return nil
+}