2019-10-03 17:00:41 -05:00
|
|
|
// 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"
|
|
|
|
"flag"
|
|
|
|
"log"
|
2019-11-16 12:44:45 -05:00
|
|
|
"strconv"
|
2019-12-31 18:57:54 -05:00
|
|
|
"time"
|
2019-10-03 17:00:41 -05:00
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2"
|
|
|
|
"github.com/caddyserver/caddy/v2/caddyconfig"
|
|
|
|
caddycmd "github.com/caddyserver/caddy/v2/cmd"
|
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
2020-03-15 00:31:52 -05:00
|
|
|
caddytpl "github.com/caddyserver/caddy/v2/modules/caddyhttp/templates"
|
2020-03-07 01:15:25 -05:00
|
|
|
"github.com/caddyserver/certmagic"
|
2022-09-16 00:10:16 -05:00
|
|
|
"go.uber.org/zap"
|
2019-10-03 17:00:41 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
caddycmd.RegisterCommand(caddycmd.Command{
|
|
|
|
Name: "file-server",
|
|
|
|
Func: cmdFileServer,
|
2020-05-26 16:04:04 -05:00
|
|
|
Usage: "[--domain <example.com>] [--root <path>] [--listen <addr>] [--browse] [--access-log]",
|
2019-10-03 17:00:41 -05:00
|
|
|
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.
|
|
|
|
|
2020-03-13 14:02:47 -05:00
|
|
|
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.
|
2019-11-16 12:44:45 -05:00
|
|
|
|
2019-10-03 17:00:41 -05:00
|
|
|
If --browse is enabled, requests for folders without an index file will
|
|
|
|
respond with a file listing.`,
|
|
|
|
Flags: func() *flag.FlagSet {
|
|
|
|
fs := flag.NewFlagSet("file-server", flag.ExitOnError)
|
|
|
|
fs.String("domain", "", "Domain name at which to serve the files")
|
|
|
|
fs.String("root", "", "The path to the root of the site")
|
|
|
|
fs.String("listen", "", "The address to which to bind the listener")
|
2020-03-15 00:31:52 -05:00
|
|
|
fs.Bool("browse", false, "Enable directory browsing")
|
|
|
|
fs.Bool("templates", false, "Enable template rendering")
|
2020-05-26 16:04:04 -05:00
|
|
|
fs.Bool("access-log", false, "Enable the access log")
|
2022-09-20 17:56:02 -05:00
|
|
|
fs.Bool("debug", false, "Enable verbose debug logs")
|
2019-10-03 17:00:41 -05:00
|
|
|
return fs
|
|
|
|
}(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func cmdFileServer(fs caddycmd.Flags) (int, error) {
|
2020-05-21 14:09:49 -05:00
|
|
|
caddy.TrapSignals()
|
|
|
|
|
2019-10-03 17:00:41 -05:00
|
|
|
domain := fs.String("domain")
|
|
|
|
root := fs.String("root")
|
|
|
|
listen := fs.String("listen")
|
|
|
|
browse := fs.Bool("browse")
|
2020-03-15 00:31:52 -05:00
|
|
|
templates := fs.Bool("templates")
|
2020-05-26 16:04:04 -05:00
|
|
|
accessLog := fs.Bool("access-log")
|
2022-09-16 00:10:16 -05:00
|
|
|
debug := fs.Bool("debug")
|
2020-03-15 00:31:52 -05:00
|
|
|
|
|
|
|
var handlers []json.RawMessage
|
|
|
|
|
|
|
|
if templates {
|
|
|
|
handler := caddytpl.Templates{FileRoot: root}
|
|
|
|
handlers = append(handlers, caddyconfig.JSONModuleObject(handler, "handler", "templates", nil))
|
|
|
|
}
|
2019-10-03 17:00:41 -05:00
|
|
|
|
|
|
|
handler := FileServer{Root: root}
|
|
|
|
if browse {
|
|
|
|
handler.Browse = new(Browse)
|
|
|
|
}
|
|
|
|
|
2020-03-15 00:31:52 -05:00
|
|
|
handlers = append(handlers, caddyconfig.JSONModuleObject(handler, "handler", "file_server", nil))
|
|
|
|
|
|
|
|
route := caddyhttp.Route{HandlersRaw: handlers}
|
|
|
|
|
2019-10-03 17:00:41 -05:00
|
|
|
if domain != "" {
|
2019-12-10 15:36:46 -05:00
|
|
|
route.MatcherSetsRaw = []caddy.ModuleMap{
|
2020-04-10 09:12:42 -05:00
|
|
|
{
|
2019-10-03 17:00:41 -05:00
|
|
|
"host": caddyconfig.JSON(caddyhttp.MatchHost{domain}, nil),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
server := &caddyhttp.Server{
|
2019-12-31 18:57:54 -05:00
|
|
|
ReadHeaderTimeout: caddy.Duration(10 * time.Second),
|
|
|
|
IdleTimeout: caddy.Duration(30 * time.Second),
|
|
|
|
MaxHeaderBytes: 1024 * 10,
|
|
|
|
Routes: caddyhttp.RouteList{route},
|
2019-10-03 17:00:41 -05:00
|
|
|
}
|
2019-10-04 18:30:51 -05:00
|
|
|
if listen == "" {
|
2020-03-13 14:02:47 -05:00
|
|
|
if domain == "" {
|
|
|
|
listen = ":80"
|
|
|
|
} else {
|
|
|
|
listen = ":" + strconv.Itoa(certmagic.HTTPSPort)
|
|
|
|
}
|
2019-10-03 17:00:41 -05:00
|
|
|
}
|
2019-10-04 18:30:51 -05:00
|
|
|
server.Listen = []string{listen}
|
2020-05-26 16:04:04 -05:00
|
|
|
if accessLog {
|
|
|
|
server.Logs = &caddyhttp.ServerLogConfig{}
|
|
|
|
}
|
2019-10-03 17:00:41 -05:00
|
|
|
|
|
|
|
httpApp := caddyhttp.App{
|
|
|
|
Servers: map[string]*caddyhttp.Server{"static": server},
|
|
|
|
}
|
|
|
|
|
2022-08-01 14:36:22 -05:00
|
|
|
var false bool
|
2019-10-03 17:00:41 -05:00
|
|
|
cfg := &caddy.Config{
|
2022-08-01 14:36:22 -05:00
|
|
|
Admin: &caddy.AdminConfig{
|
|
|
|
Disabled: true,
|
|
|
|
Config: &caddy.ConfigSettings{
|
|
|
|
Persist: &false,
|
|
|
|
},
|
|
|
|
},
|
2019-12-10 15:36:46 -05:00
|
|
|
AppsRaw: caddy.ModuleMap{
|
2019-10-03 17:00:41 -05:00
|
|
|
"http": caddyconfig.JSON(httpApp, nil),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-09-16 00:10:16 -05:00
|
|
|
if debug {
|
|
|
|
cfg.Logging = &caddy.Logging{
|
|
|
|
Logs: map[string]*caddy.CustomLog{
|
|
|
|
"default": {Level: zap.DebugLevel.CapitalString()},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-03 17:00:41 -05:00
|
|
|
err := caddy.Run(cfg)
|
|
|
|
if err != nil {
|
|
|
|
return caddy.ExitCodeFailedStartup, err
|
|
|
|
}
|
|
|
|
|
2022-04-25 11:12:10 -05:00
|
|
|
log.Printf("Caddy serving static files on %s", listen)
|
2019-10-03 17:00:41 -05:00
|
|
|
|
|
|
|
select {}
|
|
|
|
}
|