Skip to content
This repository has been archived by the owner on Mar 19, 2024. It is now read-only.

Commit

Permalink
Custom destinations for modules (#13)
Browse files Browse the repository at this point in the history
* Added destination option to configuration, so each module could be added to one or more specific destination folder
---------

Co-authored-by: Rafal Przybyla <[email protected]>
Co-authored-by: Nathaniel Ritholtz <[email protected]>
  • Loading branch information
3 people authored Feb 15, 2023
1 parent f625609 commit 2c10832
Show file tree
Hide file tree
Showing 4 changed files with 242 additions and 28 deletions.
4 changes: 3 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,10 @@

# Terrafile output
vendor/modules
testdata
terrafile

#IDEs
.idea/
dist/
dist/
.DS_Store
70 changes: 63 additions & 7 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,12 @@ curl -L https://github.com/coretech/terrafile/releases/download/v{VERSION}/terra
## How to use
Terrafile expects a file named `Terrafile` which will contain your terraform module dependencies in a yaml like format.

An example Terrafile:
There are two approaches that can be used for managing your modules depending on the structure of your terraform code:
1. The default approach: `Terrafile` is located directly in the directory where terraform is run
2. Centrally managed: `Terrafile` is located in "root" directory of your terraform code, managing modules in all subfolders / stacks

### Default Approach
An example of default approach (#1) to `Terrafile`
```
tf-aws-vpc:
source: "[email protected]:terraform-aws-modules/terraform-aws-vpc"
Expand All @@ -34,24 +39,75 @@ tf-aws-vpc-experimental:
Terrafile config file in current directory and modules exported to ./vendor/modules
```sh
$ terrafile
INFO[0000] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
```

Terrafile config file in custom directory
```sh
$ terrafile -f config/Terrafile
INFO[0000] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
```

Terraform modules exported to custom directory
```sh
$ terrafile -p custom_directory
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0001] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0000] [*] Checking out master of [email protected]:terraform-aws-modules/terraform-aws-vpc
INFO[0001] [*] Checking out v1.46.0 of [email protected]:terraform-aws-modules/terraform-aws-vpc
```

### Centrally Managed Approach
An example of using `Terrafile` in a root directory (#2):

Let's assume the following directory structure:

```
.
├── iam
│   ├── main.tf
│   └── .....tf
├── networking
│   ├── main.tf
│   └── .....tf
├── onboarding
.
.
.
├── some-other-stack
└── Terrafile
```

In the above scenario, Terrafile is not in every single folder but in the "root" of terraform code.

An example usage of centrally managed modules:

```
tf-aws-vpc:
source: "[email protected]:terraform-aws-modules/terraform-aws-vpc"
version: "v1.46.0"
destination:
- networking
tf-aws-iam:
source: "[email protected]:terraform-aws-modules/terraform-aws-iam"
version: "v5.11.1"
destination:
- iam
tf-aws-s3-bucket:
source: "[email protected]:terraform-aws-modules/terraform-aws-s3-bucket"
version: "v3.6.1"
destination:
- networking
- onboarding
- some-other-stack
```

The `destination` of module is an array of directories (stacks) where the module should be used.
The module itself is fetched once and copied over to designated destinations.
Final destination of the module is handled in a similar way as in first approach: `$destination/$module_path/$module_key`.

The output of the run is exactly the same in both options.

## TODO
* Break out the main logic into seperate commands (e.g. version, help, run)
* Update tests to include unit tests for broken out commands
Expand Down
86 changes: 72 additions & 14 deletions main.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ package main

import (
"fmt"
"io/ioutil"
"os"
"os/exec"
"path/filepath"
Expand All @@ -31,12 +30,13 @@ import (
)

type module struct {
Source string `yaml:"source"`
Version string `yaml:"version"`
Source string `yaml:"source"`
Version string `yaml:"version"`
Destinations []string `yaml:"destinations"`
}

var opts struct {
ModulePath string `short:"p" long:"module_path" default:"./vendor/modules" description:"File path to install generated terraform modules"`
ModulePath string `short:"p" long:"module_path" default:"./vendor/modules" description:"File path to install generated terraform modules, if not overridden by 'destinations:' field"`

TerrafilePath string `short:"f" long:"terrafile_file" default:"./Terrafile" description:"File path to the Terrafile file"`
}
Expand All @@ -53,47 +53,105 @@ func init() {
log.AddHook(stdemuxerhook.New(log.StandardLogger()))
}

func gitClone(repository string, version string, moduleName string) {
func gitClone(repository string, version string, moduleName string, destinationDir string) {
cleanupPath := filepath.Join(destinationDir, moduleName)
log.Printf("[*] Removing previously cloned artifacts at %s", cleanupPath)
os.RemoveAll(cleanupPath)
log.Printf("[*] Checking out %s of %s \n", version, repository)
cmd := exec.Command("git", "clone", "--single-branch", "--depth=1", "-b", version, repository, moduleName)
cmd.Dir = opts.ModulePath
err := cmd.Run()
if err != nil {
log.Fatalln(err)
cmd.Dir = destinationDir
if err := cmd.Run(); err != nil {
log.Fatalf("failed to clone repository %s due to error: %s", cmd.String(), err)
}
}

func main() {

fmt.Printf("Terrafile: version %v, commit %v, built at %v \n", version, commit, date)
_, err := flags.Parse(&opts)

// Invalid choice
if err != nil {
log.Errorf("failed to parse flags due to: %s", err)
os.Exit(1)
}

workDirAbsolutePath, err := os.Getwd()
if err != nil {
log.Errorf("failed to get working directory absolute path due to: %s", err)
}

// Read File
yamlFile, err := ioutil.ReadFile(opts.TerrafilePath)
yamlFile, err := os.ReadFile(opts.TerrafilePath)
if err != nil {
log.Fatalln(err)
log.Fatalf("failed to read configuration in file %s due to error: %s", opts.TerrafilePath, err)
}

// Parse File
var config map[string]module
if err := yaml.Unmarshal(yamlFile, &config); err != nil {
log.Fatalln(err)
log.Fatalf("failed to parse yaml file due to error: %s", err)
}

// Clone modules
var wg sync.WaitGroup
_ = os.RemoveAll(opts.ModulePath)
_ = os.MkdirAll(opts.ModulePath, os.ModePerm)

for key, mod := range config {
wg.Add(1)
go func(m module, key string) {
defer wg.Done()
gitClone(m.Source, m.Version, key)
_ = os.RemoveAll(filepath.Join(opts.ModulePath, key, ".git"))

// path to clone module
cloneDestination := opts.ModulePath
// list of paths to link module to. empty, unless Destinations are more than 1 location
var linkDestinations []string

if m.Destinations != nil && len(m.Destinations) > 0 {
// set first in Destinations as location to clone to
cloneDestination = filepath.Join(m.Destinations[0], opts.ModulePath)
// the rest of Destinations are locations to link module to
linkDestinations = m.Destinations[1:]

}

// create folder to clone into
if err := os.MkdirAll(cloneDestination, os.ModePerm); err != nil {
log.Errorf("failed to create folder %s due to error: %s", cloneDestination, err)

// no reason to continue as failed to create folder
return
}

// clone repository
gitClone(m.Source, m.Version, key, cloneDestination)

for _, d := range linkDestinations {
// the source location as folder where module was cloned and module folder name
moduleSrc := filepath.Join(workDirAbsolutePath, cloneDestination, key)
// append destination path with module path
dst := filepath.Join(d, opts.ModulePath)

log.Infof("[*] Creating folder %s", dst)
if err := os.MkdirAll(dst, os.ModePerm); err != nil {
log.Errorf("failed to create folder %s due to error: %s", dst, err)
return
}

dst = filepath.Join(dst, key)

log.Infof("[*] Remove existing artifacts at %s", dst)
if err := os.RemoveAll(dst); err != nil {
log.Errorf("failed to remove location %s due to error: %s", dst, err)
return
}

log.Infof("[*] Link %s to %s", moduleSrc, dst)
if err := os.Symlink(moduleSrc, dst); err != nil {
log.Errorf("failed to link module from %s to %s due to error: %s", moduleSrc, dst, err)
}
}
}(mod, key)
}

Expand Down
Loading

0 comments on commit 2c10832

Please sign in to comment.