2018-01-10 00:30:46 +01:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2022-02-20 10:32:38 +01:00
|
|
|
"bytes"
|
2022-08-22 20:23:16 +02:00
|
|
|
"encoding/base64"
|
2022-07-31 14:32:48 +02:00
|
|
|
"errors"
|
2019-05-16 23:26:08 +02:00
|
|
|
"fmt"
|
2018-02-01 03:18:21 +01:00
|
|
|
"os"
|
2022-08-12 12:24:52 +02:00
|
|
|
"runtime"
|
2020-06-10 23:08:28 +02:00
|
|
|
"sort"
|
2022-08-12 12:24:52 +02:00
|
|
|
"strings"
|
2018-01-10 14:35:26 +01:00
|
|
|
"time"
|
2018-01-10 01:18:19 +01:00
|
|
|
|
2019-06-18 15:42:41 +02:00
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
2018-02-01 03:18:21 +01:00
|
|
|
"github.com/mattn/go-isatty"
|
2022-02-20 10:32:38 +01:00
|
|
|
"github.com/xo/terminfo"
|
2018-02-01 03:18:21 +01:00
|
|
|
|
2021-11-05 10:19:46 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/commands"
|
|
|
|
"git.sr.ht/~rjarry/aerc/commands/account"
|
|
|
|
"git.sr.ht/~rjarry/aerc/commands/compose"
|
|
|
|
"git.sr.ht/~rjarry/aerc/commands/msg"
|
|
|
|
"git.sr.ht/~rjarry/aerc/commands/msgview"
|
|
|
|
"git.sr.ht/~rjarry/aerc/commands/terminal"
|
|
|
|
"git.sr.ht/~rjarry/aerc/config"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib"
|
2022-04-25 15:30:43 +02:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib/crypto"
|
2021-11-05 10:19:46 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib/templates"
|
|
|
|
libui "git.sr.ht/~rjarry/aerc/lib/ui"
|
2022-03-22 09:52:27 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/logging"
|
2021-11-05 10:19:46 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/widgets"
|
2018-01-10 00:30:46 +01:00
|
|
|
)
|
|
|
|
|
2019-03-21 21:30:23 +01:00
|
|
|
func getCommands(selected libui.Drawable) []*commands.Commands {
|
|
|
|
switch selected.(type) {
|
|
|
|
case *widgets.AccountView:
|
|
|
|
return []*commands.Commands{
|
|
|
|
account.AccountCommands,
|
2019-06-02 07:15:04 +02:00
|
|
|
msg.MessageCommands,
|
2019-03-21 21:32:22 +01:00
|
|
|
commands.GlobalCommands,
|
|
|
|
}
|
2019-05-12 06:06:09 +02:00
|
|
|
case *widgets.Composer:
|
|
|
|
return []*commands.Commands{
|
2019-05-12 17:21:28 +02:00
|
|
|
compose.ComposeCommands,
|
2019-05-12 06:06:09 +02:00
|
|
|
commands.GlobalCommands,
|
|
|
|
}
|
2019-03-31 03:45:41 +02:00
|
|
|
case *widgets.MessageViewer:
|
|
|
|
return []*commands.Commands{
|
|
|
|
msgview.MessageViewCommands,
|
2019-06-02 07:15:04 +02:00
|
|
|
msg.MessageCommands,
|
2019-03-31 03:45:41 +02:00
|
|
|
commands.GlobalCommands,
|
|
|
|
}
|
2019-03-30 19:12:04 +01:00
|
|
|
case *widgets.Terminal:
|
2019-03-21 21:32:22 +01:00
|
|
|
return []*commands.Commands{
|
|
|
|
terminal.TerminalCommands,
|
|
|
|
commands.GlobalCommands,
|
2019-03-21 21:30:23 +01:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
return []*commands.Commands{commands.GlobalCommands}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-21 22:01:51 +02:00
|
|
|
func execCommand(aerc *widgets.Aerc, ui *libui.UI, cmd []string) error {
|
2022-07-18 12:54:55 +02:00
|
|
|
cmds := getCommands(aerc.SelectedTabContent())
|
2019-06-27 19:33:11 +02:00
|
|
|
for i, set := range cmds {
|
|
|
|
err := set.ExecuteCommand(aerc, cmd)
|
2022-07-31 14:32:48 +02:00
|
|
|
if err != nil {
|
|
|
|
if errors.As(err, new(commands.NoSuchCommand)) {
|
|
|
|
if i == len(cmds)-1 {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if errors.As(err, new(commands.ErrorExit)) {
|
|
|
|
ui.Exit()
|
|
|
|
return nil
|
2019-06-27 19:33:11 +02:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
2022-07-31 14:32:48 +02:00
|
|
|
break
|
2019-06-27 19:33:11 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getCompletions(aerc *widgets.Aerc, cmd string) []string {
|
2019-09-05 04:30:49 +02:00
|
|
|
var completions []string
|
2022-07-18 12:54:55 +02:00
|
|
|
for _, set := range getCommands(aerc.SelectedTabContent()) {
|
2019-09-05 04:30:49 +02:00
|
|
|
completions = append(completions, set.GetCompletions(aerc, cmd)...)
|
2019-06-27 19:33:11 +02:00
|
|
|
}
|
2020-06-10 23:08:28 +02:00
|
|
|
sort.Strings(completions)
|
2019-06-27 19:33:11 +02:00
|
|
|
return completions
|
|
|
|
}
|
|
|
|
|
2022-02-19 14:06:57 +01:00
|
|
|
// set at build time
|
|
|
|
var Version string
|
2022-08-12 12:24:52 +02:00
|
|
|
var Flags string
|
|
|
|
|
|
|
|
func buildInfo() string {
|
|
|
|
info := Version
|
2022-08-22 20:23:16 +02:00
|
|
|
flags, _ := base64.StdEncoding.DecodeString(Flags)
|
|
|
|
if strings.Contains(string(flags), "notmuch") {
|
2022-08-12 12:24:52 +02:00
|
|
|
info += " +notmuch"
|
|
|
|
}
|
|
|
|
info += fmt.Sprintf(" (%s %s %s)",
|
|
|
|
runtime.Version(), runtime.GOARCH, runtime.GOOS)
|
|
|
|
return info
|
|
|
|
}
|
2019-05-22 18:35:44 +02:00
|
|
|
|
2022-07-19 22:31:51 +02:00
|
|
|
func usage(msg string) {
|
|
|
|
fmt.Fprintln(os.Stderr, msg)
|
2022-09-07 08:31:12 +02:00
|
|
|
fmt.Fprintln(os.Stderr, "usage: aerc [-v] [-a <account-name[,account-name>] [mailto:...]")
|
2022-07-19 22:31:51 +02:00
|
|
|
os.Exit(1)
|
2019-07-16 12:09:25 +02:00
|
|
|
}
|
|
|
|
|
2021-11-02 10:40:08 +01:00
|
|
|
func setWindowTitle() {
|
2022-07-19 21:11:33 +02:00
|
|
|
logging.Debugf("Parsing terminfo")
|
2022-02-20 10:32:38 +01:00
|
|
|
ti, err := terminfo.LoadFromEnv()
|
|
|
|
if err != nil {
|
2022-07-19 21:11:33 +02:00
|
|
|
logging.Warnf("Cannot get terminfo: %v", err)
|
2022-02-20 10:32:38 +01:00
|
|
|
return
|
2021-11-02 10:40:08 +01:00
|
|
|
}
|
2022-02-20 10:32:38 +01:00
|
|
|
|
|
|
|
if !ti.Has(terminfo.HasStatusLine) {
|
2022-07-19 21:11:33 +02:00
|
|
|
logging.Infof("Terminal does not have status line support")
|
2022-02-20 10:32:38 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-07-19 21:11:33 +02:00
|
|
|
logging.Infof("Setting terminal title")
|
2022-02-20 10:32:38 +01:00
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
ti.Fprintf(buf, terminfo.ToStatusLine)
|
|
|
|
fmt.Fprint(buf, "aerc")
|
|
|
|
ti.Fprintf(buf, terminfo.FromStatusLine)
|
|
|
|
os.Stderr.Write(buf.Bytes())
|
2021-11-02 10:40:08 +01:00
|
|
|
}
|
|
|
|
|
2018-01-10 00:30:46 +01:00
|
|
|
func main() {
|
2022-03-22 09:52:27 +01:00
|
|
|
defer logging.PanicHandler()
|
2022-08-22 17:38:24 +02:00
|
|
|
opts, optind, err := getopt.Getopts(os.Args, "va:")
|
2019-06-18 15:42:41 +02:00
|
|
|
if err != nil {
|
2022-07-19 22:31:51 +02:00
|
|
|
usage("error: " + err.Error())
|
2019-07-19 20:34:13 +02:00
|
|
|
return
|
2019-06-18 15:42:41 +02:00
|
|
|
}
|
2022-08-12 12:24:52 +02:00
|
|
|
logging.BuildInfo = buildInfo()
|
2022-08-22 17:38:24 +02:00
|
|
|
var accts []string
|
2019-06-18 15:42:41 +02:00
|
|
|
for _, opt := range opts {
|
2022-07-31 14:32:48 +02:00
|
|
|
if opt.Option == 'v' {
|
2022-08-12 12:24:52 +02:00
|
|
|
fmt.Println("aerc " + logging.BuildInfo)
|
2019-06-18 15:42:41 +02:00
|
|
|
return
|
|
|
|
}
|
2022-08-22 17:38:24 +02:00
|
|
|
if opt.Option == 'a' {
|
|
|
|
accts = strings.Split(opt.Value, ",")
|
|
|
|
}
|
2019-06-18 15:42:41 +02:00
|
|
|
}
|
2022-01-23 03:35:40 +01:00
|
|
|
retryExec := false
|
2019-07-19 20:34:13 +02:00
|
|
|
args := os.Args[optind:]
|
|
|
|
if len(args) > 1 {
|
2022-07-19 22:31:51 +02:00
|
|
|
usage("error: invalid arguments")
|
2019-07-19 20:34:13 +02:00
|
|
|
return
|
|
|
|
} else if len(args) == 1 {
|
2019-09-29 12:08:59 +02:00
|
|
|
arg := args[0]
|
|
|
|
err := lib.ConnectAndExec(arg)
|
|
|
|
if err == nil {
|
|
|
|
return // other aerc instance takes over
|
|
|
|
}
|
2022-01-23 03:35:40 +01:00
|
|
|
fmt.Fprintf(os.Stderr, "Failed to communicate to aerc: %v\n", err)
|
2019-09-29 12:08:59 +02:00
|
|
|
// continue with setting up a new aerc instance and retry after init
|
2022-01-23 03:35:40 +01:00
|
|
|
retryExec = true
|
2019-06-18 15:42:41 +02:00
|
|
|
}
|
|
|
|
|
2018-02-01 03:18:21 +01:00
|
|
|
if !isatty.IsTerminal(os.Stdout.Fd()) {
|
2022-07-19 22:31:51 +02:00
|
|
|
logging.Init()
|
2018-02-01 03:18:21 +01:00
|
|
|
}
|
2022-08-12 12:24:52 +02:00
|
|
|
logging.Infof("Starting up version %s", logging.BuildInfo)
|
2018-02-01 03:18:21 +01:00
|
|
|
|
2022-08-22 17:38:24 +02:00
|
|
|
conf, err := config.LoadConfigFromFile(nil, accts)
|
2018-01-10 17:19:45 +01:00
|
|
|
if err != nil {
|
2019-07-28 15:02:09 +02:00
|
|
|
fmt.Fprintf(os.Stderr, "Failed to load config: %v\n", err)
|
2022-07-31 14:32:48 +02:00
|
|
|
os.Exit(1) //nolint:gocritic // PanicHandler does not need to run as it's not a panic
|
2018-01-10 01:18:19 +01:00
|
|
|
}
|
2018-02-17 22:35:36 +01:00
|
|
|
|
2019-03-16 01:32:09 +01:00
|
|
|
var (
|
|
|
|
aerc *widgets.Aerc
|
|
|
|
ui *libui.UI
|
|
|
|
)
|
2019-06-27 19:33:11 +02:00
|
|
|
|
2022-03-24 10:47:34 +01:00
|
|
|
deferLoop := make(chan struct{})
|
|
|
|
|
2022-04-25 15:30:43 +02:00
|
|
|
c := crypto.New(conf.General.PgpProvider)
|
2022-07-29 22:31:54 +02:00
|
|
|
err = c.Init()
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("failed to initialise crypto interface: %v", err)
|
|
|
|
}
|
2022-04-25 15:30:43 +02:00
|
|
|
defer c.Close()
|
|
|
|
|
2022-07-19 22:31:51 +02:00
|
|
|
aerc = widgets.NewAerc(conf, c, func(cmd []string) error {
|
2019-07-21 22:01:51 +02:00
|
|
|
return execCommand(aerc, ui, cmd)
|
|
|
|
}, func(cmd string) []string {
|
|
|
|
return getCompletions(aerc, cmd)
|
2022-03-24 10:47:34 +01:00
|
|
|
}, &commands.CmdHistory, deferLoop)
|
2019-03-11 02:15:24 +01:00
|
|
|
|
2019-08-03 17:17:13 +02:00
|
|
|
ui, err = libui.Initialize(aerc)
|
2018-01-11 04:41:15 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-02-27 04:54:39 +01:00
|
|
|
defer ui.Close()
|
2022-03-22 09:52:27 +01:00
|
|
|
logging.UICleanup = func() {
|
|
|
|
ui.Close()
|
|
|
|
}
|
2022-03-24 10:47:34 +01:00
|
|
|
close(deferLoop)
|
2018-02-17 22:35:36 +01:00
|
|
|
|
2019-08-03 17:17:13 +02:00
|
|
|
if conf.Ui.MouseEnabled {
|
|
|
|
ui.EnableMouse()
|
|
|
|
}
|
|
|
|
|
2022-07-19 22:31:51 +02:00
|
|
|
as, err := lib.StartServer()
|
2019-07-19 20:15:48 +02:00
|
|
|
if err != nil {
|
2022-07-19 22:31:51 +02:00
|
|
|
logging.Warnf("Failed to start Unix server: %v", err)
|
2019-07-19 20:15:48 +02:00
|
|
|
} else {
|
|
|
|
defer as.Close()
|
|
|
|
as.OnMailto = aerc.Mailto
|
2022-07-11 20:11:19 +02:00
|
|
|
as.OnMbox = aerc.Mbox
|
2019-07-19 20:15:48 +02:00
|
|
|
}
|
|
|
|
|
2020-05-02 14:06:02 +02:00
|
|
|
// set the aerc version so that we can use it in the template funcs
|
|
|
|
templates.SetVersion(Version)
|
|
|
|
|
2022-01-23 03:35:40 +01:00
|
|
|
if retryExec {
|
|
|
|
// retry execution
|
|
|
|
arg := args[0]
|
|
|
|
err := lib.ConnectAndExec(arg)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Fprintf(os.Stderr, "Failed to communicate to aerc: %v\n", err)
|
2022-07-29 22:31:54 +02:00
|
|
|
err = aerc.CloseBackends()
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("failed to close backends: %v", err)
|
|
|
|
}
|
2022-01-23 03:35:40 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2019-09-29 12:08:59 +02:00
|
|
|
|
2021-11-02 10:40:08 +01:00
|
|
|
if isatty.IsTerminal(os.Stderr.Fd()) {
|
|
|
|
setWindowTitle()
|
|
|
|
}
|
|
|
|
|
2019-05-04 16:31:16 +02:00
|
|
|
for !ui.ShouldExit() {
|
2019-05-19 11:49:57 +02:00
|
|
|
for aerc.Tick() {
|
|
|
|
// Continue updating our internal state
|
|
|
|
}
|
2018-02-27 04:54:39 +01:00
|
|
|
if !ui.Tick() {
|
2018-02-27 04:41:54 +01:00
|
|
|
// ~60 FPS
|
|
|
|
time.Sleep(16 * time.Millisecond)
|
2018-01-10 14:35:26 +01:00
|
|
|
}
|
2018-01-10 02:39:00 +01:00
|
|
|
}
|
2022-07-29 22:31:54 +02:00
|
|
|
err = aerc.CloseBackends()
|
|
|
|
if err != nil {
|
|
|
|
logging.Warnf("failed to close backends: %v", err)
|
|
|
|
}
|
2018-01-10 00:30:46 +01:00
|
|
|
}
|