###### tags: `YouTube` # App Runner 入門 ## IAM ユーザの作成 ![](https://i.imgur.com/S9SGXb1.jpg) ![](https://i.imgur.com/peZspz4.jpg) ![](https://i.imgur.com/p2QVpb2.png) ![](https://i.imgur.com/OEXI3Je.jpg) ![](https://i.imgur.com/U81WnHw.png) ![](https://i.imgur.com/TL2xmWB.jpg) ![](https://i.imgur.com/pJOHgix.jpg) ![](https://i.imgur.com/NHYS2BM.jpg) ![](https://i.imgur.com/0KHn3ph.png) ```bash $ aws configure --profile apprunner-tutorial ``` 対話型の設定画面が現れる。 ``` AWS Accrss Key ID [None]: 取得したアクセスキー ID AWS Secret Access Key [None]: 取得したシークレットアクセスキー Default region name [None]: ap-northeast-1 Default output format [None]: json ``` ## AWS CDK によるリソースの作成 ### 環境構築 #### Node.js のインストール まずは、Node.js をインストールする。 Node.js のバージョン管理システムである Nodebrew を使うため、最初に **Homebrew** をインストールする。 ```bash $ /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)" ``` インストールできたら、バージョンを確認。 バージョンが表示されれば OK。 ```bash $ brew -v Homebrew 3.6.21 ``` 続いて、Nodebrew をインストールする。 ``` $ brew install nodebrew ``` インストールできたら、バージョンを確認。 バージョンが表示されれば OK。 ```bash $ nodebrew -v nodebrew 1.0.1 ``` 次に、Nodebrew を使って Node.js の安定版をインストールする。 ```bash $ nodebrew install-binary stable ``` インストールされたバージョンを確認する。 ```bash $ nodebrew ls ``` 2023/02/22 現在、安定版は v18.14.2 であるため、v18.14.2 を有効化する。 ```bash nodebrew use v18.14.0 ``` 最後に PATH を通す。 zsh を使っている場合は下記のコマンドを実行する。 ```bash $ echo 'export PATH=$HOME/.nodebrew/current/bin:$PATH' >> ~/.zshrc $ source ~/.zshrc ``` bash を使っている場合は下記のコマンドを実行する。 ```bash $ echo 'export PATH=$HOME/.nodebrew/current/bin:$PATH' >> ~/.bash_profile $ source ~/.bash_profile ``` Node.js のバージョンを確認すると、v18.14.0 が表示されるはず。 ``` $ node -v v18.14.0 ``` これで、Node.js のインストールが完了。 #### npm のインストール 続いて、下記コマンドを用いて npm をインストールする。 ```bash $ npm install -g npm ``` インストールされたバージョンを確認する。 ```bash $ npm -v 9.4.2 ``` #### AWS CDK CLI のインストール 最後に、下記コマンドを用いて AWS CDK CLI のインストールする。 ```bash $ npm install -g aws-cdk ``` インストールされたバージョンを確認する。 ```bash $ cdk --version 2.64.0 (build fb67c77) ``` ### CDK プロジェクトを作成する 新しいディレクトリの作成。 ```bash $ mkdir apprunner-tutorial && cd apprunner-tutorial ``` プログラミング言語として今回は TypeScript を採用する。 ```bash $ cdk init app --language typescript ``` ### 必要なライブラリのインストール App Runner コンストラクトライブラリをインストールする。 ```bash $ npm install @aws-cdk/aws-apprunner-alpha ``` ### Stack の設定 bin/apprunner-tutorial.ts ```typescript #!/usr/bin/env node import "source-map-support/register"; import * as cdk from "aws-cdk-lib"; import { ApprunnerTutorialStack } from "../lib/apprunner-tutorial-stack"; const app = new cdk.App(); new ApprunnerTutorialStack(app, "ApprunnerTutorialStack", { env: { account: "761817748648", region: "ap-northeast-1" }, }); ``` env の account は、IAM ユーザを作成した AWS アカウント ID を指定する。アカウント ID は、下記コマンドで確認可能。 ```bash $ aws sts get-caller-identity --profile apprunner-tutorial ``` ### Stack・Construct の作成 AWS リソースを定義する。 lib/apprunner-tutorial-stack.ts ```typescript import * as cdk from "aws-cdk-lib"; import { Construct } from "constructs"; import * as assets from "aws-cdk-lib/aws-ecr-assets"; import * as apprunner from "@aws-cdk/aws-apprunner-alpha"; export class ApprunnerTutorialStack extends cdk.Stack { constructor(scope: Construct, id: string, props?: cdk.StackProps) { super(scope, id, props); const asset = new assets.DockerImageAsset(this, "ImageAssets", { directory: "./app", platform: assets.Platform.LINUX_AMD64, }); new apprunner.Service(this, "Service", { source: apprunner.Source.fromAsset({ asset: asset, imageConfiguration: { port: 8080, }, }), }); } } ``` ## Go 言語でアプリ作成 ### Go のインストール 最初に Go をインストールする。 今回は macOS 上で動作するパッケージ管理ツール Homebrew を使って Go をインストールする。 ```bash $ brew install go ``` 上記コマンドの実行が完了したら、インストールされた Go のバージョンを確認。 ```bash $ go version go version go1.20.1 darwin/amd64 ``` ### API サーバの作成 ディレクトリを作成。先ほど、CDK のプロジェクトを作成した場所で、新たに app ディレクトリを作成する。 ```bash $ mkdir app && cd app ``` 現在のディレクトリにおけるモジュールを外部からインポートする時のパスを設定する。基本的には GitHub での公開が一般的となるため、`github.com/<ユーザ名>/<GitHub上のリポジトリ名>` といったパスを指定する。 ```bash $ go mod init github.com/FarStep131/apprunner-tutorial/app ``` `server/server.go` を作成する。 ```bash $ mkdir server && touch server/server.go ``` server/server.go ```go package server import ( "log" "net/http" "sync/atomic" "time" "github.com/gin-gonic/gin" ) type Server struct { counter int64 server *http.Server router *gin.Engine } // New creates an instance of a gin server with counter endpoint func New() *Server { router := gin.Default() server := &Server{ router: router, counter: int64(0), } router.GET("/", server.CounterHandler) return server } // CounterHandler calculates number of requests and return a json with counter number func (s *Server) CounterHandler(ctx *gin.Context) { counter := atomic.AddInt64(&s.counter, 2) ctx.JSON(200, gin.H{"counter": counter}) } // Start starts the server and listen on a provided address in a format <host>:<port> func (s *Server) Start(address string) error { s.server = &http.Server{ Addr: address, Handler: s.router, ReadTimeout: 10 * time.Second, } log.Printf("start server on %s", address) return s.server.ListenAndServe() } ``` `main.go` を作成する。 main.go ```go package main import ( "fmt" "log" "github.com/NaokiYazawa/apprunner-tutorial/app/server" ) const ( host = "0.0.0.0" port = "8080" ) func main() { s := server.New() err := s.Start(fmt.Sprintf("%s:%s", host, port)) if err != nil { log.Fatalf("server stopped with error: %s", err) } log.Printf("server stopped successfully") } ``` サーバを起動する。 ```bash $ go run main.go ``` ## 作成したアプリの Docker 化 `Dockerfile` を作成する。 ```bash $ touch Dockerfile ``` Multi-stage build を用いてイメージを作成する。 【1st Stage】 1. Go 言語の環境を用意 2. ライブラリ等のインストール 3. バイナリファイルの作成 【2nd Stage】 1. バイナリファイルの実行環境を用意 2. 1st stage から作成されたバイナリファイルをコピー 3. バイナリファイルの実行 ![](https://i.imgur.com/jDM1lbw.png) イメージサイズは小さければ小さいほどよい - アップロードもダウンロードも早くなる - 起動に要する時間が短縮される - コンテナに含まれるプログラムが少ないほど、脆弱性が軽減される Dockerfile ```dockerfile # Build stage # golang のイメージを使用 FROM golang:1.20 AS builder WORKDIR /app COPY . . RUN go mod download RUN CGO_ENABLED=0 go build -o main /app/main.go # Run stage # Debian の distroless をベースイメージに別のステージを定義 # 最終的なイメージには、distroless/static-debian11 イメージの最小限のライブラリセットとアプリの実行ファイルのみが含まれる FROM gcr.io/distroless/static-debian11 WORKDIR /app COPY --from=builder /app/main . EXPOSE 8080 CMD [ "/app/main" ] ``` Docker Desktop が起動していることを確認後、ビルドする。 ```bash $ docker build -t app . ``` `-t` オプションをつけることで、イメージの名前・タグを指定することができる。 コマンドの最後のピリオドは、`Dockerfile` が存在するディレクトリを指定。 実行する。 ```bash $ docker run --name app --rm -it -p 8080:8080 app ``` | オプション | 用途 | | -------- | -------- | | **--name** | コンテナの名前を指定する | | **--rm** | コンテナを終了したときにコンテナを削除する | | **-it** | -i -t と同じ意味。インタラクティブモードで起動する。 | | **-p** | ポートを指定する(外部のポート:コンテナ内部のポート) | コマンドの最後の app は、イメージの名前を指定。 ## デプロイ ### ブートストラップ ブートストラップは、AWS CDK アプリケーションを環境にデプロイする前にリソースをプロビジョニングするプロセスである。 これらのリソースには、ファイルを保存するための Amazon S3 バケットや、デプロイの実行に必要なアクセス権限を付与する IAM ロールが含まれる。 ```bash $ cdk bootstrap --profile apprunner-tutorial ``` ### 単一または複数のスタックを AWS にデプロイ ```bash $ cdk deploy --profile apprunner-tutorial ``` ### 単一または複数のスタックの削除 ```bash $ cdk destroy --profile apprunner-tutorial ```