361 lines
7.9 KiB
Go
361 lines
7.9 KiB
Go
package glance
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"html/template"
|
|
"log/slog"
|
|
"math"
|
|
"net/http"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
var widgetIDCounter atomic.Uint64
|
|
|
|
func newWidget(widgetType string) (widget, error) {
|
|
var w widget
|
|
|
|
switch widgetType {
|
|
case "calendar":
|
|
w = &calendarWidget{}
|
|
case "calendar-legacy":
|
|
w = &oldCalendarWidget{}
|
|
case "clock":
|
|
w = &clockWidget{}
|
|
case "weather":
|
|
w = &weatherWidget{}
|
|
case "bookmarks":
|
|
w = &bookmarksWidget{}
|
|
case "iframe":
|
|
w = &iframeWidget{}
|
|
case "html":
|
|
w = &htmlWidget{}
|
|
case "hacker-news":
|
|
w = &hackerNewsWidget{}
|
|
case "releases":
|
|
w = &releasesWidget{}
|
|
case "videos":
|
|
w = &videosWidget{}
|
|
case "markets", "stocks":
|
|
w = &marketsWidget{}
|
|
case "reddit":
|
|
w = &redditWidget{}
|
|
case "rss":
|
|
w = &rssWidget{}
|
|
case "monitor":
|
|
w = &monitorWidget{}
|
|
case "twitch-top-games":
|
|
w = &twitchGamesWidget{}
|
|
case "twitch-channels":
|
|
w = &twitchChannelsWidget{}
|
|
case "lobsters":
|
|
w = &lobstersWidget{}
|
|
case "change-detection":
|
|
w = &changeDetectionWidget{}
|
|
case "repository":
|
|
w = &repositoryWidget{}
|
|
case "search":
|
|
w = &searchWidget{}
|
|
case "extension":
|
|
w = &extensionWidget{}
|
|
case "group":
|
|
w = &groupWidget{}
|
|
case "dns-stats":
|
|
w = &dnsStatsWidget{}
|
|
case "split-column":
|
|
w = &splitColumnWidget{}
|
|
case "custom-api":
|
|
w = &customAPIWidget{}
|
|
case "docker-containers":
|
|
w = &dockerContainersWidget{}
|
|
case "server-stats":
|
|
w = &serverStatsWidget{}
|
|
default:
|
|
return nil, fmt.Errorf("unknown widget type: %s", widgetType)
|
|
}
|
|
|
|
w.setID(widgetIDCounter.Add(1))
|
|
|
|
return w, nil
|
|
}
|
|
|
|
type widgets []widget
|
|
|
|
func (w *widgets) UnmarshalYAML(node *yaml.Node) error {
|
|
var nodes []yaml.Node
|
|
|
|
if err := node.Decode(&nodes); err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, node := range nodes {
|
|
meta := struct {
|
|
Type string `yaml:"type"`
|
|
}{}
|
|
|
|
if err := node.Decode(&meta); err != nil {
|
|
return err
|
|
}
|
|
|
|
widget, err := newWidget(meta.Type)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = node.Decode(widget); err != nil {
|
|
return err
|
|
}
|
|
|
|
*w = append(*w, widget)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type widget interface {
|
|
// These need to be exported because they get called in templates
|
|
Render() template.HTML
|
|
GetType() string
|
|
GetID() uint64
|
|
|
|
initialize() error
|
|
requiresUpdate(*time.Time) bool
|
|
setProviders(*widgetProviders)
|
|
update(context.Context)
|
|
setID(uint64)
|
|
handleRequest(w http.ResponseWriter, r *http.Request)
|
|
setHideHeader(bool)
|
|
}
|
|
|
|
type cacheType int
|
|
|
|
const (
|
|
cacheTypeInfinite cacheType = iota
|
|
cacheTypeDuration
|
|
cacheTypeOnTheHour
|
|
)
|
|
|
|
type widgetBase struct {
|
|
ID uint64 `yaml:"-"`
|
|
Providers *widgetProviders `yaml:"-"`
|
|
Type string `yaml:"type"`
|
|
Title string `yaml:"title"`
|
|
TitleURL string `yaml:"title-url"`
|
|
CSSClass string `yaml:"css-class"`
|
|
CustomCacheDuration durationField `yaml:"cache"`
|
|
ContentAvailable bool `yaml:"-"`
|
|
WIP bool `yaml:"-"`
|
|
Error error `yaml:"-"`
|
|
Notice error `yaml:"-"`
|
|
templateBuffer bytes.Buffer `yaml:"-"`
|
|
cacheDuration time.Duration `yaml:"-"`
|
|
cacheType cacheType `yaml:"-"`
|
|
nextUpdate time.Time `yaml:"-"`
|
|
updateRetriedTimes int `yaml:"-"`
|
|
HideHeader bool `yaml:"hide-header"`
|
|
}
|
|
|
|
type widgetProviders struct {
|
|
assetResolver func(string) string
|
|
}
|
|
|
|
func (w *widgetBase) requiresUpdate(now *time.Time) bool {
|
|
if w.cacheType == cacheTypeInfinite {
|
|
return false
|
|
}
|
|
|
|
if w.nextUpdate.IsZero() {
|
|
return true
|
|
}
|
|
|
|
return now.After(w.nextUpdate)
|
|
}
|
|
|
|
func (w *widgetBase) IsWIP() bool {
|
|
return w.WIP
|
|
}
|
|
|
|
func (w *widgetBase) update(ctx context.Context) {
|
|
|
|
}
|
|
|
|
func (w *widgetBase) GetID() uint64 {
|
|
return w.ID
|
|
}
|
|
|
|
func (w *widgetBase) setID(id uint64) {
|
|
w.ID = id
|
|
}
|
|
|
|
func (w *widgetBase) setHideHeader(value bool) {
|
|
w.HideHeader = value
|
|
}
|
|
|
|
func (widget *widgetBase) handleRequest(w http.ResponseWriter, r *http.Request) {
|
|
http.Error(w, "not implemented", http.StatusNotImplemented)
|
|
}
|
|
|
|
func (w *widgetBase) GetType() string {
|
|
return w.Type
|
|
}
|
|
|
|
func (w *widgetBase) setProviders(providers *widgetProviders) {
|
|
w.Providers = providers
|
|
}
|
|
|
|
func (w *widgetBase) renderTemplate(data any, t *template.Template) template.HTML {
|
|
w.templateBuffer.Reset()
|
|
err := t.Execute(&w.templateBuffer, data)
|
|
if err != nil {
|
|
w.ContentAvailable = false
|
|
w.Error = err
|
|
|
|
slog.Error("Failed to render template", "error", err)
|
|
|
|
// need to immediately re-render with the error,
|
|
// otherwise risk breaking the page since the widget
|
|
// will likely be partially rendered with tags not closed.
|
|
w.templateBuffer.Reset()
|
|
err2 := t.Execute(&w.templateBuffer, data)
|
|
|
|
if err2 != nil {
|
|
slog.Error("Failed to render error within widget", "error", err2, "initial_error", err)
|
|
w.templateBuffer.Reset()
|
|
// TODO: add some kind of a generic widget error template when the widget
|
|
// failed to render, and we also failed to re-render the widget with the error
|
|
}
|
|
}
|
|
|
|
return template.HTML(w.templateBuffer.String())
|
|
}
|
|
|
|
func (w *widgetBase) withTitle(title string) *widgetBase {
|
|
if w.Title == "" {
|
|
w.Title = title
|
|
}
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) withTitleURL(titleURL string) *widgetBase {
|
|
if w.TitleURL == "" {
|
|
w.TitleURL = titleURL
|
|
}
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) withCacheDuration(duration time.Duration) *widgetBase {
|
|
w.cacheType = cacheTypeDuration
|
|
|
|
if duration == -1 || w.CustomCacheDuration == 0 {
|
|
w.cacheDuration = duration
|
|
} else {
|
|
w.cacheDuration = time.Duration(w.CustomCacheDuration)
|
|
}
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) withCacheOnTheHour() *widgetBase {
|
|
w.cacheType = cacheTypeOnTheHour
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) withNotice(err error) *widgetBase {
|
|
w.Notice = err
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) withError(err error) *widgetBase {
|
|
if err == nil && !w.ContentAvailable {
|
|
w.ContentAvailable = true
|
|
}
|
|
|
|
w.Error = err
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) canContinueUpdateAfterHandlingErr(err error) bool {
|
|
// TODO: needs covering more edge cases.
|
|
// if there's partial content and we update early there's a chance
|
|
// the early update returns even less content than the initial update.
|
|
// need some kind of mechanism that tells us whether we should update early
|
|
// or not depending on the number of things that failed during the initial
|
|
// and subsequent update and how they failed - ie whether it was server
|
|
// error (like gateway timeout, do retry early) or client error (like
|
|
// hitting a rate limit, don't retry early). will require reworking a
|
|
// good amount of code in the feed package and probably having a custom
|
|
// error type that holds more information because screw wrapping errors.
|
|
// alternatively have a resource cache and only refetch the failed resources,
|
|
// then rebuild the widget.
|
|
|
|
if err != nil {
|
|
w.scheduleEarlyUpdate()
|
|
|
|
if !errors.Is(err, errPartialContent) {
|
|
w.withError(err)
|
|
w.withNotice(nil)
|
|
return false
|
|
}
|
|
|
|
w.withError(nil)
|
|
w.withNotice(err)
|
|
return true
|
|
}
|
|
|
|
w.withNotice(nil)
|
|
w.withError(nil)
|
|
w.scheduleNextUpdate()
|
|
return true
|
|
}
|
|
|
|
func (w *widgetBase) getNextUpdateTime() time.Time {
|
|
now := time.Now()
|
|
|
|
if w.cacheType == cacheTypeDuration {
|
|
return now.Add(w.cacheDuration)
|
|
}
|
|
|
|
if w.cacheType == cacheTypeOnTheHour {
|
|
return now.Add(time.Duration(
|
|
((60-now.Minute())*60)-now.Second(),
|
|
) * time.Second)
|
|
}
|
|
|
|
return time.Time{}
|
|
}
|
|
|
|
func (w *widgetBase) scheduleNextUpdate() *widgetBase {
|
|
w.nextUpdate = w.getNextUpdateTime()
|
|
w.updateRetriedTimes = 0
|
|
|
|
return w
|
|
}
|
|
|
|
func (w *widgetBase) scheduleEarlyUpdate() *widgetBase {
|
|
w.updateRetriedTimes++
|
|
|
|
if w.updateRetriedTimes > 5 {
|
|
w.updateRetriedTimes = 5
|
|
}
|
|
|
|
nextEarlyUpdate := time.Now().Add(time.Duration(math.Pow(float64(w.updateRetriedTimes), 2)) * time.Minute)
|
|
nextUsualUpdate := w.getNextUpdateTime()
|
|
|
|
if nextEarlyUpdate.After(nextUsualUpdate) {
|
|
w.nextUpdate = nextUsualUpdate
|
|
} else {
|
|
w.nextUpdate = nextEarlyUpdate
|
|
}
|
|
|
|
return w
|
|
}
|