reichelt/cmd/apiserver/main.go

269 lines
5.1 KiB
Go

package main
import (
"encoding/json"
"flag"
"fmt"
"image"
"image/jpeg"
"image/png"
"log"
"net/http"
"net/url"
"strconv"
"strings"
"time"
cache "github.com/patrickmn/go-cache"
"go.dedaa.de/julixau/reichelt"
)
var (
addr = flag.String("http", ":8080", "The Address to bind to ")
)
type Handler struct {
*reichelt.Connection
cache *cache.Cache
}
func NotFound(resp http.ResponseWriter) {
resp.WriteHeader(http.StatusNotFound)
fmt.Fprint(resp, "404")
}
func InternalError(resp http.ResponseWriter) {
resp.WriteHeader(http.StatusInternalServerError)
fmt.Fprint(resp, "500")
}
func (h Handler) Search(resp http.ResponseWriter, path []string) {
if len(path) == 0 {
NotFound(resp)
return
}
log.Println("level 2 request:", path)
query, err := url.PathUnescape(path[0])
if err != nil {
log.Println("illegal query:", path[0], ":", err)
NotFound(resp)
return
}
parts, err := h.FindPart(query)
if err != nil {
log.Println("error retrieving part:", err)
InternalError(resp)
}
// insert to cache
if h.cache != nil {
for _, p := range parts {
h.cache.Set(strconv.Itoa(p.Number), &p, cache.NoExpiration)
}
}
encoder := json.NewEncoder(resp)
encoder.Encode(parts)
}
func (h Handler) Picture(resp http.ResponseWriter, path []string) {
if len(path) == 0 {
NotFound(resp)
return
}
log.Println("level 2 request:", path)
serve := func(img image.Image) {
resp.Header().Set("Content-type", "image/png")
resp.WriteHeader(http.StatusOK)
if err := png.Encode(resp, img); err != nil {
log.Println("could not encode png:", err)
}
}
number, err := strconv.Atoi(path[0])
if err != nil {
log.Println("encountered decode error:", err)
NotFound(resp)
}
if h.cache != nil {
if x, ok := h.cache.Get(path[0] + "-image"); ok {
serve(*(x.(*image.Image)))
return
}
}
img, err := h.GetImage(reichelt.Part{Number: number}, 99999, 9999)
if err != nil {
log.Println("error retrieving picture:", err)
InternalError(resp)
return
}
defer img.Close()
decodedImg, err := jpeg.Decode(img)
if err != nil {
log.Println("could no decode image:", err)
InternalError(resp)
return
}
if h.cache != nil {
h.cache.Set(path[0]+"-image", &decodedImg, cache.NoExpiration)
}
serve(decodedImg)
}
func (h Handler) Price(resp http.ResponseWriter, path []string) {
if len(path) == 0 {
NotFound(resp)
return
}
log.Println("level 2 request:", path)
number, err := strconv.Atoi(path[0])
if err != nil {
log.Println("encountered decode error:", err)
NotFound(resp)
}
var price float32
if h.cache != nil {
if x, ok := h.cache.Get(path[0] + "-price"); ok {
price = x.(float32)
goto cached
}
}
price = h.GetPrice(reichelt.Part{Number: number})
if h.cache != nil {
h.cache.Set(path[0]+"-price", price, time.Second*30)
}
cached:
encoder := json.NewEncoder(resp)
encoder.Encode(price)
}
func (h Handler) Meta(resp http.ResponseWriter, path []string) {
if len(path) == 0 {
NotFound(resp)
return
}
log.Println("level 2 request:", path)
number, err := strconv.Atoi(path[0])
if err != nil {
log.Println("encountered decode error:", err)
NotFound(resp)
}
// implement caching to avoid many queries to reichelt server
var meta reichelt.Meta
if h.cache != nil {
if x, ok := h.cache.Get(path[0] + "-meta"); ok {
meta = x.(reichelt.Meta)
goto cached
}
}
meta, err = h.GetMeta(reichelt.Part{Number: number})
if err != nil {
log.Println("encountered error:", err)
InternalError(resp)
}
if h.cache != nil {
h.cache.Set(path[0]+"-meta", meta, cache.NoExpiration)
}
cached:
if meta == nil {
// make output useful for nil maps
fmt.Fprint(resp, "[]")
return
}
encoder := json.NewEncoder(resp)
if len(path) > 1 {
if strings.ToLower(path[1]) == "overview" {
var headlines []string
for k, _ := range meta {
headlines = append(headlines, k)
}
encoder.Encode(headlines)
return
}
if query, err := url.QueryUnescape(path[1]); err != nil {
NotFound(resp)
log.Println("illegal query:", path[1])
} else {
// query is more concrete
subset, ok := meta[query]
if !ok {
NotFound(resp)
return
}
encoder.Encode(subset)
}
} else {
encoder.Encode(meta)
}
}
func (h Handler) ServeHTTP(resp http.ResponseWriter, req *http.Request) {
// find out whether there was URL encoded data in the query
path := req.URL.RawPath
if path == "" {
path = req.URL.Path
}
p := strings.Split(path, "/")
if len(p) < 2 {
NotFound(resp)
return
}
p = p[1:]
log.Println("level 1 request:", p)
switch p[0] {
case "search":
h.Search(resp, p[1:])
case "image":
h.Picture(resp, p[1:])
case "price":
h.Price(resp, p[1:])
case "meta":
h.Meta(resp, p[1:])
default:
NotFound(resp)
}
}
// a Simple request server
// exposing a simple api
// to search and retrieve
// - price
// - productimage
// for a product
func main() {
flag.Parse()
conn, err := reichelt.NewConnection()
if err != nil {
log.Fatal("could not create connection to reichelt:", err)
}
log.Println("start serving on:", *addr)
log.Fatal(http.ListenAndServe(*addr, Handler{
Connection: conn,
cache: cache.New(cache.NoExpiration, 0),
}))
}