2022-08-31 00:38:38 +02:00
|
|
|
package caddycmd
|
|
|
|
|
|
|
|
import (
|
2023-10-11 17:46:18 +02:00
|
|
|
"fmt"
|
|
|
|
|
2022-08-31 00:38:38 +02:00
|
|
|
"github.com/spf13/cobra"
|
2023-10-11 17:46:18 +02:00
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2"
|
2022-08-31 00:38:38 +02:00
|
|
|
)
|
|
|
|
|
2024-08-22 05:29:42 +02:00
|
|
|
var defaultFactory = newRootCommandFactory(func() *cobra.Command {
|
|
|
|
return &cobra.Command{
|
|
|
|
Use: "caddy",
|
|
|
|
Long: `Caddy is an extensible server platform written in Go.
|
2022-08-31 03:15:52 +02:00
|
|
|
|
|
|
|
At its core, Caddy merely manages configuration. Modules are plugged
|
|
|
|
in statically at compile-time to provide useful functionality. Caddy's
|
|
|
|
standard distribution includes common modules to serve HTTP, TLS,
|
|
|
|
and PKI applications, including the automation of certificates.
|
|
|
|
|
|
|
|
To run Caddy, use:
|
|
|
|
|
|
|
|
- 'caddy run' to run Caddy in the foreground (recommended).
|
|
|
|
- 'caddy start' to start Caddy in the background; only do this
|
|
|
|
if you will be keeping the terminal window open until you run
|
|
|
|
'caddy stop' to close the server.
|
|
|
|
|
|
|
|
When Caddy is started, it opens a locally-bound administrative socket
|
|
|
|
to which configuration can be POSTed via a restful HTTP API (see
|
|
|
|
https://caddyserver.com/docs/api).
|
|
|
|
|
|
|
|
Caddy's native configuration format is JSON. However, config adapters
|
|
|
|
can be used to convert other config formats to JSON when Caddy receives
|
|
|
|
its configuration. The Caddyfile is a built-in config adapter that is
|
|
|
|
popular for hand-written configurations due to its straightforward
|
|
|
|
syntax (see https://caddyserver.com/docs/caddyfile). Many third-party
|
|
|
|
adapters are available (see https://caddyserver.com/docs/config-adapters).
|
|
|
|
Use 'caddy adapt' to see how a config translates to JSON.
|
|
|
|
|
|
|
|
For convenience, the CLI can act as an HTTP client to give Caddy its
|
|
|
|
initial configuration for you. If a file named Caddyfile is in the
|
|
|
|
current working directory, it will do this automatically. Otherwise,
|
|
|
|
you can use the --config flag to specify the path to a config file.
|
|
|
|
|
|
|
|
Some special-purpose subcommands build and load a configuration file
|
|
|
|
for you directly from command line input; for example:
|
|
|
|
|
|
|
|
- caddy file-server
|
|
|
|
- caddy reverse-proxy
|
|
|
|
- caddy respond
|
|
|
|
|
|
|
|
These commands disable the administration endpoint because their
|
|
|
|
configuration is specified solely on the command line.
|
|
|
|
|
|
|
|
In general, the most common way to run Caddy is simply:
|
|
|
|
|
|
|
|
$ caddy run
|
|
|
|
|
|
|
|
Or, with a configuration file:
|
|
|
|
|
|
|
|
$ caddy run --config caddy.json
|
|
|
|
|
|
|
|
If running interactively in a terminal, running Caddy in the
|
|
|
|
background may be more convenient:
|
|
|
|
|
|
|
|
$ caddy start
|
|
|
|
...
|
|
|
|
$ caddy stop
|
|
|
|
|
|
|
|
This allows you to run other commands while Caddy stays running.
|
|
|
|
Be sure to stop Caddy before you close the terminal!
|
|
|
|
|
|
|
|
Depending on the system, Caddy may need permission to bind to low
|
|
|
|
ports. One way to do this on Linux is to use setcap:
|
|
|
|
|
|
|
|
$ sudo setcap cap_net_bind_service=+ep $(which caddy)
|
|
|
|
|
|
|
|
Remember to run that command again after replacing the binary.
|
|
|
|
|
|
|
|
See the Caddy website for tutorials, configuration structure,
|
|
|
|
syntax, and module documentation: https://caddyserver.com/docs/
|
|
|
|
|
|
|
|
Custom Caddy builds are available on the Caddy download page at:
|
|
|
|
https://caddyserver.com/download
|
|
|
|
|
|
|
|
The xcaddy command can be used to build Caddy from source with or
|
|
|
|
without additional plugins: https://github.com/caddyserver/xcaddy
|
|
|
|
|
|
|
|
Where possible, Caddy should be installed using officially-supported
|
|
|
|
package installers: https://caddyserver.com/docs/install
|
|
|
|
|
|
|
|
Instructions for running Caddy in production are also available:
|
|
|
|
https://caddyserver.com/docs/running
|
|
|
|
`,
|
2024-08-22 05:29:42 +02:00
|
|
|
Example: ` $ caddy run
|
2022-08-31 03:15:52 +02:00
|
|
|
$ caddy run --config caddy.json
|
|
|
|
$ caddy reload --config caddy.json
|
|
|
|
$ caddy stop`,
|
2022-09-02 05:41:09 +02:00
|
|
|
|
2024-08-22 05:29:42 +02:00
|
|
|
// kind of annoying to have all the help text printed out if
|
|
|
|
// caddy has an error provisioning its modules, for instance...
|
|
|
|
SilenceUsage: true,
|
|
|
|
Version: onlyVersionText(),
|
|
|
|
}
|
|
|
|
})
|
2022-08-31 00:38:38 +02:00
|
|
|
|
|
|
|
const fullDocsFooter = `Full documentation is available at:
|
2022-08-31 03:15:52 +02:00
|
|
|
https://caddyserver.com/docs/command-line`
|
2022-08-31 00:38:38 +02:00
|
|
|
|
|
|
|
func init() {
|
2024-08-22 05:29:42 +02:00
|
|
|
defaultFactory.Use(func(rootCmd *cobra.Command) {
|
|
|
|
rootCmd.SetVersionTemplate("{{.Version}}\n")
|
|
|
|
rootCmd.SetHelpTemplate(rootCmd.HelpTemplate() + "\n" + fullDocsFooter + "\n")
|
|
|
|
})
|
2022-08-31 00:38:38 +02:00
|
|
|
}
|
|
|
|
|
2023-10-11 17:46:18 +02:00
|
|
|
func onlyVersionText() string {
|
|
|
|
_, f := caddy.Version()
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
2023-01-21 16:27:58 +01:00
|
|
|
func caddyCmdToCobra(caddyCmd Command) *cobra.Command {
|
2022-08-31 00:38:38 +02:00
|
|
|
cmd := &cobra.Command{
|
2024-03-01 17:00:29 +01:00
|
|
|
Use: caddyCmd.Name + " " + caddyCmd.Usage,
|
2022-08-31 00:38:38 +02:00
|
|
|
Short: caddyCmd.Short,
|
|
|
|
Long: caddyCmd.Long,
|
|
|
|
}
|
2023-02-25 00:09:12 +01:00
|
|
|
if caddyCmd.CobraFunc != nil {
|
|
|
|
caddyCmd.CobraFunc(cmd)
|
|
|
|
} else {
|
|
|
|
cmd.RunE = WrapCommandFuncForCobra(caddyCmd.Func)
|
|
|
|
cmd.Flags().AddGoFlagSet(caddyCmd.Flags)
|
|
|
|
}
|
2022-08-31 00:38:38 +02:00
|
|
|
return cmd
|
|
|
|
}
|
2023-02-25 00:09:12 +01:00
|
|
|
|
|
|
|
// WrapCommandFuncForCobra wraps a Caddy CommandFunc for use
|
|
|
|
// in a cobra command's RunE field.
|
|
|
|
func WrapCommandFuncForCobra(f CommandFunc) func(cmd *cobra.Command, _ []string) error {
|
|
|
|
return func(cmd *cobra.Command, _ []string) error {
|
2023-10-11 17:46:18 +02:00
|
|
|
status, err := f(Flags{cmd.Flags()})
|
|
|
|
if status > 1 {
|
|
|
|
cmd.SilenceErrors = true
|
|
|
|
return &exitError{ExitCode: status, Err: err}
|
|
|
|
}
|
2023-02-25 00:09:12 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2023-10-11 17:46:18 +02:00
|
|
|
|
|
|
|
// exitError carries the exit code from CommandFunc to Main()
|
|
|
|
type exitError struct {
|
|
|
|
ExitCode int
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *exitError) Error() string {
|
|
|
|
if e.Err == nil {
|
|
|
|
return fmt.Sprintf("exiting with status %d", e.ExitCode)
|
|
|
|
}
|
|
|
|
return e.Err.Error()
|
|
|
|
}
|