alice-lg/backend/api.go

268 lines
6.6 KiB
Go
Raw Normal View History

2017-05-18 14:44:50 +02:00
package main
import (
"compress/gzip"
"encoding/json"
2017-05-18 18:22:52 +02:00
"fmt"
2017-05-18 14:44:50 +02:00
"net/http"
2017-05-18 18:10:29 +02:00
"strconv"
2017-05-18 14:44:50 +02:00
"strings"
"github.com/ecix/alice-lg/backend/api"
"github.com/julienschmidt/httprouter"
)
// Alice LG Rest API
//
// The API provides endpoints for getting
// information from the routeservers / alice datasources.
//
// Endpoints:
//
// Config
// Show /api/config
//
// Routeservers
// List /api/routeservers
// Status /api/routeservers/:id/status
// Neighbours /api/routeservers/:id/neighbours
2017-05-18 18:10:29 +02:00
// Routes /api/routeservers/:id/neighbours/:neighbourId/routes
2017-05-18 14:44:50 +02:00
//
2017-05-23 13:58:58 +02:00
// Querying
// LookupPrefix /api/routeservers/:id/lookup/prefix?q=<prefix>
//
2017-05-18 14:44:50 +02:00
2017-05-18 15:26:22 +02:00
type apiEndpoint func(*http.Request, httprouter.Params) (api.Response, error)
2017-05-18 14:44:50 +02:00
// Wrap handler for access controll, throtteling and compression
func endpoint(wrapped apiEndpoint) httprouter.Handle {
return func(res http.ResponseWriter,
req *http.Request,
params httprouter.Params) {
// Get result from handler
result, err := wrapped(req, params)
if err != nil {
result = api.ErrorResponse{
Error: err.Error(),
}
payload, _ := json.Marshal(result)
http.Error(res, string(payload), http.StatusInternalServerError)
return
}
// Encode json
payload, err := json.Marshal(result)
// Set response header
res.Header().Set("Content-Type", "application/json")
// Check if compression is supported
if strings.Contains(req.Header.Get("Accept-Encoding"), "gzip") {
// Compress response
res.Header().Set("Content-Encoding", "gzip")
gz := gzip.NewWriter(res)
defer gz.Close()
gz.Write(payload)
} else {
res.Write(payload) // Fall back to uncompressed response
}
}
}
// Register api endpoints
func apiRegisterEndpoints(router *httprouter.Router) error {
2017-05-19 11:32:22 +02:00
// Meta
router.GET("/api/status", endpoint(apiStatusShow))
2017-05-18 14:44:50 +02:00
router.GET("/api/config", endpoint(apiConfigShow))
// Routeservers
2017-05-18 18:10:29 +02:00
router.GET("/api/routeservers",
endpoint(apiRouteserversList))
router.GET("/api/routeservers/:id/status",
endpoint(apiStatus))
router.GET("/api/routeservers/:id/neighbours",
endpoint(apiNeighboursList))
router.GET("/api/routeservers/:id/neighbours/:neighbourId/routes",
endpoint(apiRoutesList))
2017-05-18 14:44:50 +02:00
2017-05-23 13:58:58 +02:00
// Querying
router.GET("/api/routeservers/:id/lookup/prefix",
endpoint(apiLookupPrefix))
2017-06-19 16:44:24 +02:00
router.GET("/api/lookup/prefix",
endpoint(apiLookupPrefixGlobal))
2017-05-18 14:44:50 +02:00
return nil
}
2017-05-19 11:32:22 +02:00
// Handle Status Endpoint, this is intended for
// monitoring and service health checks
func apiStatusShow(_req *http.Request, _params httprouter.Params) (api.Response, error) {
status, err := NewAppStatus()
return status, err
}
2017-05-18 14:44:50 +02:00
// Handle Config Endpoint
2017-05-18 15:26:22 +02:00
func apiConfigShow(_req *http.Request, _params httprouter.Params) (api.Response, error) {
2017-05-18 15:23:36 +02:00
result := api.ConfigResponse{
Rejection: api.Rejection{
Asn: AliceConfig.Ui.RoutesRejections.Asn,
RejectId: AliceConfig.Ui.RoutesRejections.RejectId,
},
RejectReasons: AliceConfig.Ui.RoutesRejections.Reasons,
Noexport: api.Noexport{
Asn: AliceConfig.Ui.RoutesNoexports.Asn,
NoexportId: AliceConfig.Ui.RoutesNoexports.NoexportId,
},
NoexportReasons: AliceConfig.Ui.RoutesNoexports.Reasons,
RoutesColumns: AliceConfig.Ui.RoutesColumns,
}
return result, nil
2017-05-18 14:44:50 +02:00
}
2017-05-18 15:26:22 +02:00
// Handle Routeservers List
func apiRouteserversList(_req *http.Request, _params httprouter.Params) (api.Response, error) {
2017-05-18 17:35:27 +02:00
// Get list of sources from config,
routeservers := []api.Routeserver{}
sources := AliceConfig.Sources
for id, source := range sources {
routeservers = append(routeservers, api.Routeserver{
Id: id,
Name: source.Name,
})
}
// Make routeservers response
response := api.RouteserversResponse{
Routeservers: routeservers,
}
return response, nil
2017-05-18 15:26:22 +02:00
}
2017-05-18 18:10:29 +02:00
2017-05-18 18:22:52 +02:00
// Helper: Validate source Id
func validateSourceId(id string) (int, error) {
rsId, err := strconv.Atoi(id)
if err != nil {
return 0, err
}
if rsId < 0 {
return 0, fmt.Errorf("Source id may not be negative")
}
if rsId >= len(AliceConfig.Sources) {
return 0, fmt.Errorf("Source id not within [0, %d]", len(AliceConfig.Sources)-1)
}
return rsId, nil
}
2017-05-23 13:58:58 +02:00
// Helper: Validate query string
func validateQueryString(req *http.Request, key string) (string, error) {
query := req.URL.Query()
values, ok := query[key]
if !ok {
return "", fmt.Errorf("Query param %s is missing.", key)
}
if len(values) != 1 {
return "", fmt.Errorf("Query param %s is ambigous.", key)
}
value := values[0]
if value == "" {
return "", fmt.Errorf("Query param %s may not be empty.", key)
}
return value, nil
}
2017-05-18 18:10:29 +02:00
// Handle status
func apiStatus(_req *http.Request, params httprouter.Params) (api.Response, error) {
2017-05-18 18:22:52 +02:00
rsId, err := validateSourceId(params.ByName("id"))
if err != nil {
return nil, err
}
2017-05-18 18:10:29 +02:00
source := AliceConfig.Sources[rsId].getInstance()
result, err := source.Status()
return result, err
}
// Handle get neighbours on routeserver
func apiNeighboursList(_req *http.Request, params httprouter.Params) (api.Response, error) {
2017-05-18 18:22:52 +02:00
rsId, err := validateSourceId(params.ByName("id"))
if err != nil {
return nil, err
}
2017-05-18 18:10:29 +02:00
source := AliceConfig.Sources[rsId].getInstance()
result, err := source.Neighbours()
return result, err
}
// Handle routes
func apiRoutesList(_req *http.Request, params httprouter.Params) (api.Response, error) {
2017-05-18 18:22:52 +02:00
rsId, err := validateSourceId(params.ByName("id"))
if err != nil {
return nil, err
}
2017-05-18 18:10:29 +02:00
neighbourId := params.ByName("neighbourId")
source := AliceConfig.Sources[rsId].getInstance()
result, err := source.Routes(neighbourId)
return result, err
}
2017-05-23 13:58:58 +02:00
// Handle lookup
func apiLookupPrefix(req *http.Request, params httprouter.Params) (api.Response, error) {
rsId, err := validateSourceId(params.ByName("id"))
if err != nil {
return nil, err
}
prefix, err := validateQueryString(req, "q")
if err != nil {
return nil, err
}
source := AliceConfig.Sources[rsId].getInstance()
result, err := source.LookupPrefix(prefix)
return result, err
}
2017-06-19 16:44:24 +02:00
// Handle global lookup
func apiLookupPrefixGlobal(req *http.Request, params httprouter.Params) (api.Response, error) {
// Get prefix to query
prefix, err := validateQueryString(req, "q")
if err != nil {
return nil, err
}
// Run query on all sources
rsCount := len(AliceConfig.Sources)
responses := make(chan api.LookupResponse, rsCount)
for _, src := range AliceConfig.Sources {
go func(src SourceConfig) {
// Run query on RS
rs := src.getInstance()
result, _ := rs.LookupPrefix(prefix)
responses <- result
}(src)
}
// Collect results
routes := []api.LookupRoute{}
for i := 0; i < rsCount; i++ {
result := <-responses
routes = append(routes, result.Routes...)
}
// Make response
response := api.LookupResponseGlobal{
Routes: routes,
}
return response, nil
}