|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
"image/draw"
|
|
|
|
"image/png"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type server struct {
|
|
|
|
out io.Writer
|
|
|
|
errors io.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *server) logReceived(r *http.Request) {
|
|
|
|
fmt.Fprintf(s.out, "> %s\n", r.URL.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
s.logReceived(r)
|
|
|
|
encoder, err := getEncoding(r)
|
|
|
|
if err != nil {
|
|
|
|
s.writeError(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
width := 400
|
|
|
|
width_s := r.URL.Query().Get("w")
|
|
|
|
if width_s != "" {
|
|
|
|
n, err := strconv.Atoi(width_s)
|
|
|
|
if err == nil {
|
|
|
|
width = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
height := 400
|
|
|
|
height_s := r.URL.Query().Get("h")
|
|
|
|
if height_s != "" {
|
|
|
|
n, err := strconv.Atoi(height_s)
|
|
|
|
if err == nil {
|
|
|
|
height = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bg := parseColor(r.URL.Query().Get("bg"))
|
|
|
|
m := image.NewRGBA(image.Rect(0, 0, width, height))
|
|
|
|
draw.Draw(m, m.Bounds(), &image.Uniform{bg}, m.Bounds().Min, draw.Src)
|
|
|
|
encoder.WriteImage(w, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseColor(c_s string) color.RGBA {
|
|
|
|
f := func(s string) (n uint8) {
|
|
|
|
if len(s) != 2 {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
i, err := strconv.ParseUint(s, 16, 0)
|
|
|
|
if err != nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return uint8(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
white := color.RGBA{0xff, 0xff, 0xff, 0xff}
|
|
|
|
c_s = strings.ToLower(c_s)
|
|
|
|
switch len(c_s) {
|
|
|
|
case 2:
|
|
|
|
n := f(c_s[0:2])
|
|
|
|
return color.RGBA{n, n, n, 0xff}
|
|
|
|
case 4:
|
|
|
|
n, a := f(c_s[0:2]), f(c_s[2:4])
|
|
|
|
return color.RGBA{n, n, n, a}
|
|
|
|
case 6:
|
|
|
|
return color.RGBA{f(c_s[0:2]), f(c_s[2:4]), f(c_s[4:6]), 0xff}
|
|
|
|
case 8:
|
|
|
|
return color.RGBA{f(c_s[0:2]), f(c_s[2:4]), f(c_s[4:6]), f(c_s[6:8])}
|
|
|
|
default:
|
|
|
|
return white
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type imageWriter interface {
|
|
|
|
WriteImage(w http.ResponseWriter, m image.Image)
|
|
|
|
}
|
|
|
|
|
|
|
|
type pngWriter struct {
|
|
|
|
CompressionLevel png.CompressionLevel
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w pngWriter) WriteImage(rw http.ResponseWriter, m image.Image) {
|
|
|
|
enc := png.Encoder{CompressionLevel: w.CompressionLevel}
|
|
|
|
rw.Header().Add("Content-Type", "image/png")
|
|
|
|
enc.Encode(rw, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getEncoding(r *http.Request) (imageWriter, error) {
|
|
|
|
parts := strings.Split(r.URL.Path, ".")
|
|
|
|
if len(parts) == 0 {
|
|
|
|
return nil, fmt.Errorf("no encoding specified")
|
|
|
|
}
|
|
|
|
|
|
|
|
switch e := parts[len(parts)-1]; e {
|
|
|
|
case "png":
|
|
|
|
return pngWriter{CompressionLevel: png.DefaultCompression}, nil
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("invalid encoding: %s", e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *server) writeError(w http.ResponseWriter, err error) {
|
|
|
|
fmt.Fprintf(s.errors, "E %s\n", err)
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
w.Write([]byte(err.Error()))
|
|
|
|
}
|