1
0
mirror of https://github.com/osmarks/mycorrhiza.git synced 2024-12-12 13:30:26 +00:00
mycorrhiza/auth/web.go

226 lines
6.1 KiB
Go
Raw Normal View History

package auth
2020-11-14 13:03:06 +00:00
import (
2021-07-14 19:51:55 +00:00
"errors"
"fmt"
"io"
2020-11-14 13:03:06 +00:00
"log"
"mime"
2020-11-14 13:03:06 +00:00
"net/http"
2021-07-14 19:51:55 +00:00
"strings"
2020-11-14 13:03:06 +00:00
"github.com/bouncepaw/mycorrhiza/viewutil"
"github.com/gorilla/mux"
"github.com/bouncepaw/mycorrhiza/cfg"
2021-09-06 17:46:34 +00:00
"github.com/bouncepaw/mycorrhiza/l18n"
2020-11-14 13:03:06 +00:00
"github.com/bouncepaw/mycorrhiza/user"
2021-02-17 18:41:35 +00:00
"github.com/bouncepaw/mycorrhiza/util"
2020-11-14 13:03:06 +00:00
)
func InitAuth(r *mux.Router) {
2022-04-02 07:22:26 +00:00
r.HandleFunc("/user-list", handlerUserList)
r.HandleFunc("/lock", handlerLock)
// The check below saves a lot of extra checks and lines of codes in other places in this file.
if !cfg.UseAuth {
return
}
if cfg.AllowRegistration {
r.HandleFunc("/register", handlerRegister).Methods(http.MethodPost, http.MethodGet)
}
2021-07-14 19:51:55 +00:00
if cfg.TelegramEnabled {
r.HandleFunc("/telegram-login", handlerTelegramLogin)
2021-07-14 19:51:55 +00:00
}
r.HandleFunc("/login", handlerLogin)
r.HandleFunc("/logout", handlerLogout)
2020-11-14 13:03:06 +00:00
}
2022-04-02 07:22:26 +00:00
func handlerUserList(w http.ResponseWriter, rq *http.Request) {
lc := l18n.FromRequest(rq)
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
w.WriteHeader(http.StatusOK)
w.Write([]byte(viewutil.Base(viewutil.MetaFrom(w, rq), lc.Get("ui.users_title"), UserList(lc), map[string]string{})))
2022-04-02 07:22:26 +00:00
}
func handlerLock(w http.ResponseWriter, rq *http.Request) {
_, _ = io.WriteString(w, Lock(l18n.FromRequest(rq)))
}
// handlerRegister displays the register form (GET) or registers the user (POST).
func handlerRegister(w http.ResponseWriter, rq *http.Request) {
2021-09-06 17:46:34 +00:00
lc := l18n.FromRequest(rq)
util.PrepareRq(rq)
if rq.Method == http.MethodGet {
_, _ = io.WriteString(
w,
viewutil.Base(
2022-04-01 19:51:15 +00:00
viewutil.MetaFrom(w, rq),
2021-09-06 17:46:34 +00:00
lc.Get("auth.register_title"),
Register(rq),
map[string]string{},
),
)
return
}
var (
username = rq.PostFormValue("username")
password = rq.PostFormValue("password")
err = user.Register(username, password, "editor", "local", false)
)
if err != nil {
log.Printf("Failed to register %s: %s", username, err.Error())
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
w.WriteHeader(http.StatusBadRequest)
_, _ = io.WriteString(
w,
viewutil.Base(
viewutil.MetaFrom(w, rq),
lc.Get("auth.register_title"),
fmt.Sprintf(
`<main class="main-width"><p>%s</p><p><a href="/register">%s<a></p></main>`,
err.Error(),
lc.Get("auth.try_again"),
),
map[string]string{},
),
)
return
}
log.Printf("Successfully registered %s", username)
if err := user.LoginDataHTTP(w, username, password); err != nil {
return
}
http.Redirect(w, rq, "/"+rq.URL.RawQuery, http.StatusSeeOther)
}
// handlerLogout shows the logout form (GET) or logs the user out (POST).
2020-11-14 13:03:06 +00:00
func handlerLogout(w http.ResponseWriter, rq *http.Request) {
if rq.Method == http.MethodGet {
var (
u = user.FromRequest(rq)
can = u != nil
lc = l18n.FromRequest(rq)
)
w.Header().Set("Content-Type", "text/html;charset=utf-8")
if can {
log.Println("User", u.Name, "tries to log out")
w.WriteHeader(http.StatusOK)
} else {
log.Println("Unknown user tries to log out")
w.WriteHeader(http.StatusForbidden)
}
_, _ = io.WriteString(
w,
viewutil.Base(viewutil.MetaFrom(w, rq), lc.Get("auth.logout_title"), Logout(can, lc), map[string]string{}),
)
} else if rq.Method == http.MethodPost {
user.LogoutFromRequest(w, rq)
http.Redirect(w, rq, "/", http.StatusSeeOther)
2020-11-14 13:03:06 +00:00
}
}
// handlerLogin shows the login form (GET) or logs the user in (POST).
func handlerLogin(w http.ResponseWriter, rq *http.Request) {
lc := l18n.FromRequest(rq)
if rq.Method == http.MethodGet {
w.Header().Set("Content-Type", "text/html;charset=utf-8")
w.WriteHeader(http.StatusOK)
_, _ = io.WriteString(
w,
viewutil.Base(
2022-04-01 19:51:15 +00:00
viewutil.MetaFrom(w, rq),
lc.Get("auth.login_title"),
Login(lc),
map[string]string{},
),
)
} else if rq.Method == http.MethodPost {
var (
username = util.CanonicalName(rq.PostFormValue("username"))
password = rq.PostFormValue("password")
err = user.LoginDataHTTP(w, username, password)
)
if err != nil {
w.Header().Set("Content-Type", "text/html;charset=utf-8")
w.WriteHeader(http.StatusInternalServerError)
_, _ = io.WriteString(w, viewutil.Base(viewutil.MetaFrom(w, rq), err.Error(), LoginError(err.Error(), lc), map[string]string{}))
return
}
http.Redirect(w, rq, "/", http.StatusSeeOther)
}
}
2021-07-14 19:51:55 +00:00
func handlerTelegramLogin(w http.ResponseWriter, rq *http.Request) {
// Note there is no lock here.
2021-09-06 17:46:34 +00:00
lc := l18n.FromRequest(rq)
2021-07-14 20:04:52 +00:00
w.Header().Set("Content-Type", "text/html;charset=utf-8")
2021-07-14 19:51:55 +00:00
rq.ParseForm()
var (
values = rq.URL.Query()
username = strings.ToLower(values.Get("username"))
2021-07-14 19:51:55 +00:00
seemsValid = user.TelegramAuthParamsAreValid(values)
err = user.Register(
2021-07-14 19:51:55 +00:00
username,
"", // Password matters not
2021-07-14 21:00:35 +00:00
"editor",
2021-07-14 19:51:55 +00:00
"telegram",
false,
)
)
// If registering a user via Telegram failed, because a Telegram user with this name
// has already registered, then everything is actually ok!
if user.HasUsername(username) && user.ByName(username).Source == "telegram" {
2021-07-14 19:51:55 +00:00
err = nil
}
if !seemsValid {
err = errors.New("Wrong parameters")
}
if err != nil {
log.Printf("Failed to register %s using Telegram: %s", username, err.Error())
w.WriteHeader(http.StatusBadRequest)
_, _ = io.WriteString(
2021-07-14 19:51:55 +00:00
w,
viewutil.Base(
2022-04-01 19:51:15 +00:00
viewutil.MetaFrom(w, rq),
2021-09-06 17:46:34 +00:00
lc.Get("ui.error"),
2021-07-14 19:51:55 +00:00
fmt.Sprintf(
2021-09-06 17:46:34 +00:00
`<main class="main-width"><p>%s</p><p>%s</p><p><a href="/login">%s<a></p></main>`,
lc.Get("auth.error_telegram"),
2021-07-14 19:51:55 +00:00
err.Error(),
2021-09-06 17:46:34 +00:00
lc.Get("auth.go_login"),
2021-07-14 19:51:55 +00:00
),
map[string]string{},
2021-07-14 19:51:55 +00:00
),
)
return
}
errmsg := user.LoginDataHTTP(w, username, "")
if errmsg != nil {
2021-07-14 19:51:55 +00:00
log.Printf("Failed to login %s using Telegram: %s", username, err.Error())
w.WriteHeader(http.StatusBadRequest)
_, _ = io.WriteString(
2021-07-14 19:51:55 +00:00
w,
viewutil.Base(
2022-04-01 19:51:15 +00:00
viewutil.MetaFrom(w, rq),
2021-07-14 19:51:55 +00:00
"Error",
fmt.Sprintf(
2021-09-06 17:46:34 +00:00
`<main class="main-width"><p>%s</p><p>%s</p><p><a href="/login">%s<a></p></main>`,
lc.Get("auth.error_telegram"),
2021-07-14 19:51:55 +00:00
err.Error(),
2021-09-06 17:46:34 +00:00
lc.Get("auth.go_login"),
2021-07-14 19:51:55 +00:00
),
map[string]string{},
2021-07-14 19:51:55 +00:00
),
)
return
}
log.Printf("Authorize %s from Telegram", username)
http.Redirect(w, rq, "/", http.StatusSeeOther)
}