|
|
|
@ -1,52 +1,26 @@
|
|
|
|
|
package ratchet
|
|
|
|
|
package server
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"html/template"
|
|
|
|
|
"log/slog"
|
|
|
|
|
"net/http"
|
|
|
|
|
"os"
|
|
|
|
|
"strconv"
|
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
|
|
"git.runcible.io/learning/ratchet/internal/domain/user"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type RatchetServer struct {
|
|
|
|
|
http.Handler
|
|
|
|
|
|
|
|
|
|
logger *slog.Logger
|
|
|
|
|
//Services used by HTTP routes
|
|
|
|
|
UserService UserService
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func parseLogLevel(levelStr string) slog.Level {
|
|
|
|
|
switch strings.ToUpper(levelStr) {
|
|
|
|
|
case "DEBUG":
|
|
|
|
|
return slog.LevelDebug
|
|
|
|
|
case "INFO":
|
|
|
|
|
return slog.LevelInfo
|
|
|
|
|
case "WARN":
|
|
|
|
|
return slog.LevelWarn
|
|
|
|
|
case "ERROR":
|
|
|
|
|
return slog.LevelError
|
|
|
|
|
default:
|
|
|
|
|
return slog.LevelInfo // Default level
|
|
|
|
|
}
|
|
|
|
|
UserService user.UserService
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func InitLogging(level string) {
|
|
|
|
|
// Use os.Stderr
|
|
|
|
|
//
|
|
|
|
|
// Stderr is used for diagnostics and logging. Stdout is used for program
|
|
|
|
|
// output. Stderr also have greater likely hood of being seen if a programs
|
|
|
|
|
// output is being redirected.
|
|
|
|
|
parsedLogLevel := parseLogLevel(level)
|
|
|
|
|
loggerHandler := slog.NewJSONHandler(os.Stderr, &slog.HandlerOptions{Level: parsedLogLevel, AddSource: true})
|
|
|
|
|
logger := slog.New(loggerHandler)
|
|
|
|
|
slog.SetDefault(logger)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func NewRatchetServer() *RatchetServer {
|
|
|
|
|
r := new(RatchetServer)
|
|
|
|
|
|
|
|
|
|
func NewRatchetServer(logger *slog.Logger) *RatchetServer {
|
|
|
|
|
rs := new(RatchetServer)
|
|
|
|
|
rs.logger = logger
|
|
|
|
|
// TODO implement middleware that disables directory listings
|
|
|
|
|
fileServer := http.FileServer(http.Dir("./ui/static/"))
|
|
|
|
|
router := http.NewServeMux()
|
|
|
|
@ -58,22 +32,22 @@ func NewRatchetServer() *RatchetServer {
|
|
|
|
|
// resulting in this route requiring an exact match on "/" only
|
|
|
|
|
// You can only include one HTTP method in a route pattern if you choose
|
|
|
|
|
// GET will match GET & HEAD http request methods
|
|
|
|
|
router.HandleFunc("GET /{$}", home)
|
|
|
|
|
router.HandleFunc("GET /snippet/view/{id}", snippetView)
|
|
|
|
|
router.HandleFunc("GET /snippet/create", snippetCreate)
|
|
|
|
|
router.HandleFunc("GET /{$}", rs.home)
|
|
|
|
|
router.HandleFunc("GET /snippet/view/{id}", rs.snippetView)
|
|
|
|
|
router.HandleFunc("GET /snippet/create", rs.snippetCreate)
|
|
|
|
|
|
|
|
|
|
// FYI The http.HandlerFunc() adapter works by automatically adding a ServeHTTP() method to
|
|
|
|
|
// the passed function
|
|
|
|
|
router.HandleFunc("POST /snippet/create", snippetCreatePost)
|
|
|
|
|
router.HandleFunc("POST /snippet/create", rs.snippetCreatePost)
|
|
|
|
|
|
|
|
|
|
// Mux Router implements the Handler interface. AKA it has a ServeHTTP receiver.
|
|
|
|
|
r.Handler = router
|
|
|
|
|
return r
|
|
|
|
|
rs.Handler = router
|
|
|
|
|
return rs
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func home(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
func (rs *RatchetServer) home(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// TODO middleware should be able to print out these lines for all routes
|
|
|
|
|
slog.Info("request received", "method", "GET", "path", "/")
|
|
|
|
|
rs.logger.Info("request received", "method", "GET", "path", "/")
|
|
|
|
|
|
|
|
|
|
w.Header().Add("Server", "Go")
|
|
|
|
|
|
|
|
|
@ -89,23 +63,25 @@ func home(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// read template file into template set.
|
|
|
|
|
ts, err := template.ParseFiles(files...)
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error(err.Error())
|
|
|
|
|
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
|
|
|
|
rs.serverError(w, r, err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
// Write template content to response body
|
|
|
|
|
err = ts.ExecuteTemplate(w, "base", nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
slog.Error(err.Error())
|
|
|
|
|
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
|
|
|
|
// This is the older more verbose way of doing what RatchetServer.serverError does
|
|
|
|
|
// rs.logger.Error(err.Error())
|
|
|
|
|
// http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
|
|
|
|
rs.serverError(w, r, err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func snippetView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
func (rs *RatchetServer) snippetView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
|
|
|
|
|
id, err := strconv.Atoi(r.PathValue("id"))
|
|
|
|
|
if err != nil || id < 1 {
|
|
|
|
|
http.NotFound(w, r)
|
|
|
|
|
// http.NotFound(w, r)
|
|
|
|
|
rs.clientError(w, http.StatusNotFound)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -124,12 +100,12 @@ func snippetView(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// snippetCreate handles display of the form used to create snippets
|
|
|
|
|
func snippetCreate(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
func (rs *RatchetServer) snippetCreate(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
w.Write([]byte("Displaying snippetCreate form..."))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// snippetCreate handles display of the form used to create snippets
|
|
|
|
|
func snippetCreatePost(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
func (rs *RatchetServer) snippetCreatePost(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
// example of a custom header. Must be done before calling WriteHeader
|
|
|
|
|
// or they will fail to take effect.
|
|
|
|
|
w.Header().Add("Server", "Dirp")
|