184 lines
3.6 KiB
Go
184 lines
3.6 KiB
Go
package glance
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"html/template"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"regexp"
|
|
"slices"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
var sequentialWhitespacePattern = regexp.MustCompile(`\s+`)
|
|
|
|
func percentChange(current, previous float64) float64 {
|
|
return (current/previous - 1) * 100
|
|
}
|
|
|
|
func extractDomainFromUrl(u string) string {
|
|
if u == "" {
|
|
return ""
|
|
}
|
|
|
|
parsed, err := url.Parse(u)
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
|
|
return strings.TrimPrefix(strings.ToLower(parsed.Host), "www.")
|
|
}
|
|
|
|
func svgPolylineCoordsFromYValues(width float64, height float64, values []float64) string {
|
|
if len(values) < 2 {
|
|
return ""
|
|
}
|
|
|
|
verticalPadding := height * 0.02
|
|
height -= verticalPadding * 2
|
|
coordinates := make([]string, len(values))
|
|
distanceBetweenPoints := width / float64(len(values)-1)
|
|
min := slices.Min(values)
|
|
max := slices.Max(values)
|
|
|
|
for i := range values {
|
|
coordinates[i] = fmt.Sprintf(
|
|
"%.2f,%.2f",
|
|
float64(i)*distanceBetweenPoints,
|
|
((max-values[i])/(max-min))*height+verticalPadding,
|
|
)
|
|
}
|
|
|
|
return strings.Join(coordinates, " ")
|
|
}
|
|
|
|
func maybeCopySliceWithoutZeroValues[T int | float64](values []T) []T {
|
|
if len(values) == 0 {
|
|
return values
|
|
}
|
|
|
|
for i := range values {
|
|
if values[i] != 0 {
|
|
continue
|
|
}
|
|
|
|
c := make([]T, 0, len(values)-1)
|
|
|
|
for i := range values {
|
|
if values[i] != 0 {
|
|
c = append(c, values[i])
|
|
}
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
return values
|
|
}
|
|
|
|
var urlSchemePattern = regexp.MustCompile(`^[a-z]+:\/\/`)
|
|
|
|
func stripURLScheme(url string) string {
|
|
return urlSchemePattern.ReplaceAllString(url, "")
|
|
}
|
|
|
|
func isRunningInsideDockerContainer() bool {
|
|
_, err := os.Stat("/.dockerenv")
|
|
return err == nil
|
|
}
|
|
|
|
func prefixStringLines(prefix string, s string) string {
|
|
lines := strings.Split(s, "\n")
|
|
|
|
for i, line := range lines {
|
|
lines[i] = prefix + line
|
|
}
|
|
|
|
return strings.Join(lines, "\n")
|
|
}
|
|
|
|
func limitStringLength(s string, max int) (string, bool) {
|
|
asRunes := []rune(s)
|
|
|
|
if len(asRunes) > max {
|
|
return string(asRunes[:max]), true
|
|
}
|
|
|
|
return s, false
|
|
}
|
|
|
|
func parseRFC3339Time(t string) time.Time {
|
|
parsed, err := time.Parse(time.RFC3339, t)
|
|
if err != nil {
|
|
return time.Now()
|
|
}
|
|
|
|
return parsed
|
|
}
|
|
|
|
func normalizeVersionFormat(version string) string {
|
|
version = strings.ToLower(strings.TrimSpace(version))
|
|
|
|
if len(version) > 0 && version[0] != 'v' {
|
|
return "v" + version
|
|
}
|
|
|
|
return version
|
|
}
|
|
|
|
func titleToSlug(s string) string {
|
|
s = strings.ToLower(s)
|
|
s = sequentialWhitespacePattern.ReplaceAllString(s, "-")
|
|
s = strings.Trim(s, "-")
|
|
|
|
return s
|
|
}
|
|
|
|
func fileServerWithCache(fs http.FileSystem, cacheDuration time.Duration) http.Handler {
|
|
server := http.FileServer(fs)
|
|
cacheControlValue := fmt.Sprintf("public, max-age=%d", int(cacheDuration.Seconds()))
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
// TODO: fix always setting cache control even if the file doesn't exist
|
|
w.Header().Set("Cache-Control", cacheControlValue)
|
|
server.ServeHTTP(w, r)
|
|
})
|
|
}
|
|
|
|
func executeTemplateToHTML(t *template.Template, data interface{}) (template.HTML, error) {
|
|
var b bytes.Buffer
|
|
|
|
err := t.Execute(&b, data)
|
|
if err != nil {
|
|
return "", fmt.Errorf("executing template: %w", err)
|
|
}
|
|
|
|
return template.HTML(b.String()), nil
|
|
}
|
|
|
|
func stringToBool(s string) bool {
|
|
return s == "true" || s == "yes"
|
|
}
|
|
|
|
func itemAtIndexOrDefault[T any](items []T, index int, def T) T {
|
|
if index >= len(items) {
|
|
return def
|
|
}
|
|
|
|
return items[index]
|
|
}
|
|
|
|
func ternary[T any](condition bool, a, b T) T {
|
|
if condition {
|
|
return a
|
|
}
|
|
|
|
return b
|
|
}
|
|
|
|
// Having compile time errors about unused variables is cool and all, but I don't want to
|
|
// have to constantly comment out my code while I'm working on it and testing things out
|
|
func ItsUsedTrustMeBro(...any) {}
|