Skip to content

Commit

Permalink
feat: begins the work to support single vs multi root_module_structure (
Browse files Browse the repository at this point in the history
#17)

## what

- This is a first cut at supporting `MultiInstance` vs `SingleInstance`
root module structures.
- Adds an additional example for the single-instance work. 
- Both examples are now applied to our masterpoint Spacelift tenant.
- Work on the README to explain Multi vs Single. Please read and provide
critical feedback.
- Includes an update to renovate to automerge PRs for non major revs.

## why

- People structure their terraform and root modules in different ways.
We need to support both.

## references

- Some discussion in Slack:
https://masterpoint.slack.com/archives/C04MUCKUDKK/p1734543200130119
- Original design session with @gberenice + @oycyc on this topic:
https://app.fireflies.ai/view/Spacelift-Automation-Airtable-Design::cNXp0mkZEAUuzhLm


<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit

- **New Features**
- Introduced automated merging of specific dependency updates in
configuration.
	- Added support for AWS integration with new settings.
	- New variable for configuring the length of generated names.
	- New module for automation with AWS integration settings.
	- Enhanced stack settings for better management and description.

- **Bug Fixes**
	- Updated variable names for clarity and consistency.

- **Documentation**
- Improved README documentation for better clarity on module usage and
structure.
- Added examples and distinctions between MultiInstance and
SingleInstance structures.

- **Chores**
- Updated GitHub Actions workflow with a new environment variable for
API key endpoint.
	- Removed outdated backend configuration file.
<!-- end of auto-generated comment: release notes by coderabbit.ai -->

---------

Co-authored-by: Veronika Gnilitska <30597968+gberenice@users.noreply.github.com>
  • Loading branch information
Gowiem and gberenice authored Dec 26, 2024
1 parent 507f296 commit 598f0c7
Show file tree
Hide file tree
Showing 33 changed files with 423 additions and 101 deletions.
5 changes: 5 additions & 0 deletions .github/renovate.json5
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,11 @@
"baseBranches": ["main", "master"],
"labels": ["auto-upgrade"],
"dependencyDashboardAutoclose": true,
"packageRules": [{
// Allow auto merge if it's not a major version update
"matchUpdateTypes": ["minor", "patch", "pin", "digest"],
"automerge": true
}],
"terraform": {
"ignorePaths": [
"**/context.tf",
Expand Down
1 change: 1 addition & 0 deletions .github/workflows/tf-test.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ on:
pull_request:

env:
SPACELIFT_API_KEY_ENDPOINT: ${{ secrets.SPACELIFT_API_KEY_ENDPOINT }}
SPACELIFT_API_KEY_ID: ${{ secrets.SPACELIFT_API_KEY_ID }}
SPACELIFT_API_KEY_SECRET: ${{ secrets.SPACELIFT_API_KEY_SECRET }}

Expand Down
135 changes: 107 additions & 28 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ This Terraform child module provides infrastructure automation for projects in [

## Overview

The `spacelift-automation` root module is designed to streamline the deployment and management of all Spacelift infrastructure, including itself.
The `spacelift-automation` root module is designed to streamline the deployment and management of all Spacelift infrastructure, including creating a Spacelift Stack to manage itself.

It automates the creation of "child" stacks and all the required accompanying Spacelift resources. For each enabled root module it creates:

Expand All @@ -17,22 +17,27 @@ It automates the creation of "child" stacks and all the required accompanying Sp
3. [Spacelift AWS Integration Attachment](https://docs.spacelift.io/integrations/cloud-providers/aws#lets-explain)
Associates a specific AWS IAM role with a stack to allow it to assume that role. The IAM role typically has permissions to manage specific AWS resources, and Spacelift assumes this role to run the operations required by the stack.
4. [Spacelift Initialization Hook](https://docs.spacelift.io/concepts/run#initializing)
Prepares your environment before executing infrastructure code. This custom script copies corresponding Terraform tfvars files into a working directory before either run or task as a `spacelift.auto.tfvars` file. It's [automatically loaded](https://opentofu.org/docs/v1.7/language/values/variables/#variable-definitions-tfvars-files) into the OpenTofu/Terraform execution environment.
Prepares your environment before executing infrastructure code. This custom script copies corresponding Terraform tfvars files into a working directory before any Spacelift run or task as a `spacelift.auto.tfvars` file. This ensures your tfvars are [automatically loaded](https://opentofu.org/docs/v1.7/language/values/variables/#variable-definitions-tfvars-files) into the OpenTofu/Terraform execution environment.

## Usage

Spacelift Automation logic is opinionated and heavily relies on the Git repository structure.
This module is configured to track all the files in the provided root module directory and create the stack based on the provided configuration (if any).
Spacelift Automation logic is opinionated and heavily relies on certain repository structures.
This module is configured to track all the files in the given root module directory and create Spacelift Stacks based on the provided configuration.

Structure requirements are:
We support the following root module directory structures, which are controlled by the `var.root_modules_structure` variable:

- Stack configs are placed in `<root_module>/stacks` directory.
- Terraform variables are placed in `<root_module>/tfvars` directory.
- Stack config file and tfvars file must be equal to OpenTofu/Terraform workspace, e.g. `dev.yaml` and `dev.tfvars`.
- Common configs are placed in `<root_module>/stacks/common.yaml` file. This is useful when you know that some values should be shared across all the stacks created for a root module, e.g. all stacks that manage Spacelift Policy must be Administrative. You can override this file name using Terraform variable.
### `MultiInstance` (the default)

Let's check the example.
Input repo structure:
This is the default structure that we expect and recommend. This is intended for root modules that manage multiple state files (instances) through [workspaces](https://opentofu.org/docs/cli/workspaces/) or [Dynamic Backend configurations](https://opentofu.org/docs/intro/whats-new/#early-variablelocals-evaluation).

Structure requirements:

- Stack configs are placed in `<root_modules_path>/<root_module>/stacks` directory for each workspace / instance of that stack. e.g. `root-modules/k8s-cluster/stacks/dev.yaml` and `root-modules/k8s-cluster/stacks/stage.yaml`
- Terraform variables are placed in `<root_modules_path>/<root_module>/tfvars` directory for each workspace / instance of that stack. e.g. `root-modules/k8s-cluster/tfvars/dev.tfvars` and `root-modules/k8s-cluster/tfvars/stage.tfvars`
- Stack config files and tfvars files must be equal to OpenTofu/Terraform workspace, e.g. `stacks/dev.yaml` and `tfvars/dev.tfvars` for a workspace named `dev`.
- Common configs are placed in `<root_modules_path>/<root_module>/stacks/common.yaml` file (or `var.common_config_file` value). This is useful when you know that some values should be shared across all the stacks created for a root module. For example, all stacks that manage Spacelift Policies must use the `administrative: true` setting or all stacks must share the same labels.

We have an example of this structure in the [examples/complete](./examples/complete/components/), which looks like the following:

```sh
├── root-modules
Expand All @@ -45,6 +50,7 @@ Input repo structure:
│ │ │ └── dev.tfvars
│ │ │ └── stage.tfvars
│ │ ├── variables.tf
│ │ ├── main.tf
│ │ └── versions.tf
│ ├── k8s-cluster
│ │ ├── stacks
Expand All @@ -60,26 +66,27 @@ Input repo structure:
...
```

Root module inputs:
The `spacelift-automation/main.tf` file looks something like this:

```hcl
aws_integration_id = "ZDPP8SKNVG0G27T4"
# GitHub configuration
github_enterprise = {
namespace = "masterpointio"
}
repository = "terraform-spacelift-automation"
# Stacks configurations
root_modules_path = "root-modules"
enabled_root_modules = ["spacelift-aws-role"]
root_modules_path = "root-modules"
all_root_modules_enabled = true
aws_integration_id = "ZDPP8SKNVG0G27T4"
```

The configuration above creates the following stacks:

- `spacelift-aws-role-dev`
- `spacelift-aws-role-stage`
- `k8s-cluster-dev`
- `k8s-cluster-prod`

These stacks have the following configuration:

Expand All @@ -88,29 +95,101 @@ These stacks have the following configuration:
- Corresponding Terraform variables are generated by an [Initialization Hook](https://docs.spacelift.io/concepts/run#initializing) and placed in the root of each Stack's working directory during each run or task. For example, the content of the file `root-modules/spacelift-aws-role/tfvars/dev.tfvars` will be copied to working directory of the Stack `spacelift-aws-role-dev` as file `spacelift.auto.tfvars` allowing the OpenTofu/Terraform inputs to be automatically loaded.
- If you would like to disable this functionality, you can set `tfvars.enabled` in the Stack's YAML file to `false`.

## FAQs
### `SingleInstance`

### Why are variable values provided separately in `tfvars/` and not in the `yaml` file?
This is a special case where each root module directory only manages one state file (instance). Each time you want to create a new instance of a root module, you need to create a new directory with the same code and change your inputs. **We do not recommend this structure** as it is less flexible and easily leads to anti-patterns, but it is supported.

Structure requirements:

- Stack configs are placed in `<root_modules_path>/<root_module>/stack.yaml` directory. e.g. `root-modules/rds-cluster/stack.yaml`
- Tfvars values are not supported in this structure. In this structure, we suggest you just add your tfvars as `***.auto.tfvars` or hardcode your values directly in root module code.

This is to support easy local and outside-spacelift operations. Keeping variable values in a `tfvars` file per workspace allows you to simply pass that file to the relevant CLI command locally via the `-var-file` option so that you don't need to provide values individually.
Here is an example of this structure that we have in the [examples/single-instance](./examples/single-instance/) directory:

```sh
├── root-modules
│ ├── spacelift-automation
│ │ ├── stack.yaml
│ │ ├── variables.tf
│ │ ├── main.tf
│ │ └── versions.tf
│ ├── rds-cluster-dev
│ │ ├── stack.yaml
│ │ ├── main.tf
│ │ └── versions.tf
│ ├── rds-cluster-prod
│ │ ├── stack.yaml
│ │ ├── main.tf
│ │ └── versions.tf
│ ├── random-pet
│ │ ├── stack.yaml
│ │ ├── variables.tf
│ │ ├── main.tf
│ │ └── versions.tf
...
```

The configuration above creates the following Spacelift Stacks:

- `spacelift-automation`
- `rds-cluster-dev`
- `rds-cluster-prod`
- `random-pet`

These stacks will be configured using the settings in the `stack.yaml` file.

## FAQs

### Can I create a Spacelift Stack for Spacelift Automation? (Recommended)

Spacelift Automation can manage itself as a Stack as well, and we recommend this so you can fully automate your Stack management upon merging to your given branch. Follow these next steps to achieve that:
Spacelift Automation can manage itself as a Stack as well, and we recommend this so you can fully automate your Stack management upon merging to your given branch. Follow these steps to achieve that:

1. Create a new vanilla OpenTofu/Terraform root module in `<root_modules_path>/spacelift-automation` that consumes this child module and supplies the necessary configuration for your unique setup. e.g.

```hcl
# root-modules/spacelift-automation/main.tf
module "spacelift-automation" {
source = "masterpointio/automation/spacelift"
version = "x.x.x" # Always pin a version, use the latest version from the release page.
# GitHub configuration
github_enterprise = {
namespace = "masterpointio"
}
repository = "your-infrastructure-repo"
# Stacks configurations
root_modules_path = "../../root-modules"
all_root_modules_enabled = true
aws_integration_id = "ZDPP8SKNVG0G27T4"
}
```

1. Create a new vanilla OpenTofu/Terraform root module that consumes this child module and supplies the necessary configuration for your unique setup. In other words, it's a configuration that uses the default capabilities of either OpenTofu or Terraform without any customization, or third-party tools or plugins.
2. Optionally, create a Terraform workspace that will be used for your Automation configuration, e.g.:

```sh
tofu workspace new masterpoint
tofu workspace new main
```
Remember that Stack config and tfvars file name must be equal to the workspace, which can be `default`.
3. Apply the vanilla OpenTofu/Terraform configuration.

Remember that Stack config and tfvars file name must be equal to the workspace e.g. `main.yaml` and `main.tfvars`. If you choose not to create a new workspace, this can be `default.yaml` and `default.tfvars`.

3. Apply the `spacelift-automation` root module.
4. Move the Automation configs to the `<root-modules>/spacelift-automation/stacks` directory and push the changes to the tracked repo and branch.
5. From this moment, Spacelift Automation is tracking the changes to its Stack configs and Terraform variables.
5. After pushed to your repo's tracked branch, Spacelift Automation will track the addition of new root modules and create Stacks for them.

Check out an example of such a configuration in the [examples/complete](./examples/complete/components/spacelift-automation/tfvars/example.tfvars).
Check out an example configuration in the [examples/complete](./examples/complete/components/spacelift-automation/tfvars/example.tfvars).

<!-- NOTE to Masterpoint team: We might want to create a small wrapper to automatize this using Taskit. On hold for now. -->

### What goes in a Stack config file? e.g. `stacks/dev.yaml`, `stacks/common.yaml`, `stack.yaml`, etc

Most settings that you would set on [the Spacelift Stack resource](https://search.opentofu.org/provider/spacelift-io/spacelift/latest/docs/resources/stack) are supported. Additionally, you can include certain Stack specific settings that will override this module's defaults like `default_tf_workspace_enabled`, `tfvars.enabled`, and similar. See the code for full details.

### Why are variable values provided separately in `tfvars/` and not in the `yaml` file?

NOTE to Masterpoint team: We might want to create a small wrapper to automatize this using Taskit. On hold for now.
This is to support easy local and outside-spacelift operations. Keeping variable values in a `tfvars` file per workspace allows you to simply pass that file to the relevant CLI command locally via the `-var-file` option so that you don't need to provide values individually. e.g. `tofu plan -var-file=tfvars/dev.tfvars`

<!-- BEGINNING OF PRE-COMMIT-TERRAFORM DOCS HOOK -->

Expand Down
18 changes: 0 additions & 18 deletions examples/complete/components/spacelift-automation/backend.tf.json

This file was deleted.

3 changes: 2 additions & 1 deletion examples/complete/components/spacelift-automation/main.tf
Original file line number Diff line number Diff line change
Expand Up @@ -8,5 +8,6 @@ module "automation" {
root_modules_path = var.root_modules_path
all_root_modules_enabled = var.all_root_modules_enabled

aws_integration_id = var.aws_integration_id
aws_integration_id = var.aws_integration_id
aws_integration_enabled = true
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
stack_settings:
administrative: true
aws_integration_enabled: true
labels:
- common_label
Original file line number Diff line number Diff line change
Expand Up @@ -4,4 +4,4 @@ github_enterprise = {
repository = "terraform-spacelift-automation"
root_modules_path = "../../../../examples/complete/components"
all_root_modules_enabled = true
aws_integration_id = "01J30JBKQTCD72ATZCRWHYST3C"
aws_integration_id = "01JEC7ZACVKHTSVY4NF8QNZVVB"
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
length = 10
3 changes: 3 additions & 0 deletions examples/single-instance/root-modules/random-pet/main.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
resource "random_pet" "template" {
length = var.length
}
7 changes: 7 additions & 0 deletions examples/single-instance/root-modules/random-pet/stack.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
stack_settings:
manage_state: true
description: This stack generates random pet names
labels:
- common_label
- stack_specific_label
default_tf_workspace_enabled: true
4 changes: 4 additions & 0 deletions examples/single-instance/root-modules/random-pet/variables.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
variable "length" {
description = "The length of the random name"
type = number
}
10 changes: 10 additions & 0 deletions examples/single-instance/root-modules/random-pet/versions.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
terraform {
required_version = "~> 1.0"

required_providers {
random = {
source = "hashicorp/random"
version = "~> 3.0"
}
}
}
5 changes: 5 additions & 0 deletions examples/single-instance/root-modules/rds-cluster-dev/main.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
resource "null_resource" "example" {
triggers = {
timestamp = timestamp()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
stack_settings:
description: This is a mock root module for Dev
10 changes: 10 additions & 0 deletions examples/single-instance/root-modules/rds-cluster-dev/versions.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
terraform {
required_version = "~> 1.0.0"

required_providers {
null = {
source = "hashicorp/null"
version = "~> 3.0"
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
resource "null_resource" "example" {
triggers = {
timestamp = timestamp()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
stack_settings:
description: This is a mock root module for Prod
10 changes: 10 additions & 0 deletions examples/single-instance/root-modules/rds-cluster-prod/versions.tf
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
terraform {
required_version = "~> 1.0.0"

required_providers {
null = {
source = "hashicorp/null"
version = "~> 3.0"
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
module "automation" {
source = "../../../../"

github_enterprise = {
namespace = "masterpointio"
}
repository = "terraform-spacelift-automation"
root_modules_path = "../../../../examples/single-instance/root-modules"
all_root_modules_enabled = true

aws_integration_id = "01JEC7ZACVKHTSVY4NF8QNZVVB"
aws_integration_enabled = true

root_module_structure = "SingleInstance"
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
stack_settings:
description: This Automation stack is used for Masterpoint's testing purposes
administrative: true
aws_integration_enabled: false
labels:
- common_label
- stack_specific_label
default_tf_workspace_enabled: true
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
terraform {
required_version = "~> 1.0"

required_providers {
spacelift = {
source = "spacelift-io/spacelift"
version = "~> 1.14"
}
}
}
Loading

0 comments on commit 598f0c7

Please sign in to comment.