From 02d199fd9578f6b2f10321eacfc098386b1e51be Mon Sep 17 00:00:00 2001 From: Roeland Jago Douma Date: Fri, 27 Oct 2023 11:38:59 +0200 Subject: [PATCH] Add PKCE support For the providers that support it. Add extra security. See also https://datatracker.ietf.org/doc/html/rfc7636 Signed-off-by: Roeland Jago Douma --- lib/Controller/LoginController.php | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/lib/Controller/LoginController.php b/lib/Controller/LoginController.php index bacafd93..835236ae 100644 --- a/lib/Controller/LoginController.php +++ b/lib/Controller/LoginController.php @@ -69,6 +69,7 @@ class LoginController extends BaseOidcController { public const PROVIDERID = 'oidc.providerid'; private const REDIRECT_AFTER_LOGIN = 'oidc.redirect'; private const ID_TOKEN = 'oidc.id_token'; + private const CODE_VERIFIER = 'oidc.code_verifier'; /** @var ISecureRandom */ private $random; @@ -229,6 +230,10 @@ public function login(int $providerId, string $redirectUrl = null) { $nonce = $this->random->generate(32, ISecureRandom::CHAR_DIGITS . ISecureRandom::CHAR_UPPER); $this->session->set(self::NONCE, $nonce); + // PKCE code_challenge see https://datatracker.ietf.org/doc/html/rfc7636 + $code_verifier = $this->random->generage(128, ISecureRandom::CHAR_DIGITS . ISecureRandom::CHAR_UPPER . ISecureRandom::CHAR_LOWER); + $this->session->set(self::CODE_VERIFIER, $code_verifier); + $this->session->set(self::PROVIDERID, $providerId); $this->session->close(); @@ -280,6 +285,8 @@ public function login(int $providerId, string $redirectUrl = null) { 'claims' => json_encode($claims), 'state' => $state, 'nonce' => $nonce, + 'code_challenge' => $this->toCodeChallenge($code_verifier), + 'code_challenge_method' => 'S256', ]; // pass discovery query parameters also on to the authentication $discoveryUrl = parse_url($provider->getDiscoveryEndpoint()); @@ -368,6 +375,8 @@ public function code(string $state = '', string $code = '', string $scope = '', return $this->buildErrorTemplateResponse($message, Http::STATUS_BAD_REQUEST, [], false); } + $code_verifier = $this->session->get(self::CODE_VERIFIER); + $discovery = $this->discoveryService->obtainDiscovery($provider); $this->logger->debug('Obtainting data from: ' . $discovery['token_endpoint']); @@ -383,6 +392,7 @@ public function code(string $state = '', string $code = '', string $scope = '', 'client_secret' => $providerClientSecret, 'redirect_uri' => $this->urlGenerator->linkToRouteAbsolute(Application::APP_ID . '.login.code'), 'grant_type' => 'authorization_code', + 'code_verifier' => $code_verifier, // Set for the PKCE flow ], ] ); @@ -730,4 +740,14 @@ private function getBackchannelLogoutErrorResponse(string $error, string $descri } return $response; } + + private function toCodeChallenge(string $data): string { + // Basically one big work around for the base64url decode being weird + $h = pack('H*', hash('sha256', $data)); + $s = base64_encode($h); // Regular base64 encoder + $s = explode('=', $s)[0]; // Remove any trailing '='s + $s = str_replace('+', '-', $s); // 62nd char of encoding + $s = str_replace('/', '_', $s); // 63rd char of encoding + return $s; + } }