-
Notifications
You must be signed in to change notification settings - Fork 310
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Add WebAuthn plugin component to handle WebAuthn get assertion …
…request (#1464)
- Loading branch information
1 parent
93d681e
commit e25f336
Showing
6 changed files
with
625 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
import abc | ||
import os | ||
import struct | ||
import subprocess | ||
|
||
from google.auth import exceptions | ||
from google.oauth2.webauthn_types import GetRequest, GetResponse | ||
|
||
|
||
class WebAuthnHandler(abc.ABC): | ||
@abc.abstractmethod | ||
def is_available(self) -> bool: | ||
"""Check whether this WebAuthn handler is available""" | ||
raise NotImplementedError("is_available method must be implemented") | ||
|
||
@abc.abstractmethod | ||
def get(self, get_request: GetRequest) -> GetResponse: | ||
"""WebAuthn get (assertion)""" | ||
raise NotImplementedError("get method must be implemented") | ||
|
||
|
||
class PluginHandler(WebAuthnHandler): | ||
"""Offloads WebAuthn get reqeust to a pluggable command-line tool. | ||
Offloads WebAuthn get to a plugin which takes the form of a | ||
command-line tool. The command-line tool is configurable via the | ||
PluginHandler._ENV_VAR environment variable. | ||
The WebAuthn plugin should implement the following interface: | ||
Communication occurs over stdin/stdout, and messages are both sent and | ||
received in the form: | ||
[4 bytes - payload size (little-endian)][variable bytes - json payload] | ||
""" | ||
|
||
_ENV_VAR = "GOOGLE_AUTH_WEBAUTHN_PLUGIN" | ||
|
||
def is_available(self) -> bool: | ||
try: | ||
self._find_plugin() | ||
except Exception: | ||
return False | ||
else: | ||
return True | ||
|
||
def get(self, get_request: GetRequest) -> GetResponse: | ||
request_json = get_request.to_json() | ||
cmd = self._find_plugin() | ||
response_json = self._call_plugin(cmd, request_json) | ||
return GetResponse.from_json(response_json) | ||
|
||
def _call_plugin(self, cmd: str, input_json: str) -> str: | ||
# Calculate length of input | ||
input_length = len(input_json) | ||
length_bytes_le = struct.pack("<I", input_length) | ||
request = length_bytes_le + input_json.encode() | ||
|
||
# Call plugin | ||
process_result = subprocess.run( | ||
[cmd], input=request, capture_output=True, check=True | ||
) | ||
|
||
# Check length of response | ||
response_len_le = process_result.stdout[:4] | ||
response_len = struct.unpack("<I", response_len_le)[0] | ||
response = process_result.stdout[4:] | ||
if response_len != len(response): | ||
raise exceptions.MalformedError( | ||
"Plugin response length {} does not match data {}".format( | ||
response_len, len(response) | ||
) | ||
) | ||
return response.decode() | ||
|
||
def _find_plugin(self) -> str: | ||
plugin_cmd = os.environ.get(PluginHandler._ENV_VAR) | ||
if plugin_cmd is None: | ||
raise exceptions.InvalidResource( | ||
"{} env var is not set".format(PluginHandler._ENV_VAR) | ||
) | ||
return plugin_cmd |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,156 @@ | ||
from dataclasses import dataclass | ||
import json | ||
from typing import Any, Dict, List, Optional | ||
|
||
from google.auth import exceptions | ||
|
||
|
||
@dataclass(frozen=True) | ||
class PublicKeyCredentialDescriptor: | ||
"""Descriptor for a security key based credential. | ||
https://www.w3.org/TR/webauthn-3/#dictionary-credential-descriptor | ||
Args: | ||
id: <url-safe base64-encoded> credential id (key handle). | ||
transports: <'usb'|'nfc'|'ble'|'internal'> List of supported transports. | ||
""" | ||
|
||
id: str | ||
transports: Optional[List[str]] = None | ||
|
||
def to_dict(self): | ||
cred = {"type": "public-key", "id": self.id} | ||
if self.transports: | ||
cred["transports"] = self.transports | ||
return cred | ||
|
||
|
||
@dataclass | ||
class AuthenticationExtensionsClientInputs: | ||
"""Client extensions inputs for WebAuthn extensions. | ||
Args: | ||
appid: app id that can be asserted with in addition to rpid. | ||
https://www.w3.org/TR/webauthn-3/#sctn-appid-extension | ||
""" | ||
|
||
appid: Optional[str] = None | ||
|
||
def to_dict(self): | ||
extensions = {} | ||
if self.appid: | ||
extensions["appid"] = self.appid | ||
return extensions | ||
|
||
|
||
@dataclass | ||
class GetRequest: | ||
"""WebAuthn get request | ||
Args: | ||
origin: Origin where the WebAuthn get assertion takes place. | ||
rpid: Relying Party ID. | ||
challenge: <url-safe base64-encoded> raw challenge. | ||
timeout_ms: Timeout number in millisecond. | ||
allow_credentials: List of allowed credentials. | ||
user_verification: <'required'|'preferred'|'discouraged'> User verification requirement. | ||
extensions: WebAuthn authentication extensions inputs. | ||
""" | ||
|
||
origin: str | ||
rpid: str | ||
challenge: str | ||
timeout_ms: Optional[int] = None | ||
allow_credentials: Optional[List[PublicKeyCredentialDescriptor]] = None | ||
user_verification: Optional[str] = None | ||
extensions: Optional[AuthenticationExtensionsClientInputs] = None | ||
|
||
def to_json(self) -> str: | ||
req_options: Dict[str, Any] = {"rpid": self.rpid, "challenge": self.challenge} | ||
if self.timeout_ms: | ||
req_options["timeout"] = self.timeout_ms | ||
if self.allow_credentials: | ||
req_options["allowCredentials"] = [ | ||
c.to_dict() for c in self.allow_credentials | ||
] | ||
if self.user_verification: | ||
req_options["userVerification"] = self.user_verification | ||
if self.extensions: | ||
req_options["extensions"] = self.extensions.to_dict() | ||
return json.dumps( | ||
{"type": "get", "origin": self.origin, "requestData": req_options} | ||
) | ||
|
||
|
||
@dataclass(frozen=True) | ||
class AuthenticatorAssertionResponse: | ||
"""Authenticator response to a WebAuthn get (assertion) request. | ||
https://www.w3.org/TR/webauthn-3/#authenticatorassertionresponse | ||
Args: | ||
client_data_json: <url-safe base64-encoded> client data JSON. | ||
authenticator_data: <url-safe base64-encoded> authenticator data. | ||
signature: <url-safe base64-encoded> signature. | ||
user_handle: <url-safe base64-encoded> user handle. | ||
""" | ||
|
||
client_data_json: str | ||
authenticator_data: str | ||
signature: str | ||
user_handle: Optional[str] | ||
|
||
|
||
@dataclass(frozen=True) | ||
class GetResponse: | ||
"""WebAuthn get (assertion) response. | ||
Args: | ||
id: <url-safe base64-encoded> credential id (key handle). | ||
response: The authenticator assertion response. | ||
authenticator_attachment: <'cross-platform'|'platform'> The attachment status of the authenticator. | ||
client_extension_results: WebAuthn authentication extensions output results in a dictionary. | ||
""" | ||
|
||
id: str | ||
response: AuthenticatorAssertionResponse | ||
authenticator_attachment: Optional[str] | ||
client_extension_results: Optional[Dict] | ||
|
||
@staticmethod | ||
def from_json(json_str: str): | ||
"""Verify and construct GetResponse from a JSON string.""" | ||
try: | ||
resp_json = json.loads(json_str) | ||
except ValueError: | ||
raise exceptions.MalformedError("Invalid Get JSON response") | ||
if resp_json.get("type") != "getResponse": | ||
raise exceptions.MalformedError( | ||
"Invalid Get response type: {}".format(resp_json.get("type")) | ||
) | ||
pk_cred = resp_json.get("responseData") | ||
if pk_cred is None: | ||
if resp_json.get("error"): | ||
raise exceptions.ReauthFailError( | ||
"WebAuthn.get failure: {}".format(resp_json["error"]) | ||
) | ||
else: | ||
raise exceptions.MalformedError("Get response is empty") | ||
if pk_cred.get("type") != "public-key": | ||
raise exceptions.MalformedError( | ||
"Invalid credential type: {}".format(pk_cred.get("type")) | ||
) | ||
assertion_json = pk_cred["response"] | ||
assertion_resp = AuthenticatorAssertionResponse( | ||
client_data_json=assertion_json["clientDataJSON"], | ||
authenticator_data=assertion_json["authenticatorData"], | ||
signature=assertion_json["signature"], | ||
user_handle=assertion_json.get("userHandle"), | ||
) | ||
return GetResponse( | ||
id=pk_cred["id"], | ||
response=assertion_resp, | ||
authenticator_attachment=pk_cred.get("authenticatorAttachment"), | ||
client_extension_results=pk_cred.get("clientExtensionResults"), | ||
) |
Oops, something went wrong.