AWS API Gatewayチュートリアルの手順4

手順4 - Amazon API Gatewayで複数のロールを使用する

この手順では、認証情報に基づいて、ユーザーに異なるAWS IAMロールを割り当てます。

  • ソーシャル接続で認証するユーザーは、購入者として扱われます。

  • データベース接続で認証するユーザーは、管理者として扱われます。

このロール割り当てロジックは、2つの方法で実行します。

  • JavaScript

  • Auth0ルール

多くのAuth0アプリケーションでは、異なるユーザーに対して異なるレベルのアクセスを付与する必要があり、サービスロジックで使用するために、指定されたIDに関する追加情報が必要になります。APIレベルでアクセスをロックするだけで十分な場合は、異なるAWS IAMロールを使用できます(たとえば、管理者は更新機能を使ってペットを追加・削除できるのに対し、ソーシャルユーザーはペットを購入することしかできない)。

以下の図では、ソーシャル接続経由で認証されたユーザーと、データベース接続で認証されたユーザーの2種類のユーザークラスについて、AWS IAMロールの割り当てを示しています。また、AWS IAMロールは、その他のエンティティ(たとえば、AWS Lamdba関数)に割り当てることができ、それらのエンティティが割り当てられているアカウントへのアクセス許可を制御できることも示しています。つまり、IAMロールは、1つ以上のポリシーで定義され、エンティティに割り当てられる、AWS機能へのアクセス許可の集合と言えます。

コード内で決定したい場合(たとえば、ペットを購入しようとしているユーザーの信用調査を行いたい場合)は、IDのフローも行います。この手続きは、後述の「手順5 - IDフローにIDトークンを使用する」でご説明します。

1.PetPurchase APIリソースを作成する

Amazon API Gatewayコンソールを使用して、Pets APIを選択します。[Resources(リソース)]ページが表示されます。

[Actions(アクション)][Create Resource(リソースを作成)]をクリックします。新しい子リソースに「Purchase(購入)」という名前を付けます。[Create Resource(リソースを作成)]をクリックします。

以前、pets用に「手順2:CORSをセットアップしてAPIを導入する」の「Amazon API Gatewayの保護と導入」で説明した方法で、OPTIONSメソッドをpurchaseリソースに追加します。

ペットを購入するために、PetPurchaseという新しいAWS Lambda関数を作成します。これにより、以下のように、isSoldsoldToの属性がペットに追加されます。

var AWS = require('aws-sdk');
var DOC = require('dynamodb-doc');
var dynamo = new DOC.DynamoDB();

exports.handler = function(event, context) {
   var petId = event.petId;
   var user = event.userName;
   var pets = {};
   console.log('start PetsPurchase, petId', petId, ' userName', user);

   var writecb = function(err, data) {
      if(!err) {
          context.done(null, pets);
      } else {
          console.log('error on GetPetsInfo: ',err);
          context.done('failed on update', null);
      }
   };

   var readcb = function(err, data) {
      if(err) {
          console.log('error on GetPetsInfo: ',err);
          context.done('failed to retrieve pet information', null);
      } else {
          // make sure we have pets
          if(data.Item && data.Item.pets) {
              pets = data.Item.pets;
              var found = false;

              for(var i = 0; i < pets.length && !found; i++) {
                  if(pets[i].id === petId) {
                     if(!pets[i].isSold) {
                        pets[i].isSold = true;
                        pets[i].soldTo = user;
                        var item = { username:"default",pets: pets};
                        dynamo.putItem({TableName:"Pets", Item:item}, writecb);
                        found = true;
                     }
                  }
               }
               if(!found) {
                 console.log('pet not found');
                 context.done('That pet is not available.', null);
               }
           } else {
              console.log('pet already sold');
              context.done('That pet is not available.', null);
           }
       }
   };

   dynamo.getItem({TableName:"Pets", Key:{username:"default"}}, readcb);
};

Was this helpful?

/

Lambda関数を定義し終えたら、POSTメソッドをpurchaseリソースに追加します。これは、PetPurchase Lambdaと呼ばれます。値が*Access-Control-Allow-Originヘッダーも忘れずにPOSTメソッドに追加します。これには、「手順2:CORSをセットアップしてAPIを導入する」の「Amazon API Gatewayの保護と導入」で説明されている、メソッド応答/統合応答の構成を使用します。

入力メッセージとして以下を指定して、API Gatewayメソッドをテストします。

{
    "petId": 1,
    "userName": "fred flintstone"
 }

Was this helpful?

/

テスト応答で、ID「1」のペットが、「Fred Flintstone」に販売されたことがわかります。

[
  {
    "id": 1,
    "price": 249.99,
    "type": "dog",
    "isSold": true,
    "soldTo": "fred flintstone"
  },

  ...

Was this helpful?

/

2.IAMを使ってPurchasePet APIを保護する

IAMを更新する

APIを保護するため、本チュートリアルのパート2で実行した、新しいロールの追加プロセスを繰り返してください。新しいロールはauth0-api-social-roleとします。

IAMポリシーで保護するこのメソッドのARNは、次のようになります。

arn:aws:execute-api:us-east-1:your-accountid:your-api-id/*/pets/purchase

Was this helpful?

/

信頼ポリシーも必ず更新してください。

Amazon API Gatewayコンソールに移動して、/pets/purchaseリソースのPOSTメソッドを選択します。[Method Request(メソッド要求)]を選択して、[Authorization Type(認可タイプ)]を「AWS_IAM」に変更します。チェックマークをクリックして設定を保存します。

この時点で2つのロールの定義が終わり、API Gatewayに使用できるようになりました。

  • auth0-api-role:ペットの更新を許可

  • auth0-api-social-role:ペットの購入を許可

Login with Amazonを構成して、Auth0を更新する

Login with Amazon(LWA)を使用してソーシャルロールを作成することができます。

このチュートリアルではLogin with Amazonの使い方をご説明しますが、他のソーシャルプロバイダーも利用できます。

  1. [Auth0 Dashboard]>[Authentication(認証)]>[Social(ソーシャル)]に移動して、[Create Connection(接続の作成)]を選択します。

  2. セットアップしたい接続を選択して、同意します。

  3. ソーシャルIDプロバイダーでクライアントIDクライアントシークレットをコピーして、貼り付けます。[Attributes(属性)](と、該当する場合は[Permissions(アクセス許可)])を選択して、[Save(保存)]をクリックします。

  4. [Applications(アプリケーション)]ビューを選択して、この接続を使用する各Auth0アプリケーションの切替を有効にして、[Save(保存)]を選択します。

適切な情報を入力したら、[Try Connection(接続を試す)]を選択して間違いがないかを確認します。

Amazonコンソールを使用してLWAを構成する際には、[Allowed Return URLs(許可されているリターンURL)]に、https://johndoe.auth0.com/login/callbackのような、Auth0アプリケーションのコールバックURLを必ず入力してください。入力する具体的なURLは、Auth0のヘルプページに記載されています。

[Auth0 Dashboard]>[Applications(アプリケーション)]>[Applications(アプリケーション)]に移動し、設定を表示するアプリケーションを選択します。[Connections(接続)]ビューを選択して、[Social(ソーシャル)]セクションに進み、[Amazon]が有効になっていることを確認します。

APIを導入して、シングルページアプリケーションを更新する

APIを導入する

Amazon API Gatewayコンソールを使用して、再びAPIを導入して新しいJavaScript SDKを生成します。

この時点で、ペット購入を可能にするために必要な構成変更は完了しています。これを運用環境に反映させるには、新たにダウンロードしたSDKをコピーして、petsフォルダーにある以前のものおよびAmazon S3バケットと置き換えます。

ログインコントローラーロジックを更新して、異なるタイプのユーザーに異なるロールを選択する

ログインコントローラーロジックは、getOptionsForRoleを使用して、異なるユーザーに異なるロールを選択します。委任トークンを取得したら、使用するロール(すなわち、ユーザーが管理者か否か)をAuth0に伝えることができます。

pets/login/login.jsファイルで、先ほど作成したソーシャルユーザーIAMロールの非管理者ユーザーのroleprincipalの値を変更します。

この時点で、Amazon資格情報または先ほど作成したデータベースユーザーを使ってログインできるはずです。UIを確認すると、ソーシャルユーザーにはペットの購入が、管理者ユーザーにはペットの追加と削除ができるようになっていることが分かります。

この機能性をテストするには、/pets/home/home.htmlng-show="isAdmin"を削除することで、UIの削除ボタンを一時的に非表示にできます。

<button ng-show="isAdmin" class="btn delete-btn" ng-click="removePet(pet.id)">remove</button>

Was this helpful?

/

この変更をS3バケットにコピーしたら、ソーシャルユーザーとしてログインしている状態でペットの削除を試みます。

ホームコントローラーロジックを更新して、ソーシャルユーザーにペット購入を許可する

home.jsbuyPet機能を変更して、ペット購入ができるようにします。

function buyPet(user, id) {
    var apigClient = getSecureApiClient();

    apigClient.petsPurchasePost({},{userName:user, petId:id})
      .then(function(response) {
        console.log(response);
        $scope.pets = response.data;
        $scope.$apply();
      }).catch(function (response) {
        alert('buy pets failed');
        showError(response);
    });
}
…

Was this helpful?

/

コードをS3バケットにコピーして、ログアウトし、LockのログインダイアログにあるAmazonアイコンをクリックする方法で、再びソーシャルユーザーとしてログインし直します。Lockペインに以前のログインが残っている場合は、[SHOW ALL(すべて表示)]をクリックする必要があります。

Amazonユーザーとしては、ペットを購入することはできても、ペットを追加・削除することはできません。一方、データベース接続に紐づけられたユーザーとしてログインした場合は、ペットを追加・削除することはできても、ペットを購入することはできません。

Auth0ルールでロール割り当てを適用する

場合によっては、アプリケーションを使用して適切なロールを決定しているけれど(こちらを参照)、セキュリティ上の理由で(ユーザーが必要以上に権限のあるロールにつくのを防ぎたい場合)、ユーザー権限をサーバー側で決定しなければならないことがあります。

Auth0では、この設定をルールを使って行えます。ルールは、自分で定義するサービスロジックのステートメントで、Auth0認証のプロセス中に実行されます。たとえば、以下を行うルールを作成できます。

  • ブラウザーからアプリケーションにロール情報が渡されないようにする

  • 認証ソースに基づいて、委任要求にロール情報を挿入する

ロール割り当てを適用する

ユーザーによって要求されたロールが許可されたものかを確認するルールを追加します。これは、紐づけられているのがソーシャル接続なのか、データベース接続なのかに応じて異なります。

  1. [Auth0 Dashboard]>[Auth Pipeline(Authパイプライン)]>[Rules(ルール)]に移動して、[Create Rule(ルールの作成)]を選択します。

  2. [Empty rule(空のルール)]テンプレートを選択します。

  3. ルールに「AWS Pets」のような名前を付けてから、以下のJavaScriptコードを使ってルールの本文を入力します。

    function (user, context, callback) {
      if(context.clientID === '{yourClientId}') {
        var socialRoleInfo = {
          role:"arn:aws:iam::<your account>:role/auth0-api-social-role",
          principal: "arn:aws:iam::your account>:saml-provider/auth0"
        };
    
        var adminRoleInfo = {
          role:"arn:aws:iam::<your account>:role/auth0-api-role",
          principal: "arn:aws:iam::<your account>:saml-provider/auth0"
        };
    
        var requestRole = context.request.body.role;
        var requestPrincipal = context.request.body.principal;
        var allowedRole = null;
    
        if(user.identities[0].isSocial === false) {
          allowedRole = adminRoleInfo;
        } else {
          allowedRole = socialRoleInfo;
        }
    
        if((requestRole && requestRole !== allowedRole.role) ||
           (requestPrincipal && requestPrincipal !== allowedRole.principal)) {
            console.log('mismatch in requested role:',requestRole, ':', requestPrincipal);
            console.log('overridding');
        } else {
          console.log('valid or no role requested for delegation');
        }
    
        context.addonConfiguration = context.addonConfiguration || {};
        context.addonConfiguration.aws = context.addonConfiguration.aws || {};
        context.addonConfiguration.aws.role = allowedRole.role;
        context.addonConfiguration.aws.principal = allowedRole.principal;
        callback(null, user, context);
    
      } else {
        callback(null, user, context);
      }
    }

    Was this helpful?

    /
    上記のコードは、必ず、実際の統合に合わせて正しい値に調整してください。フィールドは、[Princial ARN(プリンシパルARN)][Role ARN(ロールARN)]、および[Client Secret(クライアントシークレット)]となります。

  4. 変更を保存します

注意事項

  • ルールは、すべての認証のグローバルスコープで実行されます。特定のアプリケーションに紐づけられた認証要求に対してのみ、ロジックを実行する必要があります(これが、使用されているスクリプトがclientIDを求める理由です)。この情報がなければ、Auth0アカウントに紐づけられたすべての認証要求に対してロジックが実行されてしまいます。

  • 情報は、コンテキストとユーザーと併せてルールに渡されます。

  • ルールに渡されたオブジェクトは拡張できます。上記のコードでは、ルールが要求の本文でロール情報を確認します。ロールは、許可されたロールのaddonConfigurationコンテキストに設定され、常に要求本文の設定に優先されます。

ルールをデバッグする

ルールをデバッグする準備ができました。[Try this Rule(このルールを試す)]を選択すると、ルールのロジックを試行するスクリプトが表示されます。[Try(試す)]を選択します。

ルールを実行した結果が表示されます。