2022-11-03 13:54:49 +00:00
|
|
|
// Package xmain provides a standard stub for the main of a command handling logging,
|
|
|
|
|
// flags, signals and shutdown.
|
|
|
|
|
package xmain
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"errors"
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"os"
|
|
|
|
|
"os/signal"
|
|
|
|
|
"syscall"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"cdr.dev/slog"
|
|
|
|
|
"cdr.dev/slog/sloggers/sloghuman"
|
|
|
|
|
|
|
|
|
|
"oss.terrastruct.com/xos"
|
|
|
|
|
|
|
|
|
|
"oss.terrastruct.com/cmdlog"
|
|
|
|
|
|
|
|
|
|
ctxlog "oss.terrastruct.com/d2/lib/log"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type RunFunc func(context.Context, *State) error
|
|
|
|
|
|
|
|
|
|
func Main(run RunFunc) {
|
|
|
|
|
name := ""
|
|
|
|
|
args := []string(nil)
|
|
|
|
|
if len(os.Args) > 0 {
|
|
|
|
|
name = os.Args[0]
|
|
|
|
|
args = os.Args[1:]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ms := &State{
|
|
|
|
|
Name: name,
|
|
|
|
|
|
|
|
|
|
Stdin: os.Stdin,
|
|
|
|
|
Stdout: os.Stdout,
|
|
|
|
|
Stderr: os.Stderr,
|
|
|
|
|
|
2022-11-15 00:35:22 +00:00
|
|
|
Env: xos.NewEnv(os.Environ()),
|
2022-11-03 13:54:49 +00:00
|
|
|
}
|
|
|
|
|
ms.Log = cmdlog.Log(ms.Env, os.Stderr)
|
2022-11-15 00:35:22 +00:00
|
|
|
ms.Opts = NewOpts(ms.Env, args, ms.Log)
|
2022-11-03 13:54:49 +00:00
|
|
|
|
|
|
|
|
sigs := make(chan os.Signal, 1)
|
|
|
|
|
signal.Notify(sigs, os.Interrupt, syscall.SIGTERM)
|
|
|
|
|
|
|
|
|
|
err := ms.Main(context.Background(), sigs, run)
|
|
|
|
|
if err != nil {
|
|
|
|
|
code := 1
|
|
|
|
|
msg := ""
|
|
|
|
|
usage := false
|
|
|
|
|
|
|
|
|
|
var eerr ExitError
|
|
|
|
|
var uerr UsageError
|
|
|
|
|
if errors.As(err, &eerr) {
|
|
|
|
|
code = eerr.Code
|
|
|
|
|
msg = eerr.Message
|
|
|
|
|
} else if errors.As(err, &uerr) {
|
|
|
|
|
msg = err.Error()
|
|
|
|
|
usage = true
|
|
|
|
|
} else {
|
|
|
|
|
msg = err.Error()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if msg != "" {
|
|
|
|
|
if usage {
|
|
|
|
|
msg = fmt.Sprintf("%s\n%s", msg, "Run with --help to see usage.")
|
|
|
|
|
}
|
|
|
|
|
ms.Log.Error.Print(msg)
|
|
|
|
|
}
|
|
|
|
|
os.Exit(code)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type State struct {
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
Stdin io.Reader
|
|
|
|
|
Stdout io.WriteCloser
|
|
|
|
|
Stderr io.WriteCloser
|
|
|
|
|
|
2022-11-15 00:35:22 +00:00
|
|
|
Log *cmdlog.Logger
|
|
|
|
|
Env *xos.Env
|
|
|
|
|
Opts *Opts
|
2022-11-03 13:54:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ms *State) Main(ctx context.Context, sigs <-chan os.Signal, run func(context.Context, *State) error) error {
|
|
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
|
|
done := make(chan error, 1)
|
|
|
|
|
go func() {
|
|
|
|
|
defer close(done)
|
|
|
|
|
done <- run(ctx, ms)
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
select {
|
|
|
|
|
case err := <-done:
|
|
|
|
|
return err
|
|
|
|
|
case sig := <-sigs:
|
|
|
|
|
ms.Log.Warn.Printf("received signal %v: shutting down...", sig)
|
|
|
|
|
cancel()
|
|
|
|
|
select {
|
|
|
|
|
case err := <-done:
|
|
|
|
|
if err != nil && !errors.Is(err, context.Canceled) {
|
|
|
|
|
return fmt.Errorf("failed to shutdown: %w", err)
|
|
|
|
|
}
|
|
|
|
|
if sig == syscall.SIGTERM {
|
|
|
|
|
// We successfully shutdown.
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
return ExitError{Code: 1}
|
|
|
|
|
case <-time.After(time.Minute):
|
|
|
|
|
return ExitError{
|
|
|
|
|
Code: 1,
|
|
|
|
|
Message: "took longer than 1 minute to shutdown: exiting forcefully",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type ExitError struct {
|
|
|
|
|
Code int `json:"code"`
|
|
|
|
|
Message string `json:"message"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func ExitErrorf(code int, msg string, v ...interface{}) ExitError {
|
|
|
|
|
return ExitError{
|
|
|
|
|
Code: code,
|
|
|
|
|
Message: fmt.Sprintf(msg, v...),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ee ExitError) Error() string {
|
|
|
|
|
s := fmt.Sprintf("exiting with code %d", ee.Code)
|
|
|
|
|
if ee.Message != "" {
|
|
|
|
|
s += ": " + ee.Message
|
|
|
|
|
}
|
|
|
|
|
return s
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type UsageError struct {
|
|
|
|
|
Message string `json:"message"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func UsageErrorf(msg string, v ...interface{}) UsageError {
|
|
|
|
|
return UsageError{
|
|
|
|
|
Message: fmt.Sprintf(msg, v...),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ue UsageError) Error() string {
|
|
|
|
|
return fmt.Sprintf("bad usage: %s", ue.Message)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ms *State) ReadPath(fp string) ([]byte, error) {
|
|
|
|
|
if fp == "-" {
|
|
|
|
|
return io.ReadAll(ms.Stdin)
|
|
|
|
|
}
|
|
|
|
|
return os.ReadFile(fp)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ms *State) WritePath(fp string, p []byte) error {
|
|
|
|
|
if fp == "-" {
|
|
|
|
|
_, err := ms.Stdout.Write(p)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return ms.Stdout.Close()
|
|
|
|
|
}
|
|
|
|
|
return os.WriteFile(fp, p, 0644)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: remove after removing slog
|
|
|
|
|
func DiscardSlog(ctx context.Context) context.Context {
|
|
|
|
|
return ctxlog.With(ctx, slog.Make(sloghuman.Sink(io.Discard)))
|
|
|
|
|
}
|