aicra/internal/request/request.go

133 lines
2.9 KiB
Go
Raw Normal View History

2018-06-01 08:51:51 +00:00
package request
import (
"encoding/json"
"fmt"
"git.xdrm.io/go/aicra/err"
2018-07-08 22:15:29 +00:00
"git.xdrm.io/go/aicra/response"
2018-06-01 08:51:51 +00:00
"log"
"net/http"
"plugin"
"strings"
"time"
)
2018-07-07 17:21:00 +00:00
// BuildFromHttpRequest builds an interface request from a http.Request
func BuildFromHttpRequest(req *http.Request) (*Request, error) {
2018-06-01 08:51:51 +00:00
/* (1) Get useful data */
uri := normaliseUri(req.URL.Path)
uriparts := strings.Split(uri, "/")
/* (2) Init request */
inst := &Request{
Uri: uriparts,
Path: make([]string, 0, len(uriparts)),
Data: NewDataset(),
}
/* (3) Build dataset */
inst.Data.Build(req)
return inst, nil
}
// FetchFormData extracts FORM data
//
// - parse 'form-data' if not supported (not POST requests)
// - parse 'x-www-form-urlencoded'
// - parse 'application/json'
func FetchFormData(req *http.Request) map[string]interface{} {
res := make(map[string]interface{})
// Abort if GET request
if req.Method == "GET" {
return res
}
ct := req.Header.Get("Content-Type")
if strings.HasPrefix(ct, "application/json") {
receiver := make(map[string]interface{}, 0)
// 1. Init JSON reader
decoder := json.NewDecoder(req.Body)
if err := decoder.Decode(&receiver); err != nil {
log.Printf("[parse.json] %s\n", err)
return res
}
// 2. Return result
return receiver
} else if strings.HasPrefix(ct, "application/x-www-form-urlencoded") {
// 1. Parse url encoded data
req.ParseForm()
// 2. Extract values
for name, value := range req.PostForm {
res[name] = value
}
} else { // form-data or anything
startn := time.Now().UnixNano()
// 1. Parse form-data
if err := req.ParseMultipartForm(req.ContentLength + 1); err != nil {
log.Printf("[read.multipart] %s\n", err)
return res
}
// 2. Extract values
for name, value := range req.PostForm {
res[name] = value
}
fmt.Printf("* %.3f us\n", float64(time.Now().UnixNano()-startn)/1e3)
}
return res
}
// LoadController tries to load a controller from its uri
// checks for its given method ('Get', 'Post', 'Put', or 'Delete')
func (i *Request) LoadController(method string) (func(response.Arguments, *response.Response) response.Response, err.Error) {
2018-06-01 08:51:51 +00:00
/* (1) Build controller path */
path := strings.Join(i.Path, "-")
if len(path) == 0 {
2018-07-08 09:44:39 +00:00
path = fmt.Sprintf(".build/controller/ROOT.so")
} else {
2018-07-08 09:44:39 +00:00
path = fmt.Sprintf(".build/controller/%s.so", path)
}
2018-06-01 08:51:51 +00:00
/* (2) Format url */
tmp := []byte(strings.ToLower(method))
tmp[0] = tmp[0] - ('a' - 'A')
method = string(tmp)
/* (2) Try to load plugin */
p, err2 := plugin.Open(path)
if err2 != nil {
return nil, err.UncallableController
2018-06-01 08:51:51 +00:00
}
/* (3) Try to extract method */
m, err2 := p.Lookup(method)
if err2 != nil {
return nil, err.UncallableMethod
2018-06-01 08:51:51 +00:00
}
/* (4) Check signature */
2018-07-08 22:15:29 +00:00
callable, validSignature := m.(func(response.Arguments, *response.Response) response.Response)
2018-06-01 08:51:51 +00:00
if !validSignature {
return nil, err.UncallableMethod
2018-06-01 08:51:51 +00:00
}
return callable, err.Success
2018-06-01 08:51:51 +00:00
}