121 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Example static file server.
 | |
| //
 | |
| // Serves static files from the given directory.
 | |
| // Exports various stats at /stats .
 | |
| package main
 | |
| 
 | |
| import (
 | |
| 	"expvar"
 | |
| 	"flag"
 | |
| 	"log"
 | |
| 
 | |
| 	"github.com/valyala/fasthttp"
 | |
| 	"github.com/valyala/fasthttp/expvarhandler"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	addr               = flag.String("addr", "localhost:8080", "TCP address to listen to")
 | |
| 	addrTLS            = flag.String("addrTLS", "", "TCP address to listen to TLS (aka SSL or HTTPS) requests. Leave empty for disabling TLS")
 | |
| 	byteRange          = flag.Bool("byteRange", false, "Enables byte range requests if set to true")
 | |
| 	certFile           = flag.String("certFile", "./ssl-cert-snakeoil.pem", "Path to TLS certificate file")
 | |
| 	compress           = flag.Bool("compress", false, "Enables transparent response compression if set to true")
 | |
| 	dir                = flag.String("dir", "/usr/share/nginx/html", "Directory to serve static files from")
 | |
| 	generateIndexPages = flag.Bool("generateIndexPages", true, "Whether to generate directory index pages")
 | |
| 	keyFile            = flag.String("keyFile", "./ssl-cert-snakeoil.key", "Path to TLS key file")
 | |
| 	vhost              = flag.Bool("vhost", false, "Enables virtual hosting by prepending the requested path with the requested hostname")
 | |
| )
 | |
| 
 | |
| func main() {
 | |
| 	// Parse command-line flags.
 | |
| 	flag.Parse()
 | |
| 
 | |
| 	// Setup FS handler
 | |
| 	fs := &fasthttp.FS{
 | |
| 		Root:               *dir,
 | |
| 		IndexNames:         []string{"index.html"},
 | |
| 		GenerateIndexPages: *generateIndexPages,
 | |
| 		Compress:           *compress,
 | |
| 		AcceptByteRange:    *byteRange,
 | |
| 	}
 | |
| 	if *vhost {
 | |
| 		fs.PathRewrite = fasthttp.NewVHostPathRewriter(0)
 | |
| 	}
 | |
| 	fsHandler := fs.NewRequestHandler()
 | |
| 
 | |
| 	// Create RequestHandler serving server stats on /stats and files
 | |
| 	// on other requested paths.
 | |
| 	// /stats output may be filtered using regexps. For example:
 | |
| 	//
 | |
| 	//   * /stats?r=fs will show only stats (expvars) containing 'fs'
 | |
| 	//     in their names.
 | |
| 	requestHandler := func(ctx *fasthttp.RequestCtx) {
 | |
| 		switch string(ctx.Path()) {
 | |
| 		case "/stats":
 | |
| 			expvarhandler.ExpvarHandler(ctx)
 | |
| 		default:
 | |
| 			fsHandler(ctx)
 | |
| 			updateFSCounters(ctx)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Start HTTP server.
 | |
| 	if len(*addr) > 0 {
 | |
| 		log.Printf("Starting HTTP server on %q", *addr)
 | |
| 		go func() {
 | |
| 			if err := fasthttp.ListenAndServe(*addr, requestHandler); err != nil {
 | |
| 				log.Fatalf("error in ListenAndServe: %s", err)
 | |
| 			}
 | |
| 		}()
 | |
| 	}
 | |
| 
 | |
| 	// Start HTTPS server.
 | |
| 	if len(*addrTLS) > 0 {
 | |
| 		log.Printf("Starting HTTPS server on %q", *addrTLS)
 | |
| 		go func() {
 | |
| 			if err := fasthttp.ListenAndServeTLS(*addrTLS, *certFile, *keyFile, requestHandler); err != nil {
 | |
| 				log.Fatalf("error in ListenAndServeTLS: %s", err)
 | |
| 			}
 | |
| 		}()
 | |
| 	}
 | |
| 
 | |
| 	log.Printf("Serving files from directory %q", *dir)
 | |
| 	log.Printf("See stats at http://%s/stats", *addr)
 | |
| 
 | |
| 	// Wait forever.
 | |
| 	select {}
 | |
| }
 | |
| 
 | |
| func updateFSCounters(ctx *fasthttp.RequestCtx) {
 | |
| 	// Increment the number of fsHandler calls.
 | |
| 	fsCalls.Add(1)
 | |
| 
 | |
| 	// Update other stats counters
 | |
| 	resp := &ctx.Response
 | |
| 	switch resp.StatusCode() {
 | |
| 	case fasthttp.StatusOK:
 | |
| 		fsOKResponses.Add(1)
 | |
| 		fsResponseBodyBytes.Add(int64(resp.Header.ContentLength()))
 | |
| 	case fasthttp.StatusNotModified:
 | |
| 		fsNotModifiedResponses.Add(1)
 | |
| 	case fasthttp.StatusNotFound:
 | |
| 		fsNotFoundResponses.Add(1)
 | |
| 	default:
 | |
| 		fsOtherResponses.Add(1)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Various counters - see https://golang.org/pkg/expvar/ for details.
 | |
| var (
 | |
| 	// Counter for total number of fs calls
 | |
| 	fsCalls = expvar.NewInt("fsCalls")
 | |
| 
 | |
| 	// Counters for various response status codes
 | |
| 	fsOKResponses          = expvar.NewInt("fsOKResponses")
 | |
| 	fsNotModifiedResponses = expvar.NewInt("fsNotModifiedResponses")
 | |
| 	fsNotFoundResponses    = expvar.NewInt("fsNotFoundResponses")
 | |
| 	fsOtherResponses       = expvar.NewInt("fsOtherResponses")
 | |
| 
 | |
| 	// Total size in bytes for OK response bodies served.
 | |
| 	fsResponseBodyBytes = expvar.NewInt("fsResponseBodyBytes")
 | |
| )
 |