次のものを AWS に移行しました。
- ドメイン移管(53ningen.com): ムームードメインから Route 53 へ
- ドメインネームサーバー移行: ムームードメインから Route 53 へ
また、もとの運用では存在しなかった、次のようなものを導入しました。
- L7ヘルスチェック
- DNSフェイルオーバー
この資料は自分向けの作業ログであるため、実際には各ステップ冒頭に記された公式ドキュメントを理解した上で読むことを強く推奨します。
手元環境
作業は基本的には terraform を使って行いました。ただし、ドメインの移管に関しては Route 53 のマネジメントコンソールとムームードメインのコントロールパネルを操作する必要があります。
% terraform -v
Terraform v0.10.7
1. DNS を Route 53 へ移行
ドメインも Route 53 に移管しますが、その際に元のレジストラ(ムームードメイン)が提供するネームサーバーからドメインが削除されてしまうので、前もって DNS を Route 53 に切り替えておきます。
1.1. ホストゾーン(Hosted Zone)の作成
公式ドキュメント: 既存ドメインの DNS サービスを Amazon Route 53 に移行する
ホストゾーンとはドメインに紐づくリソースレコードのセットです。おおよそゾーンファイルと対応していると考えて良いと思います。実際にゾーンファイルを取り込むこともできるようです(公式ドキュメント: ゾーンファイルをインポートしてリソースレコードセットを作成する)。
ホストゾーンには、パブリックなものとVPC向けのプライベートなものがあります。マネジメントコンソールからドメイン名を入力すれば作成が完了しますが、今回はこのリソースを terraform
で管理します。
resource "aws_route53_zone" "main" {
name = "53ningen.com"
}
terraform plan
は以下のとおり
An execution plan has been generated and is shown below.
Resource actions are indicated with the following symbols:
+ create
Terraform will perform the following actions:
+ aws_route53_zone.main
id: <computed>
comment: "Managed by Terraform"
force_destroy: "false"
name: "53ningen.com"
name_servers.#: <computed>
vpc_region: <computed>
zone_id: <computed>
Plan: 1 to add, 0 to change, 0 to destroy.
terraform apply
してホストゾーンを作成すると SOA レコード
と NS レコード
も自動的に作成されます。今回はドメイン名とIPアドレスを紐づけるために A レコード
の追加が必要になります。そのため、その記述を追加します。メールサーバーの移行が必要であれば MX レコード
、サブドメインなどの運用をやっているのであれば CNAME レコード
など適当な感じに必要なものも移行しておきましょう。ひとまず今回は A レコード
のみを作ります(Route 53 が対応しているDNSリソースは サポートされる DNS リソースレコードタイプ を参照)。terraform apply
を実行するとレコードが作成されます。
An execution plan has been generated and is shown below.
Resource actions are indicated with the following symbols:
+ create
Terraform will perform the following actions:
+ aws_route53_record.www
id: <computed>
fqdn: <computed>
name: "53ningen.com"
records.#: "1"
records.2910563735: "******************"
ttl: "300"
type: "A"
zone_id: "******************"
1.2. DNSの変更
マネジメントコンソールの NS レコード欄の DNS サーバーを、ムームードメインのコントロールパネルで指定します。反映が有効になるまで少し(TTL分)時間がかかるため、少し待って dig
で確認すると良いとおもいます。
% dig 53ningen.com +trace
; <<>> DiG 9.8.3-P1 <<>> 53ningen.com +trace
;; global options: +cmd
. 171428 IN NS b.root-servers.net.
. 171428 IN NS c.root-servers.net.
. 171428 IN NS d.root-servers.net.
. 171428 IN NS e.root-servers.net.
. 171428 IN NS f.root-servers.net.
. 171428 IN NS g.root-servers.net.
. 171428 IN NS h.root-servers.net.
. 171428 IN NS i.root-servers.net.
. 171428 IN NS j.root-servers.net.
. 171428 IN NS k.root-servers.net.
. 171428 IN NS l.root-servers.net.
. 171428 IN NS m.root-servers.net.
. 171428 IN NS a.root-servers.net.
;; Received 508 bytes from 192.168.1.1#53(192.168.1.1) in 75 ms
com. 172800 IN NS d.gtld-servers.net.
com. 172800 IN NS l.gtld-servers.net.
com. 172800 IN NS a.gtld-servers.net.
com. 172800 IN NS g.gtld-servers.net.
com. 172800 IN NS b.gtld-servers.net.
com. 172800 IN NS f.gtld-servers.net.
com. 172800 IN NS c.gtld-servers.net.
com. 172800 IN NS h.gtld-servers.net.
com. 172800 IN NS j.gtld-servers.net.
com. 172800 IN NS e.gtld-servers.net.
com. 172800 IN NS i.gtld-servers.net.
com. 172800 IN NS k.gtld-servers.net.
com. 172800 IN NS m.gtld-servers.net.
;; Received 506 bytes from 2001:500:2::c#53(2001:500:2::c) in 179 ms
53ningen.com. 172800 IN NS ns-1431.awsdns-50.org.
53ningen.com. 172800 IN NS ns-771.awsdns-32.net.
53ningen.com. 172800 IN NS ns-2010.awsdns-59.co.uk.
53ningen.com. 172800 IN NS ns-393.awsdns-49.com.
;; Received 199 bytes from 192.43.172.30#53(192.43.172.30) in 258 ms
53ningen.com. 300 IN A 153.126.136.244
53ningen.com. 172800 IN NS ns-1431.awsdns-50.org.
53ningen.com. 172800 IN NS ns-2010.awsdns-59.co.uk.
53ningen.com. 172800 IN NS ns-393.awsdns-49.com.
53ningen.com. 172800 IN NS ns-771.awsdns-32.net.
;; Received 183 bytes from 2600:9000:5301:8900::1#53(2600:9000:5301:8900::1) in 112 ms
1.3. コストの計算
公式ドキュメント: 料金
お仕事ならこの計算作業する前にやるべきですが、個人のお遊びサーバーなので作業後に計算しました。
- ホストゾーン
- 0.50 USD(ホストゾーンごと)/月 – 最初の25のホストゾーン
- 0.10 USD(ホストゾーンごと)/月 – それ以上のホストゾーン
- 標準的クエリ
- 0.400 USD(100 万クエリごと) – 最初の 10 億クエリ/月
- 0.200 USD(100 万クエリごと) – 10 億クエリ以上/月
おおよそ 1ホストゾーン + 最初の10億クエリ = 0.50 + 0.40 = 0.90 USD
1.4. ヘルスチェックの設定 (Optional)
公式ドキュメント: Amazon Route 53 ヘルスチェックの作成と DNS フェイルオーバーの設定
ヘルスチェック(L7)とDNSフェイルオーバーが利用できます。ヘルスチェックで障害を検知した場合に Cloudwatch Alarm と連携をさせたり、DNSフェイルオーバーを利用して待機系に参照を向けたりすることが可能です。
ひとまずベーシックなウェブサーバーのヘルスチェックは以下のように設定できます。その他諸々 terraform を利用する際にオプション記述については terraform 公式ドキュメント: aws_route53_health_check を参照してください。
resource "aws_route53_health_check" "www" {
reference_name = "www"
ip_address = "153.126.136.244"
port = 80
type = "HTTP"
resource_path = "/"
failure_threshold = "3"
request_interval = "30"
measure_latency = true
tags = {
Name = "www"
}
}
ヘルスチェックに関しては、
- AWSエンドポイント
- ヘルスチェック 1 件につき 0.50 USD*/月
- オプション機能 1 件につき 1.00 USD/月
- AWS以外のエンドポイント:
- ヘルスチェック 1 件につき 0.75 USD*/月
- オプション機能 1 件につき 2.00 USD/月
- 任意のヘルスチェック機能は以下の通りです。
- HTTPS
- 文字列マッチング
- 短インターバル
- レイテンシー計測
となっています。
「AWS エンドポイント」とは、AWS の中で稼働するリソース(たとえば Amazon EC2 インスタンス)のうち、ヘルスチェックと同じ AWS アカウントの中でプロビジョニングされるか、ヘルスチェックと同じアカウントが請求先となっているものを指します。計算済みヘルスチェックとメトリクスベースのヘルスチェックは AWS エンドポイントのヘルスチェックとして請求されます。Elastic Load Balancing のリソースまたは Amazon S3 ウェブサイトバケットがエンドポイントであるヘルスチェックについては、お客様への請求は発生しません。Elastic Load Balancing のリソースおよび Amazon S3 ウェブサイトバケットをエンドポイントとするヘルスチェックについては、AWS によって自動的にプロビジョニングされ、Amazon Route 53 の一部として追加料金なしでご利用いただけます。
1.5 DNSフェイルオーバーの設定 (Optional)
公式ドキュメント: DNS フェイルオーバーの設定
terraform 公式ドキュメント: aws_route53_record
ヘルスチェックを入れたついでに、障害発生時に S3 の静的ホスティング機能 + Cloudfront によるメンテナンスページのほうへ参照を向けるようなDNSフェイルオーバー設定を入れてみます。DNSフェイルオーバー設定については無料で利用することができます。最終的な構成図は以下のような形になります。
ディザスタリカバリを想定した構成として、ap-northeast-1 リージョンで稼働させるアプリケーションサーバーに対して、メンテナンスページを静的ホスティングする際の S3 リージョンは地理・物理的に違う系統の適当なリージョンを選択します。今回は us-east-1 を選択しました。S3 の前段に置いた Cloudfront は主に S3 からの配信をキャッシュする役割と、メンテナンスを表すステータスコード 503 でリソースを返す役割を持っています。S3 バケットの作成とファイルの配置、静的ホスティング機能の有効化および Cloudfront の設定は以下のように terraform で記述することができます。
terraform 公式ドキュメント: aws_s3_bucket
terraform 公式ドキュメント: aws_cloudfront_distribution
provider "aws" {
region = "us-east-1"
alias = "us-east-1"
}
resource "aws_s3_bucket" "error" {
provider = "aws.us-east-1"
bucket = "error.53ningen.com"
acl = "public-read"
policy = "${file("./policies/error.53ningen.com_s3_policy.json")}"
website {
index_document = "index.html"
error_document = "error.html"
}
}
resource "aws_s3_bucket_object" "error_html" {
provider = "aws.us-east-1"
bucket = "${aws_s3_bucket.error.bucket}"
content_type = "text/html"
key = "error.html"
source = "./resources/error.html"
etag = "${md5(file("./resources/error.html"))}"
}
resource "aws_cloudfront_distribution" "error_s3_distribution" {
origin {
domain_name = "${aws_s3_bucket.error.bucket_domain_name}"
origin_id = "S3-error.53ningen.com"
}
custom_error_response {
error_code = "404"
response_code = "503"
response_page_path = "/error.html"
error_caching_min_ttl = "300"
}
enabled = true
is_ipv6_enabled = true
default_root_object = "index.html"
default_cache_behavior {
allowed_methods = ["GET", "HEAD"]
cached_methods = ["GET", "HEAD"]
target_origin_id = "S3-error.53ningen.com"
forwarded_values {
query_string = true
cookies {
forward = "none"
}
}
viewer_protocol_policy = "allow-all"
min_ttl = 0
default_ttl = 3600
max_ttl = 86400
}
price_class = "PriceClass_All"
restrictions {
geo_restriction {
restriction_type = "none"
}
}
tags {
Environment = "production"
}
viewer_certificate {
cloudfront_default_certificate = true
}
}
resources/error
には、適当なメンテナンス表示 html ファイルを、 policies/maintenance.53ningen.com_s3_policy.json
には以下のようにバケットのファイルの読み取りを許可するポリシーを配置します。
{
"Version":"2012-10-17",
"Statement":[
{
"Sid":"PublicReadForGetBucketObjects",
"Effect":"Allow",
"Principal": "*",
"Action":["s3:GetObject"],
"Resource":["arn:aws:s3:::error.53ningen.com/*"]
}
]
}
つづいて Route 53 ホストゾーンにフェイルオーバーの設定をします。通常時は既存の A レコードが有効になるようにしつつ、ヘルスチェックに失敗した場合にメンテナンスページを静的にホスティングしている S3 のほうに参照を向けるような設定を入れます。また ttl を最初の設定より短く 60 秒に変更します。 terraform を用いて次のように記述できます。
resource "aws_route53_record" "www" {
zone_id = "${aws_route53_zone.main.zone_id}"
name = "53ningen.com"
type = "A"
ttl = "60"
records = ["153.126.136.244"]
health_check_id = "${aws_route53_health_check.www.id}"
set_identifier = "www"
failover_routing_policy {
type = "PRIMARY"
}
}
resource "aws_route53_record" "www-error" {
zone_id = "${aws_route53_zone.main.zone_id}"
name = "53ningen.com"
type = "A"
alias = {
name = "${aws_cloudfront_distribution.error_s3_distribution.domain_name}"
# cloudfront hosted_zone_id
zone_id = "Z2FDTNDATAQYW2"
evaluate_target_health = false
}
set_identifier = "www-secondary"
failover_routing_policy {
type = "SECONDARY"
}
}
最後に動作確認として、動いているウェブサーバーを停止し、きちんとメンテナンス表示に切り替わるか確認します。またウェブサーバーを起動したときに、ヘルスチェックが通るようになり、きちんとアプリケーションサーバーに参照が戻るかも合わせてチェックしておきましょう。
2. ドメインを Route 53 に移管する
公式ドキュメント: ドメインの移管
ムームードメインで購入した 53ningen.com ドメインを Route 53 に移管します。ステップとしては以下のとおりです。
- WHOIS 情報としてレジストラの情報を代理公開している場合は、ムームードメインのコントロールパネルから自分のものに変更する
- 確認のメールが飛んでくるので踏む
- AUTH_CODE をメモ
- Route 53 の
Domain registration
からTransfer Domain to Route 53
に進みドメイン名を入力 - AUTH_CODE を入力
- 移管後のDNSとして先ほど作った Route 53 の Hosted Zone を指定する
- AWSから移管確認のメールが飛んでくるので踏む
- そのうち移管が完了する
Waiting for the current registrar to complete the transfer (step 7 of 14)
ドメインが移管の要件を満たしているか現在のレジストラが確認しています。このステップは、ドメインの TLD に応じて最大 10 日かかる場合があります。
結構待ったりいろんな確認メールのリンクを踏んだりする必要がありますが、あまりテクニカルな手順はないはず...。