はじめに
今回はAWS CDKでEC2の構築を行いたいと思います。AWS CDKはコードでリソースの操作ができるIacツールです。様々なプログラミング言語をサポートしているので、使い慣れた言語を利用することで学習コストを抑えることができます。今回はPython boto3を使って構築していきます。
コードでリソースの操作をするのは最初はとっつきにくいですが、一度構築してしまえば、何度も同じ環境をコマンド一つで再現することができます。
例えば、本番環境と検証環境で同じ環境を用意したいというニーズがあった場合。コンソールからポチポチ画面を操作して同じ環境を構築するのは骨の折れる作業です。また、人が作業する以上ミスが発生するリスクがあります。Iacを導入することで確実に同じ環境を構築することができミスも削減できます。
作成するリソースの構成図
サブネット1つにEC2を1台たてるシンプルな構成で構築していきたいと思います。
実装
それでは実際にコードを書いて実装してみます。以下のコードで構成図通りの環境を構築することができます。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 |
ec2.py #boto3インポート import boto3 #インスタンスの作成 ec2 = boto3.client('ec2') #vpc作成 vpc = ec2.create_vpc( CidrBlock = '192.0.0.0/24', TagSpecifications=[ { 'ResourceType': 'vpc', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-vpc' }, ] } ] ) vpcId = vpc['Vpc']['VpcId'] #サブネット作成 publicSubnet1 = ec2.create_subnet( AvailabilityZone = 'ap-northeast-1a', CidrBlock = '192.0.0.0/28', VpcId = vpcId ) publicSubnet1Id = publicSubnet1['Subnet']['SubnetId'] #インターネットゲートウェイの作成 igw = ec2.create_internet_gateway( TagSpecifications=[ { 'ResourceType': 'internet-gateway', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-igw' }, ] } ] ) igwId = igw['InternetGateway']['InternetGatewayId'] #インターネットゲートウェイをvpcにアタッチ ec2.attach_internet_gateway( InternetGatewayId= igwId, VpcId =vpcId ) #ルートテーブルの作成 routeTable = ec2.create_route_table( VpcId = vpcId, TagSpecifications=[ { 'ResourceType': 'route-table', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-route-table' }, ] } ] ) routeTableId = routeTable['RouteTable']['RouteTableId'] route1 = ec2.create_route( DestinationCidrBlock= '0.0.0.0/0', GatewayId= igwId, RouteTableId= routeTableId ) #サブネットとルートテーブルの紐づけ ec2.associate_route_table( RouteTableId = routeTableId, SubnetId = publicSubnet1Id, ) #セキュリティグループの作成 securityGroup = ec2.create_security_group( Description='test-boto3', GroupName='boto3-test', VpcId= vpcId, ) sgId = securityGroup['GroupId'] #sshの設定 authorizeSsh = ec2.authorize_security_group_ingress( GroupId = sgId, IpPermissions=[ { 'FromPort': 22, 'IpProtocol': 'tcp', 'IpRanges': [ { 'CidrIp': 'myip', 'Description': 'SSH access from the office', }, ], 'ToPort': 22, }, ], ) #キーペアの作成 keyName = "boto3-test" keyPair = ec2.create_key_pair( KeyName=keyName, KeyFormat="ppk" ) #キーペアのファイルを作成する keyData = keyPair['KeyMaterial'] #with命令でfileオブジェクトの生成、書き込み専用(ファイルが存在しなければ新規作成) with open(keyName + ".ppk",mode='w') as f: f.write(keyData) #ec2インスタンスの起動 instance1 = ec2.run_instances( ImageId="ami-05a56ce08feadf9c4", MinCount=1, MaxCount=1, InstanceType="t2.micro", KeyName=keyName, SecurityGroupIds=[sgId], SubnetId=publicSubnet1Id, TagSpecifications=[ { 'ResourceType': 'instance', 'Tags': [ { 'Key': "Name", 'Value': "boto3" }, ] } ] ) |
それではコードの解説をしていきます。
1 2 3 4 |
ec2.py # boto3インポート import boto3 |
まずはライブラリをインポートします。
1 2 3 4 |
ec2.py # インスタンスの作成 ec2 = boto3.client('ec2') |
EC2サービスへのクライアントインターフェースを作成しています。これにより、EC2インスタンスに関連する操作をプログラムから行うことができます。
VPCの作成
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
ec2.py # vpc作成 vpc = ec2.create_vpc( CidrBlock = '192.0.0.0/24', TagSpecifications=[ { 'ResourceType': 'vpc', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-vpc' }, ] } ] ) |
新しいVPC(Virtual Private Cloud)を作成しています。CIDRブロックとして192.0.0.0/24をしていします。このVPCはNameタグをboto3-test-vpcとして設定します。
1 2 3 |
ec2.py vpcId = vpc['Vpc']['VpcId'] |
boto3のclientは戻り値としてdict型を返します。上記のコードでVPC IDを取得しています。
サブネットの作成
1 2 3 4 5 6 7 8 |
ec2.py # サブネット作成 publicSubnet1 = ec2.create_subnet( AvailabilityZone = 'ap-northeast-1a', CidrBlock = '192.0.0.0/28', VpcId = vpcId ) |
作成したVPC内に新しいサブネットを作成しています。このサブネットはアベイラビリティーゾーンap-northeast-1a(東京リージョンのAZ)に配置され、CIDRブロックとしては192.0.0.0/28が割り当てられています。先ほど取得したVPCIDを指定します。
1 2 3 |
ec2.py publicSubnet1Id = publicSubnet1['Subnet']['SubnetId'] |
作成したサブネットからサブネットIDを取得しています。
インターネットゲートウェイの作成
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
ec2.py # インターネットゲートウェイの作成 igw = ec2.create_internet_gateway( TagSpecifications=[ { 'ResourceType': 'internet-gateway', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-igw' }, ] } ] ) |
インターネットゲートウェイを作成し、Nameタグでboto3-test-igwと命名しています。
1 2 3 |
ec2.py igwId = igw['InternetGateway']['InternetGatewayId'] |
作成したインターネットゲートウェイからIDを取得しています。
1 2 3 4 5 6 7 |
ec2.py # インターネットゲートウェイをvpcにアタッチ ec2.attach_internet_gateway( InternetGatewayId= igwId, VpcId =vpcId ) |
VPCに対してインターネットゲートウェイをアタッチ(接続)しています。これにより、VPC内のインスタンスがインターネットと通信できるようになります。
ルートテーブルの作成
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
ec2.py # ルートテーブルの作成 routeTable = ec2.create_route_table( VpcId = vpcId, TagSpecifications=[ { 'ResourceType': 'route-table', 'Tags': [ { 'Key': 'Name', 'Value': 'boto3-test-route-table' }, ] } ] ) |
新しいルートテーブルを作成し、Nameタグでboto3-test-route-tableと命名しています。
1 2 3 |
ec2.py routeTableId = routeTable['RouteTable']['RouteTableId'] |
作成したルートテーブルからIDを取得しています。
1 2 3 4 5 6 7 |
ec2.py route1 = ec2.create_route( DestinationCidrBlock= '0.0.0.0/0', GatewayId= igwId, RouteTableId= routeTableId ) |
インターネットへのルート(全てのトラフィックを0.0.0.0/0でキャッチしてインターネットゲートウェイへと流すためのルール)をルートテーブルに追加しています。
1 2 3 4 5 6 7 |
ec2.py # サブネットとルートテーブルの紐づけ ec2.associate_route_table( RouteTableId = routeTableId, SubnetId = publicSubnet1Id, ) |
ルートテーブルをサブネットに関連付けています。これによりサブネット内のインスタンスが作成したルートテーブルを利用してインターネットに出ることが可能になります。
セキュリティグループの作成
1 2 3 4 5 6 7 8 |
ec2.py # セキュリティグループの作成 securityGroup = ec2.create_security_group( Description='test-boto3', GroupName='boto3-test', VpcId= vpcId, ) |
新しいセキュリティグループを作成しています。インスタンスに適用されるファイアウォールです。
1 2 3 |
ec2.py sgId = securityGroup['GroupId'] |
作成したセキュリティグループからIDを取得します。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
ec2.py # sshの設定 authorizeSsh = ec2.authorize_security_group_ingress( GroupId = sgId, IpPermissions=[ { 'FromPort': 22, 'IpProtocol': 'tcp', 'IpRanges': [ { 'CidrIp': 'myip', 'Description': 'SSH access from the office', }, ], 'ToPort': 22, }, ], ) |
セキュリティグループのインバウンドルールを設定しており、指定のIPアドレス(myipからのSSH(ポート22)接続を許可しています。ここは自身の環境のIPアドレスを設定してください。
キーペアの作成
1 2 3 4 5 6 7 8 9 |
ec2.py # キーペアの作成 keyName = "boto3-test" keyPair = ec2.create_key_pair( KeyName=keyName, KeyFormat="ppk" ) |
キーペアを作成します。フォーマットはppk(PuTTY用のフォーマット)をしていします。これによりプライベートキーが生成され、セキュリティで保護された方法でインスタンスに接続する際に使用します。
1 2 3 |
ec2.py keyData = keyPair['KeyMaterial'] |
生成したキーペアから実際のキー情報を取得しています。
1 2 3 4 5 6 |
ec2.py # キーペアのファイルを作成する # with命令でfileオブジェクトの生成、書き込み専用(ファイルが存在しなければ新規作成) with open(keyName + ".ppk", mode='w') as f: f.write(keyData) |
プライベートキーをファイルとして保存しています。このファイルはSSHログインに使用します。ファイル名は先ほど設定したキーペア名に.ppk拡張子をつけたものです。
EC2インスタンスの起動
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
ec2.py # ec2インスタンスの起動 instance1 = ec2.run_instances( ImageId="ami-05a56ce08feadf9c4", MinCount=1, MaxCount=1, InstanceType="t2.micro", KeyName=keyName, SecurityGroupIds=[sgId], SubnetId=publicSubnet1Id, TagSpecifications=[ { 'ResourceType': 'instance', 'Tags': [ { 'Key': "Name", 'Value': "boto3" }, ] } ] ) |
最後に、EC2インスタンスを起動します。使用するAMIはAmazon Linux2023を指定しています。、インスタンスタイプ、キーペア、セキュリティグループ、サブネットを指定し、最低1つ、最高1つのインスタンスを起動するように設定します。インスタンスにはboto3という名前タグを設定しました。
おわりに
今回はEC2インスタンス1台を起動するという簡単なものでしたがカスタマイズしていくことで独自のニーズに合わせたインフラの構築が可能です。ご活用いただけたら幸いです。
===
ナレッジコミュニケーションでは「Musubite」というエンジニア同士のカジュアルトークサービスを利用しています。
生成 AI 技術を使ったプロジェクトに携わるメンバーと直接話せるサービスですので興味がある方は是非利用を検討してください!
今後もクラウド AI の最新情報を皆さんに届けていきたいと思います。お楽しみに。