aicra/internal/request/dataset.go

248 lines
4.7 KiB
Go
Raw Permalink Normal View History

package request
import (
"encoding/json"
"fmt"
"git.xdrm.io/go/aicra/internal/multipart"
"log"
"net/http"
"strings"
)
2018-09-13 08:21:35 +00:00
// NewDataset creates an empty request dataset
2018-06-01 08:51:51 +00:00
func NewDataset() *DataSet {
return &DataSet{
2018-07-08 23:34:21 +00:00
URI: make([]*Parameter, 0),
2018-06-01 08:51:51 +00:00
Get: make(map[string]*Parameter),
Form: make(map[string]*Parameter),
Set: make(map[string]*Parameter),
}
2018-06-01 08:51:51 +00:00
}
// Build builds a 'DataSet' from an http request
func (i *DataSet) Build(req *http.Request) {
2018-06-01 08:51:51 +00:00
/* (1) GET (query) data */
i.fetchGet(req)
2018-06-01 08:51:51 +00:00
/* (2) We are done if GET method */
if req.Method == "GET" {
2018-06-01 08:51:51 +00:00
return
}
2018-06-01 08:51:51 +00:00
/* (3) POST (body) data */
i.fetchForm(req)
}
2018-07-08 23:34:21 +00:00
// SetURI stores URL data and fills 'Set'
// with creating pointers inside 'Url'
2018-07-08 23:34:21 +00:00
func (i *DataSet) SetURI(data []string) {
for index, value := range data {
// create set index
setindex := fmt.Sprintf("URL#%d", index)
// store value in 'Set'
2018-06-01 08:51:51 +00:00
i.Set[setindex] = &Parameter{
Parsed: false,
Value: value,
}
// create link in 'Url'
2018-07-08 23:34:21 +00:00
i.URI = append(i.URI, i.Set[setindex])
}
}
// fetchGet stores data from the QUERY (in url parameters)
2018-06-01 08:51:51 +00:00
func (i *DataSet) fetchGet(req *http.Request) {
for name, value := range req.URL.Query() {
// prevent invalid names
if !validName(name) {
log.Printf("invalid variable name: '%s'\n", name)
continue
}
// prevent injections
2018-06-01 08:51:51 +00:00
if nameInjection(name) {
log.Printf("get.injection: '%s'\n", name)
continue
}
// create set index
setindex := fmt.Sprintf("GET@%s", name)
// store value in 'Set'
2018-06-01 08:51:51 +00:00
i.Set[setindex] = &Parameter{
Parsed: false,
Value: value,
}
// create link in 'Get'
i.Get[name] = i.Set[setindex]
}
}
// fetchForm stores FORM data
//
// - parse 'form-data' if not supported (not POST requests)
// - parse 'x-www-form-urlencoded'
// - parse 'application/json'
2018-06-01 08:51:51 +00:00
func (i *DataSet) fetchForm(req *http.Request) {
contentType := req.Header.Get("Content-Type")
// parse json
if strings.HasPrefix(contentType, "application/json") {
2018-07-08 23:34:21 +00:00
i.parseJSON(req)
return
}
// parse urlencoded
if strings.HasPrefix(contentType, "application/x-www-form-urlencoded") {
2018-06-01 08:51:51 +00:00
i.parseUrlencoded(req)
return
}
// parse multipart
if strings.HasPrefix(contentType, "multipart/form-data; boundary=") {
2018-06-01 08:51:51 +00:00
i.parseMultipart(req)
return
}
// if unknown type store nothing
}
2018-07-08 23:34:21 +00:00
// parseJSON parses JSON from the request body inside 'Form'
// and 'Set'
2018-07-08 23:34:21 +00:00
func (i *DataSet) parseJSON(req *http.Request) {
parsed := make(map[string]interface{}, 0)
decoder := json.NewDecoder(req.Body)
// if parse error: do nothing
if err := decoder.Decode(&parsed); err != nil {
2018-09-26 05:35:53 +00:00
log.Printf("json.parse() %s\n", err)
return
}
// else store values 'parsed' values
for name, value := range parsed {
// prevent invalid names
if !validName(name) {
log.Printf("invalid variable name: '%s'\n", name)
continue
}
// prevent injections
2018-06-01 08:51:51 +00:00
if nameInjection(name) {
log.Printf("post.injection: '%s'\n", name)
continue
}
// store value in 'Set'
2018-06-01 08:51:51 +00:00
i.Set[name] = &Parameter{
Parsed: true,
Value: value,
}
// create link in 'Form'
i.Form[name] = i.Set[name]
}
}
2018-06-01 08:51:51 +00:00
// parseUrlencoded parses urlencoded from the request body inside 'Form'
// and 'Set'
2018-06-01 08:51:51 +00:00
func (i *DataSet) parseUrlencoded(req *http.Request) {
// use http.Request interface
2018-09-26 05:35:53 +00:00
if err := req.ParseForm(); err != nil {
log.Printf("urlencoded.parse() %s\n", err)
return
}
for name, value := range req.PostForm {
// prevent invalid names
if !validName(name) {
log.Printf("invalid variable name: '%s'\n", name)
continue
}
// prevent injections
2018-06-01 08:51:51 +00:00
if nameInjection(name) {
log.Printf("post.injection: '%s'\n", name)
continue
}
// store value in 'Set'
2018-06-01 08:51:51 +00:00
i.Set[name] = &Parameter{
Parsed: false,
Value: value,
}
// create link in 'Form'
i.Form[name] = i.Set[name]
}
}
2018-06-01 08:51:51 +00:00
// parseMultipart parses multi-part from the request body inside 'Form'
// and 'Set'
2018-06-01 08:51:51 +00:00
func (i *DataSet) parseMultipart(req *http.Request) {
/* (1) Create reader */
2018-09-25 19:22:25 +00:00
boundary := req.Header.Get("Content-Type")[len("multipart/form-data; boundary="):]
mpr, err := multipart.NewReader(req.Body, boundary)
if err != nil {
return
}
/* (2) Parse multipart */
2018-09-26 05:35:53 +00:00
if err = mpr.Parse(); err != nil {
log.Printf("multipart.parse() %s\n", err)
return
}
/* (3) Store data into 'Form' and 'Set */
2018-09-25 19:22:25 +00:00
for name, data := range mpr.Data {
// prevent invalid names
if !validName(name) {
log.Printf("invalid variable name: '%s'\n", name)
continue
}
// prevent injections
2018-06-01 08:51:51 +00:00
if nameInjection(name) {
log.Printf("post.injection: '%s'\n", name)
continue
}
// store value in 'Set'
2018-06-01 08:51:51 +00:00
i.Set[name] = &Parameter{
Parsed: false,
2018-09-25 19:22:25 +00:00
File: len(data.GetHeader("filename")) > 0,
Value: string(data.Data),
}
// create link in 'Form'
i.Form[name] = i.Set[name]
}
return
}