0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

S3へのレプリケーションをトリガーにLambdaのコード更新・新規バージョン発行を実行する

Last updated at Posted at 2025-01-21

はじめに

バックエンドのBuild成果物(jar)がDeploy環境にレプリケートされた事をトリガーに、Lambdaのコード更新・新規バージョン発行を行うCDの構築を行ったので、それに関するTipsのようなものを書き残しておこうと思う。
(前々回はCloud Frontのキャッシュ削除についての記事、前回はECSのサービス更新についての記事、だったが、今回でレプリケーション後のDeployシリーズ3つ目となる、Lambdaのコード更新・新規バージョン発行についての記事。)

構成イメージ

何らかのトリガーでCode BuildがStartすると、S3にBuild成果物であるjarがpushされ、それをトリガーにレプリケートが走る。レプリケーション先の環境でレプリケート後に自動でDeploy(Lambdaのコード更新・新規バージョン発行)を実行させる。

図で示すと以下のようになる。
image.png

Lambdaのコード更新・新規バージョン発行

今回のDeployは、S3へのレプリケーションでjarがPUTされたら、Lambdaのソースコード更新・新規バージョン発行を行うというもの。
Deploy自体は、Cloud Frontのキャッシュ削除の時と同様に、レプリケーションでS3のBucketに新規でオブジェクトが作成されるのでそれをトリガーにLambda関数を実行し、以下のAWS CLIコマンドで行っている事と同じ事をAWS SDK(今回はJavaScript)で実行させる、という感じで行う。

aws lambda update-function-code --function {function名} --s3-bucket {jarのあるS3 Bucket名} --s3-key {jarのファイル名}
aws lambda publish-version --function-name {function名}

{jarのファイル名} について、Bucketの中でフォルダ階層を作っているならルートディレクトリからのパスで書く必要がある
 ex) --s3-key artifact/lambda/hoge.jar

・参考:aws.lambda.update-function-code
・参考:aws.lambda.publish-version

IAM

S3へのレプリケーションをトリガーにCloud Frontのキャッシュ削除を実行する#iam と全く同じになるのでそれを参照。

Lambda関数のトリガーを設定する

S3へのレプリケーションをトリガーにCloud Frontのキャッシュ削除を実行する#lambda関数のトリガーを設定する と全く同じになるのでそれを参照。

Lambda関数を実装する

ここも基本的には、S3へのレプリケーションをトリガーにCloud Frontのキャッシュ削除を実行する#lambda関数を実装する とほぼ同じだが、Eventオブジェクトの中身からDeploy(Lambda UpdateFunctionCode, PublishVersion)を行うのに必要な情報を抜き出す部分が少し異なってくる。

Lambdaに渡ってくるEventオブジェクトの中身

S3へのレプリケーションをトリガーにCloud Frontのキャッシュ削除を実行する#lambdaに渡ってくるeventオブジェクトの中身 と全く同じになるのでそれを参照。

Lambda関数(index.handler)の実装

今回はNode.jsのLambda関数でDeployを実行させるので、実装としては以下のようした。

const { LambdaClient, UpdateFunctionCodeCommand, PublishVersionCommand } = require("@aws-sdk/client-lambda");
const client = new LambdaClient({ region: process.env.REGION });

exports.handler = async (event) => {
    try {
        const bucket = event.Records[0].s3.bucket.name;
        const object = event.Records[0].s3.object.key;

        if (!object.includes(process.env.FILE_NAME)) {
            console.log("not applicable. object is ", object);
            return "";
        }

        const updateInput = {
            FunctionName: process.env.FUNCTION_NAME,
            S3Bucket: bucket,
            S3Key: object
        };

        const updateCommand = new UpdateFunctionCodeCommand(updateInput);
        const updateResponse = await client.send(updateCommand);

        console.log("UpdateFunctionCodeCommand status", updateResponse.$metadata.httpStatusCode);
        console.log("UpdateFunctionCodeCommand state", updateResponse.State);

        if (process.env.PUBLISH) {
            const publishInput = {
                FunctionName: process.env.FUNCTION_NAME
            };

            const publishCommand = new PublishVersionCommand(publishInput);
            const publishResponse = await client.send(publishCommand);

            console.log("PublishVersionCommand status", publishResponse.$metadata.httpStatusCode);
            console.log("PublishVersionCommand state", publishResponse.State);

            return "";
        } else {
            return "";
        }
    } catch (error) {
        return errorHandler(error);
    }
}

const errorHandler = (error) => {
    const obj = {};
    obj["status"] = 500;
    obj["message"] = error.message;
    obj["stack"] = error.stack;
    obj["result"] = "ng";
    if (error.$metadata) {
        obj["status"] = error.$metadata.httpStatusCode;
    }

    console.log("errorHandler", obj);
    return "";
}

※実装について何点か補足すると、

if (process.env.PUBLISH)
 PubishVersionを行わないパターンも想定し、環境変数にPUBLISH(値は何でもいい)がある時のみPublishVersionを行うようにしている

if (object.includes("hoge.jar"))
 Lambdaのトリガー設定時に「プレフィックス・サフィックス」を設定できるので、ここで敢えて条件分岐は不要かもしれないが念のために実装してた

console.log()
 今回はCloud Watch Events(EventBridge)がLambda関数のトリガーであり、returnされた結果を人が見る事ができないので、何が起きているか分かるように敢えてconsole.log()でlogに出力させるようにした

return "";
 これも今回S3がトリガーでreturnされた結果を人が見れないので""(空文字)を返すようにした(何かJSONを返しても意味ないと判断)。

・参考:Lambda Client - AWS SDK for JavaScript v3
・参考:Class UpdateFunctionCodeCommand
・参考:Class PublishVersionCommand

Lambda関数の実行ロールの設定

Lambda関数が他のAWSサービスに対して何らかの操作を行うには、実行ロールでその権限を設定する必要がある。
今回はLambdaのUpdateFunctionCode・PublishVersionが実行できればいいので以下のようなインラインポリシーを実行ロールに追加でアタッチするればOK。

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "VisualEditor0",
            "Effect": "Allow",
            "Action": [
                 "lambda:UpdateFunctionCode",
                 "lambda:PublishVersion"
            ],
            "Resource": "arn:aws:lambda:ap-northeast-1:xxxxxxxxxxxx:function:{lambda関数名}"
        }
    ]
}

まとめとして

前々回はCloud Frontのキャッシュ削除について、前回はECSのサービス更新について、過去2回で扱ったが、Lambdaについても同じような考え方でレプリケーション後にそれをトリガーにしてDeployを実行させる事ができた。

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?