13

私は AWS にいて、Node.js で AWS SDK for JavaScript を使用しています。AWS Lambda 関数を構築しようとしていて、内部ですべての Amazon EC2 インスタンスのリストを取得したいのですが、機能していないようです。誰かが私が間違っていることを見つけることができますか?

これが私のLambda関数コードです:

var AWS = require('aws-sdk');
AWS.config.region = 'us-west-1';

exports.handler = function(event, context) {
    console.log("\n\nLoading handler\n\n");
    var ec2 = new AWS.EC2();
    ec2.describeInstances( function(err, data) {
        console.log("\nIn describe instances:\n");
      if (err) console.log(err, err.stack); // an error occurred
      else     console.log("\n\n" + data + "\n\n"); // successful response
    });
    context.done(null, 'Function Finished!');  
};

これが私のポリシーです(正しいと思いますか?)

{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Action": [
        "logs:*"
      ],
      "Resource": "arn:aws:logs:*:*:*"
    },
    {
    "Effect": "Allow",
    "Action": [
      "ec2:*"
    ],
    "Resource": "arn:aws:ec2:*"
  },
    {
      "Effect": "Allow",
      "Action": [
        "s3:GetObject",
        "s3:PutObject"
      ],
      "Resource": [
        "arn:aws:s3:::*"
      ]
    }
  ]
}

そして、「ec2」で console.log を実行すると、次のようになります。

{ config: 
   { credentials: 
      { expired: false,
        expireTime: null,
        accessKeyId: 'XXXXXXXXXXXXXXXXXX',
        sessionToken: 'XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
        envPrefix: 'AWS' },
     credentialProvider: { providers: [Object] },
     region: 'us-west-1',
     logger: null,
     apiVersions: {},
     apiVersion: null,
     endpoint: 'ec2.us-west-1.amazonaws.com',
     httpOptions: { timeout: 120000 },
     maxRetries: undefined,
     maxRedirects: 10,
     paramValidation: true,
     sslEnabled: true,
     s3ForcePathStyle: false,
     s3BucketEndpoint: false,
     computeChecksums: true,
     convertResponseTypes: true,
     dynamoDbCrc32: true,
     systemClockOffset: 0,
     signatureVersion: 'v4' },
  isGlobalEndpoint: false,
  endpoint: 
   { protocol: 'https:',
     host: 'ec2.us-west-1.amazonaws.com',
     port: 443,
     hostname: 'ec2.us-west-1.amazonaws.com',
     pathname: '/',
     path: '/',
     href: 'https://ec2.us-west-1.amazonaws.com/' } }
4

2 に答える 2

13

最も可能性の高い原因は、EC2 DescribeInstances API への呼び出しが完了する前に、Lambda 関数を明示的に終了していることです。

その理由は、Lambda は、 を呼び出すとすぐにコードの実行が終了したと想定するためですcontext.done(...)そして、これは電話の前に起こっていconsole.log(... data ...)ます。

この奇妙な順序付けは、NodeJS の仕組みと AWS SDK for JavaScript の仕組みが原因で発生します。NodeJS では、実行を決してブロックしないでください。Web サービス (EC2 など) を呼び出すと、実行がブロックされます。したがって、AWS SDK for JavaScript (およびほとんどの NodeJS ライブラリ) は、非同期呼び出しを行うことによって機能します。

ほとんどの場合、非同期呼び出しがある場合、その呼び出しにコールバック関数を渡します。結果の準備が整うと、NodeJS はコールバック関数を実行します。

あなたのコードでは、それfunction(err, data) {...}コールバック関数です。これはすぐには実行されませんが、呼び出しが結果を受け取ったことを NodeJS が確認したときに実行がスケジュールされec2.describeInstancesます。

コールバックの実行をスケジュールするとすぐにが呼び出されcontext.done(...)、Lambdaに「I'm done, you can kill me 」と伝えられます。そして、EC2 の DescribeInstances 呼び出しがそのデータを受け取り、それをコールバック関数に渡す前に、喜んで関数に従い、中断します。

問題を解決するには?

答えは今では明らかです。呼び出しを含む if/else ブロックの直後のコールバックcontext.done(...)関数内に呼び出しを移動するだけです。console.log(...data...)

ec2.describeInstances( function(err, data) {
  console.log("\nIn describe instances:\n");
  if (err) console.log(err, err.stack); // an error occurred
  else     console.log("\n\n" + data + "\n\n"); // successful response
  context.done(null, 'Function Finished!');  
});
于 2014-12-29T17:49:10.757 に答える