mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-26 08:12:43 -04: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 {}
 | |
| }
 |