assetserver

package module
v0.1.4 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Sep 23, 2024 License: MIT Imports: 13 Imported by: 0

README

assetserver

Go Reference

NOTE: this package is still under active development and has not yet reached version 1.0.

The assetserver package provides a file server designed for static assets.

Usage

The simplest way to use it is as a replacement for http.FileServer:

assets := assetserver.New(assetFS)

where assetFS is an embed.FS or comes from os.DirFS. Then the main handler or muxer routes all requests with a prefix (say, /static/) to assetSrv. For example:

mux := http.NewServeMux()
mux.Handle("/static/", http.StripPrefix("/static/", assets))
// ... set up remaining routes ...

Asset files that are served this way have two caching-related headers in the response:

  • ETag contains a hash of the file contents
  • Cache-Control is set to public, max-age=60 (1 minute)

This means that clients might cache the files for up to a minute before revalidating. Revalidation with If-None-Match is based on the file hash and the server will always be able to respond with 304 Not Modified unless the actual file content has changed.

Better caching with tags

By writing a bit more code, assetserver can be used to implement a near-optimal caching strategy. Asset files may be referred to by tagged names: instead of, for example, style.css, the client might request style.EI7Zfw9kFp.css. The tag (EI7Zfw9kFp) is the same file hash in the ETag header. When the file changes, the tag changes. When a tagged file is served, the Cache-Control header is

public, max-age=31536000, immutable

This means that files will generally be cached for as long as possible because the content corresponding to a tagged name never changes.

Tagged asset names are constructed with (*assetserver.Server).Tag. This method inserts a tag into a filename before the first dot.

Continuing our example from above, one might expose the Tag method as a template function:

funcs := template.FuncMap{
	"tag": assets.Tag,
}
tmpl, err := template.New("").Funcs(funcs).Parse(tmplText)
...

Then the HTML template might contain:

<link rel="stylesheet" href="/static/{{tag "css/style.css"}}">
<script src="/static/{{tag "js/main.js"}}"></script>

Comparison with http.FileServer

The caching strategies described above are a significant advantage that assetserver has over the standard library's file server when it comes to static assets. Using http.FileServer for these types of files can result in both under- and over-caching:

  • When files are served from disk, the only caching hint the client gets is in the form of the Last-Modified header. The details depend on the particular heuristics of each client, but typically in this case clients cache the file for some fraction of the time since last modification. This means that if the file mtimes are changed on deploy, many clients may immediately re-request the full contents. It also means that if it has been a long time since the mtimes changed, the clients might cache the assets for a significant length of time and not notice that the server has updated the assets in the meantime.
  • When files are served from an embed.FS, there is no mtime, no Last-Modified header, and clients generally won't cache the files at all.

Compared with http.FileServer, assetserver also leaks less internal server information and this also makes it better-suited for serving static assets:

  • It does not serve directory listings (it returns 404 Not Found for directories)
  • It does not serve index.html redirects
  • Internal errors (such as permission errors) are not exposed as HTTP errors; any internal error other than "not found" becomes a 500.

Local development with NewNoCache

When running a server for local development, having assets be cached for even a minute is generally undesirable. For this use case, a "no-cache" Server can be constructed using NewNoCache. A Server created this way serves files with Cache-Control: no-cache so that the browser will revalidate the assets on every page load.

Caveats

The assetserver package is designed specifically to serve static web assets that rarely or never change. Internally, it caches a bit of information about each file that it has ever served. This means that there are a few usages to be avoided:

  • The server is not appropriate for serving a very large number of different files (say, millions)
  • The internal cache never shrinks, so if new files are being created over time, memory usage will grow without bound even if old files are deleted.
  • The internal cache uses {mtime, size} as a proxy to determine whether a file has changed (and therefore whether we need to recompute the hash). If a file changes without altering the mtime or size, or if a file is altered non-atomically, incorrect caching information will be served.

Documentation

Overview

Package assetserver provides a file server for web assets.

Index

Constants

This section is empty.

Variables

This section is empty.

Functions

This section is empty.

Types

type Server

type Server struct {
	// contains filtered or unexported fields
}

A Server serves HTTP requests with the contents of a file system.

The response headers are set appropriately for web assets so that they are cached effectively:

  • The ETag header is set with a hash of the file contents
  • The Cache-Control header is set to "public, max-age=60"

If the requested file is tagged (see Server.Tag), the tag is removed before the file is retrieved. Tagged files are served with a Cache-Control header of

public, max-age=31536000, immutable

If the Server was created with NewNoCache, all assets are instead served with Cache-Control: no-cache.

In the following cases, Server sends a 404 Not Found response:

  • If the requested file doesn't exist in the file system
  • If the requested file is a directory
  • If the requested name is tagged but the tag does not match the corresponding file

For other errors, Server sends a 500 Internal Server Error response.

func New

func New(fsys fs.FS) *Server

New creates a Server from a file system.

The files that are opened from the file system must implement io.Seeker. The fs.FS implementations which satisfy this requirement include embed.FS and the result of calling os.DirFS.

func NewNoCache added in v0.1.4

func NewNoCache(fsys fs.FS) *Server

NewNoCache is like New, but the returned Server serves all assets with Cache-Control: no-cache.

NewNoCache is intended for non-production settings (such as local development).

func (*Server) ServeHTTP

func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request)

ServeHTTP serves file system contents matching the request.

func (*Server) Tag

func (s *Server) Tag(name string) (string, error)

Tag modifies the provided file name to include an asset tag preceding the first dot. The tag is based on a hash of the file contents. File names are slash-separated paths as given to the underlying fs.FS. If the name starts with a slash, it is removed before retrieving the file.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL