mirror of
https://github.com/chubin/wttr.in
synced 2025-01-26 10:45:01 +00:00
131 lines
2.9 KiB
Go
131 lines
2.9 KiB
Go
|
package main
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"log"
|
||
|
"math/rand"
|
||
|
"net"
|
||
|
"net/http"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
func processRequest(r *http.Request) responseWithHeader {
|
||
|
var response responseWithHeader
|
||
|
|
||
|
foundInCache := false
|
||
|
cacheDigest := getCacheDigest(r)
|
||
|
|
||
|
savePeakRequest(cacheDigest, r)
|
||
|
|
||
|
cacheBody, ok := lruCache.Get(cacheDigest)
|
||
|
if ok {
|
||
|
cacheEntry := cacheBody.(responseWithHeader)
|
||
|
|
||
|
// 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)
|
||
|
cacheBody, ok = lruCache.Get(cacheDigest)
|
||
|
cacheEntry = cacheBody.(responseWithHeader)
|
||
|
}
|
||
|
if cacheEntry.InProgress {
|
||
|
log.Printf("TIMEOUT: %s\n", cacheDigest)
|
||
|
}
|
||
|
if ok && !cacheEntry.InProgress && cacheEntry.Expires.After(time.Now()) {
|
||
|
response = cacheEntry
|
||
|
foundInCache = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if !foundInCache {
|
||
|
lruCache.Add(cacheDigest, responseWithHeader{InProgress: true})
|
||
|
response = get(r)
|
||
|
if response.StatusCode == 200 || response.StatusCode == 304 {
|
||
|
lruCache.Add(cacheDigest, response)
|
||
|
} else {
|
||
|
log.Printf("REMOVE: %d response for %s from cache\n", response.StatusCode, cacheDigest)
|
||
|
lruCache.Remove(cacheDigest)
|
||
|
}
|
||
|
}
|
||
|
return response
|
||
|
}
|
||
|
|
||
|
func get(req *http.Request) responseWithHeader {
|
||
|
|
||
|
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 {
|
||
|
log.Printf("Request: %s\n", err)
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
res, err := client.Do(proxyReq)
|
||
|
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
body, err := ioutil.ReadAll(res.Body)
|
||
|
if err != nil {
|
||
|
log.Println(err)
|
||
|
}
|
||
|
|
||
|
return responseWithHeader{
|
||
|
InProgress: false,
|
||
|
Expires: time.Now().Add(time.Duration(randInt(1000, 1500)) * time.Second),
|
||
|
Body: body,
|
||
|
Header: res.Header,
|
||
|
StatusCode: res.StatusCode,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
}
|
||
|
|
||
|
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)
|
||
|
}
|