web-dev-qa-db-fra.com

"KeyError: 'Records'" dans AWS S3 - Déclencheur Lambda

J'ai le code de fonction lambda suivant pour imprimer simplement l'auteur et les métadonnées d'un événement téléchargé d'un compartiment 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'])

Cependant, j'obtiens l'erreur suivante lors des tests:

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

Suis-je en train de faire quelque chose de mal lors de l'accès au nom du compartiment et au nom de clé de l'objet S3? Sinon, qu'est-ce que je fais de mal ici?

8
Dawny33

un peu tard pour la fête. Mais voici mon premier post!

EXPLICATION:

Lorsque vous testez dans le panneau lambda -> def lambda_handler (événement, contexte) <- l'événement est injecté directement.

Cependant, dans l'API AWS, son nécessaire pour ajouter un modèle de mappage ou autre -> événement <- est vide, ce qui entraîne un quiz:

"errorType": "KeyError", "errorMessage": "'Records'"

c'est un pointeur nul. Les enregistrements n'existent pas, car -> l'événement <- n'existe pas.

SOLUTION:

Vous devez configurer la demande d'intégration dans l'API AWS. Cliquez sur Modèles de cartographie corporelle . Ensuite ajoutez le modèle de mappage Définissez le type de contenu sur application/json Ensuite, modifiez la génération générée modèle de mappage:

{
  "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
  }  
}

Et éditez la fonction Lambda :

remplacer:

pour enregistrement dans l'événement ['Records']:

avec:

pour l'enregistrement dans l'événement ['query'] ['Records']

je ne sais pas si la pile vous cinglera avec cette réponse - alors je vous appelle @ Dawny33 @KevinOelen @franklinsijo

Quant aux explications, je les ai trouvées moi-même. Cependant, le "modèle de mappage" provient de https://medium.com/simple-oughtts-amplified/passing-variables-from-aws-api-gateway-to-lambda-3c5d8602081b

6
esr