887 lines
23 KiB
Go
887 lines
23 KiB
Go
package app
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"net/url"
|
|
"os"
|
|
"os/exec"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/emersion/go-message/mail"
|
|
"github.com/go-ini/ini"
|
|
"golang.org/x/sys/unix"
|
|
|
|
"git.sr.ht/~rjarry/aerc/config"
|
|
"git.sr.ht/~rjarry/aerc/lib/format"
|
|
"git.sr.ht/~rjarry/aerc/lib/log"
|
|
"git.sr.ht/~rjarry/aerc/lib/ui"
|
|
"git.sr.ht/~rjarry/aerc/lib/xdg"
|
|
"git.sr.ht/~rockorager/vaxis"
|
|
)
|
|
|
|
const (
|
|
CONFIGURE_BASICS = iota
|
|
CONFIGURE_SOURCE = iota
|
|
CONFIGURE_OUTGOING = iota
|
|
CONFIGURE_COMPLETE = iota
|
|
)
|
|
|
|
type AccountWizard struct {
|
|
step int
|
|
steps []*ui.Grid
|
|
focus int
|
|
temporary bool
|
|
// CONFIGURE_BASICS
|
|
accountName *ui.TextInput
|
|
email *ui.TextInput
|
|
discovered map[string]string
|
|
fullName *ui.TextInput
|
|
basics []ui.Interactive
|
|
// CONFIGURE_SOURCE
|
|
sourceProtocol *Selector
|
|
sourceTransport *Selector
|
|
|
|
sourceUsername *ui.TextInput
|
|
sourcePassword *ui.TextInput
|
|
sourceServer *ui.TextInput
|
|
sourceStr *ui.Text
|
|
sourceUrl url.URL
|
|
source []ui.Interactive
|
|
// CONFIGURE_OUTGOING
|
|
outgoingProtocol *Selector
|
|
outgoingTransport *Selector
|
|
|
|
outgoingUsername *ui.TextInput
|
|
outgoingPassword *ui.TextInput
|
|
outgoingServer *ui.TextInput
|
|
outgoingStr *ui.Text
|
|
outgoingUrl url.URL
|
|
outgoingCopyTo *ui.TextInput
|
|
outgoing []ui.Interactive
|
|
// CONFIGURE_COMPLETE
|
|
complete []ui.Interactive
|
|
}
|
|
|
|
func showPasswordWarning() {
|
|
title := "ATTENTION"
|
|
text := `
|
|
The Wizard will store your passwords as clear text in:
|
|
|
|
~/.config/aerc/accounts.conf
|
|
|
|
It is recommended to remove the clear text passwords and configure
|
|
'source-cred-cmd' and 'outgoing-cred-cmd' using your own password store
|
|
after the setup.
|
|
`
|
|
warning := NewSelectorDialog(
|
|
title, text, []string{"OK"}, 0,
|
|
SelectedAccountUiConfig(),
|
|
func(_ string, _ error) {
|
|
CloseDialog()
|
|
},
|
|
)
|
|
AddDialog(warning)
|
|
}
|
|
|
|
type configStep struct {
|
|
introduction string
|
|
labels []string
|
|
fields []ui.Drawable
|
|
interactive *[]ui.Interactive
|
|
}
|
|
|
|
func NewConfigStep(intro string, interactive *[]ui.Interactive) configStep {
|
|
return configStep{introduction: intro, interactive: interactive}
|
|
}
|
|
|
|
func (s *configStep) AddField(label string, field ui.Drawable) {
|
|
s.labels = append(s.labels, label)
|
|
s.fields = append(s.fields, field)
|
|
if i, ok := field.(ui.Interactive); ok {
|
|
*s.interactive = append(*s.interactive, i)
|
|
}
|
|
}
|
|
|
|
func (s *configStep) Grid() *ui.Grid {
|
|
introduction := strings.TrimSpace(s.introduction)
|
|
h := strings.Count(introduction, "\n") + 1
|
|
spec := []ui.GridSpec{
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)}, // padding
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(h)}, // intro text
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)}, // padding
|
|
}
|
|
for range s.fields {
|
|
spec = append(spec, []ui.GridSpec{
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)}, // label
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)}, // field
|
|
{Strategy: ui.SIZE_EXACT, Size: ui.Const(1)}, // padding
|
|
}...)
|
|
}
|
|
justify := ui.GridSpec{Strategy: ui.SIZE_WEIGHT, Size: ui.Const(1)}
|
|
spec = append(spec, justify)
|
|
grid := ui.NewGrid().Rows(spec).Columns([]ui.GridSpec{justify})
|
|
|
|
intro := ui.NewText(introduction, config.Ui.GetStyle(config.STYLE_DEFAULT))
|
|
fill := ui.NewFill(' ', vaxis.Style{})
|
|
|
|
grid.AddChild(fill).At(0, 0)
|
|
grid.AddChild(intro).At(1, 0)
|
|
grid.AddChild(fill).At(2, 0)
|
|
|
|
row := 3
|
|
for i, field := range s.fields {
|
|
label := ui.NewText(s.labels[i], config.Ui.GetStyle(config.STYLE_HEADER))
|
|
grid.AddChild(label).At(row, 0)
|
|
grid.AddChild(field).At(row+1, 0)
|
|
grid.AddChild(fill).At(row+2, 0)
|
|
row += 3
|
|
}
|
|
|
|
grid.AddChild(fill).At(row, 0)
|
|
|
|
return grid
|
|
}
|
|
|
|
const (
|
|
// protocols
|
|
IMAP = "IMAP"
|
|
JMAP = "JMAP"
|
|
MAILDIR = "Maildir"
|
|
MAILDIRPP = "Maildir++"
|
|
NOTMUCH = "notmuch"
|
|
SMTP = "SMTP"
|
|
SENDMAIL = "sendmail"
|
|
// transports
|
|
SSL_TLS = "SSL/TLS"
|
|
OAUTH = "SSL/TLS+OAUTHBEARER"
|
|
XOAUTH = "SSL/TLS+XOAUTH2"
|
|
STARTTLS = "STARTTLS"
|
|
INSECURE = "Insecure"
|
|
)
|
|
|
|
var (
|
|
sources = []string{IMAP, JMAP, MAILDIR, MAILDIRPP, NOTMUCH}
|
|
outgoings = []string{SMTP, JMAP, SENDMAIL}
|
|
transports = []string{SSL_TLS, OAUTH, XOAUTH, STARTTLS, INSECURE}
|
|
)
|
|
|
|
func NewAccountWizard() *AccountWizard {
|
|
wizard := &AccountWizard{
|
|
accountName: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
temporary: false,
|
|
email: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
fullName: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
sourcePassword: ui.NewTextInput("", config.Ui).Prompt("] ").Password(true),
|
|
sourceServer: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
sourceStr: ui.NewText("", config.Ui.GetStyle(config.STYLE_DEFAULT)),
|
|
sourceUsername: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
outgoingPassword: ui.NewTextInput("", config.Ui).Prompt("] ").Password(true),
|
|
outgoingServer: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
outgoingStr: ui.NewText("", config.Ui.GetStyle(config.STYLE_DEFAULT)),
|
|
outgoingUsername: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
outgoingCopyTo: ui.NewTextInput("", config.Ui).Prompt("> "),
|
|
|
|
sourceProtocol: NewSelector(sources, 0, config.Ui).Chooser(true),
|
|
sourceTransport: NewSelector(transports, 0, config.Ui).Chooser(true),
|
|
outgoingProtocol: NewSelector(outgoings, 0, config.Ui).Chooser(true),
|
|
outgoingTransport: NewSelector(transports, 0, config.Ui).Chooser(true),
|
|
}
|
|
|
|
// Autofill some stuff for the user
|
|
wizard.email.OnFocusLost(func(_ *ui.TextInput) {
|
|
value := wizard.email.String()
|
|
if wizard.sourceUsername.String() == "" {
|
|
wizard.sourceUsername.Set(value)
|
|
}
|
|
if wizard.outgoingUsername.String() == "" {
|
|
wizard.outgoingUsername.Set(value)
|
|
}
|
|
wizard.sourceUri()
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.sourceProtocol.OnSelect(func(option string) {
|
|
wizard.sourceServer.Set("")
|
|
wizard.autofill()
|
|
wizard.sourceUri()
|
|
})
|
|
wizard.sourceServer.OnChange(func(_ *ui.TextInput) {
|
|
wizard.sourceUri()
|
|
})
|
|
wizard.sourceServer.OnFocusLost(func(_ *ui.TextInput) {
|
|
src := wizard.sourceServer.String()
|
|
out := wizard.outgoingServer.String()
|
|
if out == "" && strings.HasPrefix(src, "imap.") {
|
|
out = strings.Replace(src, "imap.", "smtp.", 1)
|
|
wizard.outgoingServer.Set(out)
|
|
}
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.sourceUsername.OnChange(func(_ *ui.TextInput) {
|
|
wizard.sourceUri()
|
|
})
|
|
wizard.sourceUsername.OnFocusLost(func(_ *ui.TextInput) {
|
|
if wizard.outgoingUsername.String() == "" {
|
|
wizard.outgoingUsername.Set(wizard.sourceUsername.String())
|
|
wizard.outgoingUri()
|
|
}
|
|
})
|
|
wizard.sourceTransport.OnSelect(func(option string) {
|
|
wizard.sourceUri()
|
|
})
|
|
var once sync.Once
|
|
wizard.sourcePassword.OnChange(func(_ *ui.TextInput) {
|
|
wizard.outgoingPassword.Set(wizard.sourcePassword.String())
|
|
wizard.sourceUri()
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.sourcePassword.OnFocusLost(func(_ *ui.TextInput) {
|
|
if wizard.sourcePassword.String() != "" {
|
|
once.Do(func() {
|
|
showPasswordWarning()
|
|
})
|
|
}
|
|
})
|
|
wizard.outgoingProtocol.OnSelect(func(option string) {
|
|
wizard.outgoingServer.Set("")
|
|
wizard.autofill()
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.outgoingServer.OnChange(func(_ *ui.TextInput) {
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.outgoingUsername.OnChange(func(_ *ui.TextInput) {
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.outgoingPassword.OnChange(func(_ *ui.TextInput) {
|
|
if wizard.outgoingPassword.String() != "" {
|
|
once.Do(func() {
|
|
showPasswordWarning()
|
|
})
|
|
}
|
|
wizard.outgoingUri()
|
|
})
|
|
wizard.outgoingTransport.OnSelect(func(option string) {
|
|
wizard.outgoingUri()
|
|
})
|
|
|
|
// CONFIGURE_BASICS
|
|
basics := NewConfigStep(
|
|
`
|
|
Welcome to aerc! Let's configure your account.
|
|
|
|
Key bindings:
|
|
|
|
<Tab>, <Down> or <Ctrl+j> Next field
|
|
<Shift+Tab>, <Up> or <Ctrl+k> Previous field
|
|
<Ctrl+q> Exit aerc
|
|
`,
|
|
&wizard.basics,
|
|
)
|
|
basics.AddField(
|
|
"Name for this account? (e.g. 'Personal' or 'Work')",
|
|
wizard.accountName,
|
|
)
|
|
basics.AddField(
|
|
"Full name for outgoing emails? (e.g. 'John Doe')",
|
|
wizard.fullName,
|
|
)
|
|
basics.AddField(
|
|
"Your email address? (e.g. 'john@example.org')",
|
|
wizard.email,
|
|
)
|
|
basics.AddField("", NewSelector([]string{"Next"}, 0, config.Ui).
|
|
OnChoose(func(option string) {
|
|
wizard.discoverServices()
|
|
wizard.autofill()
|
|
wizard.sourceUri()
|
|
wizard.outgoingUri()
|
|
wizard.advance(option)
|
|
}),
|
|
)
|
|
|
|
// CONFIGURE_SOURCE
|
|
source := NewConfigStep("Configure email source", &wizard.source)
|
|
source.AddField("Protocol", wizard.sourceProtocol)
|
|
source.AddField("Username", wizard.sourceUsername)
|
|
source.AddField("Password", wizard.sourcePassword)
|
|
source.AddField(
|
|
"Server address (or path to email store)",
|
|
wizard.sourceServer,
|
|
)
|
|
source.AddField("Transport security", wizard.sourceTransport)
|
|
source.AddField("Connection URL", wizard.sourceStr)
|
|
source.AddField(
|
|
"", NewSelector([]string{"Previous", "Next"}, 1, config.Ui).
|
|
OnChoose(wizard.advance),
|
|
)
|
|
|
|
// CONFIGURE_OUTGOING
|
|
outgoing := NewConfigStep("Configure outgoing mail", &wizard.outgoing)
|
|
outgoing.AddField("Protocol", wizard.outgoingProtocol)
|
|
outgoing.AddField("Username", wizard.outgoingUsername)
|
|
outgoing.AddField("Password", wizard.outgoingPassword)
|
|
outgoing.AddField(
|
|
"Server address (or path to sendmail)",
|
|
wizard.outgoingServer,
|
|
)
|
|
outgoing.AddField("Transport security", wizard.outgoingTransport)
|
|
outgoing.AddField("Connection URL", wizard.outgoingStr)
|
|
outgoing.AddField(
|
|
"Copy sent messages to folder (leave empty to disable)",
|
|
wizard.outgoingCopyTo,
|
|
)
|
|
outgoing.AddField(
|
|
"", NewSelector([]string{"Previous", "Next"}, 1, config.Ui).
|
|
OnChoose(wizard.advance),
|
|
)
|
|
|
|
// CONFIGURE_COMPLETE
|
|
complete := NewConfigStep(
|
|
fmt.Sprintf(`
|
|
Configuration complete!
|
|
|
|
You can go back and double check your settings, or choose [Finish] to
|
|
save your settings to %s/accounts.conf.
|
|
|
|
Make sure to review the contents of this file and read the
|
|
aerc-accounts(5) man page for guidance and further tweaking.
|
|
|
|
To add another account in the future, run ':new-account'.
|
|
`, xdg.TildeHome(xdg.ConfigPath("aerc"))),
|
|
&wizard.complete,
|
|
)
|
|
complete.AddField(
|
|
"", NewSelector([]string{
|
|
"Previous",
|
|
"Finish & open tutorial",
|
|
"Finish",
|
|
}, 1, config.Ui).OnChoose(func(option string) {
|
|
switch option {
|
|
case "Previous":
|
|
wizard.advance("Previous")
|
|
case "Finish & open tutorial":
|
|
wizard.finish(true)
|
|
case "Finish":
|
|
wizard.finish(false)
|
|
}
|
|
}),
|
|
)
|
|
|
|
wizard.steps = []*ui.Grid{
|
|
basics.Grid(), source.Grid(), outgoing.Grid(), complete.Grid(),
|
|
}
|
|
|
|
return wizard
|
|
}
|
|
|
|
func (wizard *AccountWizard) ConfigureTemporaryAccount(temporary bool) {
|
|
wizard.temporary = temporary
|
|
}
|
|
|
|
func (wizard *AccountWizard) errorFor(d ui.Interactive, err error) {
|
|
if d == nil {
|
|
PushError(err.Error())
|
|
wizard.Invalidate()
|
|
return
|
|
}
|
|
for step, interactives := range [][]ui.Interactive{
|
|
wizard.basics,
|
|
wizard.source,
|
|
wizard.outgoing,
|
|
} {
|
|
for focus, item := range interactives {
|
|
if item == d {
|
|
wizard.Focus(false)
|
|
wizard.step = step
|
|
wizard.focus = focus
|
|
wizard.Focus(true)
|
|
PushError(err.Error())
|
|
wizard.Invalidate()
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (wizard *AccountWizard) finish(tutorial bool) {
|
|
accountsConf := xdg.ConfigPath("aerc", "accounts.conf")
|
|
|
|
// Validation
|
|
if wizard.accountName.String() == "" {
|
|
wizard.errorFor(wizard.accountName,
|
|
errors.New("Account name is required"))
|
|
return
|
|
}
|
|
if wizard.email.String() == "" {
|
|
wizard.errorFor(wizard.email,
|
|
errors.New("Email address is required"))
|
|
return
|
|
}
|
|
if wizard.sourceServer.String() == "" {
|
|
wizard.errorFor(wizard.sourceServer,
|
|
errors.New("Email source configuration is required"))
|
|
return
|
|
}
|
|
if wizard.outgoingServer.String() == "" &&
|
|
wizard.outgoingProtocol.Selected() != JMAP {
|
|
wizard.errorFor(wizard.outgoingServer,
|
|
errors.New("Outgoing mail configuration is required"))
|
|
return
|
|
}
|
|
switch wizard.sourceProtocol.Selected() {
|
|
case MAILDIR, MAILDIRPP, NOTMUCH:
|
|
path := xdg.ExpandHome(wizard.sourceServer.String())
|
|
s, err := os.Stat(path)
|
|
if err == nil && !s.IsDir() {
|
|
err = fmt.Errorf("%s: Not a directory", s.Name())
|
|
}
|
|
if err == nil {
|
|
err = unix.Access(path, unix.X_OK)
|
|
}
|
|
if err != nil {
|
|
wizard.errorFor(wizard.sourceServer, err)
|
|
return
|
|
}
|
|
}
|
|
if wizard.outgoingProtocol.Selected() == SENDMAIL {
|
|
path := xdg.ExpandHome(wizard.outgoingServer.String())
|
|
s, err := os.Stat(path)
|
|
if err == nil && !s.Mode().IsRegular() {
|
|
err = fmt.Errorf("%s: Not a regular file", s.Name())
|
|
}
|
|
if err == nil {
|
|
err = unix.Access(path, unix.X_OK)
|
|
}
|
|
if err != nil {
|
|
wizard.errorFor(wizard.outgoingServer, err)
|
|
return
|
|
}
|
|
}
|
|
|
|
file, err := ini.Load(accountsConf)
|
|
if err != nil {
|
|
file = ini.Empty()
|
|
}
|
|
|
|
var sec *ini.Section
|
|
if sec, _ = file.GetSection(wizard.accountName.String()); sec != nil {
|
|
wizard.errorFor(wizard.accountName,
|
|
errors.New("An account by this name already exists"))
|
|
return
|
|
}
|
|
sec, _ = file.NewSection(wizard.accountName.String())
|
|
// these can't fail
|
|
_, _ = sec.NewKey("source", wizard.sourceUrl.String())
|
|
_, _ = sec.NewKey("outgoing", wizard.outgoingUrl.String())
|
|
_, _ = sec.NewKey("default", "INBOX")
|
|
from := mail.Address{
|
|
Name: wizard.fullName.String(),
|
|
Address: wizard.email.String(),
|
|
}
|
|
_, _ = sec.NewKey("from", format.AddressForHumans(&from))
|
|
if wizard.outgoingCopyTo.String() != "" {
|
|
_, _ = sec.NewKey("copy-to", wizard.outgoingCopyTo.String())
|
|
}
|
|
|
|
switch wizard.sourceProtocol.Selected() {
|
|
case IMAP:
|
|
_, _ = sec.NewKey("cache-headers", "true")
|
|
case JMAP:
|
|
_, _ = sec.NewKey("use-labels", "true")
|
|
_, _ = sec.NewKey("cache-state", "true")
|
|
_, _ = sec.NewKey("cache-blobs", "false")
|
|
case NOTMUCH:
|
|
cmd := exec.Command("notmuch", "config", "get", "database.mail_root")
|
|
out, err := cmd.Output()
|
|
if err == nil {
|
|
root := strings.TrimSpace(string(out))
|
|
_, _ = sec.NewKey("maildir-store", xdg.TildeHome(root))
|
|
}
|
|
querymap := ini.Empty()
|
|
def := querymap.Section("")
|
|
cmd = exec.Command("notmuch", "config", "list")
|
|
out, err = cmd.Output()
|
|
if err == nil {
|
|
re := regexp.MustCompile(`(?m)^query\.([^=]+)=(.+)$`)
|
|
for _, m := range re.FindAllStringSubmatch(string(out), -1) {
|
|
_, _ = def.NewKey(m[1], m[2])
|
|
}
|
|
}
|
|
if len(def.Keys()) == 0 {
|
|
_, _ = def.NewKey("INBOX", "tag:inbox and not tag:archived")
|
|
}
|
|
if !wizard.temporary {
|
|
qmapPath := xdg.ConfigPath("aerc",
|
|
wizard.accountName.String()+".qmap")
|
|
f, err := os.OpenFile(qmapPath, os.O_WRONLY|os.O_CREATE, 0o600)
|
|
if err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
defer f.Close()
|
|
if _, err = querymap.WriteTo(f); err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
_, _ = sec.NewKey("query-map", xdg.TildeHome(qmapPath))
|
|
}
|
|
}
|
|
|
|
if !wizard.temporary {
|
|
f, err := os.OpenFile(accountsConf, os.O_WRONLY|os.O_CREATE, 0o600)
|
|
if err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
defer f.Close()
|
|
if _, err = file.WriteTo(f); err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
}
|
|
|
|
account, err := config.ParseAccountConfig(sec.Name(), sec)
|
|
if err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
config.Accounts = append(config.Accounts, account)
|
|
|
|
view, err := NewAccountView(account, nil)
|
|
if err != nil {
|
|
NewTab(errorScreen(err.Error()), account.Name)
|
|
return
|
|
}
|
|
aerc.accounts[account.Name] = view
|
|
NewTab(view, account.Name)
|
|
|
|
if tutorial {
|
|
name := "aerc-tutorial"
|
|
if _, err := os.Stat("./aerc-tutorial.7"); !os.IsNotExist(err) {
|
|
// For development
|
|
name = "./aerc-tutorial.7"
|
|
}
|
|
term, err := NewTerminal(exec.Command("man", name))
|
|
if err != nil {
|
|
wizard.errorFor(nil, err)
|
|
return
|
|
}
|
|
NewTab(term, "Tutorial")
|
|
term.OnClose = func(err error) {
|
|
RemoveTab(term, false)
|
|
if err != nil {
|
|
PushError(err.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
RemoveTab(wizard, false)
|
|
}
|
|
|
|
func splitHostPath(server string) (string, string) {
|
|
host, path, found := strings.Cut(server, "/")
|
|
if found {
|
|
path = "/" + path
|
|
}
|
|
return host, path
|
|
}
|
|
|
|
func makeURLs(scheme, host, path, user, pass string) (url.URL, url.URL) {
|
|
var opaque string
|
|
|
|
// If everything is unset, the rendered URL is '<scheme>:'.
|
|
// Force a '//' opaque suffix so that it is rendered as '<scheme>://'.
|
|
if scheme != "" && host == "" && path == "" && user == "" && pass == "" {
|
|
opaque = "//"
|
|
}
|
|
|
|
uri := url.URL{Scheme: scheme, Host: host, Path: path, Opaque: opaque}
|
|
clean := uri
|
|
|
|
switch {
|
|
case pass != "":
|
|
uri.User = url.UserPassword(user, pass)
|
|
clean.User = url.UserPassword(user, strings.Repeat("*", len(pass)))
|
|
case user != "":
|
|
uri.User = url.User(user)
|
|
clean.User = url.User(user)
|
|
}
|
|
|
|
return uri, clean
|
|
}
|
|
|
|
func (wizard *AccountWizard) sourceUri() url.URL {
|
|
host, path := splitHostPath(wizard.sourceServer.String())
|
|
user := wizard.sourceUsername.String()
|
|
pass := wizard.sourcePassword.String()
|
|
var scheme string
|
|
switch wizard.sourceProtocol.Selected() {
|
|
case IMAP:
|
|
switch wizard.sourceTransport.Selected() {
|
|
case STARTTLS:
|
|
scheme = "imap"
|
|
case INSECURE:
|
|
scheme = "imap+insecure"
|
|
case OAUTH:
|
|
scheme = "imaps+oauthbearer"
|
|
case XOAUTH:
|
|
scheme = "imaps+xoauth2"
|
|
default:
|
|
scheme = "imaps"
|
|
}
|
|
case JMAP:
|
|
switch wizard.sourceTransport.Selected() {
|
|
case OAUTH:
|
|
scheme = "jmap+oauthbearer"
|
|
default:
|
|
scheme = "jmap"
|
|
}
|
|
case MAILDIR:
|
|
scheme = "maildir"
|
|
case MAILDIRPP:
|
|
scheme = "maildirpp"
|
|
case NOTMUCH:
|
|
scheme = "notmuch"
|
|
}
|
|
switch wizard.sourceProtocol.Selected() {
|
|
case MAILDIR, MAILDIRPP, NOTMUCH:
|
|
path = host + path
|
|
host = ""
|
|
user = ""
|
|
pass = ""
|
|
}
|
|
|
|
uri, clean := makeURLs(scheme, host, path, user, pass)
|
|
|
|
wizard.sourceStr.Text(
|
|
" " + strings.ReplaceAll(clean.String(), "%2A", "*"))
|
|
wizard.sourceUrl = uri
|
|
return uri
|
|
}
|
|
|
|
func (wizard *AccountWizard) outgoingUri() url.URL {
|
|
host, path := splitHostPath(wizard.outgoingServer.String())
|
|
user := wizard.outgoingUsername.String()
|
|
pass := wizard.outgoingPassword.String()
|
|
var scheme string
|
|
switch wizard.outgoingProtocol.Selected() {
|
|
case SMTP:
|
|
switch wizard.outgoingTransport.Selected() {
|
|
case OAUTH:
|
|
scheme = "smtps+oauthbearer"
|
|
case XOAUTH:
|
|
scheme = "smtps+xoauth2"
|
|
case INSECURE:
|
|
scheme = "smtp+insecure"
|
|
case STARTTLS:
|
|
scheme = "smtp"
|
|
default:
|
|
scheme = "smtps"
|
|
}
|
|
case JMAP:
|
|
switch wizard.outgoingTransport.Selected() {
|
|
case OAUTH:
|
|
scheme = "jmap+oauthbearer"
|
|
default:
|
|
scheme = "jmap"
|
|
}
|
|
case SENDMAIL:
|
|
scheme = ""
|
|
path = host + path
|
|
host = ""
|
|
user = ""
|
|
pass = ""
|
|
}
|
|
|
|
uri, clean := makeURLs(scheme, host, path, user, pass)
|
|
|
|
wizard.outgoingStr.Text(
|
|
" " + strings.ReplaceAll(clean.String(), "%2A", "*"))
|
|
wizard.outgoingUrl = uri
|
|
return uri
|
|
}
|
|
|
|
func (wizard *AccountWizard) Invalidate() {
|
|
ui.Invalidate()
|
|
}
|
|
|
|
func (wizard *AccountWizard) Draw(ctx *ui.Context) {
|
|
wizard.steps[wizard.step].Draw(ctx)
|
|
}
|
|
|
|
func (wizard *AccountWizard) getInteractive() []ui.Interactive {
|
|
switch wizard.step {
|
|
case CONFIGURE_BASICS:
|
|
return wizard.basics
|
|
case CONFIGURE_SOURCE:
|
|
return wizard.source
|
|
case CONFIGURE_OUTGOING:
|
|
return wizard.outgoing
|
|
case CONFIGURE_COMPLETE:
|
|
return wizard.complete
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (wizard *AccountWizard) advance(direction string) {
|
|
wizard.Focus(false)
|
|
if direction == "Next" && wizard.step < len(wizard.steps)-1 {
|
|
wizard.step++
|
|
}
|
|
if direction == "Previous" && wizard.step > 0 {
|
|
wizard.step--
|
|
}
|
|
wizard.focus = 0
|
|
wizard.Focus(true)
|
|
wizard.Invalidate()
|
|
}
|
|
|
|
func (wizard *AccountWizard) Focus(focus bool) {
|
|
if interactive := wizard.getInteractive(); interactive != nil {
|
|
interactive[wizard.focus].Focus(focus)
|
|
}
|
|
}
|
|
|
|
func (wizard *AccountWizard) Event(event vaxis.Event) bool {
|
|
interactive := wizard.getInteractive()
|
|
if key, ok := event.(vaxis.Key); ok {
|
|
switch {
|
|
case key.Matches('k', vaxis.ModCtrl),
|
|
key.Matches(vaxis.KeyTab, vaxis.ModShift),
|
|
key.Matches(vaxis.KeyUp):
|
|
if interactive != nil {
|
|
interactive[wizard.focus].Focus(false)
|
|
wizard.focus--
|
|
if wizard.focus < 0 {
|
|
wizard.focus = len(interactive) - 1
|
|
}
|
|
interactive[wizard.focus].Focus(true)
|
|
}
|
|
wizard.Invalidate()
|
|
return true
|
|
case key.Matches('j', vaxis.ModCtrl),
|
|
key.Matches(vaxis.KeyTab),
|
|
key.Matches(vaxis.KeyDown):
|
|
|
|
if interactive != nil {
|
|
interactive[wizard.focus].Focus(false)
|
|
wizard.focus++
|
|
if wizard.focus >= len(interactive) {
|
|
wizard.focus = 0
|
|
}
|
|
interactive[wizard.focus].Focus(true)
|
|
}
|
|
wizard.Invalidate()
|
|
return true
|
|
}
|
|
}
|
|
if interactive != nil {
|
|
return interactive[wizard.focus].Event(event)
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (wizard *AccountWizard) discoverServices() {
|
|
email := wizard.email.String()
|
|
if !strings.ContainsRune(email, '@') {
|
|
return
|
|
}
|
|
domain := email[strings.IndexRune(email, '@')+1:]
|
|
var wg sync.WaitGroup
|
|
type Service struct{ srv, hostport string }
|
|
services := make(chan Service)
|
|
|
|
for _, service := range []string{"imaps", "imap", "submission", "jmap"} {
|
|
wg.Add(1)
|
|
go func(srv string) {
|
|
defer log.PanicHandler()
|
|
defer wg.Done()
|
|
_, addrs, err := net.LookupSRV(srv, "tcp", domain)
|
|
if err != nil {
|
|
log.Tracef("SRV lookup for _%s._tcp.%s failed: %s",
|
|
srv, domain, err)
|
|
} else if addrs[0].Target != "" && addrs[0].Port > 0 {
|
|
services <- Service{
|
|
srv: srv,
|
|
hostport: net.JoinHostPort(
|
|
strings.TrimSuffix(addrs[0].Target, "."),
|
|
strconv.Itoa(int(addrs[0].Port))),
|
|
}
|
|
}
|
|
}(service)
|
|
}
|
|
go func() {
|
|
defer log.PanicHandler()
|
|
wg.Wait()
|
|
close(services)
|
|
}()
|
|
|
|
wizard.discovered = make(map[string]string)
|
|
for s := range services {
|
|
wizard.discovered[s.srv] = s.hostport
|
|
}
|
|
}
|
|
|
|
func (wizard *AccountWizard) autofill() {
|
|
if wizard.sourceServer.String() == "" {
|
|
switch wizard.sourceProtocol.Selected() {
|
|
case IMAP:
|
|
if s, ok := wizard.discovered["imaps"]; ok {
|
|
wizard.sourceServer.Set(s)
|
|
wizard.sourceTransport.Select(SSL_TLS)
|
|
} else if s, ok := wizard.discovered["imap"]; ok {
|
|
wizard.sourceServer.Set(s)
|
|
wizard.sourceTransport.Select(STARTTLS)
|
|
}
|
|
case JMAP:
|
|
if s, ok := wizard.discovered["jmap"]; ok {
|
|
s = strings.TrimSuffix(s, ":443")
|
|
wizard.sourceServer.Set(s + "/.well-known/jmap")
|
|
wizard.sourceTransport.Select(SSL_TLS)
|
|
}
|
|
case MAILDIR, MAILDIRPP:
|
|
wizard.sourceServer.Set("~/mail")
|
|
wizard.sourceUsername.Set("")
|
|
wizard.sourcePassword.Set("")
|
|
case NOTMUCH:
|
|
cmd := exec.Command("notmuch", "config", "get", "database.path")
|
|
out, err := cmd.Output()
|
|
if err == nil {
|
|
db := strings.TrimSpace(string(out))
|
|
wizard.sourceServer.Set(xdg.TildeHome(db))
|
|
} else {
|
|
wizard.sourceServer.Set("~/mail")
|
|
}
|
|
wizard.sourceUsername.Set("")
|
|
wizard.sourcePassword.Set("")
|
|
}
|
|
}
|
|
if wizard.outgoingServer.String() == "" {
|
|
switch wizard.outgoingProtocol.Selected() {
|
|
case SMTP:
|
|
if s, ok := wizard.discovered["submission"]; ok {
|
|
switch {
|
|
case strings.HasSuffix(s, ":587"):
|
|
wizard.outgoingTransport.Select(SSL_TLS)
|
|
case strings.HasSuffix(s, ":465"):
|
|
wizard.outgoingTransport.Select(STARTTLS)
|
|
default:
|
|
wizard.outgoingTransport.Select(INSECURE)
|
|
}
|
|
wizard.outgoingServer.Set(s)
|
|
}
|
|
case JMAP:
|
|
wizard.outgoingTransport.Select(SSL_TLS)
|
|
case SENDMAIL:
|
|
wizard.outgoingServer.Set("/usr/sbin/sendmail")
|
|
wizard.outgoingUsername.Set("")
|
|
wizard.outgoingPassword.Set("")
|
|
}
|
|
}
|
|
}
|