You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

369 lines
9.4 KiB

//
// DISCLAIMER
//
// Copyright 2017 ArangoDB GmbH, Cologne, Germany
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// Copyright holder is ArangoDB GmbH, Cologne, Germany
//
// Author Ewout Prangsma
//
package http
import (
"bytes"
"encoding/json"
"fmt"
"io"
"net/http"
"net/http/httptrace"
"net/url"
"reflect"
"strconv"
"strings"
driver "github.com/arangodb/go-driver"
)
// httpRequest implements driver.Request using standard golang http requests.
type httpRequest struct {
method string
path string
q url.Values
hdr map[string]string
written bool
bodyBuilder driver.BodyBuilder
velocyPack bool
}
// Path returns the Request path
func (r *httpRequest) Path() string {
return r.path
}
// Method returns the Request method
func (r *httpRequest) Method() string {
return r.method
}
// Clone creates a new request containing the same data as this request
func (r *httpRequest) Clone() driver.Request {
clone := *r
clone.q = url.Values{}
for k, v := range r.q {
for _, x := range v {
clone.q.Add(k, x)
}
}
if clone.hdr != nil {
clone.hdr = make(map[string]string)
for k, v := range r.hdr {
clone.hdr[k] = v
}
}
clone.bodyBuilder = r.bodyBuilder.Clone()
return &clone
}
// SetQuery sets a single query argument of the request.
// Any existing query argument with the same key is overwritten.
func (r *httpRequest) SetQuery(key, value string) driver.Request {
if r.q == nil {
r.q = url.Values{}
}
r.q.Set(key, value)
return r
}
// SetBody sets the content of the request.
// The protocol of the connection determines what kinds of marshalling is taking place.
func (r *httpRequest) SetBody(body ...interface{}) (driver.Request, error) {
return r, r.bodyBuilder.SetBody(body...)
}
// SetBodyArray sets the content of the request as an array.
// If the given mergeArray is not nil, its elements are merged with the elements in the body array (mergeArray data overrides bodyArray data).
// The protocol of the connection determines what kinds of marshalling is taking place.
func (r *httpRequest) SetBodyArray(bodyArray interface{}, mergeArray []map[string]interface{}) (driver.Request, error) {
return r, r.bodyBuilder.SetBodyArray(bodyArray, mergeArray)
}
// SetBodyImportArray sets the content of the request as an array formatted for importing documents.
// The protocol of the connection determines what kinds of marshalling is taking place.
func (r *httpRequest) SetBodyImportArray(bodyArray interface{}) (driver.Request, error) {
err := r.bodyBuilder.SetBodyImportArray(bodyArray)
if err == nil {
if r.velocyPack {
r.SetQuery("type", "list")
}
}
return r, err
}
func isNil(v reflect.Value) bool {
switch v.Kind() {
case reflect.Chan, reflect.Func, reflect.Map, reflect.Ptr, reflect.Interface, reflect.Slice:
return v.IsNil()
default:
return false
}
}
// SetHeader sets a single header arguments of the request.
// Any existing header argument with the same key is overwritten.
func (r *httpRequest) SetHeader(key, value string) driver.Request {
if r.hdr == nil {
r.hdr = make(map[string]string)
}
if strings.EqualFold(key, "Content-Type") {
switch strings.ToLower(value) {
case "application/octet-stream":
case "application/zip":
r.bodyBuilder = NewBinaryBodyBuilder(strings.ToLower(value))
}
}
r.hdr[key] = value
return r
}
// Written returns true as soon as this request has been written completely to the network.
// This does not guarantee that the server has received or processed the request.
func (r *httpRequest) Written() bool {
return r.written
}
// WroteRequest implements the WroteRequest function of an httptrace.
// It sets written to true.
func (r *httpRequest) WroteRequest(httptrace.WroteRequestInfo) {
r.written = true
}
// createHTTPRequest creates a golang http.Request based on the configured arguments.
func (r *httpRequest) createHTTPRequest(endpoint url.URL) (*http.Request, error) {
r.written = false
u := endpoint
u.Path = ""
url := u.String()
if !strings.HasSuffix(url, "/") {
url = url + "/"
}
p := r.path
if strings.HasPrefix(p, "/") {
p = p[1:]
}
url = url + p
if r.q != nil {
q := r.q.Encode()
if len(q) > 0 {
url = url + "?" + q
}
}
var bodyReader io.Reader
body := r.bodyBuilder.GetBody()
if body != nil {
bodyReader = bytes.NewReader(body)
}
req, err := http.NewRequest(r.method, url, bodyReader)
if err != nil {
return nil, driver.WithStack(err)
}
if r.hdr != nil {
for k, v := range r.hdr {
req.Header.Set(k, v)
}
}
if r.velocyPack {
req.Header.Set("Accept", "application/x-velocypack")
}
if body != nil {
req.Header.Set("Content-Length", strconv.Itoa(len(body)))
req.Header.Set("Content-Type", r.bodyBuilder.GetContentType())
}
return req, nil
}
type jsonBody struct {
body []byte
}
func NewJsonBodyBuilder() *jsonBody {
return &jsonBody{}
}
// SetBody sets the content of the request.
// The protocol of the connection determines what kinds of marshalling is taking place.
func (b *jsonBody) SetBody(body ...interface{}) error {
switch len(body) {
case 0:
return driver.WithStack(fmt.Errorf("Must provide at least 1 body"))
case 1:
if data, err := json.Marshal(body[0]); err != nil {
return driver.WithStack(err)
} else {
b.body = data
}
return nil
case 2:
mo := mergeObject{Object: body[1], Merge: body[0]}
if data, err := json.Marshal(mo); err != nil {
return driver.WithStack(err)
} else {
b.body = data
}
return nil
default:
return driver.WithStack(fmt.Errorf("Must provide at most 2 bodies"))
}
}
// SetBodyArray sets the content of the request as an array.
// If the given mergeArray is not nil, its elements are merged with the elements in the body array (mergeArray data overrides bodyArray data).
// The protocol of the connection determines what kinds of marshalling is taking place.
func (b *jsonBody) SetBodyArray(bodyArray interface{}, mergeArray []map[string]interface{}) error {
bodyArrayVal := reflect.ValueOf(bodyArray)
switch bodyArrayVal.Kind() {
case reflect.Array, reflect.Slice:
// OK
default:
return driver.WithStack(driver.InvalidArgumentError{Message: fmt.Sprintf("bodyArray must be slice, got %s", bodyArrayVal.Kind())})
}
if mergeArray == nil {
// Simple case; just marshal bodyArray directly.
if data, err := json.Marshal(bodyArray); err != nil {
return driver.WithStack(err)
} else {
b.body = data
}
return nil
}
// Complex case, mergeArray is not nil
elementCount := bodyArrayVal.Len()
mergeObjects := make([]mergeObject, elementCount)
for i := 0; i < elementCount; i++ {
mergeObjects[i] = mergeObject{
Object: bodyArrayVal.Index(i).Interface(),
Merge: mergeArray[i],
}
}
// Now marshal merged array
if data, err := json.Marshal(mergeObjects); err != nil {
return driver.WithStack(err)
} else {
b.body = data
}
return nil
}
// SetBodyImportArray sets the content of the request as an array formatted for importing documents.
// The protocol of the connection determines what kinds of marshalling is taking place.
func (b *jsonBody) SetBodyImportArray(bodyArray interface{}) error {
bodyArrayVal := reflect.ValueOf(bodyArray)
switch bodyArrayVal.Kind() {
case reflect.Array, reflect.Slice:
// OK
default:
return driver.WithStack(driver.InvalidArgumentError{Message: fmt.Sprintf("bodyArray must be slice, got %s", bodyArrayVal.Kind())})
}
// Render elements
elementCount := bodyArrayVal.Len()
buf := &bytes.Buffer{}
encoder := json.NewEncoder(buf)
for i := 0; i < elementCount; i++ {
entryVal := bodyArrayVal.Index(i)
if isNil(entryVal) {
buf.WriteString("\n")
} else {
if err := encoder.Encode(entryVal.Interface()); err != nil {
return driver.WithStack(err)
}
}
}
b.body = buf.Bytes()
return nil
}
func (b *jsonBody) GetBody() []byte {
return b.body
}
func (b *jsonBody) GetContentType() string {
return "application/json"
}
func (b *jsonBody) Clone() driver.BodyBuilder {
return &jsonBody{
body: b.GetBody(),
}
}
type binaryBody struct {
body []byte
contentType string
}
func NewBinaryBodyBuilder(contentType string) *binaryBody {
b := binaryBody{
contentType: contentType,
}
return &b
}
// SetBody sets the content of the request.
// The protocol of the connection determines what kinds of marshalling is taking place.
func (b *binaryBody) SetBody(body ...interface{}) error {
if len(body) == 0 {
return driver.WithStack(fmt.Errorf("must provide at least 1 body"))
}
if data, ok := body[0].([]byte); ok {
b.body = data
return nil
}
return driver.WithStack(fmt.Errorf("must provide body as a []byte type"))
}
func (b *binaryBody) SetBodyArray(_ interface{}, _ []map[string]interface{}) error {
return nil
}
func (b *binaryBody) SetBodyImportArray(_ interface{}) error {
return nil
}
func (b *binaryBody) GetBody() []byte {
return b.body
}
func (b *binaryBody) GetContentType() string {
return b.contentType
}
func (b *binaryBody) Clone() driver.BodyBuilder {
return &binaryBody{
body: b.GetBody(),
contentType: b.GetContentType(),
}
}