mirror of
https://github.com/caddyserver/xcaddy.git
synced 2024-11-03 07:50:30 +00:00
2887af6a01
* feat: allow fs embedding with `--embed` * code review * remove utils/io.go * don't shadow `err` * add syntax and examples to README
465 lines
12 KiB
Go
465 lines
12 KiB
Go
// Copyright 2020 Matthew Holt
|
|
//
|
|
// 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 xcaddycmd
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"os/signal"
|
|
"path"
|
|
"path/filepath"
|
|
"runtime"
|
|
"runtime/debug"
|
|
"strings"
|
|
|
|
"github.com/caddyserver/xcaddy"
|
|
"github.com/caddyserver/xcaddy/internal/utils"
|
|
)
|
|
|
|
var (
|
|
caddyVersion = os.Getenv("CADDY_VERSION")
|
|
raceDetector = os.Getenv("XCADDY_RACE_DETECTOR") == "1"
|
|
skipBuild = os.Getenv("XCADDY_SKIP_BUILD") == "1"
|
|
skipCleanup = os.Getenv("XCADDY_SKIP_CLEANUP") == "1" || skipBuild
|
|
buildDebugOutput = os.Getenv("XCADDY_DEBUG") == "1"
|
|
buildFlags = os.Getenv("XCADDY_GO_BUILD_FLAGS")
|
|
modFlags = os.Getenv("XCADDY_GO_MOD_FLAGS")
|
|
)
|
|
|
|
func Main() {
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
go trapSignals(ctx, cancel)
|
|
|
|
if len(os.Args) > 1 && os.Args[1] == "build" {
|
|
if err := runBuild(ctx, os.Args[2:]); err != nil {
|
|
log.Fatalf("[ERROR] %v", err)
|
|
}
|
|
return
|
|
}
|
|
|
|
if len(os.Args) > 1 && os.Args[1] == "version" {
|
|
fmt.Println(xcaddyVersion())
|
|
return
|
|
}
|
|
|
|
if err := runDev(ctx, os.Args[1:]); err != nil {
|
|
log.Fatalf("[ERROR] %v", err)
|
|
}
|
|
}
|
|
|
|
func runBuild(ctx context.Context, args []string) error {
|
|
// parse the command line args... rather primitively
|
|
var argCaddyVersion, output string
|
|
var plugins []xcaddy.Dependency
|
|
var replacements []xcaddy.Replace
|
|
var embedDir []string
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--with":
|
|
if i == len(args)-1 {
|
|
return fmt.Errorf("expected value after --with flag")
|
|
}
|
|
i++
|
|
mod, ver, repl, err := splitWith(args[i])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
mod = strings.TrimSuffix(mod, "/") // easy to accidentally leave a trailing slash if pasting from a URL, but is invalid for Go modules
|
|
plugins = append(plugins, xcaddy.Dependency{
|
|
PackagePath: mod,
|
|
Version: ver,
|
|
})
|
|
if repl != "" {
|
|
// adjust relative replacements in current working directory since our temporary module is in a different directory
|
|
if strings.HasPrefix(repl, ".") {
|
|
repl, err = filepath.Abs(repl)
|
|
if err != nil {
|
|
log.Fatalf("[FATAL] %v", err)
|
|
}
|
|
log.Printf("[INFO] Resolved relative replacement %s to %s", args[i], repl)
|
|
}
|
|
replacements = append(replacements, xcaddy.NewReplace(mod, repl))
|
|
}
|
|
|
|
case "--output":
|
|
if i == len(args)-1 {
|
|
return fmt.Errorf("expected value after --output flag")
|
|
}
|
|
i++
|
|
output = args[i]
|
|
case "--embed":
|
|
if i == len(args)-1 {
|
|
return fmt.Errorf("expected value after --embed flag")
|
|
}
|
|
i++
|
|
embedDir = append(embedDir, args[i])
|
|
default:
|
|
if argCaddyVersion != "" {
|
|
return fmt.Errorf("missing flag; caddy version already set at %s", argCaddyVersion)
|
|
}
|
|
argCaddyVersion = args[i]
|
|
}
|
|
}
|
|
|
|
// prefer caddy version from command line argument over env var
|
|
if argCaddyVersion != "" {
|
|
caddyVersion = argCaddyVersion
|
|
}
|
|
|
|
// ensure an output file is always specified
|
|
if output == "" {
|
|
output = getCaddyOutputFile()
|
|
}
|
|
|
|
// perform the build
|
|
builder := xcaddy.Builder{
|
|
Compile: xcaddy.Compile{
|
|
Cgo: os.Getenv("CGO_ENABLED") == "1",
|
|
},
|
|
CaddyVersion: caddyVersion,
|
|
Plugins: plugins,
|
|
Replacements: replacements,
|
|
RaceDetector: raceDetector,
|
|
SkipBuild: skipBuild,
|
|
SkipCleanup: skipCleanup,
|
|
Debug: buildDebugOutput,
|
|
BuildFlags: buildFlags,
|
|
ModFlags: modFlags,
|
|
}
|
|
for _, md := range embedDir {
|
|
if before, after, found := strings.Cut(md, ":"); found {
|
|
builder.EmbedDirs = append(builder.EmbedDirs, struct {
|
|
Dir string `json:"dir,omitempty"`
|
|
Name string `json:"name,omitempty"`
|
|
}{
|
|
after, before,
|
|
})
|
|
} else {
|
|
builder.EmbedDirs = append(builder.EmbedDirs, struct {
|
|
Dir string `json:"dir,omitempty"`
|
|
Name string `json:"name,omitempty"`
|
|
}{
|
|
before, "",
|
|
})
|
|
}
|
|
}
|
|
err := builder.Build(ctx, output)
|
|
if err != nil {
|
|
log.Fatalf("[FATAL] %v", err)
|
|
}
|
|
|
|
// if requested, run setcap to allow binding to low ports
|
|
err = setcapIfRequested(output)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// prove the build is working by printing the version
|
|
if runtime.GOOS == os.Getenv("GOOS") && runtime.GOARCH == os.Getenv("GOARCH") {
|
|
if !filepath.IsAbs(output) {
|
|
output = "." + string(filepath.Separator) + output
|
|
}
|
|
fmt.Println()
|
|
fmt.Printf("%s version\n", output)
|
|
cmd := exec.Command(output, "version")
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
err = cmd.Run()
|
|
if err != nil {
|
|
log.Fatalf("[FATAL] %v", err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getCaddyOutputFile() string {
|
|
f := "." + string(filepath.Separator) + "caddy"
|
|
if runtime.GOOS == "windows" {
|
|
f += ".exe"
|
|
}
|
|
return f
|
|
}
|
|
|
|
func runDev(ctx context.Context, args []string) error {
|
|
binOutput := getCaddyOutputFile()
|
|
|
|
// get current/main module name and the root directory of the main module
|
|
//
|
|
// make sure the module being developed is replaced
|
|
// so that the local copy is used
|
|
//
|
|
// replace directives only apply to the top-level/main go.mod,
|
|
// and since this tool is a carry-through for the user's actual
|
|
// go.mod, we need to transfer their replace directives through
|
|
// to the one we're making
|
|
cmd := exec.Command(utils.GetGo(), "list", "-mod=readonly", "-m", "-json", "all")
|
|
cmd.Stderr = os.Stderr
|
|
out, err := cmd.Output()
|
|
if err != nil {
|
|
return fmt.Errorf("exec %v: %v: %s", cmd.Args, err, string(out))
|
|
}
|
|
currentModule, moduleDir, replacements, err := parseGoListJson(out)
|
|
if err != nil {
|
|
return fmt.Errorf("json parse error: %v", err)
|
|
}
|
|
|
|
// reconcile remaining path segments; for example if a module foo/a
|
|
// is rooted at directory path /home/foo/a, but the current directory
|
|
// is /home/foo/a/b, then the package to import should be foo/a/b
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
return fmt.Errorf("unable to determine current directory: %v", err)
|
|
}
|
|
importPath := normalizeImportPath(currentModule, cwd, moduleDir)
|
|
|
|
// build caddy with this module plugged in
|
|
builder := xcaddy.Builder{
|
|
Compile: xcaddy.Compile{
|
|
Cgo: os.Getenv("CGO_ENABLED") == "1",
|
|
},
|
|
CaddyVersion: caddyVersion,
|
|
Plugins: []xcaddy.Dependency{
|
|
{PackagePath: importPath},
|
|
},
|
|
Replacements: replacements,
|
|
RaceDetector: raceDetector,
|
|
SkipBuild: skipBuild,
|
|
SkipCleanup: skipCleanup,
|
|
Debug: buildDebugOutput,
|
|
}
|
|
err = builder.Build(ctx, binOutput)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// if requested, run setcap to allow binding to low ports
|
|
err = setcapIfRequested(binOutput)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
log.Printf("[INFO] Running %v\n\n", append([]string{binOutput}, args...))
|
|
|
|
cmd = exec.Command(binOutput, args...)
|
|
cmd.Stdin = os.Stdin
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
err = cmd.Start()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if skipCleanup {
|
|
log.Printf("[INFO] Skipping cleanup as requested; leaving artifact: %s", binOutput)
|
|
return
|
|
}
|
|
err = os.Remove(binOutput)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
log.Printf("[ERROR] Deleting temporary binary %s: %v", binOutput, err)
|
|
}
|
|
}()
|
|
|
|
return cmd.Wait()
|
|
}
|
|
|
|
func setcapIfRequested(output string) error {
|
|
if os.Getenv("XCADDY_SETCAP") != "1" {
|
|
return nil
|
|
}
|
|
|
|
args := []string{"setcap", "cap_net_bind_service=+ep", output}
|
|
|
|
// check if sudo isn't available, or we were instructed not to use it
|
|
_, sudoNotFound := exec.LookPath("sudo")
|
|
skipSudo := sudoNotFound != nil || os.Getenv("XCADDY_SUDO") == "0"
|
|
|
|
var cmd *exec.Cmd
|
|
if skipSudo {
|
|
cmd = exec.Command(args[0], args[1:]...)
|
|
} else {
|
|
cmd = exec.Command("sudo", args...)
|
|
}
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
|
|
log.Printf("[INFO] Setting capabilities (requires admin privileges): %v", cmd.Args)
|
|
if err := cmd.Run(); err != nil {
|
|
return fmt.Errorf("failed to setcap on the binary: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type module struct {
|
|
Path string // module path
|
|
Version string // module version
|
|
Replace *module // replaced by this module
|
|
Main bool // is this the main module?
|
|
Dir string // directory holding files for this module, if any
|
|
}
|
|
|
|
func parseGoListJson(out []byte) (currentModule, moduleDir string, replacements []xcaddy.Replace, err error) {
|
|
var unjoinedReplaces []int
|
|
|
|
decoder := json.NewDecoder(bytes.NewReader(out))
|
|
for {
|
|
var mod module
|
|
if err = decoder.Decode(&mod); err == io.EOF {
|
|
err = nil
|
|
break
|
|
} else if err != nil {
|
|
return
|
|
}
|
|
|
|
if mod.Main {
|
|
// Current module is main module, retrieve the main module name and
|
|
// root directory path of the main module
|
|
currentModule = mod.Path
|
|
moduleDir = mod.Dir
|
|
replacements = append(replacements, xcaddy.NewReplace(currentModule, moduleDir))
|
|
continue
|
|
}
|
|
|
|
// Skip if current module is not replacement
|
|
if mod.Replace == nil {
|
|
continue
|
|
}
|
|
|
|
// 1. Target is module, version is required in this case
|
|
// 2A. Target is absolute path
|
|
// 2B. Target is relative path, proper handling is required in this case
|
|
dstPath := mod.Replace.Path
|
|
dstVersion := mod.Replace.Version
|
|
var dst string
|
|
if dstVersion != "" {
|
|
dst = dstPath + "@" + dstVersion
|
|
} else if filepath.IsAbs(dstPath) {
|
|
dst = dstPath
|
|
} else {
|
|
if moduleDir != "" {
|
|
dst = filepath.Join(moduleDir, dstPath)
|
|
log.Printf("[INFO] Resolved relative replacement %s to %s", dstPath, dst)
|
|
} else {
|
|
// moduleDir is not parsed yet, defer to later
|
|
dst = dstPath
|
|
unjoinedReplaces = append(unjoinedReplaces, len(replacements))
|
|
}
|
|
}
|
|
|
|
replacements = append(replacements, xcaddy.NewReplace(mod.Path, dst))
|
|
}
|
|
for _, idx := range unjoinedReplaces {
|
|
unresolved := string(replacements[idx].New)
|
|
resolved := filepath.Join(moduleDir, unresolved)
|
|
log.Printf("[INFO] Resolved previously-unjoined relative replacement %s to %s", unresolved, resolved)
|
|
replacements[idx].New = xcaddy.ReplacementPath(resolved)
|
|
}
|
|
return
|
|
}
|
|
|
|
func normalizeImportPath(currentModule, cwd, moduleDir string) string {
|
|
return path.Join(currentModule, filepath.ToSlash(strings.TrimPrefix(cwd, moduleDir)))
|
|
}
|
|
|
|
func trapSignals(ctx context.Context, cancel context.CancelFunc) {
|
|
sig := make(chan os.Signal, 1)
|
|
signal.Notify(sig, os.Interrupt)
|
|
|
|
select {
|
|
case <-sig:
|
|
log.Printf("[INFO] SIGINT: Shutting down")
|
|
cancel()
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
|
|
func splitWith(arg string) (module, version, replace string, err error) {
|
|
const versionSplit, replaceSplit = "@", "="
|
|
|
|
parts := strings.SplitN(arg, replaceSplit, 2)
|
|
if len(parts) > 1 {
|
|
replace = parts[1]
|
|
}
|
|
module = parts[0]
|
|
|
|
// accommodate module paths that have @ in them, but we can only tolerate that if there's also
|
|
// a version, otherwise we don't know if it's a version separator or part of the file path (see #109)
|
|
lastVersionSplit := strings.LastIndex(module, versionSplit)
|
|
if lastVersionSplit < 0 {
|
|
if replaceIdx := strings.Index(module, replaceSplit); replaceIdx >= 0 {
|
|
module, replace = module[:replaceIdx], module[replaceIdx+1:]
|
|
}
|
|
} else {
|
|
module, version = module[:lastVersionSplit], module[lastVersionSplit+1:]
|
|
if replaceIdx := strings.Index(version, replaceSplit); replaceIdx >= 0 {
|
|
version, replace = module[:replaceIdx], module[replaceIdx+1:]
|
|
}
|
|
}
|
|
|
|
if module == "" {
|
|
err = fmt.Errorf("module name is required")
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// xcaddyVersion returns a detailed version string, if available.
|
|
func xcaddyVersion() string {
|
|
mod := goModule()
|
|
ver := mod.Version
|
|
if mod.Sum != "" {
|
|
ver += " " + mod.Sum
|
|
}
|
|
if mod.Replace != nil {
|
|
ver += " => " + mod.Replace.Path
|
|
if mod.Replace.Version != "" {
|
|
ver += "@" + mod.Replace.Version
|
|
}
|
|
if mod.Replace.Sum != "" {
|
|
ver += " " + mod.Replace.Sum
|
|
}
|
|
}
|
|
return ver
|
|
}
|
|
|
|
func goModule() *debug.Module {
|
|
mod := &debug.Module{}
|
|
mod.Version = "unknown"
|
|
bi, ok := debug.ReadBuildInfo()
|
|
if ok {
|
|
mod.Path = bi.Main.Path
|
|
// The recommended way to build xcaddy involves
|
|
// creating a separate main module, which
|
|
// TODO: track related Go issue: https://github.com/golang/go/issues/29228
|
|
// once that issue is fixed, we should just be able to use bi.Main... hopefully.
|
|
for _, dep := range bi.Deps {
|
|
if dep.Path == "github.com/caddyserver/xcaddy" {
|
|
return dep
|
|
}
|
|
}
|
|
return &bi.Main
|
|
}
|
|
return mod
|
|
}
|