-
Notifications
You must be signed in to change notification settings - Fork 1.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add new resource to support IAM custom organization roles (#735)
* Add new resource to support IAM custom organization roles * Add documentation
- Loading branch information
Showing
8 changed files
with
547 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
package google | ||
|
||
import ( | ||
"github.com/hashicorp/terraform/helper/acctest" | ||
"github.com/hashicorp/terraform/helper/resource" | ||
"testing" | ||
) | ||
|
||
func TestAccGoogleOrganizationIamCustomRole_import(t *testing.T) { | ||
t.Parallel() | ||
|
||
skipIfEnvNotSet(t, "GOOGLE_ORG") | ||
roleId := "tfIamRole" + acctest.RandString(10) | ||
|
||
resource.Test(t, resource.TestCase{ | ||
PreCheck: func() { testAccPreCheck(t) }, | ||
Providers: testAccProviders, | ||
CheckDestroy: testAccCheckGoogleOrganizationIamCustomRoleDestroy, | ||
Steps: []resource.TestStep{ | ||
{ | ||
Config: testAccCheckGoogleOrganizationIamCustomRole_update(org, roleId), | ||
}, | ||
{ | ||
ResourceName: "google_organization_iam_custom_role.foo", | ||
ImportState: true, | ||
ImportStateVerify: true, | ||
}, | ||
}, | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,171 @@ | ||
package google | ||
|
||
import ( | ||
"fmt" | ||
"github.com/hashicorp/terraform/helper/schema" | ||
"github.com/hashicorp/terraform/helper/validation" | ||
"google.golang.org/api/iam/v1" | ||
) | ||
|
||
func resourceGoogleOrganizationIamCustomRole() *schema.Resource { | ||
return &schema.Resource{ | ||
Create: resourceGoogleOrganizationIamCustomRoleCreate, | ||
Read: resourceGoogleOrganizationIamCustomRoleRead, | ||
Update: resourceGoogleOrganizationIamCustomRoleUpdate, | ||
Delete: resourceGoogleOrganizationIamCustomRoleDelete, | ||
|
||
Importer: &schema.ResourceImporter{ | ||
State: schema.ImportStatePassthrough, | ||
}, | ||
|
||
Schema: map[string]*schema.Schema{ | ||
"role_id": { | ||
Type: schema.TypeString, | ||
Required: true, | ||
ForceNew: true, | ||
}, | ||
"org_id": { | ||
Type: schema.TypeString, | ||
Required: true, | ||
ForceNew: true, | ||
}, | ||
"title": { | ||
Type: schema.TypeString, | ||
Required: true, | ||
}, | ||
"permissions": { | ||
Type: schema.TypeSet, | ||
Required: true, | ||
MinItems: 1, | ||
Elem: &schema.Schema{Type: schema.TypeString}, | ||
}, | ||
"stage": { | ||
Type: schema.TypeString, | ||
Optional: true, | ||
Default: "GA", | ||
ValidateFunc: validation.StringInSlice([]string{"ALPHA", "BETA", "GA", "DEPRECATED", "DISABLED", "EAP"}, false), | ||
}, | ||
"description": { | ||
Type: schema.TypeString, | ||
Optional: true, | ||
}, | ||
"deleted": { | ||
Type: schema.TypeBool, | ||
Optional: true, | ||
Default: false, | ||
}, | ||
}, | ||
} | ||
} | ||
|
||
func resourceGoogleOrganizationIamCustomRoleCreate(d *schema.ResourceData, meta interface{}) error { | ||
config := meta.(*Config) | ||
|
||
if d.Get("deleted").(bool) { | ||
return fmt.Errorf("Cannot create a custom organization role with a deleted state. `deleted` field should be false.") | ||
} | ||
|
||
role, err := config.clientIAM.Organizations.Roles.Create("organizations/"+d.Get("org_id").(string), &iam.CreateRoleRequest{ | ||
RoleId: d.Get("role_id").(string), | ||
Role: &iam.Role{ | ||
Title: d.Get("title").(string), | ||
Description: d.Get("description").(string), | ||
Stage: d.Get("stage").(string), | ||
IncludedPermissions: convertStringSet(d.Get("permissions").(*schema.Set)), | ||
}, | ||
}).Do() | ||
|
||
if err != nil { | ||
return fmt.Errorf("Error creating the custom organization role %s: %s", d.Get("title").(string), err) | ||
} | ||
|
||
d.SetId(role.Name) | ||
|
||
return resourceGoogleOrganizationIamCustomRoleRead(d, meta) | ||
} | ||
|
||
func resourceGoogleOrganizationIamCustomRoleRead(d *schema.ResourceData, meta interface{}) error { | ||
config := meta.(*Config) | ||
|
||
role, err := config.clientIAM.Organizations.Roles.Get(d.Id()).Do() | ||
if err != nil { | ||
return handleNotFoundError(err, d, d.Id()) | ||
} | ||
|
||
parsedRoleName, err := ParseOrganizationCustomRoleName(role.Name) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
d.Set("role_id", parsedRoleName.Name) | ||
d.Set("org_id", parsedRoleName.OrgId) | ||
d.Set("title", role.Title) | ||
d.Set("description", role.Description) | ||
d.Set("permissions", role.IncludedPermissions) | ||
d.Set("stage", role.Stage) | ||
d.Set("deleted", role.Deleted) | ||
|
||
return nil | ||
} | ||
|
||
func resourceGoogleOrganizationIamCustomRoleUpdate(d *schema.ResourceData, meta interface{}) error { | ||
config := meta.(*Config) | ||
|
||
d.Partial(true) | ||
|
||
if d.HasChange("deleted") { | ||
if d.Get("deleted").(bool) { | ||
if err := resourceGoogleOrganizationIamCustomRoleDelete(d, meta); err != nil { | ||
return err | ||
} | ||
} else { | ||
if err := resourceGoogleOrganizationIamCustomRoleUndelete(d, meta); err != nil { | ||
return err | ||
} | ||
} | ||
d.SetPartial("deleted") | ||
} | ||
|
||
if d.HasChange("title") || d.HasChange("description") || d.HasChange("stage") || d.HasChange("permissions") { | ||
_, err := config.clientIAM.Organizations.Roles.Patch(d.Id(), &iam.Role{ | ||
Title: d.Get("title").(string), | ||
Description: d.Get("description").(string), | ||
Stage: d.Get("stage").(string), | ||
IncludedPermissions: convertStringSet(d.Get("permissions").(*schema.Set)), | ||
}).Do() | ||
|
||
if err != nil { | ||
return fmt.Errorf("Error updating the custom organization role %s: %s", d.Get("title").(string), err) | ||
} | ||
d.SetPartial("title") | ||
d.SetPartial("description") | ||
d.SetPartial("stage") | ||
d.SetPartial("permissions") | ||
} | ||
|
||
d.Partial(false) | ||
|
||
return nil | ||
} | ||
|
||
func resourceGoogleOrganizationIamCustomRoleDelete(d *schema.ResourceData, meta interface{}) error { | ||
config := meta.(*Config) | ||
|
||
_, err := config.clientIAM.Organizations.Roles.Delete(d.Id()).Do() | ||
if err != nil { | ||
return fmt.Errorf("Error deleting the custom organization role %s: %s", d.Get("title").(string), err) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func resourceGoogleOrganizationIamCustomRoleUndelete(d *schema.ResourceData, meta interface{}) error { | ||
config := meta.(*Config) | ||
|
||
_, err := config.clientIAM.Organizations.Roles.Undelete(d.Id(), &iam.UndeleteRoleRequest{}).Do() | ||
if err != nil { | ||
return fmt.Errorf("Error undeleting the custom organization role %s: %s", d.Get("title").(string), err) | ||
} | ||
|
||
return nil | ||
} |
Oops, something went wrong.