はじめに
AWS初心者です。AWSの基本知識をキャッチアップすべく、色々触りながら学習中です。
今回は、アプリの認証やユーザー管理の機能を提供するサービス「Amazon Cognito」を実際に触ってみた内容を記事にしました。自分の理解度確認が主目的になっておりますので、温かい目で見て頂ければ幸いです。
目的
今回触るサービスはいわゆるID管理サービスになります。これまで、MicrosoftのEntra ID/AzureAD B2CやOktaのCustomer Identity Cloud(CIC)を触ったことはあるので、概念や仕組みは何となく理解しています。アプリ開発ではMUSTな技術領域のため、AWSでのアプリ開発を見据えて、実際にAmazon Cognitoを触りながら機能や使い勝手を確認してみることにしました。
概要
本ページでは、サンプルアプリ(ローカル)からCognitoに接続してサインインする仕組みを構築した手順をまとめています。
基礎知識
ID管理サービス(IDaaS)とは
ざっくりいうと認証系機能を提供するサービスです。この記事を書くうえで復習しようと思って色々サイトを見ていましたが、個人的には「IDaaSとは何か?」が端的にまとまっていて分かりやすかったです。
チュートリアル
Amazon Cognitoへのアクセス
AWSコンソールにサインインし、Amazon Cognitoのページへ移動します。
無料で開始できるボタンを押下します。
※環境は12ヵ月無料枠を利用しています。以下の通り、基本的に無料枠の範囲内で利用可能です。
アプリケーションの設定
アプリケーションタイプ、アプリケーション名、サインイン識別子のオプション、サインアップのための必須属性を選択・入力します。
今回は、メールアドレス/パスワードでログインするSPAとして、アプリケーションを定義します。
作成すると、「ユーザープール」とクライアントアプリケーションが作成されました。
メモ
クライアントアプリを作成すると、ユーザープールが自動的に作成され、紐づけされました。ユーザープールはユーザーにサインイン/サインアップ機能を提供するユーザーディレクトリです。Entra IDではディレクトリが一つ、AzureAD B2Cではインスタンス単位でディレクトリが一つ紐づきますが、Amazon Cognitoではユーザープール(ディレクトリ)を複数作成することが可能なので、Webアプリ単位でディレクトリを分けたいユースケースに対応しています。
ユーザーの作成
ツリーメニューの「ユーザー管理」からログイン用のテストユーザーを作成します。
作成されると、入力したメールアドレスに初期パスワードを記載したメールが飛んできます。
これでCognito側の設定は完了です。
サンプルアプリの作成
QuickStepガイドの確認
前述で作成したクライアントアプリケーションのページでアプリに追加する手順とサンプルコードが用意されているのでこれを活用します(サンプルコードにはCognitoの認証エンドポイントやクライアントアプリIDなどが定義されているため、そのままコピって使うことができます)。
ベースとなるローカルアプリの準備
今回はReactで作るので、以下のコマンドでシングルアプリのプロジェクトを用意します。
npx create-react-app my-react-app
OIDCライブラリのインストール
CognitoのQuickStepガイドにも記載されていますが、以下のコマンドを実行してoidc-client-ts
とreact-oidc-context
をインストールします。
npm install oidc-client-ts react-oidc-context --save
コードの変更
CognitoのQuickStepガイドにあるサンプルコードを参考に以下のようにコードを書き換えます。<xxx>
の部分は自分が作成したユーザープールのエンドポイントやクライアントアプリケーションIDの値に置き換えてください。サンプルコードをコピーする場合は、値を変更する必要はありません。
import React from 'react';
import ReactDOM from 'react-dom/client';
import './index.css';
import App from './App';
import reportWebVitals from './reportWebVitals';
// 認証するために追加したコード
import { AuthProvider } from "react-oidc-context";
const cognitoAuthConfig = {
authority: "<cognitoユーザープールのURI>",
client_id: "<クライアントアプリケーションID>",
redirect_uri: "<リダイレクトURI>",
response_type: "code",
scope: "email openid phone",
};
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
<React.StrictMode>
<AuthProvider {...cognitoAuthConfig}>
<App />
</AuthProvider>
</React.StrictMode>
);
import logo from './logo.svg';
import './App.css';
import { useAuth } from "react-oidc-context";
function App() {
const auth = useAuth();
const signOutRedirect = () => {
const clientId = "<クライアントアプリケーションID>";
const logoutUri = "<logout uri>";
const cognitoDomain = "<cognitoドメイン>";
window.location.href = `${cognitoDomain}/logout?client_id=${clientId}&logout_uri=${encodeURIComponent(logoutUri)}`;
};
if (auth.isLoading) {
return <div>Loading...</div>;
}
if (auth.error) {
return <div>Encountering error... {auth.error.message}</div>;
}
if (auth.isAuthenticated) {
return (
<div>
<pre> Hello: {auth.user?.profile.email} </pre>
<pre> ID Token: {auth.user?.id_token} </pre>
<pre> Access Token: {auth.user?.access_token} </pre>
<pre> Refresh Token: {auth.user?.refresh_token} </pre>
<button onClick={() => auth.removeUser()}>Sign out</button>
</div>
);
}
return (
<div>
<button onClick={() => auth.signinRedirect()}>Sign in</button>
<button onClick={() => signOutRedirect()}>Sign out</button>
</div>
);
}
export default App;
アプリの実行
ローカルでアプリを起動します。
cd my-react-app
npm start
「Sign in」ボタンを押下すると、未認証のためCognitoが提供するデフォルトのログイン画面へ遷移します。
前述で作成したテストユーザーのメールアドレスを入力後、初期パスワードを入力します。
ログインに成功し、Cognitoで設定したリダイレクトURI(デフォルト)へアクセスすることができました。
まとめ
今回は特に下調べなどはせず、とりあえずCognitoを触って理解することを目的にシンプルなログイン機能を作ってみました。次回は、ログイン設定やMFA、ログインページのカスタマイズなどステップアップしてみようと思います。