AWS Lambda: Clarification on retrieving data from event object

后端 未结 2 2072
太阳男子
太阳男子 2021-01-12 11:48

At the moment I think I have a foundational misunderstanding of how to extract/pass data from lambda functions & I\'m seeking clarification

Example: Let\'s say I

2条回答
  •  广开言路
    2021-01-12 12:33

    Building on insights from Matus, I now can better answer the rest of the questions from above:

    • "What is an example of how to pass data "directly" on the event object in a Lambda function like in case #3?"

    When setting up API Gateway with Lambda to send-through specific pieces of data you need to use a Mapping Template which gets setup in AWS API Gateway.

    That mapping template itself is written in the Velocity Template Language (VTL) from the Apache Foundation. Ex for attaching only the "hello" data so it is accessible as const data = event.hello; // world, ex:

        { 
          "hello": $input.params('$hello')
        }
    

    Note: VTL is very powerful, the above is not a realistic example of usage but smallest amount of VTL just to communicate the idea, see here and here for more detail

    Jumping through the interface to set that Mapping Template is a bit of an ordeal for a starter example so here's illustrated steps:

    Warning: Be sure to double-check what is actually displayed in the editor, since there might occasionally occur some funky/unexpected behavior on the Mapping Template when you change the dropdown. Sometimes you also reach unrecoverable issues involving the mapping template, it's best when starting out to go ahead and delete the method and restart from API Gateway.

    • "For a simple Node script, case 2 appears to have the "overhead" of parsing event body from stringified JSON so that's an understandable downside, but why or when would Case 3 be a more desirable approach?

    It all depends on how your Lambda is receiving its data.

    The key insight from Matus is that this is all ultimately an implementation detail. Lambdas in this contrived example (ie set up a REST API via AWS Gateway, send a GET/POST to that API and have lambda do something with the data) can retrieve data 3 ways:

    (1) URL parameters const data = event.queryStringParameters.hello; // world

    (2) request body const data = event.body.hello; // world (see note below)

    (3) directly on the event object const data = event.hello; // world

    Note on 2: This requires selecting (a) Lambda Proxy Integration instead of using a mapping template and in your code you'll need JSON.parse the event body before accessing the data, see this answer for further details

    All this depends on what it is being fed from API Gateway. In this specific example, I'm talking about doing a REST request to pass data on an API Gateway endpoint that then is processed by Lambda-- but lots of other services/triggers can send data to a Lambda script for analysis.

    Other helpful resources:

    • https://kennbrodhagen.net/2015/12/06/how-to-create-a-request-object-for-your-lambda-event-from-api-gateway/

    • Getting json body in aws Lambda via API gateway

    • Could not parse request body into json: Unexpected character (\'-\' (code 45)) AWS Lambda + API + Postman

    • https://medium.com/@lakshmanLD/lambda-proxy-vs-lambda-integration-in-aws-api-gateway-3a9397af0e6d

    • https://github.com/valgaze/documentdb-fun

    Ex. a toy Lambda function echos back whatever is POST'd:

    let client = null; // Data outside function handler (ex a DB connection or an incrementing value) can change within the handler code below and persist between Lamba invocations so long as the container is still "warm", see more for details: https://docs.aws.amazon.com/lambda/latest/dg/running-lambda-code.html
    
    exports.handler = async (event, context) => {
        let data = {};
    
        // Lambda Proxy Integration
        if (event && event.body) {
            try {
                data = JSON.parse(event.body);
            } catch(e) { 
                return {
                    statusCode: 400,
                    body: JSON.stringify({message: 'There was an error parsing the JSON data posted to this endpoint', error:e})
                }
            }
        }
    
        try {
    
            // Echo back the data or perform transformations, pass on to another service, etc
            const response = {
                statusCode: 200,
                body: JSON.stringify({message: 'Data submitted', data})
            };
    
            return response;
    
        } catch(e) {
            // Report errors related with connection, auth, DB write, etc
            return {
              statusCode: 409,
              body: JSON.stringify({message: 'There was some type of catastrophic error', error:e})
          }
       }
    };
    

提交回复
热议问题