// Copyright 2017-2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance with the License. A copy of the License is located at // // http://aws.amazon.com/apache2.0/ // // or in the "license" file accompanying this file. This file 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. package awsv2 import ( "context" "encoding/json" "fmt" "net/http" "net/http/httptest" "strings" "testing" "time" "github.com/aws/aws-sdk-go-v2/aws" "github.com/aws/aws-sdk-go-v2/service/route53" "github.com/aws/aws-sdk-go-v2/service/route53/types" "github.com/aws/aws-xray-sdk-go/strategy/ctxmissing" "github.com/aws/aws-xray-sdk-go/xray" ) func TestAWSV2(t *testing.T) { cases := map[string]struct { responseStatus int responseBody []byte expectedRegion string expectedError string expectedRequestID string expectedStatusCode int }{ "fault response": { responseStatus: 500, responseBody: []byte(` Tried to create resource record set duplicate.example.com. type A, but it already exists b25f48e8-84fd-11e6-80d9-574e0c4664cb `), expectedRegion: "us-east-1", expectedError: "Error", expectedRequestID: "b25f48e8-84fd-11e6-80d9-574e0c4664cb", expectedStatusCode: 500, }, "error response": { responseStatus: 404, responseBody: []byte(` Sender MalformedXML 1 validation error detected: Value null at 'route53#ChangeSet' failed to satisfy constraint: Member must not be null 1234567890A `), expectedRegion: "us-west-1", expectedError: "Error", expectedRequestID: "1234567890A", expectedStatusCode: 404, }, "success response": { responseStatus: 200, responseBody: []byte(` mockComment mockID `), expectedRegion: "us-west-2", expectedStatusCode: 200, }, } for name, c := range cases { server := httptest.NewServer(http.HandlerFunc( func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(c.responseStatus) _, err := w.Write(c.responseBody) if err != nil { t.Fatal(err) } })) defer server.Close() t.Run(name, func(t *testing.T) { ctx, root := xray.BeginSegment(context.Background(), "AWSSDKV2_Route53") svc := route53.NewFromConfig(aws.Config{ Region: c.expectedRegion, EndpointResolver: aws.EndpointResolverFunc(func(service, region string) (aws.Endpoint, error) { return aws.Endpoint{ URL: server.URL, SigningName: "route53", }, nil }), Retryer: func() aws.Retryer { return aws.NopRetryer{} }, }) _, _ = svc.ChangeResourceRecordSets(ctx, &route53.ChangeResourceRecordSetsInput{ ChangeBatch: &types.ChangeBatch{ Changes: []types.Change{}, Comment: aws.String("mock"), }, HostedZoneId: aws.String("zone"), }, func(options *route53.Options) { AWSV2Instrumentor(&options.APIOptions) }) root.Close(nil) seg := xray.GetSegment(ctx) var subseg *xray.Segment _ = json.Unmarshal(seg.Subsegments[0], &subseg) if e, a := "Route 53", subseg.Name; !strings.EqualFold(e, a) { t.Errorf("expected segment name to be %s, got %s", e, a) } if e, a := c.expectedRegion, fmt.Sprintf("%v", subseg.GetAWS()["region"]); !strings.EqualFold(e, a) { t.Errorf("expected subsegment name to be %s, got %s", e, a) } if e, a := "ChangeResourceRecordSets", fmt.Sprintf("%v", subseg.GetAWS()["operation"]); !strings.EqualFold(e, a) { t.Errorf("expected operation to be %s, got %s", e, a) } if e, a := fmt.Sprint(c.expectedStatusCode), fmt.Sprintf("%v", subseg.GetHTTP().GetResponse().Status); !strings.EqualFold(e, a) { t.Errorf("expected status code to be %s, got %s", e, a) } if e, a := "aws", subseg.Namespace; !strings.EqualFold(e, a) { t.Errorf("expected namespace to be %s, got %s", e, a) } if subseg.GetAWS()[xray.RequestIDKey] != nil { if e, a := c.expectedRequestID, fmt.Sprintf("%v", subseg.GetAWS()[xray.RequestIDKey]); !strings.EqualFold(e, a) { t.Errorf("expected request id to be %s, got %s", e, a) } } }) time.Sleep(1 * time.Second) } } func TestAWSV2WithoutSegment(t *testing.T) { cases := map[string]struct { responseStatus int responseBody []byte }{ "fault response": { responseStatus: 500, responseBody: []byte(` Tried to create resource record set duplicate.example.com. type A, but it already exists b25f48e8-84fd-11e6-80d9-574e0c4664cb `), }, "error response": { responseStatus: 404, responseBody: []byte(` Sender MalformedXML 1 validation error detected: Value null at 'route53#ChangeSet' failed to satisfy constraint: Member must not be null 1234567890A `), }, "success response": { responseStatus: 200, responseBody: []byte(` mockComment mockID `), }, } for name, c := range cases { server := httptest.NewServer(http.HandlerFunc( func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(c.responseStatus) _, err := w.Write(c.responseBody) if err != nil { t.Fatal(err) } })) defer server.Close() t.Run(name, func(t *testing.T) { // Ignore errors when segment cannot be found. ctx, err := xray.ContextWithConfig( context.Background(), xray.Config{ContextMissingStrategy: ctxmissing.NewDefaultIgnoreErrorStrategy()}, ) if err != nil { t.Fatal(err) } svc := route53.NewFromConfig(aws.Config{ EndpointResolver: aws.EndpointResolverFunc(func(service, region string) (aws.Endpoint, error) { return aws.Endpoint{ URL: server.URL, SigningName: "route53", }, nil }), Retryer: func() aws.Retryer { return aws.NopRetryer{} }, }) _, _ = svc.ChangeResourceRecordSets(ctx, &route53.ChangeResourceRecordSetsInput{ ChangeBatch: &types.ChangeBatch{ Changes: []types.Change{}, Comment: aws.String("mock"), }, HostedZoneId: aws.String("zone"), }, func(options *route53.Options) { AWSV2Instrumentor(&options.APIOptions) }) }) time.Sleep(1 * time.Second) } }