Skip to content

Commit

Permalink
cache rss feed (#19)
Browse files Browse the repository at this point in the history
  • Loading branch information
wybiral authored Aug 8, 2019
1 parent 02762c8 commit 6b09ccd
Show file tree
Hide file tree
Showing 3 changed files with 84 additions and 62 deletions.
65 changes: 3 additions & 62 deletions pkg/app/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,9 @@ import (
"log"
"net"
"net/http"
"net/url"
"os"
"path"
"strconv"
"time"

"github.com/fsnotify/fsnotify"
"github.com/gorilla/feeds"
"github.com/gorilla/mux"
"github.com/wybiral/tube/pkg/media"
"github.com/wybiral/tube/pkg/onionkey"
Expand All @@ -27,6 +22,7 @@ type App struct {
Library *media.Library
Watcher *fsnotify.Watcher
Templates *template.Template
Feed []byte
Tor *tor
Listener net.Listener
Router *mux.Router
Expand Down Expand Up @@ -123,6 +119,7 @@ func (a *App) Run() error {
}
a.Watcher.Add(p.Path)
}
buildFeed(a)
go startWatcher(a)
return http.Serve(a.Listener, a.Router)
}
Expand Down Expand Up @@ -219,63 +216,7 @@ func (a *App) thumbHandler(w http.ResponseWriter, r *http.Request) {

// HTTP handler for /feed.xml
func (a *App) rssHandler(w http.ResponseWriter, r *http.Request) {
cfg := a.Config.Feed
now := time.Now()
f := &feeds.Feed{
Title: cfg.Title,
Link: &feeds.Link{Href: cfg.Link},
Description: cfg.Description,
Author: &feeds.Author{
Name: cfg.Author.Name,
Email: cfg.Author.Email,
},
Created: now,
Copyright: cfg.Copyright,
}
var externalURL string
if len(cfg.ExternalURL) > 0 {
externalURL = cfg.ExternalURL
} else if a.Tor != nil {
onion, err := a.Tor.OnionKey.Onion()
if err != nil {
return
}
externalURL = fmt.Sprintf("http://%s.onion", onion.ServiceID)
} else {
hostname, err := os.Hostname()
if err != nil {
host := a.Config.Server.Host
port := a.Config.Server.Port
externalURL = fmt.Sprintf("http://%s:%d", host, port)
} else {
externalURL = fmt.Sprintf("http://%s", hostname)
}
}
for _, v := range a.Library.Playlist() {
u, err := url.Parse(externalURL)
if err != nil {
return
}
u.Path = path.Join(u.Path, "v", v.ID)
id := u.String()
f.Items = append(f.Items, &feeds.Item{
Id: id,
Title: v.Title,
Link: &feeds.Link{Href: id},
Description: v.Description,
Enclosure: &feeds.Enclosure{
Url: id + ".mp4",
Length: strconv.FormatInt(v.Size, 10),
Type: "video/mp4",
},
Author: &feeds.Author{
Name: cfg.Author.Name,
Email: cfg.Author.Email,
},
Created: v.Timestamp,
})
}
w.Header().Set("Cache-Control", "public, max-age=7776000")
w.Header().Set("Content-Type", "text/xml")
f.WriteRss(w)
w.Write(a.Feed)
}
77 changes: 77 additions & 0 deletions pkg/app/feed.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
package app

import (
"fmt"
"net/url"
"os"
"path"
"strconv"
"time"

"github.com/wybiral/feeds"
)

// buildFeed creates RSS feed attribute for App based on Library contents.
func buildFeed(a *App) {
cfg := a.Config.Feed
now := time.Now()
f := &feeds.Feed{
Title: cfg.Title,
Link: &feeds.Link{Href: cfg.Link},
Description: cfg.Description,
Author: &feeds.Author{
Name: cfg.Author.Name,
Email: cfg.Author.Email,
},
Created: now,
Copyright: cfg.Copyright,
}
var externalURL string
if len(cfg.ExternalURL) > 0 {
externalURL = cfg.ExternalURL
} else if a.Tor != nil {
onion, err := a.Tor.OnionKey.Onion()
if err != nil {
return
}
externalURL = fmt.Sprintf("http://%s.onion", onion.ServiceID)
} else {
hostname, err := os.Hostname()
if err != nil {
host := a.Config.Server.Host
port := a.Config.Server.Port
externalURL = fmt.Sprintf("http://%s:%d", host, port)
} else {
externalURL = fmt.Sprintf("http://%s", hostname)
}
}
for _, v := range a.Library.Playlist() {
u, err := url.Parse(externalURL)
if err != nil {
return
}
u.Path = path.Join(u.Path, "v", v.ID)
id := u.String()
f.Items = append(f.Items, &feeds.Item{
Id: id,
Title: v.Title,
Link: &feeds.Link{Href: id},
Description: v.Description,
Enclosure: &feeds.Enclosure{
Url: id + ".mp4",
Length: strconv.FormatInt(v.Size, 10),
Type: "video/mp4",
},
Author: &feeds.Author{
Name: cfg.Author.Name,
Email: cfg.Author.Email,
},
Created: v.Timestamp,
})
}
feed, err := f.ToRss()
if err != nil {
return
}
a.Feed = []byte(feed)
}
4 changes: 4 additions & 0 deletions pkg/app/watcher.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ func startWatcher(a *App) {
// reset timer
timer.Reset(debounceTimeout)
case <-timer.C:
eventCount := len(removeEvents) + len(addEvents)
// handle remove events first
if len(removeEvents) > 0 {
for p := range removeEvents {
Expand All @@ -52,6 +53,9 @@ func startWatcher(a *App) {
// clear map
addEvents = make(map[string]struct{})
}
if eventCount > 0 {
buildFeed(a)
}
// reset timer
timer.Reset(debounceTimeout)
}
Expand Down

0 comments on commit 6b09ccd

Please sign in to comment.