From 9b065f32e23f22a96c2ba9cc930475d8ea75d18f Mon Sep 17 00:00:00 2001 From: "jorge.marey" Date: Fri, 13 Nov 2015 12:50:31 +0100 Subject: [PATCH 1/3] Include content option for file provisioner - Include new option in file provisioner. Now content or source can be provided. Content will create a temp file and copy there the contents. - Later that file will be used as source. - Include test to check that changes are working correctly. --- .../provisioners/file/resource_provisioner.go | 64 ++++++++++++++----- .../file/resource_provisioner_test.go | 35 +++++++++- 2 files changed, 82 insertions(+), 17 deletions(-) diff --git a/builtin/provisioners/file/resource_provisioner.go b/builtin/provisioners/file/resource_provisioner.go index 9484d3f18f4f..ac87d7583a67 100644 --- a/builtin/provisioners/file/resource_provisioner.go +++ b/builtin/provisioners/file/resource_provisioner.go @@ -2,12 +2,12 @@ package file import ( "fmt" + "io/ioutil" "log" "os" "time" "github.com/hashicorp/terraform/communicator" - "github.com/hashicorp/terraform/helper/config" "github.com/hashicorp/terraform/terraform" "github.com/mitchellh/go-homedir" ) @@ -26,18 +26,13 @@ func (p *ResourceProvisioner) Apply( return err } - // Get the source and destination - sRaw := c.Config["source"] - src, ok := sRaw.(string) - if !ok { - return fmt.Errorf("Unsupported 'source' type! Must be string.") - } - - src, err = homedir.Expand(src) + // Get the source + src, err := p.getSrc(c) if err != nil { return err } + // Get destination dRaw := c.Config["destination"] dst, ok := dRaw.(string) if !ok { @@ -48,13 +43,52 @@ func (p *ResourceProvisioner) Apply( // Validate checks if the required arguments are configured func (p *ResourceProvisioner) Validate(c *terraform.ResourceConfig) (ws []string, es []error) { - v := &config.Validator{ - Required: []string{ - "source", - "destination", - }, + numDst := 0 + numSrc := 0 + for name := range c.Raw { + switch name { + case "destination": + numDst++ + case "source", "content": + numSrc++ + default: + es = append(es, fmt.Errorf("Unknown configuration '%s'", name)) + } + } + if numSrc != 1 || numDst != 1 { + es = append(es, fmt.Errorf("Must provide one of 'content' or 'source' and 'destination' to file")) + } + return +} + +// getSrc returns the file to use as source +func (p *ResourceProvisioner) getSrc(c *terraform.ResourceConfig) (string, error) { + var src string + + sRaw, ok := c.Config["source"] + if ok { + if src, ok = sRaw.(string); !ok { + return "", fmt.Errorf("Unsupported 'source' type! Must be string.") + } + } + + content, ok := c.Config["content"] + if ok { + file, err := ioutil.TempFile("", "tf-file-content") + if err != nil { + return "", err + } + contentStr, ok := content.(string) + if !ok { + return "", fmt.Errorf("Unsupported 'content' type! Must be string.") + } + if _, err = file.WriteString(contentStr); err != nil { + return "", err + } + src = file.Name() } - return v.Validate(c) + + return homedir.Expand(src) } // copyFiles is used to copy the files from a source to a destination diff --git a/builtin/provisioners/file/resource_provisioner_test.go b/builtin/provisioners/file/resource_provisioner_test.go index 0fc990cb4cd1..713c8aa0df8e 100644 --- a/builtin/provisioners/file/resource_provisioner_test.go +++ b/builtin/provisioners/file/resource_provisioner_test.go @@ -11,7 +11,7 @@ func TestResourceProvisioner_impl(t *testing.T) { var _ terraform.ResourceProvisioner = new(ResourceProvisioner) } -func TestResourceProvider_Validate_good(t *testing.T) { +func TestResourceProvider_Validate_good_source(t *testing.T) { c := testConfig(t, map[string]interface{}{ "source": "/tmp/foo", "destination": "/tmp/bar", @@ -26,7 +26,22 @@ func TestResourceProvider_Validate_good(t *testing.T) { } } -func TestResourceProvider_Validate_bad(t *testing.T) { +func TestResourceProvider_Validate_good_content(t *testing.T) { + c := testConfig(t, map[string]interface{}{ + "content": "value to copy", + "destination": "/tmp/bar", + }) + p := new(ResourceProvisioner) + warn, errs := p.Validate(c) + if len(warn) > 0 { + t.Fatalf("Warnings: %v", warn) + } + if len(errs) > 0 { + t.Fatalf("Errors: %v", errs) + } +} + +func TestResourceProvider_Validate_bad_not_destination(t *testing.T) { c := testConfig(t, map[string]interface{}{ "source": "nope", }) @@ -40,6 +55,22 @@ func TestResourceProvider_Validate_bad(t *testing.T) { } } +func TestResourceProvider_Validate_bad_to_many_src(t *testing.T) { + c := testConfig(t, map[string]interface{}{ + "source": "nope", + "content": "value to copy", + "destination": "/tmp/bar", + }) + p := new(ResourceProvisioner) + warn, errs := p.Validate(c) + if len(warn) > 0 { + t.Fatalf("Warnings: %v", warn) + } + if len(errs) == 0 { + t.Fatalf("Should have errors") + } +} + func testConfig( t *testing.T, c map[string]interface{}) *terraform.ResourceConfig { From 800f7d2e06ec3ecff20e774dd2234fb24e2ad1ea Mon Sep 17 00:00:00 2001 From: "jorge.marey" Date: Mon, 22 Feb 2016 17:45:15 +0100 Subject: [PATCH 2/3] Update documentation --- .../source/docs/provisioners/file.html.markdown | 16 ++++++++++++++-- 1 file changed, 14 insertions(+), 2 deletions(-) diff --git a/website/source/docs/provisioners/file.html.markdown b/website/source/docs/provisioners/file.html.markdown index be50ed374d83..368219baa344 100644 --- a/website/source/docs/provisioners/file.html.markdown +++ b/website/source/docs/provisioners/file.html.markdown @@ -24,6 +24,12 @@ resource "aws_instance" "web" { destination = "/etc/myapp.conf" } + # Copies the string in content into /tmp/file.log + provisioner "file" { + content = "ami used: ${self.ami}" + destination = "/tmp/file.log" + } + # Copies the configs.d folder to /etc/configs.d provisioner "file" { source = "conf/configs.d" @@ -42,8 +48,14 @@ resource "aws_instance" "web" { The following arguments are supported: -* `source` - (Required) This is the source file or folder. It can be specified as relative - to the current working directory or as an absolute path. +* `source` - This is the source file or folder. It can be specified as relative + to the current working directory or as an absolute path. This cannot be provided with `content`. + +* `content` - This is the content to copy on the destination. If destination is a file, + the content will be written on that file, in case of a directory a file named + *tf-file-content* is created. It's recommended to use a file as destination. A + [`template_file`](/docs/providers/template/r/file.html) might be referenced in here, or + any interpolation syntax for that matter. This cannot be provided with `source`. * `destination` - (Required) This is the destination path. It must be specified as an absolute path. From 8beafe25ae5d126cc355b9d17e030d1e38db7744 Mon Sep 17 00:00:00 2001 From: James Nugent Date: Fri, 8 Jul 2016 19:34:37 +0100 Subject: [PATCH 3/3] provisioner/file: Clean up temporary files --- .../provisioners/file/resource_provisioner.go | 22 ++++++++++++------- 1 file changed, 14 insertions(+), 8 deletions(-) diff --git a/builtin/provisioners/file/resource_provisioner.go b/builtin/provisioners/file/resource_provisioner.go index ac87d7583a67..2cd060b63e43 100644 --- a/builtin/provisioners/file/resource_provisioner.go +++ b/builtin/provisioners/file/resource_provisioner.go @@ -27,10 +27,13 @@ func (p *ResourceProvisioner) Apply( } // Get the source - src, err := p.getSrc(c) + src, deleteSource, err := p.getSrc(c) if err != nil { return err } + if deleteSource { + defer os.Remove(src) + } // Get destination dRaw := c.Config["destination"] @@ -62,13 +65,13 @@ func (p *ResourceProvisioner) Validate(c *terraform.ResourceConfig) (ws []string } // getSrc returns the file to use as source -func (p *ResourceProvisioner) getSrc(c *terraform.ResourceConfig) (string, error) { +func (p *ResourceProvisioner) getSrc(c *terraform.ResourceConfig) (string, bool, error) { var src string sRaw, ok := c.Config["source"] if ok { if src, ok = sRaw.(string); !ok { - return "", fmt.Errorf("Unsupported 'source' type! Must be string.") + return "", false, fmt.Errorf("Unsupported 'source' type! Must be string.") } } @@ -76,19 +79,22 @@ func (p *ResourceProvisioner) getSrc(c *terraform.ResourceConfig) (string, error if ok { file, err := ioutil.TempFile("", "tf-file-content") if err != nil { - return "", err + return "", true, err } + contentStr, ok := content.(string) if !ok { - return "", fmt.Errorf("Unsupported 'content' type! Must be string.") + return "", true, fmt.Errorf("Unsupported 'content' type! Must be string.") } if _, err = file.WriteString(contentStr); err != nil { - return "", err + return "", true, err } - src = file.Name() + + return file.Name(), true, nil } - return homedir.Expand(src) + expansion, err := homedir.Expand(src) + return expansion, false, err } // copyFiles is used to copy the files from a source to a destination