1
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

1台のEC2を低コストで冗長化するシンプルな仕組み。AutoScalingのLifecycle hookでENIをLambdaに移動させます。

Last updated at Posted at 2019-05-10

SecondENI.gif

##EC2がダウンした場合、同じIPをアサインして通信を継続できるようにするにはどうしたらいい?

  • テーマ

EC2に予期せぬ障害が発生した際、自動復旧して同じプライベートIPアドレスで処理を継続する方法を解説します。

使用サービス

  • Lambda

  • CloudWatch Event

  • AutoScaling

  • 結論
    ・EC2には予め2つ目のENIをアタッチし、これを移動することで実現可能
    ・AutoScaling発動時に任意のイベントを指定できる「Lifecyclehook」機能を使用する
    ・「Lifecyclehook」のイベントにLambdaの起動を指定する
    ・LambdaにはENI (Elastic Network Interface)を新EC2にアタッチするようプログラミングすることで可能

  • 信頼性
    2019年5月10日にAWS公式サイト「knowledge-center」に再掲された内容に基づくので、信頼性は高いと思います。
    記事のリンクを張っておきます。「How do I automatically attach a second elastic network interface (ENI) to an instance launched through Auto Scaling?

この記事では「AutoScaling機能で自動起動したEC2に2つ目のENIをアタッチする」というもので、プライベートIPを指定できずランダムにアサインされます。

そこで、少しコードを改変し、「ENIのIDを明記し、元々EC2にアタッチされていたENIを新EC2の2つ目のENIとしてアタッチする」ようにしました。

他の環境での再現性も高いです。

ただし、設定にもよりますが通信が4分ほど遮断されるのでオンラインの処理にはLoad Blancerの使用を推奨します。

それでは、以下に設定手順を記載します。

##事前準備
image.png

AutoScaling Groupを作成します。
公式ドキュメント「Amazon EC2 Auto Scaling の使用開始

ENIを作成します。
公式ドキュメント「ネットワークインターフェイスを作成する

稼働中EC2にENIを追加でアタッチします。
公式ドキュメント「インスタンスの起動時にネットワークインターフェイスをアタッチする

##Lambdaの設定
image.png

Python 2.7でLambdaを作成します。
LambdaのIAMロールに以下を指定します。

IAMについてはclassmethodさんの「AWS IAMポリシーを理解する」の記事が分かりやすかったので、参考にしてみて下さい。

{
                "Statement": [
                    {
                        "Action": [
                            "logs:CreateLogGroup",
                            "logs:CreateLogStream",
                            "logs:PutLogEvents"
                        ],
                        "Effect": "Allow",
                        "Resource": "arn:aws:logs:*:*:*"
                    },
                    {
                        "Action": [
                            "ec2:CreateNetworkInterface",
                            "ec2:DescribeNetworkInterfaces",
                            "ec2:DetachNetworkInterface",
                            "ec2:DeleteNetworkInterface",
                            "ec2:AttachNetworkInterface",
                            "ec2:DescribeInstances",
                            "autoscaling:CompleteLifecycleAction"
                        ],
                        "Effect": "Allow",
                        "Resource": "*"
                    }
                ],
                "Version": "2012-10-17"
            }

Lambdaの関数に以下をペーストします。

「def create_interface(subnet_id):」のeni-0123456789abcdefgをアタッチしたいENIのIDに変更して下さい。(前後のダブルコーテーションは削除しないで下さい)


import boto3
import botocore
from datetime import datetime

ec2_client = boto3.client('ec2')
asg_client = boto3.client('autoscaling')


def lambda_handler(event, context):
    if event["detail-type"] == "EC2 Instance-launch Lifecycle Action":
        instance_id = event['detail']['EC2InstanceId']
        LifecycleHookName=event['detail']['LifecycleHookName']
        AutoScalingGroupName=event['detail']['AutoScalingGroupName']
        subnet_id = get_subnet_id(instance_id)
        interface_id = create_interface(subnet_id)
        attachment = attach_interface(interface_id, instance_id)
        if not interface_id:
            complete_lifecycle_action_failure(LifecycleHookName,AutoScalingGroupName,instance_id)
        elif not attachment:
            complete_lifecycle_action_failure(LifecycleHookName,AutoScalingGroupName,instance_id)
            delete_interface(interface_id)
        else:
            complete_lifecycle_action_success(LifecycleHookName,AutoScalingGroupName,instance_id)
       
        
def get_subnet_id(instance_id):
    try:
        result = ec2_client.describe_instances(InstanceIds=[instance_id])
        vpc_subnet_id = result['Reservations'][0]['Instances'][0]['SubnetId']
        log("Subnet id: {}".format(vpc_subnet_id))

    except botocore.exceptions.ClientError as e:
        log("Error describing the instance {}: {}".format(instance_id, e.response['Error']))
        vpc_subnet_id = None

    return vpc_subnet_id


def create_interface(subnet_id):
    network_interface_id = None

    if subnet_id:
        try:
            network_interface = ec2_client.create_network_interface(SubnetId=subnet_id)
            network_interface_id = "eni-0123456789abcdefg"
            log("Created network interface: {}".format(network_interface_id))
        except botocore.exceptions.ClientError as e:
            log("Error creating network interface: {}".format(e.response['Error']))

    return network_interface_id


def attach_interface(network_interface_id, instance_id):
    attachment = None

    if network_interface_id and instance_id:
        try:
            attach_interface = ec2_client.attach_network_interface(
                NetworkInterfaceId=network_interface_id,
                InstanceId=instance_id,
                DeviceIndex=1
            )
            attachment = attach_interface['AttachmentId']
            log("Created network attachment: {}".format(attachment))
        except botocore.exceptions.ClientError as e:
            log("Error attaching network interface: {}".format(e.response['Error']))

    return attachment


def delete_interface(network_interface_id):
    try:
        ec2_client.delete_network_interface(
            NetworkInterfaceId=network_interface_id
        )
        log("Deleted network interface: {}".format(network_interface_id))
        return True

    except botocore.exceptions.ClientError as e:
        log("Error deleting interface {}: {}".format(network_interface_id, e.response['Error']))
        
        
def complete_lifecycle_action_success(hookname,groupname,instance_id):
    try:
        asg_client.complete_lifecycle_action(
                LifecycleHookName=hookname,
                AutoScalingGroupName=groupname,
                InstanceId=instance_id,
                LifecycleActionResult='CONTINUE'
            )
        log("Lifecycle hook CONTINUEd for: {}".format(instance_id))
    except botocore.exceptions.ClientError as e:
            log("Error completing life cycle hook for instance {}: {}".format(instance_id, e.response['Error']))
            log('{"Error": "1"}')    
            
def complete_lifecycle_action_failure(hookname,groupname,instance_id):
    try:
        asg_client.complete_lifecycle_action(
                LifecycleHookName=hookname,
                AutoScalingGroupName=groupname,
                InstanceId=instance_id,
                LifecycleActionResult='ABANDON'
            )
        log("Lifecycle hook ABANDONed for: {}".format(instance_id))
    except botocore.exceptions.ClientError as e:
            log("Error completing life cycle hook for instance {}: {}".format(instance_id, e.response['Error']))
            log('{"Error": "1"}')    
    

def log(error):
    print('{}Z {}'.format(datetime.utcnow().isoformat(), error))

以下のAWS CLIコマンドを実行し、Lifecycle-hookイベントを作成します。
任意のEC2インスタンスにログインし、実行してください。

aws autoscaling put-lifecycle-hook \
    --lifecycle-hook-name my-lifecycle-launch-hook \
    --auto-scaling-group-name my-asg \
    --lifecycle-transition autoscaling:EC2_INSTANCE_LAUNCHING \
    --heartbeat-timeout 300 \
    --default-result ABANDON

##Cloudwatch eventの設定
image.png

CloudWatch consoleでEventを設定します。

公式ドキュメントにも設定方法が載っているので一応リンクを貼っておきます。「イベントでトリガーする CloudWatch イベント ルールの作成

Service Nameを Auto Scalingにする

Event Typeを Instance Launch and Terminateにする

Specific instance event(s),を選択し、 EC2 Instance-launch Lifecycle Actionとする

Specific group name(s),を選択し、設定したAutoScaling名を選択する(my-asg)

右側のペインの設定↓

Add targetを選択し、作成したLabmdaを指定します。

##では、EC2をターミネートして試してみましょう

IPは「172.24.0.150」と「172.24.0.136」 2つあります。 末尾が「22a」のEC2です。
15.png

「22a」さん、さようなら
6.png

新しいものが起動してきました。末尾「c27」です。
10.png

172.24.0.136」のIPが引き継がれましたね。 (今回、偶然150のIPも一緒でした。)
20.png

##発展

今回のように、ENIを使えば「シングルインスタンスでバッチ処理をコスト低で冗長運用できる」のでおすすめです。

このようにENIは様々な使い方ができて大変便利です。

・例えば2つのSSL証明書をプライベートIPアドレスに1つずつ割り当てることで1つのインスタンスで2つのサイトを運営するテクニック

・AWS公式サイトで紹介されている**「Management Network」テクニック**で、一つのWEBサーバが、インターネットに対面しているパブリックなENIとSSHアクセス向けのプライベートなENIを持ち、オンプレはWEBサーバーにSSHアクセスし、インターネット通信を使用して安全にfacebookを見たりできたり

このあたりのテクニックをまとめたVPCについての記事リンクを貼っておきます。
VPCとは何か?それはAWSに張り巡らされた1つの血管であった。 〜AWS Virtual Private Cloud 〜

今回はAutoScalingのイベントを契機にしましたが、おそらく「任意の監視で引っかかった場合に発動」といった方が便利そうです。

このあたりの監視と絡めたテクニックもまとめてQiitaに投稿したいと思います。

ありがとうございました。

1
6
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?