2020-11-14 12:14:03 +08:00
|
|
|
// Package prog provides the entry point to Elvish. Its subpackages correspond
|
|
|
|
// to subprograms of Elvish.
|
2020-04-25 06:33:18 +08:00
|
|
|
package prog
|
2017-12-05 03:55:49 +08:00
|
|
|
|
|
|
|
// This package sets up the basic environment and calls the appropriate
|
|
|
|
// "subprogram", one of the daemon, the terminal interface, or the web
|
|
|
|
// interface.
|
|
|
|
|
|
|
|
import (
|
2021-09-12 07:35:43 +08:00
|
|
|
"errors"
|
2017-12-05 03:55:49 +08:00
|
|
|
"flag"
|
|
|
|
"fmt"
|
2017-12-08 07:55:33 +08:00
|
|
|
"io"
|
2017-12-05 03:55:49 +08:00
|
|
|
"os"
|
|
|
|
|
2021-01-27 09:28:38 +08:00
|
|
|
"src.elv.sh/pkg/logutil"
|
2017-12-05 03:55:49 +08:00
|
|
|
)
|
|
|
|
|
2021-01-20 05:37:36 +08:00
|
|
|
// DeprecationLevel is a global flag that controls which deprecations to show.
|
|
|
|
// If its value is X, Elvish shows deprecations that should be shown for version
|
|
|
|
// 0.X.
|
2022-03-19 06:11:35 +08:00
|
|
|
var DeprecationLevel = 18
|
2020-04-27 04:24:30 +08:00
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
// Program represents a subprogram.
|
|
|
|
type Program interface {
|
|
|
|
RegisterFlags(fs *FlagSet)
|
|
|
|
// Run runs the subprogram.
|
|
|
|
Run(fds [3]*os.File, args []string) error
|
2020-04-25 03:30:04 +08:00
|
|
|
}
|
2017-12-08 07:55:33 +08:00
|
|
|
|
2021-09-12 07:35:43 +08:00
|
|
|
func usage(out io.Writer, fs *flag.FlagSet) {
|
2022-03-03 08:29:49 +08:00
|
|
|
fmt.Fprintln(out, "Usage: elvish [flags] [script] [args]")
|
2020-04-25 03:30:04 +08:00
|
|
|
fmt.Fprintln(out, "Supported flags:")
|
2021-09-12 07:35:43 +08:00
|
|
|
fs.SetOutput(out)
|
|
|
|
fs.PrintDefaults()
|
2017-12-08 07:55:33 +08:00
|
|
|
}
|
|
|
|
|
2020-04-25 06:43:39 +08:00
|
|
|
// Run parses command-line flags and runs the first applicable subprogram. It
|
|
|
|
// returns the exit status of the program.
|
2021-09-12 07:35:43 +08:00
|
|
|
func Run(fds [3]*os.File, args []string, p Program) int {
|
2022-02-05 22:48:55 +08:00
|
|
|
fs := flag.NewFlagSet("elvish", flag.ContinueOnError)
|
|
|
|
// Error and usage will be printed explicitly.
|
|
|
|
fs.SetOutput(io.Discard)
|
|
|
|
|
|
|
|
var log string
|
|
|
|
var help bool
|
2022-03-03 08:29:49 +08:00
|
|
|
fs.StringVar(&log, "log", "",
|
|
|
|
"Path to a file to write debug logs")
|
|
|
|
fs.BoolVar(&help, "help", false,
|
|
|
|
"Show usage help and quit")
|
|
|
|
fs.IntVar(&DeprecationLevel, "deprecation-level", DeprecationLevel,
|
|
|
|
"Show warnings for all features deprecated as of version 0.X")
|
2022-02-05 22:48:55 +08:00
|
|
|
|
|
|
|
p.RegisterFlags(&FlagSet{FlagSet: fs})
|
|
|
|
|
2020-04-25 03:30:04 +08:00
|
|
|
err := fs.Parse(args[1:])
|
2017-12-08 08:23:56 +08:00
|
|
|
if err != nil {
|
2021-07-09 08:16:19 +08:00
|
|
|
if err == flag.ErrHelp {
|
|
|
|
// (*flag.FlagSet).Parse returns ErrHelp when -h or -help was
|
|
|
|
// requested but *not* defined. Elvish defines -help, but not -h; so
|
|
|
|
// this means that -h has been requested. Handle this by printing
|
|
|
|
// the same message as an undefined flag.
|
|
|
|
fmt.Fprintln(fds[2], "flag provided but not defined: -h")
|
|
|
|
} else {
|
|
|
|
fmt.Fprintln(fds[2], err)
|
|
|
|
}
|
|
|
|
usage(fds[2], fs)
|
2017-12-08 08:23:56 +08:00
|
|
|
return 2
|
|
|
|
}
|
2017-12-07 07:27:34 +08:00
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
if log != "" {
|
|
|
|
err = logutil.SetOutputFile(log)
|
2020-12-06 10:17:19 +08:00
|
|
|
if err != nil {
|
|
|
|
fmt.Fprintln(fds[2], err)
|
|
|
|
}
|
2017-12-07 07:27:34 +08:00
|
|
|
}
|
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
if help {
|
2020-04-25 03:30:04 +08:00
|
|
|
usage(fds[1], fs)
|
|
|
|
return 0
|
|
|
|
}
|
2017-12-05 04:29:53 +08:00
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
err = p.Run(fds, fs.Args())
|
2020-04-25 03:30:04 +08:00
|
|
|
if err == nil {
|
|
|
|
return 0
|
|
|
|
}
|
2022-02-05 22:48:55 +08:00
|
|
|
if err == ErrNextProgram {
|
|
|
|
err = errNoSuitableSubprogram
|
|
|
|
}
|
2020-04-25 03:30:04 +08:00
|
|
|
if msg := err.Error(); msg != "" {
|
|
|
|
fmt.Fprintln(fds[2], msg)
|
|
|
|
}
|
|
|
|
switch err := err.(type) {
|
|
|
|
case badUsageError:
|
|
|
|
usage(fds[2], fs)
|
|
|
|
case exitError:
|
|
|
|
return err.exit
|
|
|
|
}
|
|
|
|
return 2
|
2017-12-07 07:02:15 +08:00
|
|
|
}
|
2017-12-05 03:55:49 +08:00
|
|
|
|
2021-09-12 07:35:43 +08:00
|
|
|
// Composite returns a Program that tries each of the given programs,
|
|
|
|
// terminating at the first one that doesn't return NotSuitable().
|
|
|
|
func Composite(programs ...Program) Program {
|
2022-02-05 22:48:55 +08:00
|
|
|
return composite(programs)
|
2021-09-12 07:35:43 +08:00
|
|
|
}
|
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
type composite []Program
|
|
|
|
|
|
|
|
func (cp composite) RegisterFlags(f *FlagSet) {
|
|
|
|
for _, p := range cp {
|
|
|
|
p.RegisterFlags(f)
|
|
|
|
}
|
|
|
|
}
|
2021-09-12 07:35:43 +08:00
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
func (cp composite) Run(fds [3]*os.File, args []string) error {
|
2021-09-12 07:35:43 +08:00
|
|
|
for _, p := range cp {
|
2022-02-05 22:48:55 +08:00
|
|
|
err := p.Run(fds, args)
|
|
|
|
if err != ErrNextProgram {
|
2021-09-12 07:35:43 +08:00
|
|
|
return err
|
2017-12-05 03:55:49 +08:00
|
|
|
}
|
|
|
|
}
|
2022-02-05 22:48:55 +08:00
|
|
|
// If we have reached here, all subprograms have returned ErrNextProgram
|
|
|
|
return ErrNextProgram
|
2020-04-25 03:30:04 +08:00
|
|
|
}
|
|
|
|
|
2022-02-05 22:48:55 +08:00
|
|
|
var errNoSuitableSubprogram = errors.New("internal error: no suitable subprogram")
|
|
|
|
|
|
|
|
// ErrNextProgram is a special error that may be returned by Program.Run that
|
|
|
|
// is part of a Composite program, indicating that the next program should be
|
|
|
|
// tried.
|
|
|
|
var ErrNextProgram = errors.New("next program")
|
2021-09-12 07:35:43 +08:00
|
|
|
|
|
|
|
// BadUsage returns a special error that may be returned by Program.Run. It
|
|
|
|
// causes the main function to print out a message, the usage information and
|
2020-04-25 03:30:04 +08:00
|
|
|
// exit with 2.
|
|
|
|
func BadUsage(msg string) error { return badUsageError{msg} }
|
|
|
|
|
|
|
|
type badUsageError struct{ msg string }
|
|
|
|
|
|
|
|
func (e badUsageError) Error() string { return e.msg }
|
|
|
|
|
2021-09-12 07:35:43 +08:00
|
|
|
// Exit returns a special error that may be returned by Program.Run. It causes
|
|
|
|
// the main function to exit with the given code without printing any error
|
|
|
|
// messages. Exit(0) returns nil.
|
2020-04-25 03:30:04 +08:00
|
|
|
func Exit(exit int) error {
|
|
|
|
if exit == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return exitError{exit}
|
|
|
|
}
|
|
|
|
|
|
|
|
type exitError struct{ exit int }
|
|
|
|
|
|
|
|
func (e exitError) Error() string { return "" }
|