2015-05-26 09:44:02 -05:00
|
|
|
package aws
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
|
2016-01-27 10:45:52 -06:00
|
|
|
"github.com/hashicorp/terraform/helper/acctest"
|
2015-07-30 15:17:37 -05:00
|
|
|
"github.com/hashicorp/terraform/helper/resource"
|
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
|
|
"github.com/aws/aws-sdk-go/service/s3"
|
2015-05-26 09:44:02 -05:00
|
|
|
)
|
|
|
|
|
2015-09-09 16:13:36 -05:00
|
|
|
func TestAccAWSS3BucketObject_source(t *testing.T) {
|
2016-02-24 09:04:02 -06:00
|
|
|
tmpFile, err := ioutil.TempFile("", "tf-acc-s3-obj-source")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
2016-01-27 10:45:52 -06:00
|
|
|
rInt := acctest.RandInt()
|
2015-05-26 09:44:02 -05:00
|
|
|
// first write some data to the tempfile just so it's not 0 bytes.
|
2016-02-24 09:04:02 -06:00
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("{anything will do }"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-02-24 13:18:22 -06:00
|
|
|
var obj s3.GetObjectOutput
|
2016-02-24 09:04:02 -06:00
|
|
|
|
2015-05-26 09:44:02 -05:00
|
|
|
resource.Test(t, resource.TestCase{
|
2016-02-24 09:04:02 -06:00
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
2015-05-26 09:44:02 -05:00
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
2016-02-24 09:04:02 -06:00
|
|
|
Config: testAccAWSS3BucketObjectConfigSource(rInt, tmpFile.Name()),
|
2016-02-24 13:18:22 -06:00
|
|
|
Check: testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
2015-09-09 16:13:36 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAccAWSS3BucketObject_content(t *testing.T) {
|
2016-01-27 10:45:52 -06:00
|
|
|
rInt := acctest.RandInt()
|
2016-02-24 13:18:22 -06:00
|
|
|
var obj s3.GetObjectOutput
|
2016-02-24 09:04:02 -06:00
|
|
|
|
2015-09-09 16:13:36 -05:00
|
|
|
resource.Test(t, resource.TestCase{
|
2016-02-24 09:04:02 -06:00
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
2015-09-09 16:13:36 -05:00
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
2016-02-24 09:04:02 -06:00
|
|
|
PreConfig: func() {},
|
|
|
|
Config: testAccAWSS3BucketObjectConfigContent(rInt),
|
2016-02-24 13:18:22 -06:00
|
|
|
Check: testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
2015-05-26 09:44:02 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-09-17 14:10:35 -05:00
|
|
|
func TestAccAWSS3BucketObject_withContentCharacteristics(t *testing.T) {
|
2016-02-24 09:04:02 -06:00
|
|
|
tmpFile, err := ioutil.TempFile("", "tf-acc-s3-obj-content-characteristics")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
2016-01-27 10:45:52 -06:00
|
|
|
rInt := acctest.RandInt()
|
2015-09-17 14:10:35 -05:00
|
|
|
// first write some data to the tempfile just so it's not 0 bytes.
|
2016-02-24 09:04:02 -06:00
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("{anything will do }"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2016-02-24 13:18:22 -06:00
|
|
|
var obj s3.GetObjectOutput
|
|
|
|
|
2015-09-17 14:10:35 -05:00
|
|
|
resource.Test(t, resource.TestCase{
|
2016-02-24 09:04:02 -06:00
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
2015-09-17 14:10:35 -05:00
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
2016-02-24 09:04:02 -06:00
|
|
|
Config: testAccAWSS3BucketObjectConfig_withContentCharacteristics(rInt, tmpFile.Name()),
|
2015-09-17 14:10:35 -05:00
|
|
|
Check: resource.ComposeTestCheckFunc(
|
2016-02-24 13:18:22 -06:00
|
|
|
testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
2015-09-17 14:10:35 -05:00
|
|
|
resource.TestCheckResourceAttr(
|
|
|
|
"aws_s3_bucket_object.object", "content_type", "binary/octet-stream"),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-02-24 09:17:22 -06:00
|
|
|
func TestAccAWSS3BucketObject_updates(t *testing.T) {
|
|
|
|
tmpFile, err := ioutil.TempFile("", "tf-acc-s3-obj-updates")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
|
|
|
rInt := acctest.RandInt()
|
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("initial object state"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-02-24 13:18:22 -06:00
|
|
|
var obj s3.GetObjectOutput
|
2016-02-24 09:17:22 -06:00
|
|
|
|
|
|
|
resource.Test(t, resource.TestCase{
|
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
|
|
|
Config: testAccAWSS3BucketObjectConfig_updates(rInt, tmpFile.Name()),
|
|
|
|
Check: resource.ComposeTestCheckFunc(
|
2016-02-24 13:18:22 -06:00
|
|
|
testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
2016-02-24 09:17:22 -06:00
|
|
|
resource.TestCheckResourceAttr("aws_s3_bucket_object.object", "etag", "647d1d58e1011c743ec67d5e8af87b53"),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
resource.TestStep{
|
|
|
|
PreConfig: func() {
|
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("modified object"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Config: testAccAWSS3BucketObjectConfig_updates(rInt, tmpFile.Name()),
|
|
|
|
Check: resource.ComposeTestCheckFunc(
|
2016-02-24 13:18:22 -06:00
|
|
|
testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
2016-02-24 09:17:22 -06:00
|
|
|
resource.TestCheckResourceAttr("aws_s3_bucket_object.object", "etag", "1c7fd13df1515c2a13ad9eb068931f09"),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-02-24 13:18:22 -06:00
|
|
|
func TestAccAWSS3BucketObject_updatesWithVersioning(t *testing.T) {
|
|
|
|
tmpFile, err := ioutil.TempFile("", "tf-acc-s3-obj-updates-w-versions")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
|
|
|
rInt := acctest.RandInt()
|
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("initial versioned object state"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var originalObj, modifiedObj s3.GetObjectOutput
|
|
|
|
|
|
|
|
resource.Test(t, resource.TestCase{
|
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
|
|
|
Config: testAccAWSS3BucketObjectConfig_updatesWithVersioning(rInt, tmpFile.Name()),
|
|
|
|
Check: resource.ComposeTestCheckFunc(
|
|
|
|
testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &originalObj),
|
|
|
|
resource.TestCheckResourceAttr("aws_s3_bucket_object.object", "etag", "cee4407fa91906284e2a5e5e03e86b1b"),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
resource.TestStep{
|
|
|
|
PreConfig: func() {
|
|
|
|
err = ioutil.WriteFile(tmpFile.Name(), []byte("modified versioned object"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Config: testAccAWSS3BucketObjectConfig_updatesWithVersioning(rInt, tmpFile.Name()),
|
|
|
|
Check: resource.ComposeTestCheckFunc(
|
|
|
|
testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &modifiedObj),
|
|
|
|
resource.TestCheckResourceAttr("aws_s3_bucket_object.object", "etag", "00b8c73b1b50e7cc932362c7225b8e29"),
|
|
|
|
testAccCheckAWSS3BucketObjectVersionIdDiffers(&originalObj, &modifiedObj),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func testAccCheckAWSS3BucketObjectVersionIdDiffers(first, second *s3.GetObjectOutput) resource.TestCheckFunc {
|
|
|
|
return func(s *terraform.State) error {
|
|
|
|
if first.VersionId == nil {
|
|
|
|
return fmt.Errorf("Expected first object to have VersionId: %s", first)
|
|
|
|
}
|
|
|
|
if second.VersionId == nil {
|
|
|
|
return fmt.Errorf("Expected second object to have VersionId: %s", second)
|
|
|
|
}
|
|
|
|
|
|
|
|
if *first.VersionId == *second.VersionId {
|
|
|
|
return fmt.Errorf("Expected Version IDs to differ, but they are equal (%s)", *first.VersionId)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-26 09:44:02 -05:00
|
|
|
func testAccCheckAWSS3BucketObjectDestroy(s *terraform.State) error {
|
|
|
|
s3conn := testAccProvider.Meta().(*AWSClient).s3conn
|
|
|
|
|
|
|
|
for _, rs := range s.RootModule().Resources {
|
|
|
|
if rs.Type != "aws_s3_bucket_object" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := s3conn.HeadObject(
|
|
|
|
&s3.HeadObjectInput{
|
|
|
|
Bucket: aws.String(rs.Primary.Attributes["bucket"]),
|
|
|
|
Key: aws.String(rs.Primary.Attributes["key"]),
|
2015-07-30 15:17:37 -05:00
|
|
|
IfMatch: aws.String(rs.Primary.Attributes["etag"]),
|
2015-05-26 09:44:02 -05:00
|
|
|
})
|
|
|
|
if err == nil {
|
|
|
|
return fmt.Errorf("AWS S3 Object still exists: %s", rs.Primary.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-02-24 13:18:22 -06:00
|
|
|
func testAccCheckAWSS3BucketObjectExists(n string, obj *s3.GetObjectOutput) resource.TestCheckFunc {
|
2015-05-26 09:44:02 -05:00
|
|
|
return func(s *terraform.State) error {
|
|
|
|
rs, ok := s.RootModule().Resources[n]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("Not Found: %s", n)
|
|
|
|
}
|
|
|
|
|
|
|
|
if rs.Primary.ID == "" {
|
|
|
|
return fmt.Errorf("No S3 Bucket Object ID is set")
|
|
|
|
}
|
|
|
|
|
|
|
|
s3conn := testAccProvider.Meta().(*AWSClient).s3conn
|
2016-02-24 13:18:22 -06:00
|
|
|
out, err := s3conn.GetObject(
|
2015-05-26 09:44:02 -05:00
|
|
|
&s3.GetObjectInput{
|
|
|
|
Bucket: aws.String(rs.Primary.Attributes["bucket"]),
|
|
|
|
Key: aws.String(rs.Primary.Attributes["key"]),
|
2015-07-30 15:17:37 -05:00
|
|
|
IfMatch: aws.String(rs.Primary.Attributes["etag"]),
|
2015-05-26 09:44:02 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("S3Bucket Object error: %s", err)
|
|
|
|
}
|
2016-02-24 13:18:22 -06:00
|
|
|
|
|
|
|
*obj = *out
|
|
|
|
|
2015-05-26 09:44:02 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-03 18:20:01 -06:00
|
|
|
func TestAccAWSS3BucketObject_kms(t *testing.T) {
|
|
|
|
rInt := acctest.RandInt()
|
|
|
|
var obj s3.GetObjectOutput
|
|
|
|
|
|
|
|
resource.Test(t, resource.TestCase{
|
|
|
|
PreCheck: func() { testAccPreCheck(t) },
|
|
|
|
Providers: testAccProviders,
|
|
|
|
CheckDestroy: testAccCheckAWSS3BucketObjectDestroy,
|
|
|
|
Steps: []resource.TestStep{
|
|
|
|
resource.TestStep{
|
|
|
|
PreConfig: func() {},
|
|
|
|
Config: testAccAWSS3BucketObjectConfig_withKMSId(rInt),
|
|
|
|
Check: testAccCheckAWSS3BucketObjectExists("aws_s3_bucket_object.object", &obj),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-02-24 09:04:02 -06:00
|
|
|
func testAccAWSS3BucketObjectConfigSource(randInt int, source string) string {
|
2016-01-27 10:45:52 -06:00
|
|
|
return fmt.Sprintf(`
|
2015-05-26 09:44:02 -05:00
|
|
|
resource "aws_s3_bucket" "object_bucket" {
|
2015-09-10 13:37:17 -05:00
|
|
|
bucket = "tf-object-test-bucket-%d"
|
2015-05-26 09:44:02 -05:00
|
|
|
}
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
2015-07-30 15:17:37 -05:00
|
|
|
bucket = "${aws_s3_bucket.object_bucket.bucket}"
|
2015-05-26 09:44:02 -05:00
|
|
|
key = "test-key"
|
|
|
|
source = "%s"
|
2015-09-17 14:10:35 -05:00
|
|
|
content_type = "binary/octet-stream"
|
|
|
|
}
|
2016-02-24 09:04:02 -06:00
|
|
|
`, randInt, source)
|
2016-01-27 10:45:52 -06:00
|
|
|
}
|
2015-09-17 14:10:35 -05:00
|
|
|
|
2016-02-24 09:04:02 -06:00
|
|
|
func testAccAWSS3BucketObjectConfig_withContentCharacteristics(randInt int, source string) string {
|
2016-01-27 10:45:52 -06:00
|
|
|
return fmt.Sprintf(`
|
2015-09-17 14:10:35 -05:00
|
|
|
resource "aws_s3_bucket" "object_bucket_2" {
|
|
|
|
bucket = "tf-object-test-bucket-%d"
|
|
|
|
}
|
|
|
|
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
|
|
|
bucket = "${aws_s3_bucket.object_bucket_2.bucket}"
|
|
|
|
key = "test-key"
|
|
|
|
source = "%s"
|
|
|
|
content_language = "en"
|
|
|
|
content_type = "binary/octet-stream"
|
2015-05-26 09:44:02 -05:00
|
|
|
}
|
2016-02-24 09:04:02 -06:00
|
|
|
`, randInt, source)
|
2016-01-27 10:45:52 -06:00
|
|
|
}
|
2015-09-09 16:13:36 -05:00
|
|
|
|
2016-01-27 10:45:52 -06:00
|
|
|
func testAccAWSS3BucketObjectConfigContent(randInt int) string {
|
|
|
|
return fmt.Sprintf(`
|
2015-09-09 16:13:36 -05:00
|
|
|
resource "aws_s3_bucket" "object_bucket" {
|
|
|
|
bucket = "tf-object-test-bucket-%d"
|
|
|
|
}
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
|
|
|
bucket = "${aws_s3_bucket.object_bucket.bucket}"
|
|
|
|
key = "test-key"
|
|
|
|
content = "some_bucket_content"
|
|
|
|
}
|
2016-01-27 10:45:52 -06:00
|
|
|
`, randInt)
|
|
|
|
}
|
2016-02-24 09:17:22 -06:00
|
|
|
|
|
|
|
func testAccAWSS3BucketObjectConfig_updates(randInt int, source string) string {
|
|
|
|
return fmt.Sprintf(`
|
|
|
|
resource "aws_s3_bucket" "object_bucket_3" {
|
|
|
|
bucket = "tf-object-test-bucket-%d"
|
|
|
|
}
|
|
|
|
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
|
|
|
bucket = "${aws_s3_bucket.object_bucket_3.bucket}"
|
|
|
|
key = "updateable-key"
|
|
|
|
source = "%s"
|
|
|
|
etag = "${md5(file("%s"))}"
|
|
|
|
}
|
|
|
|
`, randInt, source, source)
|
|
|
|
}
|
2016-02-24 13:18:22 -06:00
|
|
|
|
|
|
|
func testAccAWSS3BucketObjectConfig_updatesWithVersioning(randInt int, source string) string {
|
|
|
|
return fmt.Sprintf(`
|
|
|
|
resource "aws_s3_bucket" "object_bucket_3" {
|
|
|
|
bucket = "tf-object-test-bucket-%d"
|
|
|
|
versioning {
|
|
|
|
enabled = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
|
|
|
bucket = "${aws_s3_bucket.object_bucket_3.bucket}"
|
|
|
|
key = "updateable-key"
|
|
|
|
source = "%s"
|
|
|
|
etag = "${md5(file("%s"))}"
|
|
|
|
}
|
|
|
|
`, randInt, source, source)
|
|
|
|
}
|
2016-03-03 18:20:01 -06:00
|
|
|
|
|
|
|
func testAccAWSS3BucketObjectConfig_withKMSId(randInt int) string {
|
|
|
|
return fmt.Sprintf(`
|
2016-03-09 21:06:36 -06:00
|
|
|
resource "aws_kms_key" "kms_key_1" {
|
|
|
|
}
|
|
|
|
|
2016-03-03 18:20:01 -06:00
|
|
|
resource "aws_s3_bucket" "object_bucket_2" {
|
|
|
|
bucket = "tf-object-test-bucket-%d"
|
|
|
|
}
|
|
|
|
|
|
|
|
resource "aws_s3_bucket_object" "object" {
|
|
|
|
bucket = "${aws_s3_bucket.object_bucket_2.bucket}"
|
|
|
|
key = "test-key"
|
|
|
|
content = "stuff"
|
2016-04-22 15:10:34 -05:00
|
|
|
kms_key_id = "${aws_kms_key.kms_key_1.arn}"
|
2016-03-03 18:20:01 -06:00
|
|
|
}
|
|
|
|
`, randInt)
|
|
|
|
}
|