2023-01-30 16:27:06 +03:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/base64"
|
|
|
|
"errors"
|
|
|
|
"io"
|
2023-02-13 15:53:47 +03:00
|
|
|
"net"
|
2023-01-30 16:27:06 +03:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strings"
|
2023-02-13 15:53:47 +03:00
|
|
|
"time"
|
2023-01-30 16:27:06 +03:00
|
|
|
|
|
|
|
_ "embed"
|
|
|
|
|
|
|
|
"github.com/mjl-/sherpa"
|
|
|
|
"github.com/mjl-/sherpaprom"
|
|
|
|
|
2023-02-11 01:47:19 +03:00
|
|
|
"github.com/mjl-/mox/config"
|
|
|
|
"github.com/mjl-/mox/dns"
|
2023-01-30 16:27:06 +03:00
|
|
|
"github.com/mjl-/mox/metrics"
|
|
|
|
"github.com/mjl-/mox/mlog"
|
|
|
|
"github.com/mjl-/mox/mox-"
|
|
|
|
"github.com/mjl-/mox/moxvar"
|
|
|
|
"github.com/mjl-/mox/store"
|
|
|
|
)
|
|
|
|
|
|
|
|
//go:embed accountapi.json
|
|
|
|
var accountapiJSON []byte
|
|
|
|
|
|
|
|
//go:embed account.html
|
|
|
|
var accountHTML []byte
|
|
|
|
|
|
|
|
var accountDoc = mustParseAPI(accountapiJSON)
|
|
|
|
|
|
|
|
var accountSherpaHandler http.Handler
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
collector, err := sherpaprom.NewCollector("moxaccount", nil)
|
|
|
|
if err != nil {
|
|
|
|
xlog.Fatalx("creating sherpa prometheus collector", err)
|
|
|
|
}
|
|
|
|
|
2023-02-13 15:53:47 +03:00
|
|
|
accountSherpaHandler, err = sherpa.NewHandler("/api/", moxvar.Version, Account{}, &accountDoc, &sherpa.HandlerOpts{Collector: collector, AdjustFunctionNames: "none"})
|
2023-01-30 16:27:06 +03:00
|
|
|
if err != nil {
|
|
|
|
xlog.Fatalx("sherpa handler", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Account exports web API functions for the account web interface. All its
|
2023-02-13 15:53:47 +03:00
|
|
|
// methods are exported under /api/. Function calls require valid HTTP
|
2023-01-30 16:27:06 +03:00
|
|
|
// Authentication credentials of a user.
|
|
|
|
type Account struct{}
|
|
|
|
|
2023-02-13 15:53:47 +03:00
|
|
|
// check http basic auth, returns account name if valid, and writes http response
|
|
|
|
// and returns empty string otherwise.
|
|
|
|
func checkAccountAuth(ctx context.Context, log *mlog.Log, w http.ResponseWriter, r *http.Request) string {
|
2023-01-30 16:27:06 +03:00
|
|
|
authResult := "error"
|
2023-02-13 15:53:47 +03:00
|
|
|
start := time.Now()
|
|
|
|
var addr *net.TCPAddr
|
2023-01-30 16:27:06 +03:00
|
|
|
defer func() {
|
|
|
|
metrics.AuthenticationInc("httpaccount", "httpbasic", authResult)
|
2023-02-13 15:53:47 +03:00
|
|
|
if authResult == "ok" && addr != nil {
|
|
|
|
mox.LimiterFailedAuth.Reset(addr.IP, start)
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
}()
|
2023-02-13 15:53:47 +03:00
|
|
|
|
|
|
|
var err error
|
|
|
|
addr, err = net.ResolveTCPAddr("tcp", r.RemoteAddr)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorx("parsing remote address", err, mlog.Field("addr", r.RemoteAddr))
|
|
|
|
}
|
|
|
|
if addr != nil && !mox.LimiterFailedAuth.Add(addr.IP, start, 1) {
|
|
|
|
metrics.AuthenticationRatelimitedInc("httpaccount")
|
|
|
|
http.Error(w, "http 429 - too many auth attempts", http.StatusTooManyRequests)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// store.OpenEmailAuth has an auth cache, so we don't bcrypt for every auth attempt.
|
2023-01-30 16:27:06 +03:00
|
|
|
if auth := r.Header.Get("Authorization"); auth == "" || !strings.HasPrefix(auth, "Basic ") {
|
|
|
|
} else if authBuf, err := base64.StdEncoding.DecodeString(strings.TrimPrefix(auth, "Basic ")); err != nil {
|
2023-02-13 15:53:47 +03:00
|
|
|
log.Debugx("parsing base64", err)
|
2023-01-30 16:27:06 +03:00
|
|
|
} else if t := strings.SplitN(string(authBuf), ":", 2); len(t) != 2 {
|
2023-02-13 15:53:47 +03:00
|
|
|
log.Debug("bad user:pass form")
|
2023-01-30 16:27:06 +03:00
|
|
|
} else if acc, err := store.OpenEmailAuth(t[0], t[1]); err != nil {
|
|
|
|
if errors.Is(err, store.ErrUnknownCredentials) {
|
|
|
|
authResult = "badcreds"
|
|
|
|
}
|
2023-02-13 15:53:47 +03:00
|
|
|
log.Errorx("open account", err)
|
2023-01-30 16:27:06 +03:00
|
|
|
} else {
|
|
|
|
authResult = "ok"
|
2023-02-13 15:53:47 +03:00
|
|
|
accName := acc.Name
|
|
|
|
acc.Close()
|
|
|
|
return accName
|
2023-01-30 16:27:06 +03:00
|
|
|
}
|
2023-02-13 15:53:47 +03:00
|
|
|
// note: browsers don't display the realm to prevent users getting confused by malicious realm messages.
|
|
|
|
w.Header().Set("WWW-Authenticate", `Basic realm="mox account - login with email address and password"`)
|
|
|
|
http.Error(w, "http 401 - unauthorized - mox account - login with email address and password", http.StatusUnauthorized)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func accountHandle(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := context.WithValue(r.Context(), mlog.CidKey, mox.Cid())
|
|
|
|
log := xlog.WithContext(ctx).Fields(mlog.Field("userauth", ""))
|
|
|
|
|
|
|
|
accName := checkAccountAuth(ctx, log, w, r)
|
|
|
|
if accName == "" {
|
|
|
|
// Response already sent.
|
2023-01-30 16:27:06 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-02-13 15:53:47 +03:00
|
|
|
if r.Method == "GET" && r.URL.Path == "/" {
|
2023-01-30 16:27:06 +03:00
|
|
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
|
|
|
w.Header().Set("Cache-Control", "no-cache; max-age=0")
|
2023-02-06 17:26:24 +03:00
|
|
|
// We typically return the embedded admin.html, but during development it's handy
|
|
|
|
// to load from disk.
|
2023-01-30 16:27:06 +03:00
|
|
|
f, err := os.Open("http/account.html")
|
|
|
|
if err == nil {
|
|
|
|
defer f.Close()
|
|
|
|
io.Copy(w, f)
|
|
|
|
} else {
|
|
|
|
w.Write(accountHTML)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2023-02-13 15:53:47 +03:00
|
|
|
accountSherpaHandler.ServeHTTP(w, r.WithContext(context.WithValue(ctx, authCtxKey, accName)))
|
2023-01-30 16:27:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type ctxKey string
|
|
|
|
|
|
|
|
var authCtxKey ctxKey = "account"
|
|
|
|
|
|
|
|
// SetPassword saves a new password for the account, invalidating the previous password.
|
|
|
|
// Sessions are not interrupted, and will keep working. New login attempts must use the new password.
|
|
|
|
// Password must be at least 8 characters.
|
|
|
|
func (Account) SetPassword(ctx context.Context, password string) {
|
|
|
|
if len(password) < 8 {
|
|
|
|
panic(&sherpa.Error{Code: "user:error", Message: "password must be at least 8 characters"})
|
|
|
|
}
|
|
|
|
accountName := ctx.Value(authCtxKey).(string)
|
|
|
|
acc, err := store.OpenAccount(accountName)
|
|
|
|
xcheckf(ctx, err, "open account")
|
|
|
|
defer acc.Close()
|
|
|
|
err = acc.SetPassword(password)
|
|
|
|
xcheckf(ctx, err, "setting password")
|
|
|
|
}
|
2023-02-11 01:47:19 +03:00
|
|
|
|
|
|
|
// Destinations returns the default domain, and the destinations (keys are email
|
|
|
|
// addresses, or localparts to the default domain).
|
|
|
|
// todo: replace with a function that returns the whole account, when sherpadoc understands unnamed struct fields.
|
|
|
|
func (Account) Destinations(ctx context.Context) (dns.Domain, map[string]config.Destination) {
|
|
|
|
accountName := ctx.Value(authCtxKey).(string)
|
|
|
|
accConf, ok := mox.Conf.Account(accountName)
|
|
|
|
if !ok {
|
|
|
|
xcheckf(ctx, errors.New("not found"), "looking up account")
|
|
|
|
}
|
|
|
|
return accConf.DNSDomain, accConf.Destinations
|
|
|
|
}
|
|
|
|
|
|
|
|
// DestinationSave updates a destination.
|
|
|
|
// OldDest is compared against the current destination. If it does not match, an
|
|
|
|
// error is returned. Otherwise newDest is saved and the configuration reloaded.
|
|
|
|
func (Account) DestinationSave(ctx context.Context, destName string, oldDest, newDest config.Destination) {
|
|
|
|
accountName := ctx.Value(authCtxKey).(string)
|
|
|
|
accConf, ok := mox.Conf.Account(accountName)
|
|
|
|
if !ok {
|
|
|
|
xcheckf(ctx, errors.New("not found"), "looking up account")
|
|
|
|
}
|
|
|
|
curDest, ok := accConf.Destinations[destName]
|
|
|
|
if !ok {
|
|
|
|
xcheckf(ctx, errors.New("not found"), "looking up destination")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !curDest.Equal(oldDest) {
|
|
|
|
xcheckf(ctx, errors.New("modified"), "checking stored destination")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep fields we manage.
|
|
|
|
newDest.DMARCReports = curDest.DMARCReports
|
|
|
|
newDest.TLSReports = curDest.TLSReports
|
|
|
|
|
|
|
|
err := mox.DestinationSave(ctx, accountName, destName, newDest)
|
|
|
|
xcheckf(ctx, err, "saving destination")
|
|
|
|
}
|