0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

AWS API Gateway プロキシ統合・カスタム統合のチュートリアルをやってみた

Posted at

初めに

以下のチュートリアルをやってみました。これまで API Gateway の基本的な部分が分からなかったのですが、カスタム統合のチュートリアルに着手することで理解できました。マッピングテンプレートやクエリ文字列パラメータ、リソース・メソッド、API のデプロイについて理解が深まりました。

Lambda プロキシ統合とは

API Gateway の API 統合の種類の一つです。バックエンドが Lambda であり、 統合リクエスト・統合レスポンスの設定が不要な統合タイプです。

統合エンドポイントのタイプ 使用可能なAPI統合タイプ
Lambda関数 Lambdaプロキシ統合 Lambdaカスタム統合
HTTPエンドポイント HTTPプロキシ統合 HTTPカスタム統合
AWSサービスアクション 非プロキシタイプの AWS 統合のみ

0.1.png

Lambda 関数を作成する

ナビゲーションペインから「関数」を選択します。

1.png

「関数の作成」をクリックします。

2.png

関数の名前、ランタイム、ロールを設定します。ポリシーテンプレートは空白にしておきます。

3.png

関数のコードに以下をペーストします。

'use strict';
console.log('Loading hello world function');
 
exports.handler = async (event) => {
    let name = "you";
    let city = 'World';
    let time = 'day';
    let day = '';
    let responseCode = 200;
    console.log("request: " + JSON.stringify(event));
    
    if (event.queryStringParameters && event.queryStringParameters.name) {
        console.log("Received name: " + event.queryStringParameters.name);
        name = event.queryStringParameters.name;
    }
    
    if (event.queryStringParameters && event.queryStringParameters.city) {
        console.log("Received city: " + event.queryStringParameters.city);
        city = event.queryStringParameters.city;
    }
    
    if (event.headers && event.headers['day']) {
        console.log("Received day: " + event.headers.day);
        day = event.headers.day;
    }
    
    if (event.body) {
        let body = JSON.parse(event.body)
        if (body.time) 
            time = body.time;
    }
 
    let greeting = `Good ${time}, ${name} of ${city}.`;
    if (day) greeting += ` Happy ${day}!`;

    let responseBody = {
        message: greeting,
        input: event
    };
    
    // The output from a Lambda proxy integration must be 
    // in the following JSON object. The 'headers' property 
    // is for custom response headers in addition to standard 
    // ones. The 'body' property  must be a JSON string. For 
    // base64-encoded payload, you must also set the 'isBase64Encoded'
    // property to 'true'.
    let response = {
        statusCode: responseCode,
        headers: {
            "x-custom-header" : "my custom header value"
        },
        body: JSON.stringify(responseBody)
    };
    console.log("response: " + JSON.stringify(response))
    return response;
};

「Deploy」をクリックします。

4.png

「Changes deployed」が表示されたことを確認します。

5.png

REST API を作成する

「APIを作成」をクリックします。

6.png

REST API の「構築」をクリックします。

7.png

以下のようにプロトコルを設定します。

8.png

「アクション」のドロップダウンから「リソースの作成」をクリックします。

9.png

リソース名を入力し、「リソースの作成」をクリックします。

10.png

作成したリソースを選択した状態で「メソッドの作成」をクリックします。

11.png

「ANY」を選択し、横のチェックマークをクリックします。

12.png

以下のように設定します。

13.png

権限を追加するかどうか聞かれるので「OK」をクリックします。

14.png

「API のデプロイ」をクリックします。

15.png

以下の URL はリクエストに使用します。

16.1.png

リクエスト

PS C:\tutorial> curl.exe -v -X POST "https://API_Gateway_ID.execute-api.ap-northeast-1.amazonaws.com/test/helloworld?name=John&city=Seattle" -H "content-type: application/json" -H "day: Thursday" -d '{ "time": "evening" }'

Lambda が受け取ったイベント

API Gateway は上記のリクエストを Lambda に渡すために次のような JSON フォーマットに変換します。

{
    "resource": "/helloworld",
    "path": "/helloworld",
    "httpMethod": "GET",
    "headers": {
        "Accept": "*/*",
        "content-type": "application/json",
        "day": "Thursday",
        "Host": "API_Gateway_ID.execute-api.ap-northeast-1.amazonaws.com",
        "User-Agent": "curl/7.55.1",
        "X-Amzn-Trace-Id": "Root=1-60fbfb9a-70429bfb1ef089820112571c",
        "X-Forwarded-For": "11.22.333.44",
        "X-Forwarded-Port": "443",
        "X-Forwarded-Proto": "https"
    },
    "multiValueHeaders": {
        "Accept": [
            "*/*"
        ],
        "content-type": [
            "application/json"
        ],
        "day": [
            "Thursday"
        ],
        "Host": [
            "API_Gateway_ID.execute-api.ap-northeast-1.amazonaws.com"
        ],
        "User-Agent": [
            "curl/7.55.1"
        ],
        "X-Amzn-Trace-Id": [
            "Root=1-60fbfb9a-70429bfb1ef089820112571c"
        ],
        "X-Forwarded-For": [
            "11.22.333.44"
        ],
        "X-Forwarded-Port": [
            "443"
        ],
        "X-Forwarded-Proto": [
            "https"
        ]
    },
    "queryStringParameters": {
        "city": "Seattle",
        "name": "John"
    },
    "multiValueQueryStringParameters": {
        "city": [
            "Seattle"
        ],
        "name": [
            "John"
        ]
    },
    "pathParameters": null,
    "stageVariables": null,
    "requestContext": {
        "resourceId": "il7kor",
        "resourcePath": "/helloworld",
        "httpMethod": "GET",
        "extendedRequestId": "C-RANGv6tjMFtiA=",
        "requestTime": "24/Jul/2021:11:38:02 +0000",
        "path": "/test/helloworld",
        "accountId": "123456789012",
        "protocol": "HTTP/1.1",
        "stage": "test",
        "domainPrefix": "API_Gateway_ID",
        "requestTimeEpoch": 1627126682797,
        "requestId": "7a6335b9-554b-43a7-a228-3a9fb9f10a73",
        "identity": {
            "cognitoIdentityPoolId": null,
            "accountId": null,
            "cognitoIdentityId": null,
            "caller": null,
            "sourceIp": "11.22.333.44",
            "principalOrgId": null,
            "accessKey": null,
            "cognitoAuthenticationType": null,
            "cognitoAuthenticationProvider": null,
            "userArn": null,
            "userAgent": "curl/7.55.1",
            "user": null
        },
        "domainName": "API_Gateway_ID.execute-api.ap-northeast-1.amazonaws.com",
        "apiId": "API_Gateway_ID"
    },
    "body": null,
    "isBase64Encoded": false
}

レスポンス

{
    "message":"Good day,John of Seattle. Happy Thursday!",
        "input":{
            "resource":"/helloworld",
            "path":"/helloworld",

「CloudWatch のログを表示」をクリックすれば、Lambda 関数のログを確認できます。

17.png

Lambda 非プロキシ統合(カスタム統合)とは

メソッドリクエストをどのように統合リクエストにマッピングするか、統合レスポンスをどのようにメソッドレスポンスにマッピングするかを設定する Lambda 統合です。

0.2.png

Lambda 関数のコードを置き換える

以下のコードで最初に作成したコードを置き換えます。

'use strict';
var days = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];            
var times = ['morning', 'afternoon', 'evening', 'night', 'day'];

console.log('Loading function');

exports.handler = function(event, context, callback) {
  // Parse the input for the name, city, time and day property values
  let name = event.name === undefined ? 'you' : event.name;
  let city = event.city === undefined ? 'World' : event.city;
  let time = times.indexOf(event.time)<0 ? 'day' : event.time;
  let day = days.indexOf(event.day)<0 ? null : event.day;

  // Generate a greeting
  let greeting = 'Good ' + time + ', ' + name + ' of ' + city + '. ';
  if (day) greeting += 'Happy ' + day + '!';
  
  // Log the greeting to CloudWatch
  console.log('Hello: ', greeting);
  
  // Return a greeting to the caller
  callback(null, {
      "greeting": greeting
  }); 
};

REST API を作成する

/ の下に city というリソースを作成します。リソースパスは波括弧を付けて {city} とします。「API Gateway CORS を有効にする」にチェックを入れます。

18.png

メソッドは「ANY」を選択し、Lambda 関数を指定します。プロキシ統合にチェックを入れずにを「保存」をクリックします。

19.png

メソッドリクエストを編集します。

20.png

「クエリ文字の追加」をクリックします。

21.png

以下のように設定します。

22.png

ヘッダーを以下のように追加します。

23.png

「モデル」を選択します。

24.png

「作成」をクリックします。

25.png

モデル名を入力し、コンテントタイプに「application/json」と入力します。モデルのスキーマは以下を貼り付けます。

{
  "$schema": "http://json-schema.org/draft-04/schema#",
  "title": "GetStartedLambdaIntegrationInputModel",
  "type": "object",
  "properties": {
    "callerName": { "type": "string" }
  }
}

26.png

リクエスト本文に作成したモデルを指定します。

27.png

統合リクエストを編集します。

28.png

「マッピングテンプレートの追加」をクリックします。「application/json」と入力し、チェックマークをクリックします。

29.png

「はい、この統合を保護します」をクリックします。

29.5.png

以下のように設定されていることを確認します。

30.png

テンプレートの生成に作成したモデルを選択します。本文は以下で置き換えます。

#set($inputRoot = $input.path('$'))
{
  "city": "$input.params('city')",
  "time": "$input.params('time')",
  "day":  "$input.params('day')",
  "name": "$inputRoot.callerName"
}

31.png

テストの稲妻マークをクリックします。

32.png

メソッドは「POST」、パスは「Seattle」、クエリ文字列は「time=morning」、ヘッダーは「day:Wednesday」、リクエスト本文は「{ "callerName":"John" }」を入力します。

33.png

ステータスが 200 となっていれば成功です。

34.png

API のデプロイをします。ステージ変数は「custom」と入力します。

34.1.png

以下の URL にリクエストを送ります。

35.png

リクエスト

PS C:\tutorial> curl.exe -v -X POST "https://API_Gateway_ID.execute-api.ap-northeast-1.amazonaws.com/custom/Seattle?time=evening" -H "content-type: application/json" -H "day: Thursday" -H "x-amz-docs-region: ap-northeast-1" -d '{"callerName": "John"}'
  • /Seattle ・・・ リソース。これが city: 'Seattle' にマッピングされる。
  • ?time=evening ・・・ クエリ文字列パラメータ。これが time: 'evening' にマッピングされる。
  • day: Thursday ・・・ HTTP リクエストヘッダー。これが day: 'Thursday' にマッピングされる。
  • {"callerName": "John"} ・・・ リクエスト本文。これが name: 'John' にマッピングされる。

上記のパラメータは以下の設定に対応しています。

image.png

Lambda が受け取ったイベント

{ city: 'Seattle', time: 'evening', day: 'Thursday', name: 'John' }

上記のイベントは統合リクエストのマッピングテンプレートによって作成されます。

image.png

レスポンス

{"greeting":"Good evening, John of Seattle. Happy Thursday!"}

参考記事

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?