package commands
import (
"bytes"
"errors"
"path"
"reflect"
"sort"
"strings"
"unicode"
"git.sr.ht/~rjarry/go-opt"
"git.sr.ht/~rjarry/aerc/app"
"git.sr.ht/~rjarry/aerc/config"
"git.sr.ht/~rjarry/aerc/lib/log"
"git.sr.ht/~rjarry/aerc/lib/state"
"git.sr.ht/~rjarry/aerc/lib/templates"
"git.sr.ht/~rjarry/aerc/models"
)
type CommandContext uint32
const (
NONE = 1 << iota
// available everywhere
GLOBAL
// only when a message list is focused
MESSAGE_LIST
// only when a message viewer is focused
MESSAGE_VIEWER
// only when a message composer is focused
COMPOSE
// only when a terminal
TERMINAL
)
func CurrentContext() CommandContext {
var context CommandContext = GLOBAL
switch app.SelectedTabContent().(type) {
case *app.AccountView:
context |= MESSAGE_LIST
case *app.Composer:
context |= COMPOSE
case *app.MessageViewer:
context |= MESSAGE_VIEWER
case *app.Terminal:
context |= TERMINAL
}
return context
}
type Command interface {
Context() CommandContext
Aliases() []string
Execute([]string) error
}
var allCommands map[string]Command
func Register(cmd Command) {
if allCommands == nil {
allCommands = make(map[string]Command)
}
for _, alias := range cmd.Aliases() {
if allCommands[alias] != nil {
panic("duplicate command alias: " + alias)
}
allCommands[alias] = cmd
}
}
func ActiveCommands() []Command {
var cmds []Command
context := CurrentContext()
for _, cmd := range allCommands {
if cmd.Context()&context != 0 {
cmds = append(cmds, cmd)
}
}
return cmds
}
func ActiveCommandNames() []string {
var names []string
context := CurrentContext()
for alias, cmd := range allCommands {
if cmd.Context()&context != 0 {
names = append(names, alias)
}
}
return names
}
type NoSuchCommand string
func (err NoSuchCommand) Error() string {
return "Unknown command " + string(err)
}
// Expand non-ambiguous command abbreviations.
//
// q --> quit
// ar --> archive
// im --> import-mbox
func ExpandAbbreviations(name string) (string, Command, error) {
context := CurrentContext()
name = strings.TrimLeft(name, ": \t")
cmd, found := allCommands[name]
if found && cmd.Context()&context != 0 {
return name, cmd, nil
}
var candidate Command
var candidateName string
for alias, cmd := range allCommands {
if cmd.Context()&context == 0 || !strings.HasPrefix(alias, name) {
continue
}
if candidate != nil {
// We have more than one command partially
// matching the input.
return name, nil, NoSuchCommand(name)
}
// We have a partial match.
candidate = cmd
candidateName = alias
}
if candidate == nil {
return name, nil, NoSuchCommand(name)
}
return candidateName, candidate, nil
}
func ResolveCommand(
cmdline string, acct *config.AccountConfig, msg *models.MessageInfo,
) (string, Command, error) {
cmdline, err := ExpandTemplates(cmdline, acct, msg)
if err != nil {
return "", nil, err
}
name, rest, didCut := strings.Cut(cmdline, " ")
name, cmd, err := ExpandAbbreviations(name)
if err != nil {
return "", nil, err
}
cmdline = name
if didCut {
cmdline += " " + rest
}
return cmdline, cmd, nil
}
func templateData(
cfg *config.AccountConfig,
msg *models.MessageInfo,
) models.TemplateData {
var folder *models.Directory
acct := app.SelectedAccount()
if acct != nil {
folder = acct.Directories().SelectedDirectory()
}
if cfg == nil && acct != nil {
cfg = acct.AccountConfig()
}
if msg == nil && acct != nil {
msg, _ = acct.SelectedMessage()
}
data := state.NewDataSetter()
data.SetAccount(cfg)
data.SetFolder(folder)
data.SetInfo(msg, 0, false)
if acct != nil {
acct.SetStatus(func(s *state.AccountState, _ string) {
data.SetState(s)
})
}
return data.Data()
}
func ExecuteCommand(cmd Command, cmdline string) error {
args := opt.LexArgs(cmdline)
if args.Count() == 0 {
return errors.New("No arguments")
}
log.Tracef("executing command %s", args.String())
// copy zeroed struct
tmp := reflect.New(reflect.TypeOf(cmd)).Interface().(Command)
if err := opt.ArgsToStruct(args.Clone(), tmp); err != nil {
return err
}
return tmp.Execute(args.Args())
}
// expand template expressions
func ExpandTemplates(
s string, cfg *config.AccountConfig, msg *models.MessageInfo,
) (string, error) {
if strings.Contains(s, "{{") && strings.Contains(s, "}}") {
t, err := templates.ParseTemplate("execute", s)
if err != nil {
return "", err
}
data := templateData(cfg, msg)
var buf bytes.Buffer
err = templates.Render(t, &buf, data)
if err != nil {
return "", err
}
s = buf.String()
}
return s, nil
}
func GetTemplateCompletion(
cmd string,
) ([]string, string, bool) {
countLeft := strings.Count(cmd, "{{")
if countLeft == 0 {
return nil, "", false
}
countRight := strings.Count(cmd, "}}")
switch {
case countLeft > countRight:
// complete template terms
var i int
for i = len(cmd) - 1; i >= 0; i-- {
if strings.ContainsRune("{()| ", rune(cmd[i])) {
break
}
}
search, prefix := cmd[i+1:], cmd[:i+1]
padding := strings.Repeat(" ",
len(search)-len(strings.TrimLeft(search, " ")))
options := FilterList(
templates.Terms(),
strings.TrimSpace(search),
nil,
)
return options, prefix + padding, true
case countLeft == countRight:
// expand template
s, err := ExpandTemplates(cmd, nil, nil)
if err != nil {
log.Warnf("template rendering failed: %v", err)
return nil, "", false
}
return []string{s}, "", true
}
return nil, "", false
}
// GetCompletions returns the completion options and the command prefix
func GetCompletions(
cmd Command, args *opt.Args,
) (options []string, prefix string) {
// copy zeroed struct
tmp := reflect.New(reflect.TypeOf(cmd)).Interface().(Command)
s, err := args.ArgSafe(0)
if err != nil {
log.Errorf("completions error: %v", err)
return options, prefix
}
spec := opt.NewCmdSpec(s, tmp)
return spec.GetCompletions(args)
}
func GetFolders(arg string) []string {
acct := app.SelectedAccount()
if acct == nil {
return make([]string, 0)
}
return FilterList(acct.Directories().List(), arg, nil)
}
func GetTemplates(arg string) []string {
templates := make(map[string]bool)
for _, dir := range config.Templates.TemplateDirs {
for _, f := range listDir(dir, false) {
if !isDir(path.Join(dir, f)) {
templates[f] = true
}
}
}
names := make([]string, len(templates))
for n := range templates {
names = append(names, n)
}
sort.Strings(names)
return FilterList(names, arg, nil)
}
func GetLabels(arg string) []string {
acct := app.SelectedAccount()
if acct == nil {
return make([]string, 0)
}
var prefix string
if arg != "" {
// + and - are used to denote tag addition / removal and need to
// be striped only the last tag should be completed, so that
// multiple labels can be selected
switch arg[0] {
case '+':
prefix = "+"
case '-':
prefix = "-"
}
arg = strings.TrimLeft(arg, "+-")
}
return FilterList(acct.Labels(), arg, func(s string) string {
return opt.QuoteArg(prefix+s) + " "
})
}
// hasCaseSmartPrefix checks whether s starts with prefix, using a case
// sensitive match if and only if prefix contains upper case letters.
func hasCaseSmartPrefix(s, prefix string) bool {
if hasUpper(prefix) {
return strings.HasPrefix(s, prefix)
}
return strings.HasPrefix(strings.ToLower(s), strings.ToLower(prefix))
}
func hasUpper(s string) bool {
for _, r := range s {
if unicode.IsUpper(r) {
return true
}
}
return false
}