// // 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 ( "fmt" "github.com/arangodb/go-velocypack" "github.com/arangodb/go-driver" ) // httpVPackResponseElement implements driver.Response for an entry of an array response. type httpVPackResponseElement struct { statusCode *int slice velocypack.Slice } // StatusCode returns an HTTP compatible status code of the response. func (r *httpVPackResponseElement) StatusCode() int { if r.statusCode == nil { statusCode := 200 // Look for "error" field if errorFieldSlice, _ := r.slice.Get("error"); !errorFieldSlice.IsNone() { if hasError, err := errorFieldSlice.GetBool(); err == nil && hasError { // We have an error, look for code field statusCode = 500 if codeFieldSlice, _ := r.slice.Get("code"); !codeFieldSlice.IsNone() { if code, err := codeFieldSlice.GetInt(); err == nil { statusCode = int(code) } } } } r.statusCode = &statusCode } return *r.statusCode } // Endpoint returns the endpoint that handled the request. func (r *httpVPackResponseElement) Endpoint() string { return "" } // CheckStatus checks if the status of the response equals to one of the given status codes. // If so, nil is returned. // If not, an attempt is made to parse an error response in the body and an error is returned. func (r *httpVPackResponseElement) CheckStatus(validStatusCodes ...int) error { statusCode := r.StatusCode() for _, x := range validStatusCodes { if x == statusCode { // Found valid status code return nil } } // Invalid status code, try to parse arango error response. var aerr driver.ArangoError if err := r.ParseBody("", &aerr); err == nil && aerr.HasError { // Found correct arango error. return aerr } // We do not have a valid error code, so we can only create one based on the HTTP status code. return driver.ArangoError{ HasError: true, Code: statusCode, ErrorMessage: fmt.Sprintf("Unexpected status code %d", statusCode), } } // Header returns the value of a response header with given key. // If no such header is found, an empty string is returned. func (r *httpVPackResponseElement) Header(key string) string { return "" } // ParseBody performs protocol specific unmarshalling of the response data into the given result. // If the given field is non-empty, the contents of that field will be parsed into the given result. func (r *httpVPackResponseElement) ParseBody(field string, result interface{}) error { slice := r.slice if field != "" { var err error slice, err = slice.Get(field) if err != nil { return driver.WithStack(err) } if slice.IsNone() { // Field not found return nil } } if result != nil { if err := velocypack.Unmarshal(slice, result); err != nil { return driver.WithStack(err) } } return nil } // ParseArrayBody performs protocol specific unmarshalling of the response array data into individual response objects. // This can only be used for requests that return an array of objects. func (r *httpVPackResponseElement) ParseArrayBody() ([]driver.Response, error) { return nil, driver.WithStack(driver.InvalidArgumentError{Message: "ParseArrayBody not allowed"}) }