2016-11-29 17:26:36 +01:00
// Copyright 2016 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package public
2017-01-28 23:14:56 +01:00
import (
2018-02-03 23:37:05 +01:00
"log"
"net/http"
2017-01-28 23:14:56 +01:00
"path"
2018-02-03 23:37:05 +01:00
"path/filepath"
"strings"
2017-01-28 23:14:56 +01:00
2020-11-17 23:44:52 +01:00
"code.gitea.io/gitea/modules/httpcache"
2017-01-28 23:14:56 +01:00
"code.gitea.io/gitea/modules/setting"
)
2021-01-29 23:35:30 +08:00
// Options represents the available options to configure the handler.
2016-11-29 17:26:36 +01:00
type Options struct {
Directory string
2018-02-03 23:37:05 +01:00
IndexFile string
2016-11-29 17:26:36 +01:00
SkipLogging bool
2020-11-17 23:44:52 +01:00
FileSystem http . FileSystem
Prefix string
2016-11-29 17:26:36 +01:00
}
2017-01-28 23:14:56 +01:00
2020-06-20 15:20:25 +02:00
// KnownPublicEntries list all direct children in the `public` directory
var KnownPublicEntries = [ ] string {
2020-04-18 23:01:06 +02:00
"css" ,
2021-03-30 20:47:34 +08:00
"fonts" ,
2020-04-18 23:01:06 +02:00
"img" ,
"js" ,
2020-06-20 15:20:25 +02:00
"serviceworker.js" ,
2020-04-18 23:01:06 +02:00
"vendor" ,
2021-01-02 06:05:45 +08:00
"favicon.ico" ,
2020-04-18 23:01:06 +02:00
}
2021-01-29 23:35:30 +08:00
// Custom implements the static handler for serving custom assets.
2020-11-13 20:51:07 +08:00
func Custom ( opts * Options ) func ( next http . Handler ) http . Handler {
2018-02-03 23:37:05 +01:00
return opts . staticHandler ( path . Join ( setting . CustomPath , "public" ) )
}
// staticFileSystem implements http.FileSystem interface.
type staticFileSystem struct {
dir * http . Dir
}
func newStaticFileSystem ( directory string ) staticFileSystem {
if ! filepath . IsAbs ( directory ) {
2020-11-13 20:51:07 +08:00
directory = filepath . Join ( setting . AppWorkPath , directory )
2018-02-03 23:37:05 +01:00
}
dir := http . Dir ( directory )
return staticFileSystem { & dir }
}
func ( fs staticFileSystem ) Open ( name string ) ( http . File , error ) {
return fs . dir . Open ( name )
}
// StaticHandler sets up a new middleware for serving static files in the
2020-11-13 20:51:07 +08:00
func StaticHandler ( dir string , opts * Options ) func ( next http . Handler ) http . Handler {
2018-02-03 23:37:05 +01:00
return opts . staticHandler ( dir )
}
2020-11-13 20:51:07 +08:00
func ( opts * Options ) staticHandler ( dir string ) func ( next http . Handler ) http . Handler {
return func ( next http . Handler ) http . Handler {
// Defaults
if len ( opts . IndexFile ) == 0 {
opts . IndexFile = "index.html"
}
// Normalize the prefix if provided
if opts . Prefix != "" {
// Ensure we have a leading '/'
if opts . Prefix [ 0 ] != '/' {
opts . Prefix = "/" + opts . Prefix
}
// Remove any trailing '/'
opts . Prefix = strings . TrimRight ( opts . Prefix , "/" )
}
if opts . FileSystem == nil {
opts . FileSystem = newStaticFileSystem ( dir )
2018-02-03 23:37:05 +01:00
}
2020-11-13 20:51:07 +08:00
return http . HandlerFunc ( func ( w http . ResponseWriter , req * http . Request ) {
if ! opts . handle ( w , req , opts ) {
next . ServeHTTP ( w , req )
}
} )
2018-02-03 23:37:05 +01:00
}
}
2020-12-24 12:25:17 +08:00
// parseAcceptEncoding parse Accept-Encoding: deflate, gzip;q=1.0, *;q=0.5 as compress methods
func parseAcceptEncoding ( val string ) map [ string ] bool {
parts := strings . Split ( val , ";" )
var types = make ( map [ string ] bool )
for _ , v := range strings . Split ( parts [ 0 ] , "," ) {
types [ strings . TrimSpace ( v ) ] = true
}
return types
}
2020-11-13 20:51:07 +08:00
func ( opts * Options ) handle ( w http . ResponseWriter , req * http . Request , opt * Options ) bool {
if req . Method != "GET" && req . Method != "HEAD" {
2018-02-03 23:37:05 +01:00
return false
}
2020-11-13 20:51:07 +08:00
file := req . URL . Path
2018-02-03 23:37:05 +01:00
// if we have a prefix, filter requests by stripping the prefix
if opt . Prefix != "" {
if ! strings . HasPrefix ( file , opt . Prefix ) {
return false
}
file = file [ len ( opt . Prefix ) : ]
if file != "" && file [ 0 ] != '/' {
return false
}
}
f , err := opt . FileSystem . Open ( file )
if err != nil {
2020-04-18 23:01:06 +02:00
// 404 requests to any known entries in `public`
if path . Base ( opts . Directory ) == "public" {
parts := strings . Split ( file , "/" )
if len ( parts ) < 2 {
return false
}
2020-06-20 15:20:25 +02:00
for _ , entry := range KnownPublicEntries {
2020-04-18 23:01:06 +02:00
if entry == parts [ 1 ] {
2020-11-13 20:51:07 +08:00
w . WriteHeader ( 404 )
2020-04-18 23:01:06 +02:00
return true
}
}
}
2018-02-03 23:37:05 +01:00
return false
}
defer f . Close ( )
fi , err := f . Stat ( )
if err != nil {
log . Printf ( "[Static] %q exists, but fails to open: %v" , file , err )
return true
}
// Try to serve index file
if fi . IsDir ( ) {
// Redirect if missing trailing slash.
2020-11-13 20:51:07 +08:00
if ! strings . HasSuffix ( req . URL . Path , "/" ) {
http . Redirect ( w , req , path . Clean ( req . URL . Path + "/" ) , http . StatusFound )
2018-02-03 23:37:05 +01:00
return true
}
f , err = opt . FileSystem . Open ( file )
if err != nil {
return false // Discard error.
}
defer f . Close ( )
fi , err = f . Stat ( )
if err != nil || fi . IsDir ( ) {
2020-11-13 20:51:07 +08:00
return false
2018-02-03 23:37:05 +01:00
}
}
if ! opt . SkipLogging {
log . Println ( "[Static] Serving " + file )
}
2020-11-17 23:44:52 +01:00
if httpcache . HandleEtagCache ( req , w , fi ) {
return true
2018-02-03 23:37:05 +01:00
}
2020-12-24 12:25:17 +08:00
ServeContent ( w , req , fi , fi . ModTime ( ) , f )
2018-02-03 23:37:05 +01:00
return true
}