2018-01-10 01:18:19 +01:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
2019-03-21 22:36:42 +01:00
|
|
|
"errors"
|
2018-01-10 03:24:50 +01:00
|
|
|
"fmt"
|
2019-05-22 18:35:44 +02:00
|
|
|
"io/ioutil"
|
2019-05-18 21:29:26 +02:00
|
|
|
"net/url"
|
2019-05-16 23:26:08 +02:00
|
|
|
"os"
|
2019-05-18 21:29:26 +02:00
|
|
|
"os/exec"
|
2018-01-10 01:18:19 +01:00
|
|
|
"path"
|
2019-03-31 20:42:18 +02:00
|
|
|
"regexp"
|
2019-06-05 19:48:00 +02:00
|
|
|
"sort"
|
2018-01-10 03:24:50 +01:00
|
|
|
"strings"
|
2018-01-10 01:18:19 +01:00
|
|
|
"unicode"
|
2018-01-10 17:19:45 +01:00
|
|
|
|
2019-05-21 22:31:04 +02:00
|
|
|
"github.com/gdamore/tcell"
|
2018-01-10 17:19:45 +01:00
|
|
|
"github.com/go-ini/ini"
|
|
|
|
"github.com/kyoh86/xdg"
|
2018-01-10 01:18:19 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type UIConfig struct {
|
2019-03-16 01:40:28 +01:00
|
|
|
IndexFormat string `ini:"index-format"`
|
|
|
|
TimestampFormat string `ini:"timestamp-format"`
|
2018-01-10 01:18:19 +01:00
|
|
|
ShowHeaders []string `delim:","`
|
2019-03-16 01:40:28 +01:00
|
|
|
RenderAccountTabs string `ini:"render-account-tabs"`
|
|
|
|
SidebarWidth int `ini:"sidebar-width"`
|
|
|
|
PreviewHeight int `ini:"preview-height"`
|
|
|
|
EmptyMessage string `ini:"empty-message"`
|
2018-01-10 01:18:19 +01:00
|
|
|
}
|
|
|
|
|
2019-03-31 20:24:53 +02:00
|
|
|
const (
|
|
|
|
FILTER_MIMETYPE = iota
|
|
|
|
FILTER_HEADER
|
|
|
|
)
|
|
|
|
|
2018-01-10 01:18:19 +01:00
|
|
|
type AccountConfig struct {
|
2019-06-08 19:41:56 +02:00
|
|
|
Archive string
|
2019-05-18 21:29:26 +02:00
|
|
|
CopyTo string
|
|
|
|
Default string
|
|
|
|
From string
|
|
|
|
Name string
|
|
|
|
Source string
|
|
|
|
SourceCredCmd string
|
|
|
|
Folders []string
|
|
|
|
Params map[string]string
|
|
|
|
Outgoing string
|
|
|
|
OutgoingCredCmd string
|
2018-01-10 01:18:19 +01:00
|
|
|
}
|
|
|
|
|
2019-03-21 22:36:42 +01:00
|
|
|
type BindingConfig struct {
|
2019-05-14 20:27:28 +02:00
|
|
|
Global *KeyBindings
|
2019-05-21 22:31:04 +02:00
|
|
|
AccountWizard *KeyBindings
|
2019-05-14 20:27:28 +02:00
|
|
|
Compose *KeyBindings
|
|
|
|
ComposeEditor *KeyBindings
|
|
|
|
ComposeReview *KeyBindings
|
|
|
|
MessageList *KeyBindings
|
|
|
|
MessageView *KeyBindings
|
|
|
|
Terminal *KeyBindings
|
2019-03-21 22:36:42 +01:00
|
|
|
}
|
|
|
|
|
2019-05-14 21:25:30 +02:00
|
|
|
type ComposeConfig struct {
|
|
|
|
Editor string `ini:"editor"`
|
|
|
|
}
|
|
|
|
|
2019-03-31 20:24:53 +02:00
|
|
|
type FilterConfig struct {
|
|
|
|
FilterType int
|
|
|
|
Filter string
|
|
|
|
Command string
|
2019-03-31 20:42:18 +02:00
|
|
|
Header string
|
|
|
|
Regex *regexp.Regexp
|
2019-03-31 20:24:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type ViewerConfig struct {
|
|
|
|
Pager string
|
|
|
|
Alternatives []string
|
2019-06-07 10:26:14 +02:00
|
|
|
ShowHeaders bool `ini:"show-headers"`
|
2019-03-31 20:24:53 +02:00
|
|
|
}
|
|
|
|
|
2018-01-10 01:18:19 +01:00
|
|
|
type AercConfig struct {
|
2019-03-21 22:36:42 +01:00
|
|
|
Bindings BindingConfig
|
2019-05-14 21:25:30 +02:00
|
|
|
Compose ComposeConfig
|
2018-01-10 01:18:19 +01:00
|
|
|
Ini *ini.File `ini:"-"`
|
|
|
|
Accounts []AccountConfig `ini:"-"`
|
2019-03-31 20:24:53 +02:00
|
|
|
Filters []FilterConfig `ini:"-"`
|
|
|
|
Viewer ViewerConfig `ini:"-"`
|
2018-01-10 01:18:19 +01:00
|
|
|
Ui UIConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
// Input: TimestampFormat
|
|
|
|
// Output: timestamp-format
|
|
|
|
func mapName(raw string) string {
|
|
|
|
newstr := make([]rune, 0, len(raw))
|
|
|
|
for i, chr := range raw {
|
|
|
|
if isUpper := 'A' <= chr && chr <= 'Z'; isUpper {
|
|
|
|
if i > 0 {
|
|
|
|
newstr = append(newstr, '-')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
newstr = append(newstr, unicode.ToLower(chr))
|
|
|
|
}
|
|
|
|
return string(newstr)
|
|
|
|
}
|
|
|
|
|
2018-01-10 03:24:50 +01:00
|
|
|
func loadAccountConfig(path string) ([]AccountConfig, error) {
|
2018-01-10 17:19:45 +01:00
|
|
|
file, err := ini.Load(path)
|
|
|
|
if err != nil {
|
2019-05-22 17:35:55 +02:00
|
|
|
// No config triggers account configuration wizard
|
|
|
|
return nil, nil
|
2018-01-10 03:24:50 +01:00
|
|
|
}
|
|
|
|
file.NameMapper = mapName
|
2018-01-10 17:19:45 +01:00
|
|
|
|
|
|
|
var accounts []AccountConfig
|
2018-01-10 03:24:50 +01:00
|
|
|
for _, _sec := range file.SectionStrings() {
|
|
|
|
if _sec == "DEFAULT" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
sec := file.Section(_sec)
|
2019-03-16 02:33:08 +01:00
|
|
|
account := AccountConfig{
|
2019-06-08 19:41:56 +02:00
|
|
|
Archive: "Archive",
|
2019-03-16 02:33:08 +01:00
|
|
|
Default: "INBOX",
|
|
|
|
Name: _sec,
|
|
|
|
Params: make(map[string]string),
|
|
|
|
}
|
2018-01-10 03:24:50 +01:00
|
|
|
if err = sec.MapTo(&account); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for key, val := range sec.KeysHash() {
|
2019-03-16 02:33:08 +01:00
|
|
|
if key == "folders" {
|
2019-06-05 19:48:00 +02:00
|
|
|
folders := strings.Split(val, ",")
|
|
|
|
sort.Strings(folders)
|
|
|
|
account.Folders = folders
|
2019-05-21 20:31:14 +02:00
|
|
|
} else if key == "source-cred-cmd" {
|
2019-05-18 21:29:26 +02:00
|
|
|
account.SourceCredCmd = val
|
2019-05-13 05:35:36 +02:00
|
|
|
} else if key == "outgoing" {
|
|
|
|
account.Outgoing = val
|
2019-05-21 20:31:14 +02:00
|
|
|
} else if key == "outgoing-cred-cmd" {
|
2019-05-18 21:29:26 +02:00
|
|
|
account.OutgoingCredCmd = val
|
2019-05-13 22:04:01 +02:00
|
|
|
} else if key == "from" {
|
|
|
|
account.From = val
|
2019-05-16 01:41:21 +02:00
|
|
|
} else if key == "copy-to" {
|
|
|
|
account.CopyTo = val
|
2019-06-08 19:41:56 +02:00
|
|
|
} else if key == "archive" {
|
|
|
|
account.Archive = val
|
2018-01-10 03:24:50 +01:00
|
|
|
} else if key != "name" {
|
|
|
|
account.Params[key] = val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if account.Source == "" {
|
|
|
|
return nil, fmt.Errorf("Expected source for account %s", _sec)
|
|
|
|
}
|
2019-05-18 21:29:26 +02:00
|
|
|
|
|
|
|
source, err := parseCredential(account.Source, account.SourceCredCmd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Invalid source credentials for %s: %s", _sec, err)
|
|
|
|
}
|
|
|
|
account.Source = source
|
|
|
|
|
|
|
|
outgoing, err := parseCredential(account.Outgoing, account.OutgoingCredCmd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Invalid outgoing credentials for %s: %s", _sec, err)
|
|
|
|
}
|
|
|
|
account.Outgoing = outgoing
|
|
|
|
|
2018-01-10 03:24:50 +01:00
|
|
|
accounts = append(accounts, account)
|
|
|
|
}
|
|
|
|
return accounts, nil
|
|
|
|
}
|
|
|
|
|
2019-05-18 21:29:26 +02:00
|
|
|
func parseCredential(cred, command string) (string, error) {
|
|
|
|
if cred == "" || command == "" {
|
|
|
|
return cred, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
u, err := url.Parse(cred)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
// ignore the command if a password is specified
|
|
|
|
if _, exists := u.User.Password(); exists {
|
|
|
|
return cred, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't attempt to parse the command if the url is a path (ie /usr/bin/sendmail)
|
|
|
|
if !u.IsAbs() {
|
|
|
|
return cred, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd := exec.Command("sh", "-c", command)
|
|
|
|
output, err := cmd.Output()
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("failed to read password: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
pw := strings.TrimSpace(string(output))
|
|
|
|
u.User = url.UserPassword(u.User.Username(), pw)
|
|
|
|
|
|
|
|
return u.String(), nil
|
|
|
|
}
|
|
|
|
|
2019-05-22 18:35:44 +02:00
|
|
|
func installTemplate(root, sharedir, name string) error {
|
|
|
|
if _, err := os.Stat(root); os.IsNotExist(err) {
|
|
|
|
err := os.MkdirAll(root, 0755)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data, err := ioutil.ReadFile(path.Join(sharedir, name))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = ioutil.WriteFile(path.Join(root, name), data, 0644)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func LoadConfig(root *string, sharedir string) (*AercConfig, error) {
|
2018-01-10 01:18:19 +01:00
|
|
|
if root == nil {
|
|
|
|
_root := path.Join(xdg.ConfigHome(), "aerc")
|
|
|
|
root = &_root
|
|
|
|
}
|
2019-05-16 23:26:08 +02:00
|
|
|
filename := path.Join(*root, "accounts.conf")
|
|
|
|
if err := checkConfigPerms(filename); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
filename = path.Join(*root, "aerc.conf")
|
|
|
|
file, err := ini.Load(filename)
|
2018-01-10 17:19:45 +01:00
|
|
|
if err != nil {
|
2019-05-22 18:35:44 +02:00
|
|
|
if err := installTemplate(*root, sharedir, "aerc.conf"); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if file, err = ini.Load(filename); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-10 01:18:19 +01:00
|
|
|
}
|
|
|
|
file.NameMapper = mapName
|
|
|
|
config := &AercConfig{
|
2019-03-21 22:36:42 +01:00
|
|
|
Bindings: BindingConfig{
|
2019-05-14 20:27:28 +02:00
|
|
|
Global: NewKeyBindings(),
|
2019-05-21 22:31:04 +02:00
|
|
|
AccountWizard: NewKeyBindings(),
|
2019-05-14 20:27:28 +02:00
|
|
|
Compose: NewKeyBindings(),
|
|
|
|
ComposeEditor: NewKeyBindings(),
|
|
|
|
ComposeReview: NewKeyBindings(),
|
|
|
|
MessageList: NewKeyBindings(),
|
|
|
|
MessageView: NewKeyBindings(),
|
|
|
|
Terminal: NewKeyBindings(),
|
2019-03-21 22:36:42 +01:00
|
|
|
},
|
|
|
|
Ini: file,
|
2019-03-15 06:12:06 +01:00
|
|
|
|
2018-01-10 01:18:19 +01:00
|
|
|
Ui: UIConfig{
|
2019-06-07 21:35:23 +02:00
|
|
|
IndexFormat: "%D %-17.17n %s",
|
|
|
|
TimestampFormat: "2006-01-02 03:04 PM",
|
2018-01-10 01:18:19 +01:00
|
|
|
ShowHeaders: []string{
|
|
|
|
"From", "To", "Cc", "Bcc", "Subject", "Date",
|
|
|
|
},
|
|
|
|
RenderAccountTabs: "auto",
|
|
|
|
SidebarWidth: 20,
|
|
|
|
PreviewHeight: 12,
|
|
|
|
EmptyMessage: "(no messages)",
|
|
|
|
},
|
|
|
|
}
|
2019-05-21 22:31:04 +02:00
|
|
|
// These bindings are not configurable
|
|
|
|
config.Bindings.AccountWizard.ExKey = KeyStroke{
|
|
|
|
Key: tcell.KeyCtrlE,
|
|
|
|
}
|
|
|
|
quit, _ := ParseBinding("<C-q>", ":quit<Enter>")
|
|
|
|
config.Bindings.AccountWizard.Add(quit)
|
2019-03-31 20:24:53 +02:00
|
|
|
if filters, err := file.GetSection("filters"); err == nil {
|
|
|
|
// TODO: Parse the filter more finely, e.g. parse the regex
|
2019-03-31 21:21:04 +02:00
|
|
|
for _, match := range filters.KeyStrings() {
|
|
|
|
cmd := filters.KeysHash()[match]
|
2019-03-31 20:24:53 +02:00
|
|
|
filter := FilterConfig{
|
|
|
|
Command: cmd,
|
|
|
|
Filter: match,
|
|
|
|
}
|
2019-03-31 20:42:18 +02:00
|
|
|
if strings.Contains(match, ",~") {
|
2019-03-31 20:24:53 +02:00
|
|
|
filter.FilterType = FILTER_HEADER
|
2019-03-31 20:42:18 +02:00
|
|
|
header := filter.Filter[:strings.Index(filter.Filter, ",")]
|
|
|
|
regex := filter.Filter[strings.Index(filter.Filter, "~")+1:]
|
|
|
|
filter.Header = strings.ToLower(header)
|
|
|
|
filter.Regex, err = regexp.Compile(regex)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
} else if strings.ContainsRune(match, ',') {
|
|
|
|
filter.FilterType = FILTER_HEADER
|
|
|
|
header := filter.Filter[:strings.Index(filter.Filter, ",")]
|
|
|
|
value := filter.Filter[strings.Index(filter.Filter, ",")+1:]
|
|
|
|
filter.Header = strings.ToLower(header)
|
|
|
|
filter.Regex, err = regexp.Compile(regexp.QuoteMeta(value))
|
2019-03-31 20:24:53 +02:00
|
|
|
} else {
|
|
|
|
filter.FilterType = FILTER_MIMETYPE
|
|
|
|
}
|
|
|
|
config.Filters = append(config.Filters, filter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if viewer, err := file.GetSection("viewer"); err == nil {
|
|
|
|
if err := viewer.MapTo(&config.Viewer); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for key, val := range viewer.KeysHash() {
|
|
|
|
switch key {
|
|
|
|
case "alternatives":
|
|
|
|
config.Viewer.Alternatives = strings.Split(val, ",")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-05-14 21:25:30 +02:00
|
|
|
if compose, err := file.GetSection("compose"); err == nil {
|
|
|
|
if err := compose.MapTo(&config.Compose); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2019-03-15 06:12:06 +01:00
|
|
|
if ui, err := file.GetSection("ui"); err == nil {
|
2019-03-16 01:40:28 +01:00
|
|
|
if err := ui.MapTo(&config.Ui); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-10 01:18:19 +01:00
|
|
|
}
|
2018-01-10 03:24:50 +01:00
|
|
|
accountsPath := path.Join(*root, "accounts.conf")
|
|
|
|
if accounts, err := loadAccountConfig(accountsPath); err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else {
|
|
|
|
config.Accounts = accounts
|
|
|
|
}
|
2019-05-22 18:35:44 +02:00
|
|
|
filename = path.Join(*root, "binds.conf")
|
|
|
|
binds, err := ini.Load(filename)
|
2019-03-21 22:36:42 +01:00
|
|
|
if err != nil {
|
2019-05-22 18:35:44 +02:00
|
|
|
if err := installTemplate(*root, sharedir, "binds.conf"); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if binds, err = ini.Load(filename); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-03-21 22:36:42 +01:00
|
|
|
}
|
|
|
|
groups := map[string]**KeyBindings{
|
|
|
|
"default": &config.Bindings.Global,
|
|
|
|
"compose": &config.Bindings.Compose,
|
|
|
|
"messages": &config.Bindings.MessageList,
|
|
|
|
"terminal": &config.Bindings.Terminal,
|
|
|
|
"view": &config.Bindings.MessageView,
|
2019-05-14 20:27:28 +02:00
|
|
|
|
|
|
|
"compose::editor": &config.Bindings.ComposeEditor,
|
|
|
|
"compose::review": &config.Bindings.ComposeReview,
|
2019-03-21 22:36:42 +01:00
|
|
|
}
|
|
|
|
for _, name := range binds.SectionStrings() {
|
|
|
|
sec, err := binds.GetSection(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
group, ok := groups[strings.ToLower(name)]
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("Unknown keybinding group " + name)
|
|
|
|
}
|
|
|
|
bindings := NewKeyBindings()
|
|
|
|
for key, value := range sec.KeysHash() {
|
|
|
|
if key == "$ex" {
|
|
|
|
strokes, err := ParseKeyStrokes(value)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(strokes) != 1 {
|
|
|
|
return nil, errors.New(
|
|
|
|
"Error: only one keystroke supported for $ex")
|
|
|
|
}
|
|
|
|
bindings.ExKey = strokes[0]
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if key == "$noinherit" {
|
|
|
|
if value == "false" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if value != "true" {
|
|
|
|
return nil, errors.New(
|
|
|
|
"Error: expected 'true' or 'false' for $noinherit")
|
|
|
|
}
|
|
|
|
bindings.Globals = false
|
|
|
|
}
|
|
|
|
binding, err := ParseBinding(key, value)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
bindings.Add(binding)
|
|
|
|
}
|
|
|
|
*group = MergeBindings(bindings, *group)
|
|
|
|
}
|
|
|
|
// Globals can't inherit from themselves
|
|
|
|
config.Bindings.Global.Globals = false
|
2018-01-10 01:18:19 +01:00
|
|
|
return config, nil
|
|
|
|
}
|
2019-05-16 23:26:08 +02:00
|
|
|
|
|
|
|
// checkConfigPerms checks for too open permissions
|
|
|
|
// printing the fix on stdout and returning an error
|
|
|
|
func checkConfigPerms(filename string) error {
|
|
|
|
info, err := os.Stat(filename)
|
|
|
|
if err != nil {
|
2019-05-22 17:35:55 +02:00
|
|
|
return nil // disregard absent files
|
2019-05-16 23:26:08 +02:00
|
|
|
}
|
|
|
|
perms := info.Mode().Perm()
|
|
|
|
goPerms := perms >> 3
|
|
|
|
// group or others have read access
|
|
|
|
if goPerms&0x44 != 0 {
|
|
|
|
fmt.Printf("The file %v has too open permissions.\n", filename)
|
|
|
|
fmt.Println("This is a security issue (it contains passwords).")
|
|
|
|
fmt.Printf("To fix it, run `chmod 600 %v`\n", filename)
|
|
|
|
return errors.New("account.conf permissions too lax")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|