Skip to content
Snippets Groups Projects
handler-index.go 2.64 KiB
Newer Older
gbe's avatar
gbe committed
package main

import (
	"errors"
	"fmt"
	"html/template"
	"net/http"
	"strconv"
gbe's avatar
gbe committed
	"github.com/go-kit/kit/log/level"

	"git.c3pb.de/gbe/invinoveritas/auth"
gbe's avatar
gbe committed
	"git.c3pb.de/gbe/invinoveritas/log"
gbe's avatar
gbe committed
	"git.c3pb.de/gbe/invinoveritas/vino"
func (h Handler) index(write WriteStorage, read ReadStorage) http.Handler {
	var (
		once sync.Once
		tpl  *template.Template
	)
gbe's avatar
gbe committed
	type templateData struct {
gbe's avatar
gbe committed
		Wines []vino.Vino
gbe's avatar
gbe committed
		User  auth.User
	}

	return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
		once.Do(func() {
			tpl = template.Must(template.ParseFS(templateFS, "templates/base.tpl", "templates/index.tpl"))
		})

		if r.Method == "GET" {
			wines, err := read.ListWines(r.Context())
			if err != nil {
gbe's avatar
gbe committed
				httpError(w, r, "can't list wines", err, http.StatusInternalServerError)
				return
			}

			w.Header().Add("content-type", "text/html")

gbe's avatar
gbe committed
			data := templateData{
				Wines: wines,
				User:  auth.Get(r),
			}

			err = tpl.ExecuteTemplate(w, "index.tpl", data)
			if err != nil {
gbe's avatar
gbe committed
				level.Error(log.Get(r)).
					Log("error", err,
						"msg", "can't execute index template")
		if r.Method != "POST" {
gbe's avatar
gbe committed
			httpError(w, r, "invalid method", errors.New(r.Method), http.StatusMethodNotAllowed)
		name := r.FormValue("name")
		if len(name) > 80 || len(name) == 0 {
			httpError(w, r, "bad name", errors.New("name must be between 1 and 80 chars"), http.StatusBadRequest)
		ratingVal := r.FormValue("rating")
		var (
			rating int
			err    error
		)
		if ratingVal != "" {
			rating, err = strconv.Atoi(ratingVal)
			if err != nil {
gbe's avatar
gbe committed
				httpError(w, r, "can't convert rating", err, http.StatusBadRequest)
		err = r.ParseMultipartForm(16 * 1024 * 1024)
gbe's avatar
gbe committed
		if err != nil {
gbe's avatar
gbe committed
			level.Warn(log.Get(r)).Log("error", err, "msg", "can't parse multipart form data")
gbe's avatar
gbe committed
		}
gbe's avatar
gbe committed
		vino := vino.Vino{
			Name:   name,
			Rating: rating,
		}

		// See if there's an image file uploaded
gbe's avatar
gbe committed
		if r.MultipartForm != nil && len(r.MultipartForm.File["picture"]) != 0 {
			picFile, picHdr, err := r.FormFile("picture")
			if err != nil {
gbe's avatar
gbe committed
				httpError(w, r, "can't open picture file", err, http.StatusInternalServerError)
				return
			}
			defer picFile.Close()

			err = vino.AddPicture(picFile, picHdr.Header.Get("Content-Type"))
			if err != nil {
gbe's avatar
gbe committed
				httpError(w, r, "can't load picture", err, http.StatusBadRequest)
gbe's avatar
gbe committed
		}

		err = write.Store(r.Context(), &vino)
gbe's avatar
gbe committed
		if err != nil {
gbe's avatar
gbe committed
			httpError(w, r, fmt.Sprintf("can't store wine %q", vino), err, http.StatusInternalServerError)
gbe's avatar
gbe committed
			return
		}
gbe's avatar
gbe committed
		http.Redirect(w, r, "/details?id="+strconv.Itoa(vino.ID), http.StatusSeeOther) // TODO: Is this the correct status?
gbe's avatar
gbe committed
}