lgtm-cat(サービス名 LGTMeow https://lgtmeow.com のインフラ管理用プロジェクトです。
tfenvとDockerの利用を紹介します。
どちらの実行環境でもいいですが、Terraformはマイナーバージョンでも破壊的な変更を行う事があるのでバージョンを気軽に切り替えられる事は必須です。
非推奨の機能が次のマイナーバージョンでいきなり削除されることも過去にはあったので、Terraformの警告を無視しない事が非常に大切になります。
詳細はtfenvを確認してください。
brew install tfenv
でインストールします。
その後以下の手順で設定を行います。
tfenv install 1.0.3
tfenv use 1.0.3
terraform --version
で Terraform v1.0.3 が表示されればOK
Docker上でTerraformを実行します。
初回起動
docker-compose up --build -d
起動
二回目以降は下記のコマンドを実行します。
docker-compose up -d
AdministratorAccess
を付与したユーザーを作成して下さい。
アクセスキーでアクセス出来るように設定しておく必要があります。
Terraformはこのアクセスキーを使ってAWSの各種Resourcesを作成・管理します。
Macの場合 brew install awscli
を実行してaws cliのインストールを行います。
その後、aws configure --profile lgtm-cat
対話形式のインターフェースに従い入力します。
AWS Access Key ID [None]: `アクセスキーIDを入力`
AWS Secret Access Key [None]: `シークレットアクセスキーを入力`
Default region name [None]: ap-northeast-1
Default output format [None]: json
ちなみに aws cli なしでもAWSクレデンシャルを設定する事は可能です。
~/.aws/credentials
という名前のファイルを作成して以下の内容を設定します。
[lgtm-cat]
aws_access_key_id = あなたのアクセスキーID
aws_secret_access_key = あなたのシークレットアクセスキー
このプロジェクトではprofile名を lgtm-cat
としています。
この名前は任意の物でも構いませんが、必ずprofile名を明示的に付けておく事が重要です。
.tfstate
というファイルに実行状態を記録します。(実体はただのJSONファイルです)
このプロジェクトでは lgtm-cat-tfstate
というS3Bucketがその保存先になります。
この設定は providers/aws/environments/○○/backend.tf
に記載されています。
以下のファイルを配置して下さい。
txt_records = ["google-site-verification=XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"]
from_email = "us-east-1のSES EmailAddressesに定義されているメールアドレスを指定"
Docker起動後にホストOS上で以下のコマンドを実行すると terraform init
が実行されます。
chmod 755 terraform-init.sh
docker-compose exec terraform ./terraform-init.sh
下記のようなディレクトリ構成になっています。
lgtm-cat-terraform/
├ modules/
│ └ aws/
└ providers/
└ aws/
└ environments/
├ stg/
│ ├ 10-network/
│ ├ 11-acm/
│ └ 22-xxxx/
└ prod/
├ 10-network/
├ 11-acm/
└ 22-xxxx/
本番環境とステージング・開発環境など複数環境に構築するケースを想定し、environments/
配下に環境ごとのディレクトリを作成しています。
providers
の頭の数字に注目して下さい。
.tfstate
はこれらのディレクトリ配下毎に存在しますが、数字の大きなディレクトリは数字が小さなディレクトリに依存しています。
その為、必ず数字が小さいディレクトリから terraform apply
を実行する必要があります。
今後このプロジェクトをベースに機能を追加する際も依存関係を意識してディレクトリ名を決める必要があります。
LGTMeow では本番環境とステージング環境など、その他の環境も同じAWSアカウント上にリソースが存在します。
その為、AWSアカウントに1つだけ存在すれば良いリソースに関しては providers/aws/environments/prod
にのみ定義されている場合があります。
ステージング用のリソースが本番用のリソースに依存しているケースもあるので、先に本番用の providers/aws/environments/prod
配下の terraform apply
を全て終わらせておく必要があります。
以下の2つに関しては先に https://github.com/nekochans/lgtm-cat-image-recognition のデプロイを実施し作成されたAPI GatewayのIDをSecretManagerの中に登録しておく必要があります。
providers/aws/environments/stg/20-api
providers/aws/environments/prod/20-api
SecretManagerはステージング用、本番用の2種類作成します。
名称は以下の通りです。
/stg/lgtm-cat/image-recognition
/prod/lgtm-cat/image-recognition
中身は以下のように https://github.com/nekochans/lgtm-cat-image-recognition のデプロイ時に生成されるAPI GatewayのIDを設定します。
{
"api_id": "xxxxxxxxxx"
}
- 今はAWSのみだが、他のproviderが増えても大丈夫なように
providers/
を作ってあります - 各moduleには特定のリージョンに依存した値はハードコードしない(AZの名前とか)
- 各moduleには特定の環境に依存した値はハードコードしない
- マルチリージョンでの運用にも耐えられるディレクトリ設計
以下の命名規則に従って命名します。
項目名 | 命名規則 |
---|---|
ファイル名 | ケバブケース |
ディレクトリ名 | ケバブケース |
リソース、データソース名 | スネークケース |
リソースID | ケバブケース |
変数名 | スネークケース |
リソース名とは resource
や data
等のTerraformの予約語に付けられる名前です。
下記の例だと ssh_from_all_to_bastion
がそれに該当します。
resource "aws_security_group_rule" "ssh_from_all_to_bastion" {}
リソースIDとはそのリソースの中で一意になっている必要がある値です。
下記の例だと key_name
がそれに該当します。
resource "aws_key_pair" "ssh" {
public_key = "${file(var.ssh_public_key_path)}"
key_name = "${terraform.workspace}-ssh-key"
}
他にもタグ名を良く付ける事がありますが、それもこちらのルールの通りケバブケースで命名します。
このようなややこしい規則になっている理由ですが、RDSCluster等、一部のリソース名で _
が禁止文字に設定されている為です。
他にもインデント等細かいルールがありますが、こちらに関しては terraform fmt -recursive
を実行すれば自動整形されるのでこれを利用すれば良いでしょう。
terraform fmt -recursive
は必ずプロジェクトルートで実行を行ってください。
そうしないと全ての .tf
ファイルに修正が適応されません。
リソース変更を行っていないにも関わらず terraform plan
で差分が発生する事があります。
Terraform以外の手段でリソースが変更された場合は差分が出るようになっている為です。
No changes. Your infrastructure matches the configuration.
Your configuration already matches the changes detected above. If you'd like to update the Terraform state to match, create and apply a refresh-only plan:
terraform apply -refresh-only
このように No changes. Your infrastructure matches the configuration.
が表示されている場合は terraform apply -refresh-only
を実行して下さい。
Provider Plugin Cache 機能を有効化しています。 機能については、以下のドキュメントを参考にしてください。
ドキュメントにも記載されていますが、キャッシュされたプラグインは Terraform によって削除されることはありません。
未使用のプラグインは手動で削除してください。