// 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)
}
}