Implement Last-Modified caching
This commit is contained in:
parent
c7faa254c5
commit
e47cb5f0e4
89
server/serve_files.go
Normal file
89
server/serve_files.go
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"compress/gzip"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var files = []File{
|
||||||
|
File{"bundle.js", "text/javascript"},
|
||||||
|
File{"bundle.css", "text/css"},
|
||||||
|
File{"font/fontello.woff", "application/font-woff"},
|
||||||
|
File{"font/fontello.ttf", "application/x-font-ttf"},
|
||||||
|
File{"font/fontello.eot", "application/vnd.ms-fontobject"},
|
||||||
|
File{"font/fontello.svg", "image/svg+xml"},
|
||||||
|
}
|
||||||
|
|
||||||
|
type File struct {
|
||||||
|
Path string
|
||||||
|
ContentType string
|
||||||
|
}
|
||||||
|
|
||||||
|
func serveFiles(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/" {
|
||||||
|
serveFile(w, r, "index.html.gz", "text/html")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, file := range files {
|
||||||
|
if strings.HasSuffix(r.URL.Path, file.Path) {
|
||||||
|
serveFile(w, r, file.Path+".gz", file.ContentType)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
serveFile(w, r, "index.html.gz", "text/html")
|
||||||
|
}
|
||||||
|
|
||||||
|
func serveFile(w http.ResponseWriter, r *http.Request, path, contentType string) {
|
||||||
|
info, err := AssetInfo(path)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !modifiedSince(w, r, info.ModTime()) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := Asset(path)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", contentType)
|
||||||
|
|
||||||
|
if strings.Contains(r.Header.Get("Accept-Encoding"), "gzip") {
|
||||||
|
w.Header().Set("Content-Encoding", "gzip")
|
||||||
|
w.Header().Set("Content-Length", strconv.Itoa(len(data)))
|
||||||
|
w.Write(data)
|
||||||
|
} else {
|
||||||
|
gzr, err := gzip.NewReader(bytes.NewReader(data))
|
||||||
|
buf, err := ioutil.ReadAll(gzr)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Length", strconv.Itoa(len(buf)))
|
||||||
|
w.Write(buf)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func modifiedSince(w http.ResponseWriter, r *http.Request, modtime time.Time) bool {
|
||||||
|
t, err := time.Parse(http.TimeFormat, r.Header.Get("If-Modified-Since"))
|
||||||
|
|
||||||
|
if err == nil && modtime.Before(t.Add(1*time.Second)) {
|
||||||
|
w.WriteHeader(http.StatusNotModified)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Last-Modified", modtime.UTC().Format(http.TimeFormat))
|
||||||
|
return true
|
||||||
|
}
|
@ -1,13 +1,9 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"compress/gzip"
|
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/khlieng/name_pending/Godeps/_workspace/src/github.com/gorilla/websocket"
|
"github.com/khlieng/name_pending/Godeps/_workspace/src/github.com/gorilla/websocket"
|
||||||
@ -21,15 +17,6 @@ var (
|
|||||||
sessions map[string]*Session
|
sessions map[string]*Session
|
||||||
sessionLock sync.Mutex
|
sessionLock sync.Mutex
|
||||||
|
|
||||||
files = []File{
|
|
||||||
File{"bundle.js", "text/javascript"},
|
|
||||||
File{"bundle.css", "text/css"},
|
|
||||||
File{"font/fontello.eot", "application/vnd.ms-fontobject"},
|
|
||||||
File{"font/fontello.svg", "image/svg+xml"},
|
|
||||||
File{"font/fontello.ttf", "application/x-font-ttf"},
|
|
||||||
File{"font/fontello.woff", "application/font-woff"},
|
|
||||||
}
|
|
||||||
|
|
||||||
upgrader = websocket.Upgrader{
|
upgrader = websocket.Upgrader{
|
||||||
ReadBufferSize: 1024,
|
ReadBufferSize: 1024,
|
||||||
WriteBufferSize: 1024,
|
WriteBufferSize: 1024,
|
||||||
@ -39,11 +26,6 @@ var (
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
type File struct {
|
|
||||||
Path string
|
|
||||||
ContentType string
|
|
||||||
}
|
|
||||||
|
|
||||||
func Run(port int) {
|
func Run(port int) {
|
||||||
defer storage.Close()
|
defer storage.Close()
|
||||||
|
|
||||||
@ -71,39 +53,6 @@ func upgradeWS(w http.ResponseWriter, r *http.Request) {
|
|||||||
handleWS(conn)
|
handleWS(conn)
|
||||||
}
|
}
|
||||||
|
|
||||||
func serveFiles(w http.ResponseWriter, r *http.Request) {
|
|
||||||
if r.URL.Path == "/" {
|
|
||||||
serveFile("index.html.gz", "text/html", w, r)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, file := range files {
|
|
||||||
if strings.HasSuffix(r.URL.Path, file.Path) {
|
|
||||||
serveFile(file.Path+".gz", file.ContentType, w, r)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
serveFile("index.html.gz", "text/html", w, r)
|
|
||||||
}
|
|
||||||
|
|
||||||
func serveFile(path, contentType string, w http.ResponseWriter, r *http.Request) {
|
|
||||||
data, _ := Asset(path)
|
|
||||||
|
|
||||||
w.Header().Set("Content-Type", contentType)
|
|
||||||
|
|
||||||
if strings.Contains(r.Header.Get("Accept-Encoding"), "gzip") {
|
|
||||||
w.Header().Set("Content-Encoding", "gzip")
|
|
||||||
w.Header().Set("Content-Length", strconv.Itoa(len(data)))
|
|
||||||
w.Write(data)
|
|
||||||
} else {
|
|
||||||
gzr, _ := gzip.NewReader(bytes.NewReader(data))
|
|
||||||
buf, _ := ioutil.ReadAll(gzr)
|
|
||||||
w.Header().Set("Content-Length", strconv.Itoa(len(buf)))
|
|
||||||
w.Write(buf)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func reconnect() {
|
func reconnect() {
|
||||||
for _, user := range storage.LoadUsers() {
|
for _, user := range storage.LoadUsers() {
|
||||||
session := NewSession()
|
session := NewSession()
|
||||||
|
Loading…
Reference in New Issue
Block a user