providers/heroku: add heroku_app_feature resource (#14035)
This commit is contained in:
parent
2af1bd9fe0
commit
e495c6bacc
|
@ -1,7 +1,9 @@
|
||||||
package heroku
|
package heroku
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/hashicorp/terraform/helper/schema"
|
"github.com/hashicorp/terraform/helper/schema"
|
||||||
"github.com/hashicorp/terraform/terraform"
|
"github.com/hashicorp/terraform/terraform"
|
||||||
|
@ -27,6 +29,7 @@ func Provider() terraform.ResourceProvider {
|
||||||
ResourcesMap: map[string]*schema.Resource{
|
ResourcesMap: map[string]*schema.Resource{
|
||||||
"heroku_addon": resourceHerokuAddon(),
|
"heroku_addon": resourceHerokuAddon(),
|
||||||
"heroku_app": resourceHerokuApp(),
|
"heroku_app": resourceHerokuApp(),
|
||||||
|
"heroku_app_feature": resourceHerokuAppFeature(),
|
||||||
"heroku_cert": resourceHerokuCert(),
|
"heroku_cert": resourceHerokuCert(),
|
||||||
"heroku_domain": resourceHerokuDomain(),
|
"heroku_domain": resourceHerokuDomain(),
|
||||||
"heroku_drain": resourceHerokuDrain(),
|
"heroku_drain": resourceHerokuDrain(),
|
||||||
|
@ -46,3 +49,12 @@ func providerConfigure(d *schema.ResourceData) (interface{}, error) {
|
||||||
log.Println("[INFO] Initializing Heroku client")
|
log.Println("[INFO] Initializing Heroku client")
|
||||||
return config.Client()
|
return config.Client()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func buildCompositeID(a, b string) string {
|
||||||
|
return fmt.Sprintf("%s:%s", a, b)
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCompositeID(id string) (string, string) {
|
||||||
|
parts := strings.SplitN(id, ":", 2)
|
||||||
|
return parts[0], parts[1]
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,101 @@
|
||||||
|
package heroku
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
|
||||||
|
heroku "github.com/cyberdelia/heroku-go/v3"
|
||||||
|
"github.com/hashicorp/terraform/helper/schema"
|
||||||
|
)
|
||||||
|
|
||||||
|
func resourceHerokuAppFeature() *schema.Resource {
|
||||||
|
return &schema.Resource{
|
||||||
|
Create: resourceAppFeatureCreate,
|
||||||
|
Update: resourceAppFeatureUpdate,
|
||||||
|
Read: resourceAppFeatureRead,
|
||||||
|
Delete: resourceAppFeatureDelete,
|
||||||
|
|
||||||
|
Schema: map[string]*schema.Schema{
|
||||||
|
"app": {
|
||||||
|
Type: schema.TypeString,
|
||||||
|
Required: true,
|
||||||
|
ForceNew: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
"name": {
|
||||||
|
Type: schema.TypeString,
|
||||||
|
Required: true,
|
||||||
|
ForceNew: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
"enabled": {
|
||||||
|
Type: schema.TypeBool,
|
||||||
|
Optional: true,
|
||||||
|
Default: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func resourceAppFeatureRead(d *schema.ResourceData, meta interface{}) error {
|
||||||
|
client := meta.(*heroku.Service)
|
||||||
|
|
||||||
|
app, id := parseCompositeID(d.Id())
|
||||||
|
|
||||||
|
feature, err := client.AppFeatureInfo(context.TODO(), app, id)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
d.Set("app", app)
|
||||||
|
d.Set("name", feature.Name)
|
||||||
|
d.Set("enabled", feature.Enabled)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func resourceAppFeatureCreate(d *schema.ResourceData, meta interface{}) error {
|
||||||
|
client := meta.(*heroku.Service)
|
||||||
|
|
||||||
|
app := d.Get("app").(string)
|
||||||
|
featureName := d.Get("name").(string)
|
||||||
|
enabled := d.Get("enabled").(bool)
|
||||||
|
|
||||||
|
opts := heroku.AppFeatureUpdateOpts{Enabled: enabled}
|
||||||
|
|
||||||
|
log.Printf("[DEBUG] Feature set configuration: %#v, %#v", featureName, opts)
|
||||||
|
|
||||||
|
feature, err := client.AppFeatureUpdate(context.TODO(), app, featureName, opts)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
d.SetId(buildCompositeID(app, feature.ID))
|
||||||
|
|
||||||
|
return resourceAppFeatureRead(d, meta)
|
||||||
|
}
|
||||||
|
|
||||||
|
func resourceAppFeatureUpdate(d *schema.ResourceData, meta interface{}) error {
|
||||||
|
if d.HasChange("enabled") {
|
||||||
|
return resourceAppFeatureCreate(d, meta)
|
||||||
|
}
|
||||||
|
|
||||||
|
return resourceAppFeatureRead(d, meta)
|
||||||
|
}
|
||||||
|
|
||||||
|
func resourceAppFeatureDelete(d *schema.ResourceData, meta interface{}) error {
|
||||||
|
client := meta.(*heroku.Service)
|
||||||
|
|
||||||
|
app, id := parseCompositeID(d.Id())
|
||||||
|
featureName := d.Get("name").(string)
|
||||||
|
|
||||||
|
log.Printf("[INFO] Deleting app feature %s (%s) for app %s", featureName, id, app)
|
||||||
|
opts := heroku.AppFeatureUpdateOpts{Enabled: false}
|
||||||
|
_, err := client.AppFeatureUpdate(context.TODO(), app, id, opts)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
d.SetId("")
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -0,0 +1,135 @@
|
||||||
|
package heroku
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
heroku "github.com/cyberdelia/heroku-go/v3"
|
||||||
|
"github.com/hashicorp/terraform/helper/acctest"
|
||||||
|
"github.com/hashicorp/terraform/helper/resource"
|
||||||
|
"github.com/hashicorp/terraform/terraform"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAccHerokuAppFeature(t *testing.T) {
|
||||||
|
var feature heroku.AppFeatureInfoResult
|
||||||
|
appName := fmt.Sprintf("tftest-%s", acctest.RandString(10))
|
||||||
|
|
||||||
|
resource.Test(t, resource.TestCase{
|
||||||
|
PreCheck: func() { testAccPreCheck(t) },
|
||||||
|
Providers: testAccProviders,
|
||||||
|
CheckDestroy: testAccCheckHerokuFeatureDestroy,
|
||||||
|
Steps: []resource.TestStep{
|
||||||
|
{
|
||||||
|
Config: testAccCheckHerokuFeature_basic(appName),
|
||||||
|
Check: resource.ComposeTestCheckFunc(
|
||||||
|
testAccCheckHerokuFeatureExists("heroku_app_feature.runtime_metrics", &feature),
|
||||||
|
testAccCheckHerokuFeatureEnabled(&feature, true),
|
||||||
|
resource.TestCheckResourceAttr(
|
||||||
|
"heroku_app_feature.runtime_metrics", "enabled", "true",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Config: testAccCheckHerokuFeature_disabled(appName),
|
||||||
|
Check: resource.ComposeTestCheckFunc(
|
||||||
|
testAccCheckHerokuFeatureExists("heroku_app_feature.runtime_metrics", &feature),
|
||||||
|
testAccCheckHerokuFeatureEnabled(&feature, false),
|
||||||
|
resource.TestCheckResourceAttr(
|
||||||
|
"heroku_app_feature.runtime_metrics", "enabled", "false",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func testAccCheckHerokuFeatureDestroy(s *terraform.State) error {
|
||||||
|
client := testAccProvider.Meta().(*heroku.Service)
|
||||||
|
|
||||||
|
for _, rs := range s.RootModule().Resources {
|
||||||
|
if rs.Type != "heroku_app_feature" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := client.AppFeatureInfo(context.TODO(), rs.Primary.Attributes["app"], rs.Primary.ID)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
return fmt.Errorf("Feature still exists")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func testAccCheckHerokuFeatureExists(n string, feature *heroku.AppFeatureInfoResult) resource.TestCheckFunc {
|
||||||
|
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 feature ID is set")
|
||||||
|
}
|
||||||
|
|
||||||
|
app, id := parseCompositeID(rs.Primary.ID)
|
||||||
|
if app != rs.Primary.Attributes["app"] {
|
||||||
|
return fmt.Errorf("Bad app: %s", app)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := testAccProvider.Meta().(*heroku.Service)
|
||||||
|
|
||||||
|
foundFeature, err := client.AppFeatureInfo(context.TODO(), app, id)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if foundFeature.ID != id {
|
||||||
|
return fmt.Errorf("Feature not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
*feature = *foundFeature
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func testAccCheckHerokuFeatureEnabled(feature *heroku.AppFeatureInfoResult, enabled bool) resource.TestCheckFunc {
|
||||||
|
return func(s *terraform.State) error {
|
||||||
|
if feature.Enabled != enabled {
|
||||||
|
return fmt.Errorf("Bad enabled: %v", feature.Enabled)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func testAccCheckHerokuFeature_basic(appName string) string {
|
||||||
|
return fmt.Sprintf(`
|
||||||
|
resource "heroku_app" "example" {
|
||||||
|
name = "%s"
|
||||||
|
region = "us"
|
||||||
|
}
|
||||||
|
|
||||||
|
resource "heroku_app_feature" "runtime_metrics" {
|
||||||
|
app = "${heroku_app.example.name}"
|
||||||
|
name = "log-runtime-metrics"
|
||||||
|
}
|
||||||
|
`, appName)
|
||||||
|
}
|
||||||
|
|
||||||
|
func testAccCheckHerokuFeature_disabled(appName string) string {
|
||||||
|
return fmt.Sprintf(`
|
||||||
|
resource "heroku_app" "example" {
|
||||||
|
name = "%s"
|
||||||
|
region = "us"
|
||||||
|
}
|
||||||
|
|
||||||
|
resource "heroku_app_feature" "runtime_metrics" {
|
||||||
|
app = "${heroku_app.example.name}"
|
||||||
|
name = "log-runtime-metrics"
|
||||||
|
enabled = false
|
||||||
|
}
|
||||||
|
`, appName)
|
||||||
|
}
|
|
@ -0,0 +1,28 @@
|
||||||
|
---
|
||||||
|
layout: "heroku"
|
||||||
|
page_title: "Heroku: heroku_app_feature"
|
||||||
|
sidebar_current: "docs-heroku-resource-app-feature"
|
||||||
|
description: |-
|
||||||
|
Provides a Heroku App Feature resource. This can be used to create and manage App Features on Heroku.
|
||||||
|
---
|
||||||
|
|
||||||
|
# heroku\_app\_feature
|
||||||
|
|
||||||
|
Provides a Heroku App Feature resource. This can be used to create and manage App Features on Heroku.
|
||||||
|
|
||||||
|
## Example Usage
|
||||||
|
|
||||||
|
```hcl
|
||||||
|
resource "heroku_app_feature" "log_runtime_metrics" {
|
||||||
|
app = "test-app"
|
||||||
|
name = "log-runtime-metrics"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Argument Reference
|
||||||
|
|
||||||
|
The following arguments are supported:
|
||||||
|
|
||||||
|
* `app` - (Required) The Heroku app to link to.
|
||||||
|
* `name` - (Required) The name of the App Feature to manage.
|
||||||
|
* `enabled` - (Optional) Whether to enable or disable the App Feature. The default value is true.
|
Loading…
Reference in New Issue