パスキーなどの FIDO 認証情報はパスワードの代替を目的としていますが、ほとんどの場合、ユーザーはユーザー名の入力も不要になります。これにより、ユーザーは現在のウェブサイトのパスキーのリストからアカウントを選択して認証できます。
以前のバージョンのセキュリティ キーは 2 段階認証方法として設計されており、認証情報の ID が必要だったため、ユーザー名の入力が必要でした。ID を知らなくてもセキュリティ キーが検出できる認証情報は、検出可能な認証情報と呼ばれます。現在作成されている FIDO 認証情報のほとんどは、検出可能な認証情報です。特に、パスワード マネージャーまたは最新のセキュリティ キーに保存されているパスキーは検出可能な認証情報です。
認証情報がパスキー(検出可能な認証情報)として作成されるようにするには、認証情報の作成時に residentKey と requireResidentKey を指定します。
証明書利用者(RP)は、パスキー認証時に
allowCredentialsを省略することで、検出可能な認証情報を使用できます。この場合、ブラウザまたはシステムには、作成時に設定された user.name プロパティで識別される、使用可能なパスキーのリストが表示されます。ユーザーが 1 つ選択すると、結果の署名に user.id 値が含まれます。サーバーは、入力されたユーザー名の代わりに、その値または返された認証情報 ID を使用してアカウントを検索できます。
前述のアカウント セレクタ UI には、検出不可能な認証情報は表示されません。
requireResidentKey と residentKey
パスキーを作成するには、navigator.credentials.create() で authenticatorSelection.residentKey と authenticatorSelection.requireResidentKey を次のように指定します。
async function register () {
// ...
const publicKeyCredentialCreationOptions = {
// ...
authenticatorSelection: {
authenticatorAttachment: 'platform',
residentKey: 'required',
requireResidentKey: true,
}
};
const credential = await navigator.credentials.create({
publicKey: publicKeyCredentialCreationOptions
});
// This does not run until the user selects a passkey.
const credential = {};
credential.id = cred.id;
credential.rawId = cred.id; // Pass a Base64URL encoded ID string.
credential.type = cred.type;
// ...
}
residentKey:
'required': 検出可能な認証情報を作成する必要があります。作成できない場合は、NotSupportedErrorが返されます。'preferred': RP は検出可能な認証情報の作成を優先しますが、検出不可能な認証情報も受け入れます。'discouraged': RP は検出不可能な認証情報の作成を優先しますが、検出可能な認証情報も受け入れます。
requireResidentKey:
- このプロパティは、古いバージョンの仕様である WebAuthn Level 1 との下位互換性を維持するために保持されています。
residentKeyが'required'の場合はtrueに設定し、それ以外の場合はfalseに設定します。
allowCredentials
RP は navigator.credentials.get() で allowCredentials を使用して、パスキー認証のエクスペリエンスを制御できます。通常、パスキー認証のエクスペリエンスには次の 3 種類があります。
モーダル アカウント セレクタを表示する
検出可能な認証情報を使用すると、RP はモーダル アカウント セレクタを表示して、ユーザーがログインするアカウントを選択できるようにし、その後ユーザー確認を行うことができます。これは、パスキー認証専用のボタンを押して開始されるパスキー認証フローに適しています。
このユーザー エクスペリエンスを実現するには、navigator.credentials.get() の allowCredentials パラメータを省略するか、空の配列を渡します。
async function authenticate() {
// ...
const publicKeyCredentialRequestOptions = {
// Server generated challenge:
challenge: ****,
// The same RP ID as used during registration:
rpId: 'example.com',
// You can omit `allowCredentials` as well:
allowCredentials: []
};
const credential = await navigator.credentials.get({
publicKey: publicKeyCredentialRequestOptions,
signal: abortController.signal
});
// This does not run until the user selects a passkey.
const credential = {};
credential.id = cred.id;
credential.rawId = cred.id; // Pass a Base64URL encoded ID string.
credential.type = cred.type;
// ...
}
パスキー フォームの自動入力を表示する
ほとんどのユーザーがパスキーを使用し、ローカル デバイスでパスキーを利用できる場合は、上記のモーダル アカウント セレクタが適切に機能します。ローカル パスキーがないユーザーの場合でも、モーダル ダイアログが表示され、別のデバイスからパスキーを提示するように求められます。ユーザーをパスキーに移行する際に、パスキーを設定していないユーザーにはその UI を表示しないようにすることをおすすめします。
代わりに、パスキーの選択を、保存されたユーザー名とパスワードとともに、従来のログイン フォームのフィールドの自動入力プロンプトに組み込むことができます。これにより、パスキーを使用しているユーザーはパスキーを選択してログイン フォームに「入力」できます。保存されたユーザー名とパスワードのペアを使用しているユーザーはそれらを選択できます。どちらも使用していないユーザーは、ユーザー名とパスワードを入力できます。
このユーザー エクスペリエンスは、RP がパスワードとパスキーを併用する移行中の場合に最適です。
このユーザー エクスペリエンスを実現するには、allowCredentials プロパティに空の配列を渡すか、パラメータを省略するだけでなく、mediation: 'conditional' で navigator.credentials.get() を指定し、HTML username 入力フィールドに autocomplete="username webauthn"、または password 入力フィールドに autocomplete="password webauthn" のアノテーションを付けます。
navigator.credentials.get() を呼び出しても UI は表示されませんが、ユーザーがアノテーション付きの入力フィールドにフォーカスすると、使用可能なパスキーが自動入力オプションに含まれます。ユーザーが 1 つ選択すると、通常のデバイスのロック解除確認が行われ、その後にのみ .get() から返された Promise が結果で解決されます。ユーザーがパスキーを選択しない場合、Promise は解決されません。
async function authenticate() {
// ...
const publicKeyCredentialRequestOptions = {
// Server generated challenge:
challenge: ****,
// The same RP ID as used during registration:
rpId: 'example.com',
// You can omit `allowCredentials` as well:
allowCredentials: []
};
const cred = await navigator.credentials.get({
publicKey: publicKeyCredentialRequestOptions,
signal: abortController.signal,
// Specify 'conditional' to activate conditional UI
mediation: 'conditional'
});
// This does not run until the user selects a passkey.
const credential = {};
credential.id = cred.id;
credential.rawId = cred.id; // Pass a Base64URL encoded ID string.
credential.type = cred.type;
// ...
}
<input type="text" name="username" autocomplete="username webauthn" ...>
このユーザー エクスペリエンスの構築方法については、フォームの自動入力でパスキーを使用してログインする、およびウェブアプリでフォームの自動入力を使用してパスキーを実装するの Codelab をご覧ください。
再認証
パスキーを再認証に使用する場合など、ユーザーの識別子がすでにわかっている場合があります。この場合、ブラウザや OS にアカウント セレクタを表示せずにパスキーを使用したいと考えられます。これは、allowCredentials パラメータに認証情報 ID のリストを渡すことで実現できます。
その場合、指定された認証情報のいずれかがローカルで利用可能な場合は、デバイスのロック解除をすぐに求められます。そうでない場合は、有効な認証情報を持つ別のデバイス(スマートフォンまたはセキュリティ キー)を提示するように求められます。
このユーザー エクスペリエンスを実現するには、ログインするユーザーの認証情報 ID のリストを指定します。ユーザーはすでにわかっているため、RP はそれらをクエリできます。navigator.credentials.get() の allowCredentials プロパティに PublicKeyCredentialDescriptor オブジェクトとして認証情報 ID を指定します。
async function authenticate() {
// ...
const publicKeyCredentialRequestOptions = {
// Server generated challenge:
challenge: ****,
// The same RP ID as used during registration:
rpId: 'example.com',
// Provide a list of PublicKeyCredentialDescriptors:
allowCredentials: [{
id: ****,
type: 'public-key',
transports: [
'internal',
'hybrid'
]
}, {
id: ****,
type: 'public-key',
transports: [
'internal',
'hybrid'
]
}, ...]
};
const credential = await navigator.credentials.get({
publicKey: publicKeyCredentialRequestOptions,
signal: abortController.signal
});
// This does not run until the user selects a passkey.
const credential = {};
credential.id = cred.id;
credential.rawId = cred.id; // Pass a Base64URL encoded ID string.
credential.type = cred.type;
// ...
}
PublicKeyCredentialDescriptor オブジェクトは次の要素で構成されます。
id: RP がパスキー登録時に取得した公開鍵認証情報の ID。type: 通常、このフィールドは'public-key'です。transports: この認証情報を持つデバイスでサポートされているトランスポートのヒント。ブラウザはこれを使用して、外部デバイスの提示を求める UI を最適化します。このリストを指定する場合は、各認証情報の登録時にgetTransports()を呼び出した結果を含める必要があります。
まとめ
検出可能な認証情報を使用すると、ユーザーはユーザー名の入力をスキップできるため、パスキーのログイン エクスペリエンスが大幅に向上します。residentKey、requireResidentKey、allowCredentials を組み合わせることで、RP は次のようなログイン エクスペリエンスを実現できます。
- モーダル アカウント セレクタを表示する。
- パスキー フォームの自動入力を表示する。
- 再認証。
検出可能な認証情報を賢く使用してください。そうすることで、ユーザーがシームレスに利用でき、エンゲージメントを高めることができる洗練されたパスキーのログイン エクスペリエンスを設計できます。