2020-05-30 16:14:17 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"math/rand"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2020-06-08 05:26:38 +00:00
|
|
|
"strings"
|
2022-11-27 14:51:41 +00:00
|
|
|
"sync"
|
2020-05-30 16:14:17 +00:00
|
|
|
"time"
|
2022-11-27 14:51:41 +00:00
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
"github.com/chubin/wttr.in/internal/routing"
|
|
|
|
|
2022-11-27 14:51:41 +00:00
|
|
|
lru "github.com/hashicorp/golang-lru"
|
2020-05-30 16:14:17 +00:00
|
|
|
)
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
type responseWithHeader struct {
|
2022-11-27 14:51:41 +00:00
|
|
|
InProgress bool // true if the request is being processed
|
|
|
|
Expires time.Time // expiration time of the cache entry
|
|
|
|
|
|
|
|
Body []byte
|
|
|
|
Header http.Header
|
|
|
|
StatusCode int // e.g. 200
|
|
|
|
}
|
|
|
|
|
|
|
|
// RequestProcessor handles incoming requests.
|
|
|
|
type RequestProcessor struct {
|
|
|
|
peakRequest30 sync.Map
|
|
|
|
peakRequest60 sync.Map
|
|
|
|
lruCache *lru.Cache
|
2022-11-27 21:16:32 +00:00
|
|
|
stats *Stats
|
2022-11-29 20:24:53 +00:00
|
|
|
router routing.Router
|
2022-11-27 14:51:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewRequestProcessor returns new RequestProcessor.
|
|
|
|
func NewRequestProcessor() (*RequestProcessor, error) {
|
|
|
|
lruCache, err := lru.New(lruCacheSize)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-11-27 21:16:32 +00:00
|
|
|
rp := &RequestProcessor{
|
2022-11-27 14:51:41 +00:00
|
|
|
lruCache: lruCache,
|
2022-11-27 21:16:32 +00:00
|
|
|
stats: NewStats(),
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize routes.
|
|
|
|
rp.router.AddPath("/:stats", rp.stats)
|
|
|
|
|
|
|
|
return rp, nil
|
2022-11-27 14:51:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Start starts async request processor jobs, such as peak handling.
|
|
|
|
func (rp *RequestProcessor) Start() {
|
|
|
|
rp.startPeakHandling()
|
|
|
|
}
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
func (rp *RequestProcessor) ProcessRequest(r *http.Request) (*responseWithHeader, error) {
|
2022-11-20 16:55:17 +00:00
|
|
|
var (
|
2022-11-29 20:24:53 +00:00
|
|
|
response *responseWithHeader
|
2022-11-20 16:55:17 +00:00
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
2022-11-27 21:16:32 +00:00
|
|
|
rp.stats.Inc("total")
|
|
|
|
|
|
|
|
// Main routing logic.
|
|
|
|
if rh := rp.router.Route(r); rh != nil {
|
|
|
|
result := rh.Response(r)
|
|
|
|
if result != nil {
|
2022-11-29 20:24:53 +00:00
|
|
|
return fromCadre(result), nil
|
2022-11-27 21:16:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-20 16:55:17 +00:00
|
|
|
if resp, ok := redirectInsecure(r); ok {
|
2022-11-27 21:16:32 +00:00
|
|
|
rp.stats.Inc("redirects")
|
2022-11-20 16:55:17 +00:00
|
|
|
return resp, nil
|
2021-11-01 11:27:27 +00:00
|
|
|
}
|
|
|
|
|
2020-06-08 05:26:38 +00:00
|
|
|
if dontCache(r) {
|
2022-11-27 21:16:32 +00:00
|
|
|
rp.stats.Inc("uncached")
|
2020-06-08 05:26:38 +00:00
|
|
|
return get(r)
|
|
|
|
}
|
|
|
|
|
2020-05-30 16:14:17 +00:00
|
|
|
cacheDigest := getCacheDigest(r)
|
|
|
|
|
2020-06-08 05:26:38 +00:00
|
|
|
foundInCache := false
|
|
|
|
|
2022-11-27 14:51:41 +00:00
|
|
|
rp.savePeakRequest(cacheDigest, r)
|
2020-05-30 16:14:17 +00:00
|
|
|
|
2022-11-27 14:51:41 +00:00
|
|
|
cacheBody, ok := rp.lruCache.Get(cacheDigest)
|
2020-05-30 16:14:17 +00:00
|
|
|
if ok {
|
2022-11-27 21:16:32 +00:00
|
|
|
rp.stats.Inc("cache1")
|
2022-11-29 20:24:53 +00:00
|
|
|
cacheEntry := cacheBody.(responseWithHeader)
|
2020-05-30 16:14:17 +00:00
|
|
|
|
|
|
|
// if after all attempts we still have no answer,
|
|
|
|
// we try to make the query on our own
|
|
|
|
for attempts := 0; attempts < 300; attempts++ {
|
|
|
|
if !ok || !cacheEntry.InProgress {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
time.Sleep(30 * time.Millisecond)
|
2022-11-27 14:51:41 +00:00
|
|
|
cacheBody, ok = rp.lruCache.Get(cacheDigest)
|
2022-11-20 16:55:17 +00:00
|
|
|
if ok && cacheBody != nil {
|
2022-11-29 20:24:53 +00:00
|
|
|
cacheEntry = cacheBody.(responseWithHeader)
|
2022-11-20 16:55:17 +00:00
|
|
|
}
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
if cacheEntry.InProgress {
|
|
|
|
log.Printf("TIMEOUT: %s\n", cacheDigest)
|
|
|
|
}
|
|
|
|
if ok && !cacheEntry.InProgress && cacheEntry.Expires.After(time.Now()) {
|
2022-11-20 16:55:17 +00:00
|
|
|
response = &cacheEntry
|
2020-05-30 16:14:17 +00:00
|
|
|
foundInCache = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !foundInCache {
|
2022-11-27 21:16:32 +00:00
|
|
|
// Handling query.
|
|
|
|
format := r.URL.Query().Get("format")
|
|
|
|
if len(format) != 0 {
|
|
|
|
rp.stats.Inc("format")
|
|
|
|
if format == "j1" {
|
|
|
|
rp.stats.Inc("format=j1")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
rp.lruCache.Add(cacheDigest, responseWithHeader{InProgress: true})
|
2022-11-27 21:16:32 +00:00
|
|
|
|
2022-11-20 16:55:17 +00:00
|
|
|
response, err = get(r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-07 10:40:06 +00:00
|
|
|
if response.StatusCode == 200 || response.StatusCode == 304 || response.StatusCode == 404 {
|
2022-11-27 14:51:41 +00:00
|
|
|
rp.lruCache.Add(cacheDigest, *response)
|
2020-05-30 16:14:17 +00:00
|
|
|
} else {
|
|
|
|
log.Printf("REMOVE: %d response for %s from cache\n", response.StatusCode, cacheDigest)
|
2022-11-27 14:51:41 +00:00
|
|
|
rp.lruCache.Remove(cacheDigest)
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
}
|
2022-11-20 16:55:17 +00:00
|
|
|
return response, nil
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
func get(req *http.Request) (*responseWithHeader, error) {
|
2020-05-30 16:14:17 +00:00
|
|
|
|
|
|
|
client := &http.Client{}
|
|
|
|
|
|
|
|
queryURL := fmt.Sprintf("http://%s%s", req.Host, req.RequestURI)
|
|
|
|
|
|
|
|
proxyReq, err := http.NewRequest(req.Method, queryURL, req.Body)
|
|
|
|
if err != nil {
|
2022-11-20 16:55:17 +00:00
|
|
|
return nil, err
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// proxyReq.Header.Set("Host", req.Host)
|
|
|
|
// proxyReq.Header.Set("X-Forwarded-For", req.RemoteAddr)
|
|
|
|
|
|
|
|
for header, values := range req.Header {
|
|
|
|
for _, value := range values {
|
|
|
|
proxyReq.Header.Add(header, value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-20 09:03:31 +00:00
|
|
|
if proxyReq.Header.Get("X-Forwarded-For") == "" {
|
|
|
|
proxyReq.Header.Set("X-Forwarded-For", ipFromAddr(req.RemoteAddr))
|
|
|
|
}
|
|
|
|
|
2020-05-30 16:14:17 +00:00
|
|
|
res, err := client.Do(proxyReq)
|
|
|
|
if err != nil {
|
2022-11-20 16:55:17 +00:00
|
|
|
return nil, err
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
body, err := ioutil.ReadAll(res.Body)
|
|
|
|
if err != nil {
|
2022-11-20 16:55:17 +00:00
|
|
|
return nil, err
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
return &responseWithHeader{
|
2020-05-30 16:14:17 +00:00
|
|
|
InProgress: false,
|
|
|
|
Expires: time.Now().Add(time.Duration(randInt(1000, 1500)) * time.Second),
|
|
|
|
Body: body,
|
|
|
|
Header: res.Header,
|
|
|
|
StatusCode: res.StatusCode,
|
2022-11-20 16:55:17 +00:00
|
|
|
}, nil
|
2020-05-30 16:14:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of the cache.get_signature of original wttr.in
|
|
|
|
func getCacheDigest(req *http.Request) string {
|
|
|
|
|
|
|
|
userAgent := req.Header.Get("User-Agent")
|
|
|
|
|
|
|
|
queryHost := req.Host
|
|
|
|
queryString := req.RequestURI
|
|
|
|
|
|
|
|
clientIPAddress := readUserIP(req)
|
|
|
|
|
|
|
|
lang := req.Header.Get("Accept-Language")
|
|
|
|
|
|
|
|
return fmt.Sprintf("%s:%s%s:%s:%s", userAgent, queryHost, queryString, clientIPAddress, lang)
|
|
|
|
}
|
|
|
|
|
2020-06-08 05:26:38 +00:00
|
|
|
// return true if request should not be cached
|
|
|
|
func dontCache(req *http.Request) bool {
|
|
|
|
|
|
|
|
// dont cache cyclic requests
|
|
|
|
loc := strings.Split(req.RequestURI, "?")[0]
|
2021-11-01 11:27:27 +00:00
|
|
|
return strings.Contains(loc, ":")
|
|
|
|
}
|
|
|
|
|
|
|
|
// redirectInsecure returns redirection response, and bool value, if redirection was needed,
|
|
|
|
// if the query comes from a browser, and it is insecure.
|
|
|
|
//
|
|
|
|
// Insecure queries are marked by the frontend web server
|
|
|
|
// with X-Forwarded-Proto header:
|
|
|
|
//
|
|
|
|
// proxy_set_header X-Forwarded-Proto $scheme;
|
|
|
|
//
|
|
|
|
//
|
2022-11-29 20:24:53 +00:00
|
|
|
func redirectInsecure(req *http.Request) (*responseWithHeader, bool) {
|
2021-11-01 11:27:27 +00:00
|
|
|
if isPlainTextAgent(req.Header.Get("User-Agent")) {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2022-11-20 13:00:59 +00:00
|
|
|
if req.TLS != nil || strings.ToLower(req.Header.Get("X-Forwarded-Proto")) == "https" {
|
2021-11-01 11:27:27 +00:00
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
target := "https://" + req.Host + req.URL.Path
|
|
|
|
if len(req.URL.RawQuery) > 0 {
|
|
|
|
target += "?" + req.URL.RawQuery
|
|
|
|
}
|
|
|
|
|
|
|
|
body := []byte(fmt.Sprintf(`<HTML><HEAD><meta http-equiv="content-type" content="text/html;charset=utf-8">
|
|
|
|
<TITLE>301 Moved</TITLE></HEAD><BODY>
|
|
|
|
<H1>301 Moved</H1>
|
|
|
|
The document has moved
|
|
|
|
<A HREF="%s">here</A>.
|
|
|
|
</BODY></HTML>
|
|
|
|
`, target))
|
|
|
|
|
2022-11-29 20:24:53 +00:00
|
|
|
return &responseWithHeader{
|
2021-11-01 11:27:27 +00:00
|
|
|
InProgress: false,
|
|
|
|
Expires: time.Now().Add(time.Duration(randInt(1000, 1500)) * time.Second),
|
|
|
|
Body: body,
|
|
|
|
Header: http.Header{"Location": []string{target}},
|
|
|
|
StatusCode: 301,
|
|
|
|
}, true
|
|
|
|
}
|
|
|
|
|
|
|
|
// isPlainTextAgent returns true if userAgent is a plain-text agent
|
|
|
|
func isPlainTextAgent(userAgent string) bool {
|
|
|
|
userAgentLower := strings.ToLower(userAgent)
|
|
|
|
for _, signature := range plainTextAgents {
|
|
|
|
if strings.Contains(userAgentLower, signature) {
|
|
|
|
return true
|
|
|
|
}
|
2020-06-08 05:26:38 +00:00
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-05-30 16:14:17 +00:00
|
|
|
func readUserIP(r *http.Request) string {
|
|
|
|
IPAddress := r.Header.Get("X-Real-Ip")
|
|
|
|
if IPAddress == "" {
|
|
|
|
IPAddress = r.Header.Get("X-Forwarded-For")
|
|
|
|
}
|
|
|
|
if IPAddress == "" {
|
|
|
|
IPAddress = r.RemoteAddr
|
|
|
|
var err error
|
|
|
|
IPAddress, _, err = net.SplitHostPort(IPAddress)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("ERROR: userip: %q is not IP:port\n", IPAddress)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return IPAddress
|
|
|
|
}
|
|
|
|
|
|
|
|
func randInt(min int, max int) int {
|
|
|
|
return min + rand.Intn(max-min)
|
|
|
|
}
|
2022-11-20 09:03:31 +00:00
|
|
|
|
|
|
|
// ipFromAddr returns IP address from a ADDR:PORT pair.
|
|
|
|
func ipFromAddr(s string) string {
|
|
|
|
pos := strings.LastIndex(s, ":")
|
|
|
|
if pos == -1 {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
return s[:pos]
|
|
|
|
}
|
2022-11-29 20:24:53 +00:00
|
|
|
|
|
|
|
// fromCadre converts Cadre into a responseWithHeader.
|
|
|
|
func fromCadre(cadre *routing.Cadre) *responseWithHeader {
|
|
|
|
return &responseWithHeader{
|
|
|
|
Body: cadre.Body,
|
|
|
|
Expires: cadre.Expires,
|
|
|
|
StatusCode: 200,
|
|
|
|
InProgress: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|