aicra/router.go

152 lines
3.5 KiB
Go
Raw Normal View History

package gfw
import (
"fmt"
2018-05-30 07:06:26 +00:00
"git.xdrm.io/xdrm-brackets/gfw/config"
2018-06-01 07:09:26 +00:00
"git.xdrm.io/xdrm-brackets/gfw/err"
2018-06-01 08:51:51 +00:00
"git.xdrm.io/xdrm-brackets/gfw/request"
"log"
"net/http"
"strings"
)
2018-06-01 08:51:51 +00:00
func (s *Server) route(res http.ResponseWriter, httpReq *http.Request) {
/* (1) Build request
---------------------------------------------------------*/
/* (1) Try to build request */
2018-06-01 08:51:51 +00:00
req, err2 := request.Build(httpReq)
2018-06-01 07:09:26 +00:00
if err2 != nil {
2018-06-01 08:51:51 +00:00
log.Fatal(err2)
}
/* (2) Find a controller
---------------------------------------------------------*/
2018-06-01 08:51:51 +00:00
controller := s.findController(req)
/* (3) Check method
---------------------------------------------------------*/
2018-06-01 08:51:51 +00:00
var method *config.Method
if method = controller.Method(httpReq.Method); method == nil {
2018-06-01 07:09:26 +00:00
Json, _ := err.UnknownMethod.MarshalJSON()
res.Header().Add("Content-Type", "application/json")
res.Write(Json)
2018-06-01 07:09:26 +00:00
log.Printf("[err] %s\n", err.UnknownMethod.Reason)
return
}
2018-05-29 17:01:20 +00:00
/* (4) Check parameters
---------------------------------------------------------*/
2018-06-01 07:09:26 +00:00
var paramError err.Error = err.Success
2018-05-29 17:01:20 +00:00
parameters := make(map[string]interface{})
for name, param := range method.Parameters {
/* (1) Extract value */
2018-06-01 08:51:51 +00:00
p, isset := req.Data.Set[name]
2018-05-29 17:01:20 +00:00
/* (2) Required & missing */
if !isset && !*param.Optional {
2018-06-01 07:09:26 +00:00
paramError = err.MissingParam
2018-05-29 17:01:20 +00:00
paramError.BindArgument(name)
break
}
2018-05-29 17:01:20 +00:00
/* (3) Optional & missing: set default value */
if !isset {
2018-06-01 08:51:51 +00:00
p = &request.Parameter{
2018-05-29 17:01:20 +00:00
Parsed: true,
File: param.Type == "FILE",
Value: nil,
}
if param.Default != nil {
p.Value = *param.Default
}
}
2018-05-29 17:01:20 +00:00
/* (4) Parse parameter if not file */
2018-06-01 08:51:51 +00:00
if !p.File {
p.Parse()
}
2018-05-29 17:01:20 +00:00
/* (4) Fail on unexpected multipart file */
waitFile, gotFile := param.Type == "FILE", p.File
if gotFile && !waitFile || !gotFile && waitFile {
2018-06-01 07:09:26 +00:00
paramError = err.InvalidParam
2018-05-29 17:01:20 +00:00
paramError.BindArgument(name)
paramError.BindArgument("FILE")
break
}
/* (5) Do not check if file */
if gotFile {
parameters[name] = p.Value
continue
}
/* (6) Check type */
if s.Checker.Run(param.Type, p.Value) != nil {
2018-06-01 07:09:26 +00:00
paramError = err.InvalidParam
paramError.BindArgument(name)
paramError.BindArgument(param.Type)
paramError.BindArgument(p.Value)
break
2018-05-29 17:01:20 +00:00
}
2018-05-29 17:01:20 +00:00
parameters[name] = p.Value
}
// Fail if argument check failed
2018-06-01 07:09:26 +00:00
if paramError.Code != err.Success.Code {
Json, _ := paramError.MarshalJSON()
res.Header().Add("Content-Type", "application/json")
res.Write(Json)
log.Printf("[err] %s\n", paramError.Reason)
return
}
2018-06-01 07:09:26 +00:00
/* (5) Load controller
---------------------------------------------------------*/
2018-06-01 08:51:51 +00:00
callable, err := req.LoadController(httpReq.Method)
2018-06-01 07:09:26 +00:00
if err != nil {
log.Printf("[err] %s\n", err)
return
}
2018-06-01 08:51:51 +00:00
fmt.Printf("OK\nplugin: '%si.so'\n", strings.Join(req.Path, "/"))
2018-05-29 17:01:20 +00:00
for name, value := range parameters {
fmt.Printf(" $%s = %v\n", name, value)
}
2018-06-01 07:09:26 +00:00
/* (6) Execute and get response
---------------------------------------------------------*/
2018-06-01 08:51:51 +00:00
resp := callable(parameters)
if resp != nil {
fmt.Printf("-- OUT --\n")
for name, value := range resp.Dump() {
fmt.Printf(" $%s = %v\n", name, value)
}
eJSON, _ := resp.Err.MarshalJSON()
fmt.Printf("-- ERR --\n%s\n", eJSON)
2018-06-01 07:09:26 +00:00
}
return
}
2018-06-01 07:09:26 +00:00
2018-06-01 08:51:51 +00:00
func (s *Server) findController(req *request.Request) *config.Controller {
2018-06-01 07:09:26 +00:00
2018-06-01 08:51:51 +00:00
/* (1) Try to browse by URI */
pathi, ctl := s.config.Browse(req.Uri)
2018-06-01 07:09:26 +00:00
2018-06-01 08:51:51 +00:00
/* (2) Set controller uri */
req.Path = make([]string, 0, pathi)
req.Path = append(req.Path, req.Uri[:pathi]...)
2018-06-01 07:09:26 +00:00
/* (3) Extract & store URI params */
2018-06-01 08:51:51 +00:00
req.Data.SetUri(req.Uri[pathi:])
2018-06-01 07:09:26 +00:00
/* (4) Return controller */
return ctl
}