package web import ( "html/template" "net/http" "git.snrd.de/Spaenny/goddit" "github.com/alexedwards/scs/v2" "github.com/google/uuid" "github.com/gorilla/csrf" "golang.org/x/crypto/bcrypt" ) type UserHandler struct { store goddit.Store sessions *scs.SessionManager } func (h *UserHandler) Register() http.HandlerFunc { type data struct { SessionData CSRF template.HTML } tmpl := template.Must(template.ParseFiles("templates/layout.html", "templates/user_register.html")) return func(w http.ResponseWriter, r *http.Request) { tmpl.Execute(w, data{ SessionData: GetSessionData(h.sessions, r.Context()), CSRF: csrf.TemplateField(r), }) } } func (h *UserHandler) RegisterSubmit() http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { form := RegisterForm{ Username: r.FormValue("username"), Password: r.FormValue("password"), UsernameTaken: false, } if _, err := h.store.UserByUsername(form.Username); err == nil { form.UsernameTaken = true } if !form.Validate() { h.sessions.Put(r.Context(), "form", form) http.Redirect(w, r, r.Referer(), http.StatusFound) return } password, err := bcrypt.GenerateFromPassword([]byte(form.Password), bcrypt.DefaultCost) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } if err := h.store.CreateUser(&goddit.User{ ID: uuid.New(), Username: form.Username, Password: string(password), }); err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } h.sessions.Put(r.Context(), "flash", "Your registration was successful. Please log in.") http.Redirect(w, r, "/", http.StatusFound) } } func (h *UserHandler) Login() http.HandlerFunc { type data struct { SessionData CSRF template.HTML } tmpl := template.Must(template.ParseFiles("templates/layout.html", "templates/user_login.html")) return func(w http.ResponseWriter, r *http.Request) { tmpl.Execute(w, data{ SessionData: GetSessionData(h.sessions, r.Context()), CSRF: csrf.TemplateField(r), }) } } func (h *UserHandler) LoginSubmit() http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { form := LoginForm{ Username: r.FormValue("username"), Password: r.FormValue("password"), IncorrectCredentials: false, } user, err := h.store.UserByUsername(form.Username) if err == nil { form.IncorrectCredentials = true } else { compareErr := bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(form.Password)) form.IncorrectCredentials = compareErr != nil } if !form.Validate() { h.sessions.Put(r.Context(), "form", form) http.Redirect(w, r, r.Referer(), http.StatusFound) return } h.sessions.Put(r.Context(), "user_id", user.ID) h.sessions.Put(r.Context(), "flash", "You have been logged in successfully.") http.Redirect(w, r, "/", http.StatusFound) } } func (h *UserHandler) Logout() http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { h.sessions.Remove(r.Context(), "user_id") h.sessions.Put(r.Context(), "flash", "You have been logged out successfully.") http.Redirect(w, r, "/", http.StatusFound) } }