91f70d2a4a
Added basic support for server sync. Re-added a few missing defers. Renamed TO-DO to TODO across the entire codebase. Renamed StaticForumStore to MemoryForumStore. The ForumStore is now built on a sync.Map with a view slice for generating /forums rather than a slice. Renamed many more functions and variables to satisfy the linter. increase_post_user_stats() and decrease_post_user_stats() are now methods on the User struct. We also fix a bug where they take the moderator's score rather than the target user's into account when recalculating their level after a post / topic is deleted. Transitioned the topic list to CSS Grid for Tempra Simple, with a float fallback. Cosmo and Cosmo Conflux are now hidden from the theme list. Fixed more data races. Added more debug data to the template compiler logs.
339 lines
9.7 KiB
Go
339 lines
9.7 KiB
Go
package main
|
|
|
|
import "fmt"
|
|
import "log"
|
|
import "bytes"
|
|
import "sync"
|
|
import "net/http"
|
|
import "runtime/debug"
|
|
|
|
// TODO: Use the error_buffer variable to construct the system log in the Control Panel. Should we log errors caused by users too? Or just collect statistics on those or do nothing? Intercept recover()? Could we intercept the logger instead here? We might get too much information, if we intercept the logger, maybe make it part of the Debug page?
|
|
var errorBufferMutex sync.RWMutex
|
|
var errorBuffer []error
|
|
|
|
//var notfound_count_per_second int
|
|
//var noperms_count_per_second int
|
|
var errorInternal []byte
|
|
var errorNotfound []byte
|
|
|
|
func initErrors() error {
|
|
var b bytes.Buffer
|
|
user := User{0, "guest", "Guest", "", 0, false, false, false, false, false, false, GuestPerms, nil, "", false, "", "", "", "", "", 0, 0, "0.0.0.0.0", 0}
|
|
pi := Page{"Internal Server Error", user, hvars, tList, "A problem has occurred in the system."}
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
errorInternal = b.Bytes()
|
|
|
|
b.Reset()
|
|
pi = Page{"Not Found", user, hvars, tList, "The requested page doesn't exist."}
|
|
err = templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
errorNotfound = b.Bytes()
|
|
return nil
|
|
}
|
|
|
|
// LogError logs internal handler errors which can't be handled with InternalError() as a wrapper for log.Fatal(), we might do more with it in the future
|
|
func LogError(err error) {
|
|
log.Print(err)
|
|
debug.PrintStack()
|
|
errorBufferMutex.Lock()
|
|
defer errorBufferMutex.Unlock()
|
|
errorBuffer = append(errorBuffer, err)
|
|
log.Fatal("")
|
|
}
|
|
|
|
// InternalError is the main function for handling internal errors, while simultaneously printing out a page for the end-user to let them know that *something* has gone wrong
|
|
func InternalError(err error, w http.ResponseWriter) {
|
|
_, _ = w.Write(errorInternal)
|
|
log.Print(err)
|
|
debug.PrintStack()
|
|
errorBufferMutex.Lock()
|
|
defer errorBufferMutex.Unlock()
|
|
errorBuffer = append(errorBuffer, err)
|
|
log.Fatal("")
|
|
}
|
|
|
|
// InternalErrorJSQ is the JSON "maybe" version of InternalError which can handle both JSON and normal requests
|
|
func InternalErrorJSQ(err error, w http.ResponseWriter, r *http.Request, isJs bool) {
|
|
w.WriteHeader(500)
|
|
if !isJs {
|
|
_, _ = w.Write(errorInternal)
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"A problem has occured in the system."}`))
|
|
}
|
|
log.Print(err)
|
|
debug.PrintStack()
|
|
errorBufferMutex.Lock()
|
|
defer errorBufferMutex.Unlock()
|
|
errorBuffer = append(errorBuffer, err)
|
|
log.Fatal("")
|
|
}
|
|
|
|
// InternalErrorJS is the JSON version of InternalError on routes we know will only be requested via JSON. E.g. An API.
|
|
func InternalErrorJS(err error, w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(500)
|
|
_, _ = w.Write([]byte(`{"errmsg":"A problem has occured in the system."}`))
|
|
errorBufferMutex.Lock()
|
|
defer errorBufferMutex.Unlock()
|
|
errorBuffer = append(errorBuffer, err)
|
|
log.Fatal(err)
|
|
}
|
|
|
|
func PreError(errmsg string, w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(500)
|
|
user := User{ID: 0, Group: 6, Perms: GuestPerms}
|
|
pi := Page{"Error", user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
func LocalError(errmsg string, w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(500)
|
|
pi := Page{"Local Error", user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
func LoginRequired(w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(401)
|
|
pi := Page{"Local Error", user, hvars, tList, "You need to login to do that."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
func PreErrorJS(errmsg string, w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(500)
|
|
_, _ = w.Write([]byte(`{"errmsg":"` + errmsg + `"}`))
|
|
}
|
|
|
|
func PreErrorJSQ(errmsg string, w http.ResponseWriter, r *http.Request, isJs bool) {
|
|
w.WriteHeader(500)
|
|
if !isJs {
|
|
user := User{ID: 0, Group: 6, Perms: GuestPerms}
|
|
pi := Page{"Local Error", user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"` + errmsg + `"}`))
|
|
}
|
|
}
|
|
|
|
func LocalErrorJSQ(errmsg string, w http.ResponseWriter, r *http.Request, user User, isJs bool) {
|
|
w.WriteHeader(500)
|
|
if !isJs {
|
|
pi := Page{"Local Error", user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"` + errmsg + `"}`))
|
|
}
|
|
}
|
|
|
|
func LocalErrorJS(errmsg string, w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(500)
|
|
_, _ = w.Write([]byte(`{'errmsg': '` + errmsg + `'}`))
|
|
}
|
|
|
|
func NoPermissions(w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(403)
|
|
pi := Page{"Local Error", user, hvars, tList, "You don't have permission to do that."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
errpage := b.String()
|
|
fmt.Fprintln(w, errpage)
|
|
}
|
|
|
|
func NoPermissionsJSQ(w http.ResponseWriter, r *http.Request, user User, isJs bool) {
|
|
w.WriteHeader(403)
|
|
if !isJs {
|
|
pi := Page{"Local Error", user, hvars, tList, "You don't have permission to do that."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"You don't have permission to do that."}`))
|
|
}
|
|
}
|
|
|
|
func Banned(w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(403)
|
|
pi := Page{"Banned", user, hvars, tList, "You have been banned from this site."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
// nolint
|
|
func BannedJSQ(w http.ResponseWriter, r *http.Request, user User, isJs bool) {
|
|
w.WriteHeader(403)
|
|
if !isJs {
|
|
pi := Page{"Banned", user, hvars, tList, "You have been banned from this site."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"You have been banned from this site."}`))
|
|
}
|
|
}
|
|
|
|
// nolint
|
|
func LoginRequiredJSQ(w http.ResponseWriter, r *http.Request, user User, isJs bool) {
|
|
w.WriteHeader(401)
|
|
if !isJs {
|
|
pi := Page{"Local Error", user, hvars, tList, "You need to login to do that."}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"You need to login to do that."}`))
|
|
}
|
|
}
|
|
|
|
func SecurityError(w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(403)
|
|
pi := Page{"Security Error", user, hvars, tList, "There was a security issue with your request."}
|
|
if preRenderHooks["pre_render_security_error"] != nil {
|
|
if runPreRenderHook("pre_render_security_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
func NotFound(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(404)
|
|
_, _ = w.Write(errorNotfound)
|
|
}
|
|
|
|
// nolint
|
|
func CustomError(errmsg string, errcode int, errtitle string, w http.ResponseWriter, r *http.Request, user User) {
|
|
w.WriteHeader(errcode)
|
|
pi := Page{errtitle, user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
}
|
|
|
|
// nolint
|
|
func CustomErrorJSQ(errmsg string, errcode int, errtitle string, w http.ResponseWriter, r *http.Request, user User, isJs bool) {
|
|
w.WriteHeader(errcode)
|
|
if !isJs {
|
|
pi := Page{errtitle, user, hvars, tList, errmsg}
|
|
if preRenderHooks["pre_render_error"] != nil {
|
|
if runPreRenderHook("pre_render_error", w, r, &user, &pi) {
|
|
return
|
|
}
|
|
}
|
|
var b bytes.Buffer
|
|
err := templates.ExecuteTemplate(&b, "error.html", pi)
|
|
if err != nil {
|
|
LogError(err)
|
|
}
|
|
fmt.Fprintln(w, b.String())
|
|
} else {
|
|
_, _ = w.Write([]byte(`{"errmsg":"` + errmsg + `"}`))
|
|
}
|
|
}
|