mirror of
https://github.com/stevenhowes/osgrid-server.git
synced 2026-05-26 15:53:42 +01:00
110 lines
2.8 KiB
Go
110 lines
2.8 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/paulcager/go-http-middleware"
|
|
"github.com/paulcager/osgridref"
|
|
flag "github.com/spf13/pflag"
|
|
)
|
|
|
|
const (
|
|
apiVersion = "v4"
|
|
)
|
|
|
|
var (
|
|
staticCache time.Duration
|
|
listenPort string
|
|
)
|
|
|
|
func main() {
|
|
flag.StringVar(&listenPort, "port", ":9090", "Port to listen on")
|
|
flag.DurationVar(&staticCache, "static-cache-max-age", 1*time.Hour, "If not zero, the max-age property to set in Cache-Control for responses")
|
|
flag.Parse()
|
|
|
|
server := makeHTTPServer(listenPort)
|
|
log.Fatal(server.ListenAndServe())
|
|
}
|
|
|
|
type Reply struct {
|
|
OSGridRef string `json:"osGridRef"`
|
|
Easting int `json:"easting"`
|
|
Northing int `json:"northing"`
|
|
Lat float64 `json:"lat"`
|
|
Lon float64 `json:"lon"`
|
|
}
|
|
|
|
func makeHTTPServer(listenPort string) *http.Server {
|
|
http.Handle("/"+apiVersion+"/gridref/", middleware.MakeCachingHandler(1*time.Hour, http.HandlerFunc(
|
|
func(w http.ResponseWriter, r *http.Request) {
|
|
gridRefStr := r.URL.Path[len("/"+apiVersion+"/gridref/"):]
|
|
gridRef, err := osgridref.ParseOsGridRef(gridRefStr)
|
|
if err != nil {
|
|
handleError(w, r, gridRefStr, err)
|
|
return
|
|
}
|
|
|
|
lat, lon := gridRef.ToLatLon()
|
|
handle(w, r, gridRef, lat, lon)
|
|
})))
|
|
|
|
http.Handle("/"+apiVersion+"/latlon/", middleware.MakeCachingHandler(1*time.Hour, http.HandlerFunc(
|
|
func(w http.ResponseWriter, r *http.Request) {
|
|
latLonStr := r.URL.Path[len("/"+apiVersion+"/latlon/"):]
|
|
latLon, err := osgridref.ParseLatLon(latLonStr, 0, osgridref.WGS84)
|
|
if err != nil {
|
|
handleError(w, r, latLonStr, err)
|
|
return
|
|
}
|
|
|
|
gridRef := latLon.ToOsGridRef()
|
|
handle(w, r, gridRef, latLon.Lat, latLon.Lon)
|
|
})))
|
|
|
|
if !strings.Contains(listenPort, ":") {
|
|
listenPort = ":" + listenPort
|
|
}
|
|
|
|
log.Println("Starting HTTP server on " + listenPort)
|
|
s := &http.Server{
|
|
ReadHeaderTimeout: 20 * time.Second,
|
|
WriteTimeout: 2 * time.Minute,
|
|
IdleTimeout: 10 * time.Minute,
|
|
Handler: middleware.MakeLoggingHandler(http.DefaultServeMux),
|
|
Addr: listenPort,
|
|
}
|
|
|
|
return s
|
|
}
|
|
|
|
func handleError(w http.ResponseWriter, _ *http.Request, str string, _ error) {
|
|
w.Header().Add("Access-Control-Allow-Origin", "*")
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
fmt.Fprintf(w, "Invalid request: %q\n", str)
|
|
}
|
|
|
|
func handle(w http.ResponseWriter, _ *http.Request, ref osgridref.OsGridRef, lat float64, lon float64) {
|
|
reply := Reply{
|
|
OSGridRef: ref.StringNCompact(8),
|
|
Easting: ref.Easting,
|
|
Northing: ref.Northing,
|
|
Lat: lat,
|
|
Lon: lon,
|
|
}
|
|
|
|
w.Header().Add("Content-Type", "application/json")
|
|
w.Header().Add("Access-Control-Allow-Origin", "*")
|
|
enc := json.NewEncoder(w)
|
|
enc.SetEscapeHTML(false)
|
|
err := enc.Encode(reply)
|
|
if err != nil {
|
|
log.Printf("Failed to write response: %s", err)
|
|
w.WriteHeader(http.StatusBadGateway)
|
|
}
|
|
}
|