mirror of
https://github.com/osmarks/mycorrhiza.git
synced 2025-01-08 10:51:09 +00:00
49b0a35304
They're not perfect, I still don't like them, but I can't think of a good solution right now. I'm going to thinking about the best way of doing web stuff for some time.
173 lines
4.8 KiB
Go
173 lines
4.8 KiB
Go
package web
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"io"
|
|
"log"
|
|
"mime"
|
|
"net/http"
|
|
"sort"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/bouncepaw/mycorrhiza/cfg"
|
|
"github.com/bouncepaw/mycorrhiza/user"
|
|
"github.com/bouncepaw/mycorrhiza/util"
|
|
"github.com/bouncepaw/mycorrhiza/views"
|
|
)
|
|
|
|
// initAdmin sets up /admin routes if auth is used. Call it after you have decided if you want to use auth.
|
|
func initAdmin(r *mux.Router) {
|
|
r.HandleFunc("/shutdown", handlerAdminShutdown).Methods(http.MethodPost)
|
|
r.HandleFunc("/reindex-users", handlerAdminReindexUsers).Methods(http.MethodPost)
|
|
|
|
r.HandleFunc("/user/new", handlerAdminUserNew).Methods(http.MethodGet, http.MethodPost)
|
|
r.HandleFunc("/users/{username}/edit", handlerAdminUserEdit).Methods(http.MethodGet, http.MethodPost)
|
|
r.HandleFunc("/users/{username}/delete", handlerAdminUserDelete).Methods(http.MethodGet, http.MethodPost)
|
|
r.HandleFunc("/users", handlerAdminUsers)
|
|
|
|
r.HandleFunc("/", handlerAdmin)
|
|
}
|
|
|
|
// handlerAdmin provides the admin panel.
|
|
func handlerAdmin(w http.ResponseWriter, rq *http.Request) {
|
|
w.Header().Set("Content-Type", "text/html;charset=utf-8")
|
|
w.WriteHeader(http.StatusOK)
|
|
io.WriteString(w, views.BaseHTML("Admin panel", views.AdminPanelHTML(), user.FromRequest(rq)))
|
|
}
|
|
|
|
// handlerAdminShutdown kills the wiki.
|
|
func handlerAdminShutdown(w http.ResponseWriter, rq *http.Request) {
|
|
if user.CanProceed(rq, "admin/shutdown") {
|
|
log.Println("An admin commanded the wiki to shutdown")
|
|
os.Exit(0)
|
|
}
|
|
}
|
|
|
|
// handlerAdminReindexUsers reinitialises the user system.
|
|
func handlerAdminReindexUsers(w http.ResponseWriter, rq *http.Request) {
|
|
user.ReadUsersFromFilesystem()
|
|
redirectTo := rq.Referer()
|
|
if redirectTo == "" {
|
|
redirectTo = "/hypha/" + cfg.UserHypha
|
|
}
|
|
http.Redirect(w, rq, redirectTo, http.StatusSeeOther)
|
|
}
|
|
|
|
func handlerAdminUsers(w http.ResponseWriter, rq *http.Request) {
|
|
// Get a sorted list of users
|
|
var userList []*user.User
|
|
for u := range user.YieldUsers() {
|
|
userList = append(userList, u)
|
|
}
|
|
|
|
sort.Slice(userList, func(i, j int) bool {
|
|
less := userList[i].RegisteredAt.Before(userList[j].RegisteredAt)
|
|
return less
|
|
})
|
|
|
|
html := views.AdminUsersPanelHTML(userList)
|
|
html = views.BaseHTML("Manage users", html, user.FromRequest(rq))
|
|
|
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
|
io.WriteString(w, html)
|
|
}
|
|
|
|
func handlerAdminUserEdit(w http.ResponseWriter, rq *http.Request) {
|
|
vars := mux.Vars(rq)
|
|
u := user.UserByName(vars["username"])
|
|
if u == nil {
|
|
util.HTTP404Page(w, "404 page not found")
|
|
return
|
|
}
|
|
|
|
f := util.FormDataFromRequest(rq, []string{"group"})
|
|
|
|
if rq.Method == http.MethodPost {
|
|
oldGroup := u.Group
|
|
newGroup := f.Get("group")
|
|
|
|
if user.ValidGroup(newGroup) {
|
|
u.Group = newGroup
|
|
if err := user.SaveUserDatabase(); err != nil {
|
|
u.Group = oldGroup
|
|
log.Println(err)
|
|
f = f.WithError(err)
|
|
} else {
|
|
http.Redirect(w, rq, "/admin/users/", http.StatusSeeOther)
|
|
return
|
|
}
|
|
} else {
|
|
f = f.WithError(fmt.Errorf("invalid group \"%s\"", newGroup))
|
|
}
|
|
}
|
|
|
|
f.Put("group", u.Group)
|
|
|
|
html := views.AdminUserEditHTML(u, f)
|
|
html = views.BaseHTML(fmt.Sprintf("User %s", u.Name), html, user.FromRequest(rq))
|
|
|
|
if f.HasError() {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
}
|
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
|
io.WriteString(w, html)
|
|
}
|
|
|
|
func handlerAdminUserDelete(w http.ResponseWriter, rq *http.Request) {
|
|
vars := mux.Vars(rq)
|
|
u := user.UserByName(vars["username"])
|
|
if u == nil {
|
|
util.HTTP404Page(w, "404 page not found")
|
|
return
|
|
}
|
|
|
|
f := util.NewFormData()
|
|
|
|
if rq.Method == http.MethodPost {
|
|
f = f.WithError(user.DeleteUser(u.Name))
|
|
if !f.HasError() {
|
|
http.Redirect(w, rq, "/admin/users/", http.StatusSeeOther)
|
|
} else {
|
|
log.Println(f.Error())
|
|
}
|
|
}
|
|
|
|
html := views.AdminUserDeleteHTML(u, util.NewFormData())
|
|
html = views.BaseHTML(fmt.Sprintf("User %s", u.Name), html, user.FromRequest(rq))
|
|
|
|
if f.HasError() {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
}
|
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
|
io.WriteString(w, html)
|
|
}
|
|
|
|
func handlerAdminUserNew(w http.ResponseWriter, rq *http.Request) {
|
|
if rq.Method == http.MethodGet {
|
|
// New user form
|
|
html := views.AdminUserNewHTML(util.NewFormData())
|
|
html = views.BaseHTML("New user", html, user.FromRequest(rq))
|
|
|
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
|
io.WriteString(w, html)
|
|
} else if rq.Method == http.MethodPost {
|
|
// Create a user
|
|
f := util.FormDataFromRequest(rq, []string{"name", "password", "group"})
|
|
|
|
err := user.Register(f.Get("name"), f.Get("password"), f.Get("group"), "local", true)
|
|
|
|
if err != nil {
|
|
html := views.AdminUserNewHTML(f.WithError(err))
|
|
html = views.BaseHTML("New user", html, user.FromRequest(rq))
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
|
io.WriteString(w, html)
|
|
} else {
|
|
http.Redirect(w, rq, "/admin/users/", http.StatusSeeOther)
|
|
}
|
|
}
|
|
}
|