//go:build go1.7 // +build go1.7 package s3_test import ( "fmt" "io" "io/ioutil" "net/http" "net/http/httptest" "strconv" "strings" "testing" "time" "github.com/aws/aws-sdk-go/aws" "github.com/aws/aws-sdk-go/aws/awserr" "github.com/aws/aws-sdk-go/aws/corehandlers" "github.com/aws/aws-sdk-go/aws/request" "github.com/aws/aws-sdk-go/awstesting/unit" "github.com/aws/aws-sdk-go/service/s3" ) const errMsg = `ErrorCodemessage bodyrequestIDhostID=` const xmlPreambleMsg = `` var lastModifiedTime = time.Date(2009, 11, 23, 0, 0, 0, 0, time.UTC) func TestCopyObjectNoError(t *testing.T) { const successMsg = ` 2009-11-23T00:00:00Z"1da64c7f13d1e8dbeaea40b905fd586c"` var responseBodyClosed bool res, err := newCopyTestSvc(successMsg, &responseBodyClosed).CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/exists.txt"), Key: aws.String("destination.txt"), }) if err != nil { t.Fatalf("expected no error, but received %v", err) } if e, a := fmt.Sprintf(`%q`, "1da64c7f13d1e8dbeaea40b905fd586c"), *res.CopyObjectResult.ETag; e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := lastModifiedTime, *res.CopyObjectResult.LastModified; !e.Equal(a) { t.Errorf("expected %v, but received %v", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func TestCopyObjectError(t *testing.T) { var responseBodyClosed bool _, err := newCopyTestSvc(errMsg, &responseBodyClosed).CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) if err == nil { t.Error("expected error, but received none") } e := err.(awserr.Error) if e, a := "ErrorCode", e.Code(); e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "message body", e.Message(); e != a { t.Errorf("expected %s, but received %s", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func TestUploadPartCopySuccess(t *testing.T) { const successMsg = ` 2009-11-23T00:00:00Z"1da64c7f13d1e8dbeaea40b905fd586c"` var responseBodyClosed bool res, err := newCopyTestSvc(successMsg, &responseBodyClosed).UploadPartCopy(&s3.UploadPartCopyInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), PartNumber: aws.Int64(0), UploadId: aws.String("uploadID"), }) if err != nil { t.Errorf("expected no error, but received %v", err) } if e, a := fmt.Sprintf(`%q`, "1da64c7f13d1e8dbeaea40b905fd586c"), *res.CopyPartResult.ETag; e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := lastModifiedTime, *res.CopyPartResult.LastModified; !e.Equal(a) { t.Errorf("expected %v, but received %v", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func TestUploadPartCopyError(t *testing.T) { var responseBodyClosed bool _, err := newCopyTestSvc(errMsg, &responseBodyClosed).UploadPartCopy(&s3.UploadPartCopyInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), PartNumber: aws.Int64(0), UploadId: aws.String("uploadID"), }) if err == nil { t.Error("expected an error") } e := err.(awserr.Error) if e, a := "ErrorCode", e.Code(); e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "message body", e.Message(); e != a { t.Errorf("expected %s, but received %s", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func TestCompleteMultipartUploadSuccess(t *testing.T) { const successMsg = ` locationNamebucketNamekeyName"etagVal"` var responseBodyClosed bool res, err := newCopyTestSvc(successMsg, &responseBodyClosed).CompleteMultipartUpload(&s3.CompleteMultipartUploadInput{ Bucket: aws.String("bucketname"), Key: aws.String("key"), UploadId: aws.String("uploadID"), }) if err != nil { t.Errorf("expected no error, but received %v", err) } if e, a := `"etagVal"`, *res.ETag; e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "bucketName", *res.Bucket; e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "keyName", *res.Key; e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "locationName", *res.Location; e != a { t.Errorf("expected %s, but received %s", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func TestCompleteMultipartUploadError(t *testing.T) { var responseBodyClosed bool _, err := newCopyTestSvc(errMsg, &responseBodyClosed).CompleteMultipartUpload(&s3.CompleteMultipartUploadInput{ Bucket: aws.String("bucketname"), Key: aws.String("key"), UploadId: aws.String("uploadID"), }) if err == nil { t.Error("expected an error") } e := err.(awserr.Error) if e, a := "ErrorCode", e.Code(); e != a { t.Errorf("expected %s, but received %s", e, a) } if e, a := "message body", e.Message(); e != a { t.Errorf("expected %s, but received %s", e, a) } if !responseBodyClosed { t.Error("http response body is not closed") } } func newCopyTestSvc(errMsg string, responseBodyClosed *bool) *s3.S3 { const statusCode = http.StatusOK svc := s3.New(unit.Session, &aws.Config{ MaxRetries: aws.Int(0), SleepDelay: func(time.Duration) {}, }) closeChecker := func() error { if responseBodyClosed != nil { *responseBodyClosed = true } return nil } svc.Handlers.Send.Swap(corehandlers.SendHandler.Name, request.NamedHandler{ Name: "newCopyTestSvc", Fn: func(r *request.Request) { io.Copy(ioutil.Discard, r.HTTPRequest.Body) r.HTTPRequest.Body.Close() r.HTTPResponse = &http.Response{ Status: http.StatusText(statusCode), StatusCode: statusCode, Header: http.Header{}, Body: newFuncCloser(strings.NewReader(errMsg), closeChecker), } }, }) return svc } // funcCloser converts io.Reader into io.ReadCloser by adding a custom function that is called on Close() type funcCloser struct { io.Reader closeFn func() error } func newFuncCloser(reader io.Reader, closeFn func() error) *funcCloser { return &funcCloser{Reader: reader, closeFn: closeFn} } func (f funcCloser) Close() error { return f.closeFn() } func TestStatusOKPayloadHandling(t *testing.T) { cases := map[string]struct { Header http.Header Payloads [][]byte OpCall func(*s3.S3) error Err string }{ "200 error": { Header: http.Header{ "Content-Length": []string{strconv.Itoa(len(errMsg))}, }, Payloads: [][]byte{[]byte(errMsg)}, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "ErrorCode: message body", }, "200 error partial response": { Header: http.Header{ "Content-Length": []string{strconv.Itoa(len(errMsg))}, }, Payloads: [][]byte{ []byte(errMsg[:20]), }, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "unexpected EOF", }, "200 error multipart": { Header: http.Header{ "Transfer-Encoding": []string{"chunked"}, }, Payloads: [][]byte{ []byte(errMsg[:20]), []byte(errMsg[20:]), }, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "ErrorCode: message body", }, "200 error multipart partial response": { Header: http.Header{ "Transfer-Encoding": []string{"chunked"}, }, Payloads: [][]byte{ []byte(errMsg[:20]), }, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "XML syntax error", }, "200 error multipart no payload": { Header: http.Header{ "Transfer-Encoding": []string{"chunked"}, }, Payloads: [][]byte{}, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "empty response payload", }, "response with only xml preamble": { Header: http.Header{ "Transfer-Encoding": []string{"chunked"}, }, Payloads: [][]byte{ []byte(xmlPreambleMsg), }, OpCall: func(c *s3.S3) error { _, err := c.CopyObject(&s3.CopyObjectInput{ Bucket: aws.String("bucketname"), CopySource: aws.String("bucketname/doesnotexist.txt"), Key: aws.String("destination.txt"), }) return err }, Err: "empty response payload", }, } for name, c := range cases { t.Run(name, func(t *testing.T) { srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ww := w.(interface { http.ResponseWriter http.Flusher }) for k, vs := range c.Header { for _, v := range vs { ww.Header().Add(k, v) } } ww.WriteHeader(http.StatusOK) ww.Flush() for _, p := range c.Payloads { ww.Write(p) ww.Flush() } })) defer srv.Close() client := s3.New(unit.Session, &aws.Config{ Endpoint: &srv.URL, DisableSSL: aws.Bool(true), DisableParamValidation: aws.Bool(true), S3ForcePathStyle: aws.Bool(true), }) err := c.OpCall(client) if len(c.Err) != 0 { if err == nil { t.Fatalf("expect error, got none") } if e, a := c.Err, err.Error(); !strings.Contains(a, e) { t.Fatalf("expect %v error in, %v", e, a) } } else if err != nil { t.Fatalf("expect no error, got %v", err) } }) } }