長期間使用していない AWS Lambda の inactive 状態を active 状態にする方法を備忘録として残しておく

事象

関数が数週間アイドルのままの場合、Inactive 状態になり、その状態で関数を呼び出すとエラーになる

詳細は以下を参照

対象の洗い出し

以下のスクリプトを AWS CLI にて使用することで対象の洗い出しが行える

#!/bin/sh

aws --profile <<プロフィール名>> lambda list-functions --region <<リージョン名>> --query 'Functions[?starts_with(FunctionName, `<<関数名プレフィックス>>`) == `true`].FunctionName' > lambda_list.json
lambda_names=`jq -r .[] lambda_list.json`

for lambda_name in $lambda_names; do
  aws --profile <<プロフィール名>> lambda get-function --output text --function-name $lambda_name --query 'Configuration.[FunctionName, State, LastUpdateStatus]' | sort -k 1 | column -t -s "`printf '\t'`"
done

解消方法

inactive の状態で lambda を実行するとエラーになるがしばらくすると active 状態となり実行可能となる
もしくは、以下のスクリプトを AWS Console 上で実行することで inactive => active に状態を更新できる
publish-version(※1)で active 状態に更新されない場合は invoke(※2)に切り替えて実施する

#!/bin/bash -eu

all_lambda_name_list=()
inactive_lambda_name_list=()

# 最初にアカウント上のすべての関数名を取得
function get_all_lambda_name_list() {
    echo "【Start get_all_lambda_name_list】$(date "+%Y/%m/%d %H:%M:%S")"
    readonly lambda_list=$(aws lambda list-functions \
        | jq -r '.Functions')

    len=$(echo $lambda_list | jq length)
    for i in $( seq 0 $(($len - 1)) ); do
        all_lambda_name_list+=($(echo $lambda_list | jq -r .[$i].FunctionName))
    done
    echo "【End get_all_lambda_name_list】$(date "+%Y/%m/%d %H:%M:%S")"
}

# 関数が state = Active かどうかの判定
function is_lambda_active() {
    local lambda_name=$1
    local state=$(aws lambda get-function-configuration \
        --function-name $lambda_name \
        | jq -r '.State')
    
    if [ $state = "Active" ]; then
        echo "true"
    else
        echo "false"
    fi
}

# Active でない関数のみを抽出
function filter_inactive_resources() {
    echo "【Start filter_inactive_resources】$(date "+%Y/%m/%d %H:%M:%S")"
    for lambda_name in ${all_lambda_name_list[@]}; do
        is_active=$(is_lambda_active $lambda_name)
        if [ "${is_active}" = "false" ]; then
            echo $lambda_name
            inactive_lambda_name_list+=($lambda_name)
        fi
    done
    echo "【End filter_inactive_resources】$(date "+%Y/%m/%d %H:%M:%S")"
}

# Active でない関数に対して publish-version を実行する
function publish_lambda() {
    echo "【Start publish_lambda】$(date "+%Y/%m/%d %H:%M:%S")"
    if [ ${#inactive_lambda_name_list[@]} -eq 0 ]; then
        echo "there is no lambda to publish"
        return 0
    fi
    for lambda_name in ${inactive_lambda_name_list[@]}; do
        echo "now publishing ${lambda_name}"
        aws lambda publish-version --function-name $lambda_name # ※1
        # aws lambda invoke --function-name $lambda_name lambda_invoke.log # ※2
        aws lambda wait function-active --function-name $lambda_name # 関数が Active になるまで待機
    done
    echo "【End publish_lambda】$(date "+%Y/%m/%d %H:%M:%S")"
}

echo "【Start Processing】$(date "+%Y/%m/%d %H:%M:%S")"
get_all_lambda_name_list
filter_inactive_resources
publish_lambda
echo "【End Processing】$(date "+%Y/%m/%d %H:%M:%S")"

AWS CLI で実行する場合は以下を参考にする
publish-version(※1)で active 状態に更新されない場合は invoke(※2)に切り替えて実施する

#!/bin/sh

# Shared
AWS_PROFILE_PREFIX={AWS プロファイルのプレフィックス} #自分の設定に書き換える
LOG_FOLDER=log
LOG_FILE=${LOG_FOLDER}/make_active_all_lambda.log

all_lambda_name_list=()
inactive_lambda_name_list=()

read -p "Which environment connect to? (dev/stg/prd): " input_environment

# Check the environment
case "${input_environment}" in
  "dev" | "stg" | "prd")
    echo "Connecting to ${environment} environment..." >> ${LOG_FILE} 2>&1

    if [ $input_environment = "prd" ]; then
        # AWS MFA Authentication
        aws-mfa --profile=${AWS_PROFILE_PREFIX}-${input_environment}
    fi;;
  *)
    echo "Please enter the correct environment." >> ${LOG_FILE} 2>&1
    echo "【End Processing】$(date "+%Y/%m/%d %H:%M:%S")" >> ${LOG_FILE} 2>&1
    exit;;
esac

# アカウント内全ての関数名を取得
function get_all_lambda_name_list() {

    readonly lambda_list=$(aws lambda --profile=${AWS_PROFILE_PREFIX}-${input_environment} list-functions \
        | jq -r '.Functions')

    len=$(echo $lambda_list | jq length)
    for i in $( seq 0 $(($len - 1)) ); do
        all_lambda_name_list+=($(echo $lambda_list | jq -r .[$i].FunctionName))
    done
}

# 関数の state == Active かどうかを返す
function is_lambda_active() {
    local lambda_name=$1
    local state=$(aws lambda --profile=${AWS_PROFILE_PREFIX}-${input_environment} get-function-configuration \
        --function-name $lambda_name \
        | jq -r '.State')

    if [ $state = "Active" ]; then
        echo "true"
    else
        echo "false"
    fi
}

# 関数の state == Active でないものを抽出
function filter_inactive_resources() {
    echo "== following lambda is inactive ==" >> ${LOG_FILE} 2>&1
    for lambda_name in ${all_lambda_name_list[@]}; do
        is_active=$(is_lambda_active $lambda_name)
        if [ "${is_active}" = "false" ]; then
            echo $lambda_name
            inactive_lambda_name_list+=($lambda_name)
        fi
    done
    echo "== that's all ==" >> ${LOG_FILE} 2>&1
}

# 関数を Active にする
function publish_lambda() {
    if [ ${#inactive_lambda_name_list[@]} -eq 0 ]; then
        echo "there is no lambda to publish" >> ${LOG_FILE} 2>&1
        return 0
    fi
    for lambda_name in ${inactive_lambda_name_list[@]}; do
        echo "now publishing ${lambda_name}" >> ${LOG_FILE} 2>&1
        aws lambda --profile=${AWS_PROFILE_PREFIX}-${input_environment} publish-version --function-name $lambda_name # ※1
        # aws lambda --profile=${AWS_PROFILE_PREFIX}-${input_environment} invoke --function-name $lambda_name ${LOG_FOLDER}/lambda_invoke.log >> ${LOG_FILE} 2>&1 # ※2
        aws lambda --profile=${AWS_PROFILE_PREFIX}-${input_environment} wait function-active --function-name $lambda_name # 関数が Active になるまで待つ
    done
}

echo "【Start Processing】$(date "+%Y/%m/%d %H:%M:%S")" >> ${LOG_FILE} 2>&1
get_all_lambda_name_list
filter_inactive_resources
publish_lambda
echo "【End Processing】$(date "+%Y/%m/%d %H:%M:%S")" >> ${LOG_FILE} 2>&1