透過 s3 + api gateway + lambda 實作 image resize
data:image/s3,"s3://crabby-images/eab13/eab13e96815bc266530428a261668fa53c62fb49" alt=""
首先要先新增一個 s3 bucket, 這邊不多說明如何建置 s3 bucket, 但是要記得 bucket 須為 public
s3 這邊要注意以下幾點
- 記得 bucket 要為 public
- bucket 要啟用 static website hosting
- 切換到 properties
data:image/s3,"s3://crabby-images/7ed2d/7ed2ded067c2cd28da8152296437a9dae6a7c8a8" alt=""
4. 拉到最下面應該會看到 static website hosting, 請點選 Edit
data:image/s3,"s3://crabby-images/53f7d/53f7decee487d8df5ffe10e669a3b0e5199d817a" alt=""
5. 將 static website hosting enable, 且設定 hosting type 為 host a static website
data:image/s3,"s3://crabby-images/38518/38518c58cf3b4d96f67345c0b1a6fb110127c5c4" alt=""
6. 設定 redirection rules
[
{
"Condition": {
"HttpErrorCodeReturnedEquals": "404",
"KeyPrefixEquals": ""
},
"Redirect": {
"HostName": "<api gateway hostname>",
"HttpRedirectCode": "307",
"Protocol": "https",
"ReplaceKeyPrefixWith": "test/resize?key="
}
}
]
備註:
"ReplaceKeyPrefixWith": "<api gateway path>?key="這邊的 key 不能替換掉, 因為 lambda 會抓這邊的 key 去解析
這邊 rule 的意思是倘若存取到這個 bucket 的話, 都會跳轉到 <api gateway hostname>/<api gateway path>?key=<s3 filename>, 緊接著會觸發 lambda 去 resize image
建置 Lambda
- 新增 lambda function
data:image/s3,"s3://crabby-images/007c9/007c9d8bb0752e9cdca5e689efa1e7223e36b966" alt=""
2. Function name 輸入 resize, 且選擇合適的 nodejs 版本以及架構, 所選的架構要記一下, 因為稍後再安裝 nodejs 套件的時候需要指定, 且新增一個 aws role
data:image/s3,"s3://crabby-images/af813/af8133d35b4896ea9cf964901c1547d6f4fd694f" alt=""
data:image/s3,"s3://crabby-images/e46dd/e46dd5e13208760e2517444ecf6230f07354afb3" alt=""
3. 新增完後點擊“下載” nodejs lambda 壓縮檔案
建議先砍掉 node_modules, 然後重新執行 npm install,
但要注意 sharp 檔案需看先前新增 lambda 時是使用什麽架構, 倘若是使用 x86_64, 則使用以下指令 npm install --arch=x64 --platform=linux sharp@0.23.2,
然後因為 lambda 的 function 還會用到 aws-sdk, 因此 npm 也要安裝 aws-sdk
4. 新增完後會在列表看到剛剛新增的 lambda
data:image/s3,"s3://crabby-images/c9906/c9906f77f3e026d437b1337a4e33510101280db6" alt=""
5. 接著將剛剛下載下來的 zip 檔案上傳到 lambda, 點選Upload From 接著選取.zip file, 最後選擇剛剛下載下來的 zip file
ps: 因為檔案稍大因此可能會上傳一小段時間
data:image/s3,"s3://crabby-images/e0494/e049459d28cdda24866a3367933c2475693266ba" alt=""
6. 去到 iam 尋找上面所新增的 role, 找到後點選剛剛新增的 role, 我們這邊要賦予該 role 擁有存取 aws s3 的權限, 因為在 image 在 resize 後會將 resize 的圖片上傳到 s3, 而我們這邊給予該 role aws s3 full access
data:image/s3,"s3://crabby-images/ac3ac/ac3ac491468b7578d8491dcec00d95b14e353a63" alt=""
7. 選取 attach policies 後搜尋 AmazonS3FullAccess
data:image/s3,"s3://crabby-images/fa60e/fa60e19ee20590ec61071fe9a7adf26aa2995574" alt=""
8. 編輯環境變量, 點選 Configuration 後接著點選 Environment variables
data:image/s3,"s3://crabby-images/037b4/037b46e54289e88e63481b36cd7d67bfb5d1332e" alt=""
9. 因為 lambda 有用到 BUCKET, 以及 URL env 因此要新增這兩個出來
BUCKET: s3 bucket name
URL: s3 static website hosting endpoint url
data:image/s3,"s3://crabby-images/b4c9e/b4c9eafa938ae6b749e571bda2742b2e060ba0ba" alt=""
建置 API gateway
- 點選 Create API
data:image/s3,"s3://crabby-images/d5d0c/d5d0caeba541727196d4815f00d6711ed093929e" alt=""
2. 我們這邊選取 REST API 且點選 Build
data:image/s3,"s3://crabby-images/ed0fb/ed0fbcd54ea68f7d5f5bee4b3edf9658f7296400" alt=""
3. 輸入 api 名稱, 輸入完後點選新增
data:image/s3,"s3://crabby-images/46c86/46c86b72c25f6bc828bb98e498756ddb6a1938e5" alt=""
4. 新增 resource , 我們可以把 resource 想成是路徑!!
點選 Action 後選擇 Create Resource
data:image/s3,"s3://crabby-images/4ad10/4ad10404821993aa88b2d5066043781fd8891b3e" alt=""
5. 輸入 resource name, resource name 預設會成為路徑
例如: resource name 輸入 helloworld
那麼 resource path 會是 /helloworld
data:image/s3,"s3://crabby-images/5cc8b/5cc8b0586aa3c4af721387a2db9f58bde85ce318" alt=""
6. 新增完 resource 後, 我們緊接著要新增 restful method
a. 選擇 Lambda Function
b. 選擇 Lambda 的 region
c. 選擇 or 輸入上面所新增的 Lambda function name
d. 勾選 Use Lambda Proxy integration
data:image/s3,"s3://crabby-images/2af34/2af34e60acdff0e59c6b205b59cbc949b63d0ccd" alt=""
7. deploy api (產生的 api hostname 以及相對應的 path 以及 method)
a. 點選 deploy api
data:image/s3,"s3://crabby-images/7771b/7771b6d7f8c23c717eecafc2216141c9909d9782" alt=""
b. 因為一開始沒有新增過 stage, 因此選擇 create stage
data:image/s3,"s3://crabby-images/bf447/bf4472c8fc5d2b37136f1f2e2ec56e4829334f27" alt=""
c. 新增完後可以去 stage 查看剛剛新增的資訊以及 api
應該會看到類似以下的畫面, 倘若想要查看 api host 可以點選 GET 或者其他的 method 查看
data:image/s3,"s3://crabby-images/f616c/f616c1ec241fc56d97ed535e30df08d4fc20cbe9" alt=""
測試 resize image
使用 s3 新增的 static website hosting 的 Bucket website endpoint,
假如 endpoint 為 http://test.bucket.s3, 則可以在網址列輸入
http://test.bucket.s3/100x100/<filename>
這樣的話會在 s3 新增一個 100x100 的 folder, 而裡面會有 filename 即代表成功實作 resize image lambda
參考資料:
- https://docs.aws.amazon.com/zh_tw/apigateway/latest/developerguide/api-gateway-create-api-as-simple-proxy-for-lambda.html
- https://aws.amazon.com/tw/blogs/compute/resize-images-on-the-fly-with-amazon-s3-aws-lambda-and-amazon-api-gateway/
- https://docs.aws.amazon.com/lambda/latest/dg/getting-started-create-function.html
- https://docs.aws.amazon.com/AmazonS3/latest/userguide/EnableWebsiteHosting.html
- https://docs.aws.amazon.com/zh_tw/AmazonS3/latest/userguide/how-to-page-redirect.html
- https://stackoverflow.com/questions/64598842/aws-s3-redirection-rules-not-letting-me-saving-it
- https://sharp.pixelplumbing.com/install
- https://leesonhsu.blogspot.com/2018/03/awss3-imageresize.html