Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/haoliangyu/terrac
A minimal private module registry for Terraform and OpenTofu
https://github.com/haoliangyu/terrac
azure cli gcp module opentofu private-registry s3 terraform
Last synced: 3 months ago
JSON representation
A minimal private module registry for Terraform and OpenTofu
- Host: GitHub
- URL: https://github.com/haoliangyu/terrac
- Owner: haoliangyu
- License: mit
- Created: 2022-11-25T00:03:23.000Z (about 2 years ago)
- Default Branch: main
- Last Pushed: 2024-08-31T05:19:05.000Z (4 months ago)
- Last Synced: 2024-10-13T16:43:32.882Z (3 months ago)
- Topics: azure, cli, gcp, module, opentofu, private-registry, s3, terraform
- Language: TypeScript
- Homepage:
- Size: 2.58 MB
- Stars: 30
- Watchers: 2
- Forks: 0
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG
- License: LICENSE
Awesome Lists containing this project
- awesome-opentofu - terrac - Minimal private module registry compatible with OpenTofu. (Tools / Registry)
README
# terrac
A simple CLI tool to quickly setup a minimal private terraform module registry with your cloud storage service.
[![Version](https://img.shields.io/npm/v/terrac.svg)](https://npmjs.org/package/terrac)
[![License](https://img.shields.io/npm/l/terrac.svg)](https://github.com/haoliangyu/terrac/blob/main/package.json)
[![Main](https://github.com/haoliangyu/terrac/actions/workflows/main.yaml/badge.svg)](https://github.com/haoliangyu/terrac/actions/workflows/main.yaml)* [Why](#why)
* [Design](#design)
* [Installation](#installation)
* [Tutorial](#tutorial)
* [Configuration](#configuration)
* [Commands](#commands)
* [Backends](#backends)
* [Limitations](#limitations)
* [Roadmap](#roadmap)
* [Development](#development)## Why
Sharing terraform module privately is usually necessary when the infrastructure development happens across multiple teams (DevOps vs applications) or multiple repositories (core vs. app infrastructure).
While a module can be downloaded from a git URL, it lacks the support to code versionization and storage management. While other paid solutions (like [Terraform Cloud](https://developer.hashicorp.com/terraform/cloud-docs/registry)) or open-source solutions (like [citizen](https://github.com/outsideris/citizen)) exist as a full-feature registry, they are usually very heavy for small teams, in terms of const, features, or maintenance.
The `terrac` CLI provides a thin layer on your choice of cloud storage service, such as [AWS S3](https://aws.amazon.com/s3/) or [GCP Cloud Storage](https://cloud.google.com/storage/), to publish and share private module code. It provides features:
* Publish and download with [semver](https://semver.org)
* Manage storage schema and metadata automatically
* Fully integrated with your infrastructure
* Completely serverless (nothing is hosted)
* Simple commands (npm-style)
* Free and flexibleIt is suitable to use as a private terraform registry in small teams (while limitations apply).
## Design
The desing of `terrac` consists of three components:
* **Configuration**: a JSON file to provide configurations on the module and the cloud storage service
* **Commands**: a set of commands to provide user interface and interaction
* **Backends**: a set of standard abstractions for different cloud storage services. All backends expose the same interface to the commands and encapuslate the details of interaction with the remote API.```mermaid
graph TD;
Configuration-->Command:publish;
Command:publish-->Backend:s3;
Command:publish-->Backend:gcp;
Command:publish-->Backend:azure;
Backend:gcp-->GCP;
Backend:azure-->Azure;
Backend:s3-->AWS;
```## Installation
### npm
As a node.js application, terrac can be installed with [npm](https://www.npmjs.com) or other compatible package managers, such as [yarn](https://yarnpkg.com) or [pnpm](https://pnpm.io).
```bash
npm install -g terrac
```### manual
Binaries for diefferent platforms are provided as attachment at each [GitHub release](https://github.com/haoliangyu/terrac/releases):
* `terrac-linux` for Linux
* `terrac-macos` for MacOS
* `terrac-win.exe` for WindownsYou can simply download a binary and put it in a directoy included in the `PATH` environment variable. Note that administrator permission may be required for the action.
### bash
The following script automates the manual installation with a bash script in Linux and MacOS.
``` bash
curl https://raw.githubusercontent.com/haoliangyu/terrac/main/scripts/install.sh | bash
```## Tutorial
To initialize a module directory and publish
``` bash
# Create terrac configuration
terrac init# Pack and publish the module to the remote storage service
terrac publish# Get the module source URL to use in terraform
terrac get module-name
```## Configuration
A `terrac.json` file at the module root directory is used to provide configuration for terrac. It contains two blocks:
* **backend** to provide the cloud storage configuration
* **module** to provide the module metadataThe JSON configuration can be populated interactively using the `terrac init` command and this is an example:
```json
{
"backend": {
"type": "s3",
"bucket": "team-sharing",
"region": "us-east-1"
},
"module": {
"name": "custom-rds-module",
"version": "1.6.3"
}
}
```### Backend
See the [Backends](#backends) section for more details.
### Module
The `module` object describes the meta information for the module to publish:
* **name**: module name
* **version**: module version number. This could be a sematic version or a custom version name.## Commands
* [`terrac init`](#terrac-init)
* [`terrac get`](#terrac-get)
* [`terrac list`](#terrac-list)
* [`terrac publish`](#terrac-publish)### `terrac init`
Initialize terrac configuration in a directory. It will provide an interactive questionnaire for the selected backend and create the `terrac.json` configuration file at the local directory.
```
USAGE
$ terrac init [--work-directory ]FLAGS
--work-directory= [default: .] Root directory of the terraform projectDESCRIPTION
Initialize terrac configuration in a directory.
```_See code: [src/commands/init.ts](https://github.com/haoliangyu/terrac/blob/master/src/commands/init.ts)_
### `terrac get`
Get the module source URL with the given module and version.
```
USAGE
$ terrac get NAME [VERSION] [--exact] [--work-directory ]ARGUMENTS
NAME Module name
VERSION Module version. This could be a version name (like latest), a semver, or a semver component.
By default, terrac will verify the release with the input version and generate the source URL.
If it needs to resolve to an exact version, use the --exact flag.FLAGS
--exact Whether to resolve to an exact version if a named version or a semver component is given
--work-directory= [default: .] Root directory of the module configurationDESCRIPTION
Get the module source URL of the given module and version.EXAMPLES
$ terrac get my-module$ terrac get my-module latest
$ terrac get my-module 1.3.2
$ terrac get my-module 1.3
$ terrac get --exact my-module 1.3
```The command will try to automatically resolve to the desired version. If the version is found at the backend, it will provide the download URL as output
```
The release 1.3.2 is found and available at s3::https://s3-us-east-1.amazonaws.com/test-bucket/test-module/1.3.2/module.zip
```The output URL can be used in the module `source` property. For example,
``` terraform
module "example" {
source = "s3::https://s3-us-east-1.amazonaws.com/test-bucket/test-module/1.3.2/module.zip"
}
```_See code: [src/commands/get.ts](https://github.com/haoliangyu/terrac/blob/master/src/commands/get.ts)_
### `terrac list`
List available modules and their versions.
```
USAGE
$ terrac list [NAME] [--work-directory ]ARGUMENTS
NAME Module nameFLAGS
--work-directory= [default: .] Root directory for the terrac configuration fileDESCRIPTION
List available modules and versions.EXAMPLES
$ terrac list$ terrac list my-module
```If the module name is provided, it will print a list of available versions for the given module. Otherwise, it will list all published modules in the backend.
_See code: [src/commands/list.ts](https://github.com/haoliangyu/terrac/blob/master/src/commands/list.ts)_
### `terrac publish`
Publish a terraform module using the metadata from the `terrac.json` configuration. If a version name (like `beta`) is specified, it will publish a single release with the version `beta`. If a semver is specified, it will publish multile releaes with each semver component. For example, the version `1.3.2` will create three releases (`1`, `1.3`, `1.3.2`) and each will have it own downloadable source URL.
A `latest` release will be updated at every publication.
```
USAGE
$ terrac publish [--overwrite] [--work-directory ]FLAGS
--overwrite Overwrite a published version with a new package
--work-directory= [default: .] Root directory of the module projectDESCRIPTION
Publish a terraform module.EXAMPLES
$ terrac publish$ terrac publish --overwrite
```Once the publication is successful, it will list all updated releases with the download URLs.
_See code: [src/commands/publish.ts](https://github.com/haoliangyu/terrac/blob/master/src/commands/publish.ts)_
####
## Backends
Terrac supports a variety of storage backends for artifact hosting:
* Local
* AWS S3
* GCP Storage
* Azure Block StorageTo set a backend for module publication, update the `backend` block in the `terrac.json` configuration file.
### Local Directory
The `local` backend type uses a local directory for package storage. It is typically used for local testing.
``` jsonc
// terrac.json
{
"backend": {
"type": "local",
// path to the storage directory
"path": "./"
}
}
```### AWS S3
The `s3` backend type uses an [AWS S3](https://aws.amazon.com/s3/) bucket for artifact storage. It utilizes the AWS SDK for JavaScript to communicate with AWS and requires proper authentication setup (see [documentation](https://docs.aws.amazon.com/sdk-for-javascript/v3/developer-guide/setting-credentials-node.html)).
``` jsonc
// terrac.json
{
"backend": {
"type": "s3",
// bucket name
"bucket": "module-bucket",
// bucket region
"region": "us-east-1"
}
}
```### GCP Cloud Storage
The `gcp` backend type uses a [GCP Cloud Storage](https://cloud.google.com/storage) bucket for artifact storage. It uses the GCP Node.js SDK to communicate with GCP and requires proper authentication setup (see [documentation](https://cloud.google.com/docs/authentication/provide-credentials-adc)).
``` jsonc
// terrac.json
{
"backend": {
"type": "gcp",
// bucket name
"bucket": "module-bucket",
// project id
"projectId": "my-module-registry"
}
}
```### Azure Blob Storage
The `azure` backend type uses a [Azure Blog Storage](https://azure.microsoft.com/en-us/products/storage/blobs) container for artifact storage.
``` jsonc
// terrac.json
{
"backend": {
"type": "azure",
// Azure storage account name
"account": "terrac",
// Azure storage container name
"conatiner": "terrac-test",
// [Optional] File name prefix
"fileNamePrefix": "name/prefix"
}
}
```## Limitations
The purpose of `terrac` is to provide a consistent and simple interface for terraform module hosting with a variety of storage backends. It focuses on the module publication and download. However, it doesn't provide or guarantee some advanced features of modern artifact registry, such as:
* Authentication
* High availability
* High scalability
* Atomic write
* Conflict control
* Permission controlIt may be possible to configure a storage backend for these features but this is out of the scope of this project.
## Roadmap
* Features
* [x] Add `overwrite` option to the `publish` command
* [x] Add `init` command to interatively initialize a module project
* [x] Add schema check to the terrac configuration file
* [x] Add support to any custom version name in the `get` and `publish` commands
* [x] Add support to using partial semver in the `get` command
* [ ] Install with brew
* [x] Install with bash script
* [x] Install with standalone executable in different OS
* [ ] Use backend configuration schema to configure `init` command questionnaire* Backends
* [x] GCP Cloud Storage
* [x] Azure Blob Storage* Maintenance
* [ ] Add E2E tests with `terraform` and `OpenTofu`
* [ ] Unit tests for `init` command
* [ ] Automate release process to cut GitHub release## Development
In order to run tests locally, it requires the following packages to be installed:
* [localstack](https://github.com/localstack/localstack#installation)
* [awscli-local](https://github.com/localstack/awscli-local)