![見出し画像](https://assets.st-note.com/production/uploads/images/132931497/rectangle_large_type_2_76b4030ce8fb836dfa14d9a305e551da.png?width=1200)
CloudFrontとWAFでフロントエンドとAPIを保護する
CloudFrontにWAFを設定してフロントとAPIを保護するというのをやってみました。画面としては以下のような画面で、ボタンをクリックするとユーザー情報を取得して一覧を作成する簡単なものです。
ユーザー情報は以下のAPIから取得しています。
https://jsonplaceholder.typicode.com/users
![](https://assets.st-note.com/img/1710624125272-ieI1HFpVHG.png?width=1200)
構成
構成としては以下の通り。
![](https://assets.st-note.com/img/1710651552955-K609EqJRtw.png?width=1200)
フロントエンド
フロントエンド用に作成するリソースは、フロントエンド資材を配置するS3バケットだけです。
S3バケットの作成
フロントエンドの資材の配信元となるS3バケットを作成します。
![](https://assets.st-note.com/img/1710625637592-6v9pw0OA02.png?width=1200)
バケット名を入力して、それ以外はデフォルトで大丈夫です。バケットを作成ボタンをクリックしてバケットを作成します。
![](https://assets.st-note.com/img/1710625765597-VBb0KV302m.png?width=1200)
![](https://assets.st-note.com/img/1710625894558-Dgah2YcjNk.png?width=1200)
バックエンド
バックエンド用に作成するリソースは、API GatewayとLambdaの2つです。
Lambdaの作成
ユーザー情報を取得するLambda関数を作成します。
![](https://assets.st-note.com/img/1710626503854-j8fgMxf6jn.png?width=1200)
一から作成を選択し、関数名を入力し、ランタイムはNode.js 20.xにします。その他の設定はデフォルト設定とします。
![](https://assets.st-note.com/img/1710626987688-W7vliiqMTd.png?width=1200)
作成が出来たら、コードタブに移動します。
![](https://assets.st-note.com/img/1710627877892-pue5cIU4Wd.png?width=1200)
index.mjsにユーザー情報を取得するコードを書きDeployボタンをクリックします。
![](https://assets.st-note.com/img/1710627928693-aZb6rHOOGS.png?width=1200)
コードとしては以下の通りです。
/* global fetch */
export const handler = async (event) => {
const responseBase = {
headers: {
"content-type": "application/json",
"Access-Control-Allow-Origin": "*"
}
};
try {
const users = await fetch("https://jsonplaceholder.typicode.com/users").then(
(response) => response.json()
);
return {
...responseBase,
statusCode: 200,
body: JSON.stringify({
users
})
};
} catch (err) {
console.log(err);
return {
...responseBase,
statusCode: 500,
body: JSON.stringify({
error: JSON.stringify(err)
})
};
}
};
![](https://assets.st-note.com/img/1710628014573-I5SlyLiRwi.png?width=1200)
最後にTestボタンをクリックして、正常に取得できているかを確認します。
![](https://assets.st-note.com/img/1710628094376-EMauw4WhVA.png?width=1200)
API Gatewayの作成
APIの入り口となるAPI Gatewayを作成します。REST APIの構築ボタンをクリックします。
![](https://assets.st-note.com/img/1710628229439-i2oed9suV9.png?width=1200)
新しいAPIを選択し、API名を入力し、APIエンドポイントタイプはリージョンを選択します。APIを作成ボタンをクリックして、APIを作成します。
![](https://assets.st-note.com/img/1710628435699-OFM2dBLbaI.png?width=1200)
ユーザー情報の取得のためのリソースを作成していきます。リソースを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710628581625-gYfEtauwLm.png?width=1200)
リソース名に「users」を入力し、CORS (クロスオリジンリソース共有)にチェックをして、リソースを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710628963086-SfBwtE2eiA.png?width=1200)
作成されたらメソッドを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710629046695-7foNzfWT73.png?width=1200)
以下の通り入力し、その他はデフォルトのままで、メソッドを作成ボタンをクリックします。
メソッドタイプ: GET
統合タイプ: Lambda関数
Lambdaプロキシ統合: チェック
Lambda関数: 先ほど作成したLambda関数を選択
![](https://assets.st-note.com/img/1710629334706-gxmFqLWBZk.png?width=1200)
作成が完了したらテストタブをクリックします。
![](https://assets.st-note.com/img/1710629636922-mccrpXvOUW.png?width=1200)
テストボタンをクリックして、Lambdaの呼び出しに成功するか確認します。
![](https://assets.st-note.com/img/1710629697074-E2PC5hBtLB.png?width=1200)
APIをデプロイボタンでデプロイを行います。
![](https://assets.st-note.com/img/1710629837355-E31ULsuRA5.png?width=1200)
ステージは新しいステージを選択し、ステージ名はapiを入力し、デプロイボタンをクリックします。
![](https://assets.st-note.com/img/1710629910766-68lgGxzi78.png?width=1200)
デプロイが完了したらURLを呼び出すに記載されているURLに/usersを追加して、ブラウザからアクセスが出来るか確認します。
![](https://assets.st-note.com/img/1710630042491-ZR5d0mNkll.png?width=1200)
以下の通りJSONの情報が返ってきていれば確認が完了です。
![](https://assets.st-note.com/img/1710652125210-vA6dhfptwU.png?width=1200)
WAF
アプリケーションを保護するためのWAFを作成します。
Web ACLの作成
Global (CloudFront)を選択し、Create web ACLボタンをクリックします。
![](https://assets.st-note.com/img/1710632950739-XQCAMTxlyB.png?width=1200)
Resource typeはAmazon CloudFront distributionsを選択し、Nameを入力し、その他はデフォルトのままでNextボタンをクリックします。
![](https://assets.st-note.com/img/1710633419462-naKb4MeB17.png?width=1200)
Add rulesをクリックし、Add managed rule groupsをクリックします。
※今回はマネージドルールだけを追加します。最後にあるCDKサンプルではIP制限も入れています。
![](https://assets.st-note.com/img/1710633584637-nsVLZss0Mt.png?width=1200)
AWS managed rule groupsを開き、Free rule groupsのCore rule setのAdd to web ACLをONにします。
![](https://assets.st-note.com/img/1710633872844-y2vdB4ZQy4.png?width=1200)
![](https://assets.st-note.com/img/1710633877928-KPhib6czOb.png?width=1200)
Rulesにルールが追加されたことを確認し、Default actionはAllowを選択し、Nextボタンをクリックします。
![](https://assets.st-note.com/img/1710634026573-NBODVbi21u.png?width=1200)
ルールの優先順位については、今回は1つだけしか入れていないので、そのままNextボタンをクリックします。
![](https://assets.st-note.com/img/1710634124475-d5HrBZVNH8.png?width=1200)
メトリクスの設定は、今回はデフォルトのままで、Nextボタンをクリックします。
![](https://assets.st-note.com/img/1710634441446-nObciJdFWe.png?width=1200)
設定内容を確認して、Create web ACLボタンをクリックしてWAFを作成します。
![](https://assets.st-note.com/img/1710634671083-wbLkUXksmK.png?width=1200)
![](https://assets.st-note.com/img/1710634676634-uiOxUA9nVf.png?width=1200)
![](https://assets.st-note.com/img/1710634712860-SrtnRxC2S4.png?width=1200)
CloudFront
フロントエンド用S3バケットのオリジンアクセスコントロールの作成
CloudFrontからフロントエンドのソースが配置されているS3バケットにアクセスするためのオリジンアクセスコントロール(以降OAC)を作成します。CloudFrontのコンソールから、セキュリティのオリジンアクセスに遷移します。コントロール設定を作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710634892811-jMFjS2dOsG.png?width=1200)
OACの名前を入力し、オリジンタイプはS3に設定してCreateボタンをクリックしてOACを作成します。
![](https://assets.st-note.com/img/1710634991656-esA5DbJe8o.png?width=1200)
![](https://assets.st-note.com/img/1710635026577-NSjW3aEquH.png?width=1200)
CloudFrontディストリビューションの作成
CloudFrontのディストリビューションを作成します。
![](https://assets.st-note.com/img/1710635080614-vUe26chZaM.png?width=1200)
Origin domainには、作成したS3バケットを指定します。名前は、S3バケットを選択した際に自動で設定されますので、そのままでも変えてもどちらでも良いです。本記事ではそのままで行きます。
オリジンアクセスは、Origin access control settings (recommended)を選択し、先ほど作成したOACを選択します。
![](https://assets.st-note.com/img/1710635791930-ljRMIIv5Fo.png?width=1200)
ビューワープロトコルポリシーは、Redirect HTTP to HTTPSを選択し、キャッシュキーとオリジンリクエストには、Cache policy and origin request policy (recommended)で、CachingOptimizedを選択します。
![](https://assets.st-note.com/img/1710636036996-ZYsQKyAxij.png?width=1200)
WAFは、セキュリティ保護を有効にするを選択し、既存のWAF設定を使用を選択し、先ほど作成したWAFを選択します。
![](https://assets.st-note.com/img/1710635852613-CnScmIuNyv.png?width=1200)
料金クラスは、日本向けと仮定して、北米、欧州、アジア、中東、アフリカを選択しておきます。デフォルトルートオブジェクトは、index.htmlを入力します。
![](https://assets.st-note.com/img/1710636227865-vec5Ouwqsh.png?width=1200)
その他はデフォルトのままで、ディストリビューションを作成ボタンをクリックして、ディストリビューションを作成します。
![](https://assets.st-note.com/img/1710636271203-ZP4TqE4b0M.png?width=1200)
作成されたらポリシーをコピーボタンが黄色の帯の右側にあるので、クリックしてポリシーをコピーしておきます。
![](https://assets.st-note.com/img/1710636333406-S19HH4z0yQ.png?width=1200)
オリジンS3バケットのバケットポリシーを設定
S3バケットに戻り、アクセス許可のタブのバケットポリシーの編集ボタンをクリックします。
![](https://assets.st-note.com/img/1710636387672-hScSVTlA8W.png?width=1200)
先ほど、CloudFrontのディストリビューションを作成後にコピーしたポリシーを、ポリシー欄にペーストして、変更の保存ボタンをクリックします。
![](https://assets.st-note.com/img/1710642238926-VNPAMhs5zV.png?width=1200)
![](https://assets.st-note.com/img/1710642245105-UXH08nPRLN.png?width=1200)
CloudFrontとS3の連携の確認
想定通りS3からフロントエンド資材を取得できているか確認します。作成したS3バケットに適当に作成したindex.htmlファイルをアップロードします。
![](https://assets.st-note.com/img/1710642497168-rHqhNjN1cz.png?width=1200)
作成したCloudFrontのディストリビューションのディストリビューションドメイン名をコピーしてブラウザでアクセスして表示されれば確認完了です。
![](https://assets.st-note.com/img/1710642674437-Y4I6nG8K6n.png?width=1200)
API Gatewayのオリジンとビヘイビアを追加
CloudFrontとAPI Gatewayを連携させるため、ディストリビューションにオリジンとビヘイビアを追加します。
オリジンタブのオリジン作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710642841485-tnxCEdTl7F.png?width=1200)
Origin domainに作成したAPI Gatewayを選択します。
カスタムヘッダーとしてRefererを追加し、値には適当な設定します。このカスタムヘッダーは、APIのリクエストもWAFを通したいため、CloudFront経由でしかAPI Gatewayにアクセス出来ないようにするためのものです。後でAPI Gatewayのリソースポリシーにも設定します。ただし、この値を知っていればアクセス出来てしまいますので、推測の出来ない値を設定する必要があります。
※ API Gatewayで、CloudFrontのこのディストリビューションから、みたいなリソースポリシーが設定できないか調べたのですが、見つけられずこの方法を取ることにしました。
オリジンを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710643020110-Ha1iIdKIGU.png?width=1200)
追加ができたら、ビヘイビアタブをクリックし、ビヘイビアを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710643573213-N4YJIdhxLn.png?width=1200)
パスパターンに「/api/*」オリジンとオリジングループに先ほど作成したAPI Gatewayのオリジンを選択します。ビューワープロトコルポリシーはRedirect HTTP to HTTPSを選択し、許可された HTTP メソッドはGET, HEAD, OPTIONS, PUT, POST, PATCH, DELETEを選択します。
![](https://assets.st-note.com/img/1710643847688-QNFdlItMjA.png?width=1200)
キャッシュポリシーにはCachingDisabled、オリジンリクエストポリシーには、AllViewerExceptHostHeaderを選択して、ビヘイビアを作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710643840099-xxru981aB8.png?width=1200)
![](https://assets.st-note.com/img/1710644097305-Dd5BusIt0r.png?width=1200)
API Gatewayのリソースポリシーの作成
先ほど、API Gatewayのオリジンで、Refererをカスタムヘッダーとして追加しました。それをAPI Gatewayへのアクセス制限として利用するため、API Gatewayのリソースポリシーを設定します。
![](https://assets.st-note.com/img/1710644386999-MnO81M9Ykc.png?width=1200)
ポリシーの詳細のエリアに、Refererが一致しなければ拒否するというポリシーの設定と通常は許可をする設定を入力し、変更を保存ボタンをクリックします。
![](https://assets.st-note.com/img/1710644536822-7g03TLgRKT.png?width=1200)
ポリシーとしては以下の通りです。
{
"Version": "2012-10-17",
"Statement": [
{
"Effect": "Deny",
"Principal": "*",
"Action": "execute-api:Invoke",
"Resource": "arn:aws:execute-api:ap-northeast-1:{AccountのID}:hu32j8ntf8/*/*/*",
"Condition": {
"StringNotEquals": {
"aws:Referer": "5MvvYcvcjGPHg6CP2vWA"
}
}
},
{
"Effect": "Allow",
"Principal": "*",
"Action": "execute-api:Invoke",
"Resource": "arn:aws:execute-api:ap-northeast-1:{AccountのID}:hu32j8ntf8/*/*/*"
}
]
}
設定が保存されたらリソースへ移動して、APIをデプロイボタンをクリックして、リソースポリシーの変更をデプロイします。
![](https://assets.st-note.com/img/1710644789054-RkxQzEvwiz.png?width=1200)
![](https://assets.st-note.com/img/1710644841772-afljwu35Xk.png?width=1200)
CloudFrontとAPI Gatewayの連携の確認
デプロイが完了したらURLを呼び出すに記載されているURLに/usersを追加して、ブラウザからアクセスして、想定通り拒否されるか確認します。
![](https://assets.st-note.com/img/1710645111839-rvB2B5YYi9.png?width=1200)
次に、CloudFrontのディストリビューション経由であれば、ユーザー情報が取得できることを確認します。
![](https://assets.st-note.com/img/1710645147960-W4PFzKaxqA.png?width=1200)
フロントエンド資材のデプロイ
AWS環境の方は準備ができたので、フロントエンドの資材をビルドして、アップロードします。CI/CDは用意していないので、ローカル環境でビルドを行いS3にアップロードします。
![](https://assets.st-note.com/img/1710645544158-hAd32R5xwx.png?width=1200)
![](https://assets.st-note.com/img/1710645578970-S3QbUic75Z.png?width=1200)
動作確認
CloudFrontディストリビューションのキャッシュの削除
先ほど動作確認でアクセスしているため、キャッシュを削除しておきます。キャッシュ削除タブのキャッシュ削除を作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710645715849-NEG1zD9AVi.png?width=1200)
オブジェクトパスを追加に「/*」を入力し、キャッシュ削除を作成ボタンをクリックします。
![](https://assets.st-note.com/img/1710645767610-cnt2I6L78s.png?width=1200)
スタータスが完了済みになるまで待ちます。
![](https://assets.st-note.com/img/1710645870693-aoz3v8NaBu.png?width=1200)
フロントエンドの表示確認とバックエンドとの疎通確認
作成したCloudFrontのディストリビューションのURLにブラウザでアクセスして表示され、取得ボタンをクリックしてユーザー情報が表示されれば、確認完了です。
![](https://assets.st-note.com/img/1710646087293-cF60hnxNDw.png?width=1200)
![](https://assets.st-note.com/img/1710646122268-J9RhLiHhY7.png?width=1200)
WAFのサンプリングの確認
作成したWAFのWeb ACLのSampled requestsタブに移動し、WAFを利用できているか確認します。APIのパスである「/api/users」もフロントエンドの資材もあるため、問題なく利用ができていることが確認できます。
![](https://assets.st-note.com/img/1710652465093-ml3FPRPYsP.png?width=1200)
事後処理
今回作成したリソースの削除は以下になります。
CloudFront
作成したCloudFrontのディストリビューションを削除する
無効にする
最終変更日がデプロイから日付に変わるまで待つ
削除する
作成したCloudFrontのオリジンアクセスを削除する
S3
作成したバケットを空にする
作成したバケットを削除する
API Gateway
作成したREST APIを削除する
Lambda
作成した関数を削除する
WAF
作成したWeb ACLを削除する
CloudWatch Logs
自動で作成されたLambdaのロググループを削除する(/aws/lambda/Lambda関数名)
IAM
自動で作成されたポリシーを削除する
作成時刻を見て、直近のAWSLambdaBasicExecutionRole-xxxxxxのポリシーを削除
自動で作成されたロールを削除する
cloudfront-s3-api-waf-get-users-role-xxxのロールを削除(念のため作成時刻を確認して実施する)
まとめ
今回はCloudFrontにWAFを設定して、フロントエンドとAPIを保護するということをやってみました。これである程度は悪意のある攻撃からWebアプリを守ることが出来そうです。WAFの他のマネージドルールやカスタムルールなども作成できるため、どこまで防御するのかなどを決めて、設定していけば良いというスタート地点までは出来たかなと思います。
もちろん、フロントエンド、API自体などでもセキュリティ対策は必要になるのですが、WAFがあるだけで、少し安心が出来ますね。
CDK版も作ってみました。
この記事が参加している募集
この記事が気に入ったらサポートをしてみませんか?