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
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(),
|
|
}
|
|
}
|
|
|