Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Migrate provider from cloudfoundry-community to cloudfoundry #57

Merged
merged 19 commits into from
Dec 9, 2024
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .terraform-version
Original file line number Diff line number Diff line change
@@ -1 +1 @@
1.8.5
1.9.8
73 changes: 35 additions & 38 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,15 +10,14 @@ Creates an RDS database based on the `rds_plan_name` variable and outputs the `i

```
module "database" {
source = "github.com/GSA-TTS/terraform-cloudgov//database?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//database?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = local.cf_space_name
name = "database_name"
rds_plan_name = "micro-psql"
tags = ["tag1", "tag2"]
cf_space_id = data.cloudfoundry_space.app_space.id
name = "database_name"
rds_plan_name = "micro-psql"
tags = ["tag1", "tag2"]
# See options at https://cloud.gov/docs/services/relational-database/#setting-optional-parameters-1
json_params = jsonencode(
json_params = jsonencode(
{
"storage" : 10,
}
Expand All @@ -32,17 +31,16 @@ Creates a Elasticache redis instance and outputs the `instance_id` for use elsew

```
module "redis" {
source = "github.com/GSA-TTS/terraform-cloudgov//redis?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//redis?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = local.cf_space_name
name = "redis_name"
redis_plan_name = "redis-dev"
tags = ["tag1", "tag2"]
cf_space_id = data.cloudfoundry_space.app_space.id
name = "redis_name"
redis_plan_name = "redis-dev"
tags = ["tag1", "tag2"]
# See options at https://cloud.gov/docs/services/aws-elasticache/#setting-optional-parameters
json_params = jsonencode(
json_params = jsonencode(
{
"engineVersion" : "6.2",
"engineVersion" : "7.0",
}
)
}
Expand All @@ -54,14 +52,13 @@ Creates an s3 bucket and outputs the `bucket_id` for use elsewhere.

```
module "s3" {
source = "github.com/GSA-TTS/terraform-cloudgov//s3?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//s3?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = local.cf_space_name
name = "${local.app_name}-s3-${local.env}"
tags = ["tag1", "tag2"]
cf_space_id = data.cloudfoundry_space.app_space.id
name = "${local.app_name}-s3-${local.env}"
tags = ["tag1", "tag2"]
# See options at https://cloud.gov/docs/services/s3/#setting-optional-parameters
json_params = jsonencode(
json_params = jsonencode(
{
"object_ownership" : "ObjectWriter",
}
Expand All @@ -75,19 +72,19 @@ Connects a custom domain name or domain name with CDN to an already running appl

Note that the domain must be created in cloud.gov by an OrgManager before this module is included.

`cf create-domain CLOUD_GOV_ORG my-production-domain-name`
`cf create-domain CLOUD_GOV_ORG my-production-domain.name`

```
module "domain" {
source = "github.com/GSA-TTS/terraform-cloudgov//domain?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//domain?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = local.cf_space_name
app_name_or_id = "app_name"
cdn_plan_name = "domain"
domain_name = "my-production-domain-name"
host_name = "my-production-host-name"
tags = ["tag1", "tag2"]
cf_org_name = local.cf_org_name
cf_space = data.cloudfoundry_space.app_space
app_names = ["app_name"]
cdn_plan_name = "domain"
domain_name = "my-production-domain.name"
host_name = "my-production-host-name"
tags = ["tag1", "tag2"]
}
```

Expand All @@ -101,11 +98,10 @@ Notes:

```
module "clamav" {
source = "github.com/GSA-TTS/terraform-cloudgov//clamav?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//clamav?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = local.cf_space_name
app_name_or_id = "app_name"
name = "my_clamav_name"
clamav_image = "ghcr.io/gsa-tts/clamav-rest/clamav:TAG_NAME"
max_file_size = "30M"
Expand All @@ -129,10 +125,11 @@ Creates a new cloud.gov space, such as when creating an egress space, and output

```
module "egress_space" {
source = "github.com/GSA-TTS/terraform-cloudgov//cg_space?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//cg_space?ref=v2.0.0"
mogul marked this conversation as resolved.
Show resolved Hide resolved

cf_org_name = local.cf_org_name
cf_space_name = "${local.cf_space_name}-egress"
allow_ssh = false
managers = [
"[email protected]"
]
Expand All @@ -156,12 +153,12 @@ Prerequities:

```
module "egress_proxy" {
source = "github.com/GSA-TTS/terraform-cloudgov//egress_proxy?ref=v1.1.0"
source = "github.com/GSA-TTS/terraform-cloudgov//egress_proxy?ref=v2.0.0"

cf_org_name = local.cf_org_name
cf_space_name = "${local.cf_space_name}-egress"
client_space = local.cf_space_name
name = "egress-proxy"
cf_org_name = local.cf_org_name
cf_egress_space = data.cloudfoundry_space.egress_space
cf_client_spaces = {(data.cloudfoundry_space.app_space.name) = data.cloudfoundy_space.app_space.id}
name = "egress-proxy"
allowlist = {
"source_app_name" = ["host.com:443", "otherhost.com:443"]
rahearn marked this conversation as resolved.
Show resolved Hide resolved
}
Expand Down
1 change: 1 addition & 0 deletions SECURITY.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ Only certain branches are supported with security updates.
| Version (branch) | Supported |
| ---------------- | --------- |
| main | :white_check_mark: |
| v1 | :white_check_mark: |
| other | :x: |

When using this code or reporting vulnerability please be sure to use supported branches and the most recent release tag.
Expand Down
119 changes: 119 additions & 0 deletions UPGRADING.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
# Upgrading from v1 to v2

The terraform-cloudgov modules have many backwards-incompatible changes between v1 and v2. These changes are mostly around:

1. Changing from the [cloudfoundry-community](https://registry.terraform.io/providers/cloudfoundry-community/cloudfoundry/latest/docs) provider to the [cloudfoundry](https://registry.terraform.io/providers/cloudfoundry/cloudfoundry/latest/docs) provider.
1. Changes to inputs and outputs and resources created to better fit with the new provider and as the result of lessons learned over the life of the v1 code base.

## Using v1 and v2 together

It is possible to use modules from both v1 and v2 in the same root module, to ease migration (or even remain on v1 for existing resources and use v2 for new ones). We will continue to maintain the v1 branch for awhile with bug-fixes as needed.

Specify both providers in your root module (here, `cloudfoundry-community` is probably called `cloudfoundry` in your old module):

```
terraform {
required_version = "~> 1.0"
required_providers {
cloudfoundry = {
source = "cloudfoundry/cloudfoundry"
version = "1.1.0"
}

cloudfoundry-community = {
source = "cloudfoundry-community/cloudfoundry"
version = "0.53.1"
}
}
}

provider "cloudfoundry" {
api_url = "https://api.fr.cloud.gov"
user = var.cf_user
password = var.cf_password
}

provider "cloudfoundry-community" {
api_url = "https://api.fr.cloud.gov"
user = var.cf_user
password = var.cf_password
}
```

The v1 modules should properly select the `cloudfoundry-community` provider, but if they don't you may need to [explicitely set the provider](https://developer.hashicorp.com/terraform/language/modules/develop/providers#passing-providers-explicitly):
rahearn marked this conversation as resolved.
Show resolved Hide resolved

```
module "database" {
source = "github.com/gsa-tts/terraform-cloudgov//database?ref=v1.1.0"
providers = {
cloudfoundry = cloudfoundry-community
}

# ...
}
```

## Provider Upgrades

Follow the steps in the [cloudfoundry provider migration guide](https://github.com/cloudfoundry/terraform-provider-cloudfoundry/blob/main/migration-guide/Readme.md) to migrate an existing use of the v1 module to v2. As an example, here are the steps for upgrading a database module:

1. Update source line to point to v2 module and change `cf_space_name` to `cf_space_id`
1. Verify that `terraform validate` passes
1. Run: `terraform state show module.database.cloudfoundry_service_instance.rds | grep -m 1 id` and copy the ID
1. Run: `terraform state rm module.database.cloudfoundry_service_instance.rds`
1. Run: `terraform import module.database.cloudfoundry_service_instance.rds ID_FROM_STEP_3`
1. Run: `terraform apply` to fill in new computed attributes

## Module Changes

### Common Changes

1. Check the `variables.tf` and `outputs.tf` files for each module for new names of variables and outputs. There should not be any variables or outputs that kept the same name but changed behavior.

### Egress Proxy

Egress Proxy no longer sets up network policies between the proxy and client apps, and does not create a User Provided Service Instance to deliver the credentials to the app. It is the developer's responsibility to do those things in the root module to better handle circular dependencies between creating the client app(s) and the proxy.

To setup the network policy in your root module, add:

```
resource "cloudfoundry_network_policy" "egress_policy" {
provider = cloudfoundry-community
policy {
source_app = cloudfoundry_app.client_app.id # assumes you're deploying the client app with terraform
destination_app = module.egress_proxy.app_id
port = module.egress_proxy.https_port
}
}
```

To add a UPSI:

```
resource "cloudfoundry_service_instance" "egress_proxy_credentials" {
name = "egress-proxy-credentials"
space = module.app_space.space_id
type = "user-provided"
credentials = module.egress_proxy.json_credentials
}
```

### Clamav
rahearn marked this conversation as resolved.
Show resolved Hide resolved

Clamav no longer sets up network policies between the clamav app and client apps. It is the developer's responsibility to set this up to better handle circular dependencies between the various apps.
rahearn marked this conversation as resolved.
Show resolved Hide resolved

rahearn marked this conversation as resolved.
Show resolved Hide resolved
### cg_space

Migrating the space roles is a little trickier. Here is a path forward:
rahearn marked this conversation as resolved.
Show resolved Hide resolved

1. Upgrade the space module source to v2
1. `terraform apply -target=module.space.cloudfoundry_space_users.space_permissions` to remove the old permissions resources.
1. Manually add your terraform user as a SpaceDeveloper and a SpaceManager to the space.
```
cf set-space-role CF_USER_GUID ORG SPACE SpaceDeveloper
cf set-space-role CF_USER_GUID ORG SPACE SpaceManager
```
1. Ensure that your terraform user is _not_ listed in the terraform as a deployer, manager, or developer
1. `terraform apply` to add new permissions resources.
1. Use `cf unset-space-role` to remove the manual permissions settings or destroy the service account entirely to clean up after yourself.
1. (Optional, if you didn't destroy the service account) Re-add your terraform user to the terraform permissions configuration and have a different deployer apply it (like via CI/CD)
48 changes: 16 additions & 32 deletions cg_space/main.tf
Original file line number Diff line number Diff line change
Expand Up @@ -3,46 +3,30 @@ data "cloudfoundry_org" "org" {
}

resource "cloudfoundry_space" "space" {
name = var.cf_space_name
org = data.cloudfoundry_org.org.id
name = var.cf_space_name
org = data.cloudfoundry_org.org.id
allow_ssh = var.allow_ssh
}

###
# User roles
###

data "cloudfoundry_user" "managers" {
for_each = var.managers
name = each.key
org_id = data.cloudfoundry_org.org.id
}

data "cloudfoundry_user" "developers" {
for_each = var.developers
name = each.key
org_id = data.cloudfoundry_org.org.id
}

data "cloudfoundry_user" "deployers" {
for_each = var.deployers
name = each.key
org_id = data.cloudfoundry_org.org.id
locals {
manager_names = setunion(var.managers, var.deployers)
rahearn marked this conversation as resolved.
Show resolved Hide resolved
developer_names = setunion(var.developers, var.deployers)
}


locals {
manager_ids = concat(
[for user in data.cloudfoundry_user.managers : user.id],
[for user in data.cloudfoundry_user.deployers : user.id]
)
developer_ids = concat(
[for user in data.cloudfoundry_user.developers : user.id],
[for user in data.cloudfoundry_user.deployers : user.id]
)
resource "cloudfoundry_space_role" "managers" {
for_each = local.manager_names
username = each.key
space = cloudfoundry_space.space.id
type = "space_manager"
}

resource "cloudfoundry_space_users" "space_permissions" {
space = cloudfoundry_space.space.id
managers = local.manager_ids
developers = local.developer_ids
resource "cloudfoundry_space_role" "developers" {
for_each = local.developer_names
username = each.key
space = cloudfoundry_space.space.id
type = "space_developer"
}
4 changes: 4 additions & 0 deletions cg_space/outputs.tf
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,7 @@ output "space_id" {
output "space_name" {
value = cloudfoundry_space.space.name
}

output "space" {
value = cloudfoundry_space.space
}
4 changes: 2 additions & 2 deletions cg_space/providers.tf
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,8 @@ terraform {
required_version = "~> 1.0"
required_providers {
cloudfoundry = {
source = "cloudfoundry-community/cloudfoundry"
version = ">=0.53.1"
source = "cloudfoundry/cloudfoundry"
version = ">=1.1.0"
}
}
}
Loading
Loading