0
0

API Gatewayで受けたファイルをS3にユニークな名前で保存する。

Posted at

既存Webサービスのファイルアップロード機能をサーバーレス化する

1. 背景

既存のWebサービスでは、ファイルアップロード系のAPIをサーバーレス化する必要が生じました。この記事では、その過程と実装方法について詳しく解説します。

image.png
[図1: 既存システムの構成図]

2. 構築

2.1 既存システムの構成

既存のシステムでは、ログイン、情報確認、ファイルアップロードの機能をApplication Load Balancer (ALB)、EC2インスタンス、RDSで実現していました。

image.png

[図2: 既存システムの詳細構成図]

2.2 CloudFrontによる振り分け

機能を切り出すにあたり、CloudFrontを使用してトラフィックの振り分けを行うことにしました。

2.2.1 HTTPS to HTTPSの設定

CloudFrontで振り分ける際にHTTPS to HTTPSを行う場合、ALBに別途FQDNとACMの設定が必要になります。
既存のサービスへ切り戻しや切り替え時の通信断を減らすため既存のFQDN service.sample.com を残しつつ、FQDN( 'service-alb.sample.com' )を追加する。
image.png

2.2.2 API Gatewayのカスタムドメイン設定

今回切り分けるAPI Gatewayも、カスタムドメインで同様にFQDNとACMを設定しておきます。

custom domain: service-api.sample.com
API mapping: default(/) → sample-api/prd/

2.3 API Gatewayの設定

API GatewayではREST APIのPOSTで通信を受け取り、AWSサービス統合を使用してS3にPUTリクエストを送信します。この際、パスオーバーライドを使用してS3のオブジェクトキーを動的に設定したい。

2.3.1 初期の問題点

最初は以下のように設定しましたが、これでは常に同じファイル名で上書きされてしまう問題がありました。

my-bucket/upload/file

image.png

[スクリーンショット3: 初期のAPI Gateway設定画面]

2.3.2 requestIDを使用した解決策

いろいろ試して、パスオーバーライドで変数を使う方法に行き着いた。my-bucket/upload/{key}のように設定し、requestIDでkeyを上書きすることにしました。

image.png
[スクリーンショット4: パスオーバーライドで変数定義]

マッピングテンプレートは作成後にメソッドを編集する画面でしか出てこないので、作成後に変更する。
image.png

編集画面の一番下に、作成時にはなかったマッピングテンプレートが出てくる。
受信するデータのContent-Typeに合わせて設定を作成する。下記は:Content-Type: application/jsonの例
image.png

#set($context.requestOverride.path.key=$context.requestId)
$input.body

2.3.3 日付ベースのフォルダ構造の実装

さらに使いやすくするため、日付でフォルダを分ける構造を実装しました。

エポック日を使用した方法

ただ、サンプルを見てるとエポックミリ秒を使ったものが多く、日付ユーティリティが API Gatewayには実装されていない状態だったので、エポック日に方向転換(閏年とか月毎の月末日の違いとか考えるのが面倒)

#set($EpochDay=$context.requestTimeEpoch/1000/3600/24)
#set($context.requestOverride.path.key="${EpochDay}/${context.requestId}")
$input.body
UTCの年月日を使用した方法

API Gateway開発者ガイドを参照し、$context.requestTimeを使用してUTCベースの年月日フォルダ構造を実装しました。

[コードブロック3: UTCの年月日を使用したVTLマッピングテンプレート]

#set($year=$context.requestTime.substring(7,11))
#set($monthstr=$context.requestTime.substring(3,6))
#if($monthstr=="Jan")
    #set(month=01)
#elseif($monthstr=="Feb")
    #set(month=02)
#elseif($monthstr=="Mar")
    #set(month=03)
#elseif($monthstr=="Apr")
    #set(month=04)
#elseif($monthstr=="May")
    #set(month=05)
#elseif($monthstr=="Jun")
    #set(month=06)
#elseif($monthstr=="Jul")
    #set(month=07)
#elseif($monthstr=="Aug")
    #set(month=08)
#elseif($monthstr=="Sep")
    #set(month=09)
#elseif($monthstr=="Oct")
    #set(month=10)
#elseif($monthstr=="Nov")
    #set(month=11)
#elseif($monthstr=="Dec")
    #set(month=12)
#end
#set($day=$context.requestTime.substring(0,2))
#set($context.requestOverride.path.key="${year}/${month}/${day}/${context.requestId}")
$input.body

3. 結論

この方法により、既存のWebサービスのファイルアップロード機能を効率的にサーバーレス化することができました。日付ベースのフォルダ構造により、アップロードされたファイルの管理も容易になりました。
image.png

[図3: 最終的なシステム構成図]

4. 参考資料

  1. API Gateway開発者ガイド
  2. AWS再投稿: API Gateway call S3 put object using VTL
0
0
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
0
0