197 lines
4.8 KiB
Go
197 lines
4.8 KiB
Go
package glance
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"html/template"
|
|
"log/slog"
|
|
"net/http"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
var changeDetectionWidgetTemplate = mustParseTemplate("change-detection.html", "widget-base.html")
|
|
|
|
type changeDetectionWidget struct {
|
|
widgetBase `yaml:",inline"`
|
|
ChangeDetections changeDetectionWatchList `yaml:"-"`
|
|
WatchUUIDs []string `yaml:"watches"`
|
|
InstanceURL string `yaml:"instance-url"`
|
|
Token string `yaml:"token"`
|
|
Limit int `yaml:"limit"`
|
|
CollapseAfter int `yaml:"collapse-after"`
|
|
}
|
|
|
|
func (widget *changeDetectionWidget) initialize() error {
|
|
widget.withTitle("Change Detection").withCacheDuration(1 * time.Hour)
|
|
|
|
if widget.Limit <= 0 {
|
|
widget.Limit = 10
|
|
}
|
|
|
|
if widget.CollapseAfter == 0 || widget.CollapseAfter < -1 {
|
|
widget.CollapseAfter = 5
|
|
}
|
|
|
|
if widget.InstanceURL == "" {
|
|
widget.InstanceURL = "https://www.changedetection.io"
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (widget *changeDetectionWidget) update(ctx context.Context) {
|
|
if len(widget.WatchUUIDs) == 0 {
|
|
uuids, err := fetchWatchUUIDsFromChangeDetection(widget.InstanceURL, string(widget.Token))
|
|
|
|
if !widget.canContinueUpdateAfterHandlingErr(err) {
|
|
return
|
|
}
|
|
|
|
widget.WatchUUIDs = uuids
|
|
}
|
|
|
|
watches, err := fetchWatchesFromChangeDetection(widget.InstanceURL, widget.WatchUUIDs, string(widget.Token))
|
|
|
|
if !widget.canContinueUpdateAfterHandlingErr(err) {
|
|
return
|
|
}
|
|
|
|
if len(watches) > widget.Limit {
|
|
watches = watches[:widget.Limit]
|
|
}
|
|
|
|
widget.ChangeDetections = watches
|
|
}
|
|
|
|
func (widget *changeDetectionWidget) Render() template.HTML {
|
|
return widget.renderTemplate(widget, changeDetectionWidgetTemplate)
|
|
}
|
|
|
|
type changeDetectionWatch struct {
|
|
Title string
|
|
URL string
|
|
LastChanged time.Time
|
|
DiffURL string
|
|
PreviousHash string
|
|
}
|
|
|
|
type changeDetectionWatchList []changeDetectionWatch
|
|
|
|
func (r changeDetectionWatchList) sortByNewest() changeDetectionWatchList {
|
|
sort.Slice(r, func(i, j int) bool {
|
|
return r[i].LastChanged.After(r[j].LastChanged)
|
|
})
|
|
|
|
return r
|
|
}
|
|
|
|
type changeDetectionResponseJson struct {
|
|
Title string `json:"title"`
|
|
URL string `json:"url"`
|
|
LastChanged int64 `json:"last_changed"`
|
|
DateCreated int64 `json:"date_created"`
|
|
PreviousHash string `json:"previous_md5"`
|
|
}
|
|
|
|
func fetchWatchUUIDsFromChangeDetection(instanceURL string, token string) ([]string, error) {
|
|
request, _ := http.NewRequest("GET", fmt.Sprintf("%s/api/v1/watch", instanceURL), nil)
|
|
|
|
if token != "" {
|
|
request.Header.Add("x-api-key", token)
|
|
}
|
|
|
|
uuidsMap, err := decodeJsonFromRequest[map[string]struct{}](defaultHTTPClient, request)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch list of watch UUIDs: %v", err)
|
|
}
|
|
|
|
uuids := make([]string, 0, len(uuidsMap))
|
|
|
|
for uuid := range uuidsMap {
|
|
uuids = append(uuids, uuid)
|
|
}
|
|
|
|
return uuids, nil
|
|
}
|
|
|
|
func fetchWatchesFromChangeDetection(instanceURL string, requestedWatchIDs []string, token string) (changeDetectionWatchList, error) {
|
|
watches := make(changeDetectionWatchList, 0, len(requestedWatchIDs))
|
|
|
|
if len(requestedWatchIDs) == 0 {
|
|
return watches, nil
|
|
}
|
|
|
|
requests := make([]*http.Request, len(requestedWatchIDs))
|
|
|
|
for i, repository := range requestedWatchIDs {
|
|
request, _ := http.NewRequest("GET", fmt.Sprintf("%s/api/v1/watch/%s", instanceURL, repository), nil)
|
|
|
|
if token != "" {
|
|
request.Header.Add("x-api-key", token)
|
|
}
|
|
|
|
requests[i] = request
|
|
}
|
|
|
|
task := decodeJsonFromRequestTask[changeDetectionResponseJson](defaultHTTPClient)
|
|
job := newJob(task, requests).withWorkers(15)
|
|
responses, errs, err := workerPoolDo(job)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var failed int
|
|
|
|
for i := range responses {
|
|
if errs[i] != nil {
|
|
failed++
|
|
slog.Error("Failed to fetch or parse change detection watch", "url", requests[i].URL, "error", errs[i])
|
|
continue
|
|
}
|
|
|
|
watchJson := responses[i]
|
|
|
|
watch := changeDetectionWatch{
|
|
URL: watchJson.URL,
|
|
DiffURL: fmt.Sprintf("%s/diff/%s?from_version=%d", instanceURL, requestedWatchIDs[i], watchJson.LastChanged-1),
|
|
}
|
|
|
|
if watchJson.LastChanged == 0 {
|
|
watch.LastChanged = time.Unix(watchJson.DateCreated, 0)
|
|
} else {
|
|
watch.LastChanged = time.Unix(watchJson.LastChanged, 0)
|
|
}
|
|
|
|
if watchJson.Title != "" {
|
|
watch.Title = watchJson.Title
|
|
} else {
|
|
watch.Title = strings.TrimPrefix(strings.Trim(stripURLScheme(watchJson.URL), "/"), "www.")
|
|
}
|
|
|
|
if watchJson.PreviousHash != "" {
|
|
var hashLength = 8
|
|
|
|
if len(watchJson.PreviousHash) < hashLength {
|
|
hashLength = len(watchJson.PreviousHash)
|
|
}
|
|
|
|
watch.PreviousHash = watchJson.PreviousHash[0:hashLength]
|
|
}
|
|
|
|
watches = append(watches, watch)
|
|
}
|
|
|
|
if len(watches) == 0 {
|
|
return nil, errNoContent
|
|
}
|
|
|
|
watches.sortByNewest()
|
|
|
|
if failed > 0 {
|
|
return watches, fmt.Errorf("%w: could not get %d watches", errPartialContent, failed)
|
|
}
|
|
|
|
return watches, nil
|
|
}
|