269 lines
5.1 KiB
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),
|
|
}))
|
|
}
|