89 lines
3.0 KiB
Go
89 lines
3.0 KiB
Go
package dist
|
|
|
|
//go:generate go-bindata -o dist_gen.go -ignore 'map|go' -pkg dist ../ui/build/...
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
|
|
"github.com/elazarl/go-bindata-assetfs"
|
|
)
|
|
|
|
// DebugAssets serves assets via a specified directory
|
|
type DebugAssets struct {
|
|
Dir string // Dir is a directory location of asset files
|
|
Default string // Default is the file to serve if file is not found.
|
|
}
|
|
|
|
// Handler is an http.FileServer for the Dir
|
|
func (d *DebugAssets) Handler() http.Handler {
|
|
return http.FileServer(NewDir(d.Dir, d.Default))
|
|
}
|
|
|
|
// BindataAssets serves assets from go-bindata, but, also serves Default if assent doesn't exist
|
|
// This is to support single-page react-apps with its own router.
|
|
type BindataAssets struct {
|
|
Prefix string // Prefix is prepended to the http file request
|
|
Default string // Default is the file to serve if the file is not found
|
|
DefaultContentType string // DefaultContentType is the content type of the default file
|
|
}
|
|
|
|
// Handler serves go-bindata using a go-bindata-assetfs façade
|
|
func (b *BindataAssets) Handler() http.Handler {
|
|
return b
|
|
}
|
|
|
|
// addCacheHeaders requests an hour of Cache-Control and sets an ETag based on file size and modtime
|
|
func (b *BindataAssets) addCacheHeaders(filename string, w http.ResponseWriter) error {
|
|
w.Header().Add("Cache-Control", "public, max-age=3600")
|
|
fi, err := AssetInfo(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
hour, minute, second := fi.ModTime().Clock()
|
|
etag := fmt.Sprintf(`"%d%d%d%d%d"`, fi.Size(), fi.ModTime().Day(), hour, minute, second)
|
|
|
|
w.Header().Set("ETag", etag)
|
|
return nil
|
|
}
|
|
|
|
// ServeHTTP wraps http.FileServer by returning a default asset if the asset
|
|
// doesn't exist. This supports single-page react-apps with its own
|
|
// built-in router. Additionally, we override the content-type if the
|
|
// Default file is used.
|
|
func (b *BindataAssets) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
// def wraps the assets to return the default file if the file doesn't exist
|
|
def := func(name string) ([]byte, error) {
|
|
// If the named asset exists, then return it directly.
|
|
octets, err := Asset(name)
|
|
if err != nil {
|
|
// If this is at / then we just error out so we can return a Directory
|
|
// This directory will then be redirected by go to the /index.html
|
|
if name == b.Prefix {
|
|
return nil, err
|
|
}
|
|
// If this is anything other than slash, we just return the default
|
|
// asset. This default asset will handle the routing.
|
|
// Additionally, because we know we are returning the default asset,
|
|
// we need to set the default asset's content-type.
|
|
w.Header().Set("Content-Type", b.DefaultContentType)
|
|
if err := b.addCacheHeaders(b.Default, w); err != nil {
|
|
return nil, err
|
|
}
|
|
return Asset(b.Default)
|
|
}
|
|
if err := b.addCacheHeaders(name, w); err != nil {
|
|
return nil, err
|
|
}
|
|
return octets, nil
|
|
}
|
|
var dir http.FileSystem = &assetfs.AssetFS{
|
|
Asset: def,
|
|
AssetDir: AssetDir,
|
|
AssetInfo: AssetInfo,
|
|
Prefix: b.Prefix,
|
|
}
|
|
http.FileServer(dir).ServeHTTP(w, r)
|
|
}
|