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
 | |
| }
 |