mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 13:50:01 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			258 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			258 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package rest
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"encoding/base64"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"io/ioutil"
 | |
| 	"net/http"
 | |
| 	"reflect"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/aws/aws-sdk-go/aws"
 | |
| 	"github.com/aws/aws-sdk-go/aws/awserr"
 | |
| 	"github.com/aws/aws-sdk-go/aws/request"
 | |
| 	awsStrings "github.com/aws/aws-sdk-go/internal/strings"
 | |
| 	"github.com/aws/aws-sdk-go/private/protocol"
 | |
| )
 | |
| 
 | |
| // UnmarshalHandler is a named request handler for unmarshaling rest protocol requests
 | |
| var UnmarshalHandler = request.NamedHandler{Name: "awssdk.rest.Unmarshal", Fn: Unmarshal}
 | |
| 
 | |
| // UnmarshalMetaHandler is a named request handler for unmarshaling rest protocol request metadata
 | |
| var UnmarshalMetaHandler = request.NamedHandler{Name: "awssdk.rest.UnmarshalMeta", Fn: UnmarshalMeta}
 | |
| 
 | |
| // Unmarshal unmarshals the REST component of a response in a REST service.
 | |
| func Unmarshal(r *request.Request) {
 | |
| 	if r.DataFilled() {
 | |
| 		v := reflect.Indirect(reflect.ValueOf(r.Data))
 | |
| 		if err := unmarshalBody(r, v); err != nil {
 | |
| 			r.Error = err
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // UnmarshalMeta unmarshals the REST metadata of a response in a REST service
 | |
| func UnmarshalMeta(r *request.Request) {
 | |
| 	r.RequestID = r.HTTPResponse.Header.Get("X-Amzn-Requestid")
 | |
| 	if r.RequestID == "" {
 | |
| 		// Alternative version of request id in the header
 | |
| 		r.RequestID = r.HTTPResponse.Header.Get("X-Amz-Request-Id")
 | |
| 	}
 | |
| 	if r.DataFilled() {
 | |
| 		if err := UnmarshalResponse(r.HTTPResponse, r.Data, aws.BoolValue(r.Config.LowerCaseHeaderMaps)); err != nil {
 | |
| 			r.Error = err
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // UnmarshalResponse attempts to unmarshal the REST response headers to
 | |
| // the data type passed in. The type must be a pointer. An error is returned
 | |
| // with any error unmarshaling the response into the target datatype.
 | |
| func UnmarshalResponse(resp *http.Response, data interface{}, lowerCaseHeaderMaps bool) error {
 | |
| 	v := reflect.Indirect(reflect.ValueOf(data))
 | |
| 	return unmarshalLocationElements(resp, v, lowerCaseHeaderMaps)
 | |
| }
 | |
| 
 | |
| func unmarshalBody(r *request.Request, v reflect.Value) error {
 | |
| 	if field, ok := v.Type().FieldByName("_"); ok {
 | |
| 		if payloadName := field.Tag.Get("payload"); payloadName != "" {
 | |
| 			pfield, _ := v.Type().FieldByName(payloadName)
 | |
| 			if ptag := pfield.Tag.Get("type"); ptag != "" && ptag != "structure" {
 | |
| 				payload := v.FieldByName(payloadName)
 | |
| 				if payload.IsValid() {
 | |
| 					switch payload.Interface().(type) {
 | |
| 					case []byte:
 | |
| 						defer r.HTTPResponse.Body.Close()
 | |
| 						b, err := ioutil.ReadAll(r.HTTPResponse.Body)
 | |
| 						if err != nil {
 | |
| 							return awserr.New(request.ErrCodeSerialization, "failed to decode REST response", err)
 | |
| 						}
 | |
| 
 | |
| 						payload.Set(reflect.ValueOf(b))
 | |
| 
 | |
| 					case *string:
 | |
| 						defer r.HTTPResponse.Body.Close()
 | |
| 						b, err := ioutil.ReadAll(r.HTTPResponse.Body)
 | |
| 						if err != nil {
 | |
| 							return awserr.New(request.ErrCodeSerialization, "failed to decode REST response", err)
 | |
| 						}
 | |
| 
 | |
| 						str := string(b)
 | |
| 						payload.Set(reflect.ValueOf(&str))
 | |
| 
 | |
| 					default:
 | |
| 						switch payload.Type().String() {
 | |
| 						case "io.ReadCloser":
 | |
| 							payload.Set(reflect.ValueOf(r.HTTPResponse.Body))
 | |
| 
 | |
| 						case "io.ReadSeeker":
 | |
| 							b, err := ioutil.ReadAll(r.HTTPResponse.Body)
 | |
| 							if err != nil {
 | |
| 								return awserr.New(request.ErrCodeSerialization,
 | |
| 									"failed to read response body", err)
 | |
| 							}
 | |
| 							payload.Set(reflect.ValueOf(ioutil.NopCloser(bytes.NewReader(b))))
 | |
| 
 | |
| 						default:
 | |
| 							io.Copy(ioutil.Discard, r.HTTPResponse.Body)
 | |
| 							r.HTTPResponse.Body.Close()
 | |
| 							return awserr.New(request.ErrCodeSerialization,
 | |
| 								"failed to decode REST response",
 | |
| 								fmt.Errorf("unknown payload type %s", payload.Type()))
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func unmarshalLocationElements(resp *http.Response, v reflect.Value, lowerCaseHeaderMaps bool) error {
 | |
| 	for i := 0; i < v.NumField(); i++ {
 | |
| 		m, field := v.Field(i), v.Type().Field(i)
 | |
| 		if n := field.Name; n[0:1] == strings.ToLower(n[0:1]) {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		if m.IsValid() {
 | |
| 			name := field.Tag.Get("locationName")
 | |
| 			if name == "" {
 | |
| 				name = field.Name
 | |
| 			}
 | |
| 
 | |
| 			switch field.Tag.Get("location") {
 | |
| 			case "statusCode":
 | |
| 				unmarshalStatusCode(m, resp.StatusCode)
 | |
| 
 | |
| 			case "header":
 | |
| 				err := unmarshalHeader(m, resp.Header.Get(name), field.Tag)
 | |
| 				if err != nil {
 | |
| 					return awserr.New(request.ErrCodeSerialization, "failed to decode REST response", err)
 | |
| 				}
 | |
| 
 | |
| 			case "headers":
 | |
| 				prefix := field.Tag.Get("locationName")
 | |
| 				err := unmarshalHeaderMap(m, resp.Header, prefix, lowerCaseHeaderMaps)
 | |
| 				if err != nil {
 | |
| 					awserr.New(request.ErrCodeSerialization, "failed to decode REST response", err)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func unmarshalStatusCode(v reflect.Value, statusCode int) {
 | |
| 	if !v.IsValid() {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	switch v.Interface().(type) {
 | |
| 	case *int64:
 | |
| 		s := int64(statusCode)
 | |
| 		v.Set(reflect.ValueOf(&s))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func unmarshalHeaderMap(r reflect.Value, headers http.Header, prefix string, normalize bool) error {
 | |
| 	if len(headers) == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 	switch r.Interface().(type) {
 | |
| 	case map[string]*string: // we only support string map value types
 | |
| 		out := map[string]*string{}
 | |
| 		for k, v := range headers {
 | |
| 			if awsStrings.HasPrefixFold(k, prefix) {
 | |
| 				if normalize == true {
 | |
| 					k = strings.ToLower(k)
 | |
| 				} else {
 | |
| 					k = http.CanonicalHeaderKey(k)
 | |
| 				}
 | |
| 				out[k[len(prefix):]] = &v[0]
 | |
| 			}
 | |
| 		}
 | |
| 		if len(out) != 0 {
 | |
| 			r.Set(reflect.ValueOf(out))
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func unmarshalHeader(v reflect.Value, header string, tag reflect.StructTag) error {
 | |
| 	switch tag.Get("type") {
 | |
| 	case "jsonvalue":
 | |
| 		if len(header) == 0 {
 | |
| 			return nil
 | |
| 		}
 | |
| 	case "blob":
 | |
| 		if len(header) == 0 {
 | |
| 			return nil
 | |
| 		}
 | |
| 	default:
 | |
| 		if !v.IsValid() || (header == "" && v.Elem().Kind() != reflect.String) {
 | |
| 			return nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	switch v.Interface().(type) {
 | |
| 	case *string:
 | |
| 		v.Set(reflect.ValueOf(&header))
 | |
| 	case []byte:
 | |
| 		b, err := base64.StdEncoding.DecodeString(header)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(b))
 | |
| 	case *bool:
 | |
| 		b, err := strconv.ParseBool(header)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(&b))
 | |
| 	case *int64:
 | |
| 		i, err := strconv.ParseInt(header, 10, 64)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(&i))
 | |
| 	case *float64:
 | |
| 		f, err := strconv.ParseFloat(header, 64)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(&f))
 | |
| 	case *time.Time:
 | |
| 		format := tag.Get("timestampFormat")
 | |
| 		if len(format) == 0 {
 | |
| 			format = protocol.RFC822TimeFormatName
 | |
| 		}
 | |
| 		t, err := protocol.ParseTime(format, header)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(&t))
 | |
| 	case aws.JSONValue:
 | |
| 		escaping := protocol.NoEscape
 | |
| 		if tag.Get("location") == "header" {
 | |
| 			escaping = protocol.Base64Escape
 | |
| 		}
 | |
| 		m, err := protocol.DecodeJSONValue(header, escaping)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		v.Set(reflect.ValueOf(m))
 | |
| 	default:
 | |
| 		err := fmt.Errorf("Unsupported value for param %v (%s)", v.Interface(), v.Type())
 | |
| 		return err
 | |
| 	}
 | |
| 	return nil
 | |
| }
 |