2019-05-19 20:49:27 +03:00
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
2022-12-01 12:57:07 +03:00
|
|
|
"errors"
|
2023-07-09 21:16:34 +03:00
|
|
|
"fmt"
|
2019-05-19 20:49:27 +03:00
|
|
|
"net/http"
|
2022-04-18 12:39:26 +03:00
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2019-05-19 20:49:27 +03:00
|
|
|
)
|
|
|
|
|
2022-04-18 12:39:26 +03:00
|
|
|
const (
|
|
|
|
cacheMaxAgePattern = `max-age=(\d+)`
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
cacheMaxAgeRe *regexp.Regexp
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
cacheMaxAgeRe = regexp.MustCompile(cacheMaxAgePattern)
|
|
|
|
}
|
|
|
|
|
2022-12-29 13:54:14 +03:00
|
|
|
type PageParams struct {
|
|
|
|
Page int `json:"page"`
|
|
|
|
PageSize int `json:"page_size"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *PageParams) Limit() int {
|
|
|
|
if p.PageSize < 0 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return p.PageSize
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *PageParams) Offset() int {
|
|
|
|
if p.PageSize <= 0 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return (p.Page - 1) * p.PageSize
|
|
|
|
}
|
|
|
|
|
2022-04-18 12:39:26 +03:00
|
|
|
func IsNoCache(r *http.Request, cacheTtl time.Duration) bool {
|
|
|
|
cacheControl := r.Header.Get("cache-control")
|
|
|
|
if strings.Contains(cacheControl, "no-cache") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if match := cacheMaxAgeRe.FindStringSubmatch(cacheControl); match != nil && len(match) > 1 {
|
|
|
|
if maxAge, _ := strconv.Atoi(match[1]); time.Duration(maxAge)*time.Second <= cacheTtl {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2022-10-16 19:59:19 +03:00
|
|
|
|
2022-12-29 13:54:14 +03:00
|
|
|
func ParsePageParams(r *http.Request) *PageParams {
|
|
|
|
pageParams := &PageParams{}
|
2022-10-16 19:59:19 +03:00
|
|
|
page := r.URL.Query().Get("page")
|
|
|
|
pageSize := r.URL.Query().Get("page_size")
|
|
|
|
if p, err := strconv.Atoi(page); err == nil {
|
|
|
|
pageParams.Page = p
|
|
|
|
}
|
|
|
|
if p, err := strconv.Atoi(pageSize); err == nil && pageParams.Page > 0 {
|
|
|
|
pageParams.PageSize = p
|
|
|
|
}
|
|
|
|
return pageParams
|
|
|
|
}
|
2022-10-19 19:28:30 +03:00
|
|
|
|
2022-12-29 13:54:14 +03:00
|
|
|
func ParsePageParamsWithDefault(r *http.Request, page, size int) *PageParams {
|
2022-10-19 19:28:30 +03:00
|
|
|
pageParams := ParsePageParams(r)
|
|
|
|
if pageParams.Page == 0 {
|
|
|
|
pageParams.Page = page
|
|
|
|
}
|
|
|
|
if pageParams.PageSize == 0 {
|
|
|
|
pageParams.PageSize = size
|
|
|
|
}
|
|
|
|
return pageParams
|
|
|
|
}
|
2022-12-01 12:57:07 +03:00
|
|
|
|
|
|
|
func ParseUserAgent(ua string) (string, string, error) {
|
2023-07-09 11:28:23 +03:00
|
|
|
re := regexp.MustCompile(`(?iU)^(?:(?:wakatime|chrome|firefox)\/(?:v?[\d+.]+|unset)\s)?(?:\(?(\w+)[-_].*\)?.+\s)?([^\/\s]+)-wakatime\/.+$`)
|
2022-12-01 12:57:07 +03:00
|
|
|
groups := re.FindAllStringSubmatch(ua, -1)
|
|
|
|
if len(groups) == 0 || len(groups[0]) != 3 {
|
|
|
|
return "", "", errors.New("failed to parse user agent string")
|
|
|
|
}
|
|
|
|
return groups[0][1], groups[0][2], nil
|
|
|
|
}
|
2023-07-09 21:16:34 +03:00
|
|
|
|
|
|
|
func RaiseForStatus(res *http.Response, err error) (*http.Response, error) {
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
if res.StatusCode >= 400 {
|
|
|
|
return res, fmt.Errorf("got response status %d for '%s %s'", res.StatusCode, res.Request.Method, res.Request.URL.String())
|
|
|
|
}
|
|
|
|
return res, nil
|
|
|
|
}
|