web-dev-qa-db-ja.com

AWS S3の「KeyError: 'Records'」-Lambdaトリガー

S3バケットのアップロードされたイベントの作成者とメタデータを単に出力するための次のラムダ関数コードがあります。

from __future__ import print_function
import json
import urllib
import boto3

print('Loading function')

s3 = boto3.client('s3')


def lambda_handler(event, context):

    #print("Received event: " + json.dumps(event, indent=2))
    # bucket = event['Records'][0]['s3']['bucket']['name']

    for record in event['Records']:
        bucket = record[0]['s3']['bucket']['name']
        key = record[0]['s3']['object']['key']
        response = s3.head_object(Bucket=bucket, Key=key)

        logger.info('Response: {}'.format(response))

        print("Author : " + response['Metadata']['author'])
        print("Description : " + response['Metadata']['description'])

ただし、テスト中に次のエラーが発生します。

{
  "stackTrace": [
    [
      "/var/task/lambda_function.py",
      17,
      "lambda_handler",
      "for record in event['Records']:"
    ]
  ],
  "errorType": "KeyError",
  "errorMessage": "'Records'"
}

S3オブジェクトのバケット名とキー名にアクセスしているときに何か問題がありますか?そうでない場合、私はここで何が間違っていますか?

8
Dawny33

パーティーに少し遅れました。しかし、これが私の最初の投稿です!

説明:

ラムダパネルでテストすると-> def lambda_handler(event、context)<-イベントが直接注入されます。

ただし、AWS APIでは、マッピングテンプレートを追加する必要がありますまたはそれ以外の場合->イベント<-が空であるため、クイズが発生します。

"errorType": "KeyError"、 "errorMessage": "'Records'"

これはnullポインタです。 ->イベント<-が存在しないため、レコードが存在しません。

解決策:

AWS API内でIntegration Requestを構成する必要があります。 Body Mapping Templatesをクリックします。次にマッピングテンプレートを追加しますコンテンツタイプをapplication/jsonに設定し、編集して生成しますマッピングテンプレート:

{
  "body" : $input.json('$'),
  "headers": {
    #foreach($header in $input.params().header.keySet())
    "$header": "$util.escapeJavaScript($input.params().header.get($header))" #if($foreach.hasNext),#end

    #end
  },
  "method": "$context.httpMethod",
  "params": {
    #foreach($param in $input.params().path.keySet())
    "$param": "$util.escapeJavaScript($input.params().path.get($param))" #if($foreach.hasNext),#end

    #end
  },
  "query": {
    #foreach($queryParam in $input.params().querystring.keySet())
    "$queryParam": "$util.escapeJavaScript($input.params().querystring.get($queryParam))" #if($foreach.hasNext),#end

    #end
  }  
}

そしてLambda関数を編集します

交換:

イベントのレコードの場合['Records']:

と:

イベントのレコード用['query'] ['Records']

スタックがこの回答であなたにpingするかどうかわからない-だから私はあなたに@ Dawny33 @KevinOelen @franklinsijoと呼ぶ

説明は自分で考えました。ただし、「マッピングテンプレート」は https://medium.com/simple-thoughts-amplified/passing-variables-from-aws-api-gateway-to-lambda-3c5d8602081b から取得されます

6
esr