mirror of https://github.com/statping/statping
				
				
				
			
		
			
				
	
	
		
			228 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			228 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
package handlers
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/json"
 | 
						|
	"fmt"
 | 
						|
	"html/template"
 | 
						|
	"net/http"
 | 
						|
	"path"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/statping-ng/statping-ng/source"
 | 
						|
	"github.com/statping-ng/statping-ng/types/errors"
 | 
						|
	"github.com/statping-ng/statping-ng/utils"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	cookieName = "statping_auth"
 | 
						|
 | 
						|
	timeout = time.Second * 30
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	jwtKey     []byte
 | 
						|
	httpServer *http.Server
 | 
						|
	usingSSL   bool
 | 
						|
	mainTmpl   = `{{define "main" }} {{ template "base" . }} {{ end }}`
 | 
						|
	templates  = []string{"base.gohtml"}
 | 
						|
)
 | 
						|
 | 
						|
func StopHTTPServer(err error) {
 | 
						|
	log.Infoln("Stopping HTTP Server")
 | 
						|
}
 | 
						|
 | 
						|
// RunHTTPServer will start a HTTP server on a specific IP and port
 | 
						|
func RunHTTPServer() error {
 | 
						|
	if utils.Params.GetBool("DISABLE_HTTP") {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	ip := utils.Params.GetString("SERVER_IP")
 | 
						|
	host := fmt.Sprintf("%v:%v", ip, utils.Params.GetInt("SERVER_PORT"))
 | 
						|
	key := utils.FileExists(utils.Directory + "/server.key")
 | 
						|
	cert := utils.FileExists(utils.Directory + "/server.crt")
 | 
						|
 | 
						|
	if key && cert {
 | 
						|
		log.Infoln("server.cert and server.key was found in root directory! Starting in SSL mode.")
 | 
						|
		log.Infoln(fmt.Sprintf("Statping Secure HTTPS Server running on https://%v:%v", ip, 443))
 | 
						|
		usingSSL = true
 | 
						|
	} else {
 | 
						|
		log.Infoln("Statping HTTP Server running on http://" + host + basePath)
 | 
						|
	}
 | 
						|
 | 
						|
	router = Router()
 | 
						|
	resetCookies()
 | 
						|
 | 
						|
	if utils.Params.GetBool("LETSENCRYPT_ENABLE") {
 | 
						|
		return startLetsEncryptServer(ip)
 | 
						|
	} else if usingSSL {
 | 
						|
		return startSSLServer(ip)
 | 
						|
	} else {
 | 
						|
		return startServer(host)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// IsReadAuthenticated will allow Read Only authentication for some routes
 | 
						|
func IsReadAuthenticated(r *http.Request) bool {
 | 
						|
	if ok := hasSetupEnv(); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	if ok := hasAPIQuery(r); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	if ok := hasAuthorizationHeader(r); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	_, err := getJwtToken(r)
 | 
						|
	if err == nil {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
// IsFullAuthenticated returns true if the HTTP request is authenticated. You can set the environment variable GO_ENV=test
 | 
						|
// to bypass the admin authenticate to the dashboard features.
 | 
						|
func IsFullAuthenticated(r *http.Request) bool {
 | 
						|
	if ok := hasSetupEnv(); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	if ok := hasAPIQuery(r); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	if ok := hasAuthorizationHeader(r); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	claim, err := getJwtToken(r)
 | 
						|
	if err != nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	return claim.Admin
 | 
						|
}
 | 
						|
 | 
						|
// ScopeName will show private JSON fields in the API.
 | 
						|
// It will return "admin" if request has valid admin authentication.
 | 
						|
func ScopeName(r *http.Request) string {
 | 
						|
	if ok := hasAPIQuery(r); ok {
 | 
						|
		return "admin"
 | 
						|
	}
 | 
						|
	if ok := hasAuthorizationHeader(r); ok {
 | 
						|
		return "admin"
 | 
						|
	}
 | 
						|
	claim, err := getJwtToken(r)
 | 
						|
	if err != nil {
 | 
						|
		return ""
 | 
						|
	}
 | 
						|
	if claim.Admin {
 | 
						|
		return "admin"
 | 
						|
	}
 | 
						|
	return "user"
 | 
						|
}
 | 
						|
 | 
						|
// IsAdmin returns true if the user session is an administrator
 | 
						|
func IsAdmin(r *http.Request) bool {
 | 
						|
	claim, err := getJwtToken(r)
 | 
						|
	if err != nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	return claim.Admin
 | 
						|
}
 | 
						|
 | 
						|
// IsUser returns true if the user is registered
 | 
						|
func IsUser(r *http.Request) bool {
 | 
						|
	if ok := hasSetupEnv(); ok {
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	tk, err := getJwtToken(r)
 | 
						|
	if err != nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	if err := tk.Valid(); err != nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	return true
 | 
						|
}
 | 
						|
 | 
						|
func loadTemplate(w http.ResponseWriter, r *http.Request) (*template.Template, error) {
 | 
						|
	var err error
 | 
						|
	mainTemplate := template.New("main")
 | 
						|
	mainTemplate, err = mainTemplate.Parse(mainTmpl)
 | 
						|
	if err != nil {
 | 
						|
		log.Errorln(err)
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	mainTemplate.Funcs(handlerFuncs(w, r))
 | 
						|
	// render all templates
 | 
						|
	for _, temp := range templates {
 | 
						|
		tmp, _ := source.TmplBox.String(temp)
 | 
						|
		mainTemplate, err = mainTemplate.Parse(tmp)
 | 
						|
		if err != nil {
 | 
						|
			log.Errorln(err)
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return mainTemplate, err
 | 
						|
}
 | 
						|
 | 
						|
// ExecuteResponse will render a HTTP response for the front end user
 | 
						|
func ExecuteResponse(w http.ResponseWriter, r *http.Request, file string, data interface{}, redirect interface{}) {
 | 
						|
	if url, ok := redirect.(string); ok {
 | 
						|
		http.Redirect(w, r, path.Join(basePath, url), http.StatusSeeOther)
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if usingSSL {
 | 
						|
		w.Header().Add("Strict-Transport-Security", "max-age=63072000; includeSubDomains")
 | 
						|
	}
 | 
						|
	mainTemplate, err := loadTemplate(w, r)
 | 
						|
	if err != nil {
 | 
						|
		log.Errorln(err)
 | 
						|
	}
 | 
						|
 | 
						|
	asset := file
 | 
						|
	if source.UsingAssets(utils.Directory) {
 | 
						|
 | 
						|
		asset = utils.Directory + "/assets/" + file
 | 
						|
 | 
						|
		if _, err := mainTemplate.ParseFiles(asset); err != nil {
 | 
						|
			log.Errorln(err)
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		render, err := source.TmplBox.String(asset)
 | 
						|
		if err != nil {
 | 
						|
			log.Errorln(err)
 | 
						|
		}
 | 
						|
		// render the page requested
 | 
						|
		if _, err := mainTemplate.Parse(render); err != nil {
 | 
						|
			log.Errorln(err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	// execute the template
 | 
						|
	if err := mainTemplate.Execute(w, data); err != nil {
 | 
						|
		log.Errorln(err)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func returnJson(d interface{}, w http.ResponseWriter, r *http.Request) {
 | 
						|
	w.Header().Set("Content-Type", "application/json")
 | 
						|
	if e, ok := d.(errors.Error); ok {
 | 
						|
		w.WriteHeader(e.Status())
 | 
						|
		json.NewEncoder(w).Encode(e)
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if e, ok := d.(error); ok {
 | 
						|
		w.WriteHeader(500)
 | 
						|
		json.NewEncoder(w).Encode(errors.New(e.Error()))
 | 
						|
		return
 | 
						|
	}
 | 
						|
	w.WriteHeader(http.StatusOK)
 | 
						|
	json.NewEncoder(w).Encode(d)
 | 
						|
}
 | 
						|
 | 
						|
// error404Handler is a HTTP handler for 404 error pages
 | 
						|
func error404Handler(w http.ResponseWriter, r *http.Request) {
 | 
						|
	if usingSSL {
 | 
						|
		w.Header().Add("Strict-Transport-Security", "max-age=63072000; includeSubDomains")
 | 
						|
	}
 | 
						|
	w.WriteHeader(http.StatusNotFound)
 | 
						|
	ExecuteResponse(w, r, "base.gohtml", nil, nil)
 | 
						|
}
 |