mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-11-04 18:17:08 +08:00 
			
		
		
		
	* Add file_limit option for file_server browse * Move file_limit inside browse. * add file_server_file_limit caddyfile adapt test.
		
			
				
	
	
		
			225 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			225 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2015 Matthew Holt and The Caddy Authors
 | 
						|
//
 | 
						|
// Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
// you may not use this file except in compliance with the License.
 | 
						|
// You may obtain a copy of the License at
 | 
						|
//
 | 
						|
//     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
//
 | 
						|
// Unless required by applicable law or agreed to in writing, software
 | 
						|
// distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
// See the License for the specific language governing permissions and
 | 
						|
// limitations under the License.
 | 
						|
 | 
						|
package fileserver
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/json"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"log"
 | 
						|
	"os"
 | 
						|
	"strconv"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/caddyserver/certmagic"
 | 
						|
	"github.com/spf13/cobra"
 | 
						|
	"go.uber.org/zap"
 | 
						|
 | 
						|
	caddycmd "github.com/caddyserver/caddy/v2/cmd"
 | 
						|
 | 
						|
	"github.com/caddyserver/caddy/v2"
 | 
						|
	"github.com/caddyserver/caddy/v2/caddyconfig"
 | 
						|
	"github.com/caddyserver/caddy/v2/modules/caddyhttp"
 | 
						|
	"github.com/caddyserver/caddy/v2/modules/caddyhttp/encode"
 | 
						|
	caddytpl "github.com/caddyserver/caddy/v2/modules/caddyhttp/templates"
 | 
						|
)
 | 
						|
 | 
						|
func init() {
 | 
						|
	caddycmd.RegisterCommand(caddycmd.Command{
 | 
						|
		Name:  "file-server",
 | 
						|
		Usage: "[--domain <example.com>] [--root <path>] [--listen <addr>] [--browse] [--reveal-symlinks] [--access-log] [--precompressed]",
 | 
						|
		Short: "Spins up a production-ready file server",
 | 
						|
		Long: `
 | 
						|
A simple but production-ready file server. Useful for quick deployments,
 | 
						|
demos, and development.
 | 
						|
 | 
						|
The listener's socket address can be customized with the --listen flag.
 | 
						|
 | 
						|
If a domain name is specified with --domain, the default listener address
 | 
						|
will be changed to the HTTPS port and the server will use HTTPS. If using
 | 
						|
a public domain, ensure A/AAAA records are properly configured before
 | 
						|
using this option.
 | 
						|
 | 
						|
By default, Zstandard and Gzip compression are enabled. Use --no-compress
 | 
						|
to disable compression.
 | 
						|
 | 
						|
If --browse is enabled, requests for folders without an index file will
 | 
						|
respond with a file listing.`,
 | 
						|
		CobraFunc: func(cmd *cobra.Command) {
 | 
						|
			cmd.Flags().StringP("domain", "d", "", "Domain name at which to serve the files")
 | 
						|
			cmd.Flags().StringP("root", "r", "", "The path to the root of the site")
 | 
						|
			cmd.Flags().StringP("listen", "l", "", "The address to which to bind the listener")
 | 
						|
			cmd.Flags().BoolP("browse", "b", false, "Enable directory browsing")
 | 
						|
			cmd.Flags().BoolP("reveal-symlinks", "", false, "Show symlink paths when browse is enabled.")
 | 
						|
			cmd.Flags().BoolP("templates", "t", false, "Enable template rendering")
 | 
						|
			cmd.Flags().BoolP("access-log", "a", false, "Enable the access log")
 | 
						|
			cmd.Flags().BoolP("debug", "v", false, "Enable verbose debug logs")
 | 
						|
			cmd.Flags().IntP("file-limit", "f", defaultDirEntryLimit, "Max directories to read")
 | 
						|
			cmd.Flags().BoolP("no-compress", "", false, "Disable Zstandard and Gzip compression")
 | 
						|
			cmd.Flags().StringSliceP("precompressed", "p", []string{}, "Specify precompression file extensions. Compression preference implied from flag order.")
 | 
						|
			cmd.RunE = caddycmd.WrapCommandFuncForCobra(cmdFileServer)
 | 
						|
			cmd.AddCommand(&cobra.Command{
 | 
						|
				Use:     "export-template",
 | 
						|
				Short:   "Exports the default file browser template",
 | 
						|
				Example: "caddy file-server export-template > browse.html",
 | 
						|
				RunE: func(cmd *cobra.Command, args []string) error {
 | 
						|
					_, err := io.WriteString(os.Stdout, BrowseTemplate)
 | 
						|
					return err
 | 
						|
				},
 | 
						|
			})
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func cmdFileServer(fs caddycmd.Flags) (int, error) {
 | 
						|
	caddy.TrapSignals()
 | 
						|
 | 
						|
	domain := fs.String("domain")
 | 
						|
	root := fs.String("root")
 | 
						|
	listen := fs.String("listen")
 | 
						|
	browse := fs.Bool("browse")
 | 
						|
	templates := fs.Bool("templates")
 | 
						|
	accessLog := fs.Bool("access-log")
 | 
						|
	fileLimit := fs.Int("file-limit")
 | 
						|
	debug := fs.Bool("debug")
 | 
						|
	revealSymlinks := fs.Bool("reveal-symlinks")
 | 
						|
	compress := !fs.Bool("no-compress")
 | 
						|
	precompressed, err := fs.GetStringSlice("precompressed")
 | 
						|
	if err != nil {
 | 
						|
		return caddy.ExitCodeFailedStartup, fmt.Errorf("invalid precompressed flag: %v", err)
 | 
						|
	}
 | 
						|
	var handlers []json.RawMessage
 | 
						|
 | 
						|
	if compress {
 | 
						|
		zstd, err := caddy.GetModule("http.encoders.zstd")
 | 
						|
		if err != nil {
 | 
						|
			return caddy.ExitCodeFailedStartup, err
 | 
						|
		}
 | 
						|
 | 
						|
		gzip, err := caddy.GetModule("http.encoders.gzip")
 | 
						|
		if err != nil {
 | 
						|
			return caddy.ExitCodeFailedStartup, err
 | 
						|
		}
 | 
						|
 | 
						|
		handlers = append(handlers, caddyconfig.JSONModuleObject(encode.Encode{
 | 
						|
			EncodingsRaw: caddy.ModuleMap{
 | 
						|
				"zstd": caddyconfig.JSON(zstd.New(), nil),
 | 
						|
				"gzip": caddyconfig.JSON(gzip.New(), nil),
 | 
						|
			},
 | 
						|
			Prefer: []string{"zstd", "gzip"},
 | 
						|
		}, "handler", "encode", nil))
 | 
						|
	}
 | 
						|
 | 
						|
	if templates {
 | 
						|
		handler := caddytpl.Templates{FileRoot: root}
 | 
						|
		handlers = append(handlers, caddyconfig.JSONModuleObject(handler, "handler", "templates", nil))
 | 
						|
	}
 | 
						|
 | 
						|
	handler := FileServer{Root: root}
 | 
						|
 | 
						|
	if len(precompressed) != 0 {
 | 
						|
		// logic mirrors modules/caddyhttp/fileserver/caddyfile.go case "precompressed"
 | 
						|
		var order []string
 | 
						|
		for _, compression := range precompressed {
 | 
						|
			modID := "http.precompressed." + compression
 | 
						|
			mod, err := caddy.GetModule(modID)
 | 
						|
			if err != nil {
 | 
						|
				return caddy.ExitCodeFailedStartup, fmt.Errorf("getting module named '%s': %v", modID, err)
 | 
						|
			}
 | 
						|
			inst := mod.New()
 | 
						|
			precompress, ok := inst.(encode.Precompressed)
 | 
						|
			if !ok {
 | 
						|
				return caddy.ExitCodeFailedStartup, fmt.Errorf("module %s is not a precompressor; is %T", modID, inst)
 | 
						|
			}
 | 
						|
			if handler.PrecompressedRaw == nil {
 | 
						|
				handler.PrecompressedRaw = make(caddy.ModuleMap)
 | 
						|
			}
 | 
						|
			handler.PrecompressedRaw[compression] = caddyconfig.JSON(precompress, nil)
 | 
						|
			order = append(order, compression)
 | 
						|
		}
 | 
						|
		handler.PrecompressedOrder = order
 | 
						|
	}
 | 
						|
 | 
						|
	if browse {
 | 
						|
		handler.Browse = &Browse{RevealSymlinks: revealSymlinks, FileLimit: fileLimit}
 | 
						|
	}
 | 
						|
 | 
						|
	handlers = append(handlers, caddyconfig.JSONModuleObject(handler, "handler", "file_server", nil))
 | 
						|
 | 
						|
	route := caddyhttp.Route{HandlersRaw: handlers}
 | 
						|
 | 
						|
	if domain != "" {
 | 
						|
		route.MatcherSetsRaw = []caddy.ModuleMap{
 | 
						|
			{
 | 
						|
				"host": caddyconfig.JSON(caddyhttp.MatchHost{domain}, nil),
 | 
						|
			},
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	server := &caddyhttp.Server{
 | 
						|
		ReadHeaderTimeout: caddy.Duration(10 * time.Second),
 | 
						|
		IdleTimeout:       caddy.Duration(30 * time.Second),
 | 
						|
		MaxHeaderBytes:    1024 * 10,
 | 
						|
		Routes:            caddyhttp.RouteList{route},
 | 
						|
	}
 | 
						|
	if listen == "" {
 | 
						|
		if domain == "" {
 | 
						|
			listen = ":80"
 | 
						|
		} else {
 | 
						|
			listen = ":" + strconv.Itoa(certmagic.HTTPSPort)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	server.Listen = []string{listen}
 | 
						|
	if accessLog {
 | 
						|
		server.Logs = &caddyhttp.ServerLogConfig{}
 | 
						|
	}
 | 
						|
 | 
						|
	httpApp := caddyhttp.App{
 | 
						|
		Servers: map[string]*caddyhttp.Server{"static": server},
 | 
						|
	}
 | 
						|
 | 
						|
	var false bool
 | 
						|
	cfg := &caddy.Config{
 | 
						|
		Admin: &caddy.AdminConfig{
 | 
						|
			Disabled: true,
 | 
						|
			Config: &caddy.ConfigSettings{
 | 
						|
				Persist: &false,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		AppsRaw: caddy.ModuleMap{
 | 
						|
			"http": caddyconfig.JSON(httpApp, nil),
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	if debug {
 | 
						|
		cfg.Logging = &caddy.Logging{
 | 
						|
			Logs: map[string]*caddy.CustomLog{
 | 
						|
				"default": {
 | 
						|
					BaseLog: caddy.BaseLog{Level: zap.DebugLevel.CapitalString()},
 | 
						|
				},
 | 
						|
			},
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	err = caddy.Run(cfg)
 | 
						|
	if err != nil {
 | 
						|
		return caddy.ExitCodeFailedStartup, err
 | 
						|
	}
 | 
						|
 | 
						|
	log.Printf("Caddy serving static files on %s", listen)
 | 
						|
 | 
						|
	select {}
 | 
						|
}
 |