*

mwindey

  • *****
  • 496 posts
Google login API outdated
« on: March 23, 2025, 12:32:53 PM »
Google Login is outdated the API needs an update:

[Sun Mar 23 12:27:16.411582 2025] [lsapi:error] [pid 134478:tid 139976540747520] [client 2a02:1811:b055:6200:6734:dd4f:2156:30e7:54513] [host ***.be] Backend fatal error: PHP Fatal error:  Uncaught Google_AuthException: Error fetching OAuth2 access token, message: 'invalid_grant' in /home/u****/domains/w***.be/public_html/oc-content/plugins/google_login/src/auth/Google_OAuth2.php:112

*

MB Themes

Re: Google login API outdated
« Reply #1 on: March 26, 2025, 02:37:16 PM »
How you came to conclusion it's outdated?
  To get fast support, we need following details: Detail description, URL to reproduce problem, Screenshots

*

mwindey

  • *****
  • 496 posts
Re: Google login API outdated
« Reply #2 on: March 26, 2025, 02:54:57 PM »
ChatGPT and errors [Sun Mar 23 12:27:16.411582 2025] [lsapi:error] [pid 134478:tid 139976540747520] [client 2a02:1811:b055:6200:6734:dd4f:2156:30e7:54513] [host ***.be] Backend fatal error: PHP Fatal error:  Uncaught Google_AuthException: Error fetching OAuth2 access token, message: 'invalid_grant' in /home/u****/domains/w***.be/public_html/oc-content/plugins/google_login/src/auth/Google_OAuth2.php:112

All is ok with settings in google....server clock is correct .... etc..

*

MB Themes

Re: Google login API outdated
« Reply #3 on: March 26, 2025, 03:48:01 PM »
But it's telling that grant is not ok. It would make sense to first check with google support, they usually do not change existing features.
  To get fast support, we need following details: Detail description, URL to reproduce problem, Screenshots

*

mwindey

  • *****
  • 496 posts
Re: Google login API outdated
« Reply #4 on: March 26, 2025, 06:09:51 PM »
Problem is php 8.3 the code says deprecated in 2 files... By changing to the next code it works without invalid grant errors now.
File plugin....plugins/google_login/src/contrib/Google_Oauth2Service.php
From:

Code: [Select]
* @author Google, Inc.
 */
class Google_Oauth2Service extends Google_Service {
  public $userinfo;
  public $userinfo_v2_me;
  /**
   * Constructs the internal representation of the Oauth2 service.
   *
   * @param Google_Client $client
   */
  public function __construct(Google_Client $client) {
    $this->servicePath = '';
    $this->version = 'v2';
    $this->serviceName = 'oauth2';

    $client->addService($this->serviceName, $this->version);
    $this->userinfo = new Google_UserinfoServiceResource($this, $this->serviceName, 'userinfo', json_decode('{"methods": {"get": {"path": "oauth2/v2/userinfo", "scopes": ["https://www.googleapis.com/auth/userinfo.email", "https://www.googleapis.com/auth/userinfo.profile"], "id": "oauth2.userinfo.get", "httpMethod": "GET", "response": {"$ref": "Userinfo"}}}}', true));
    $this->userinfo_v2_me = new Google_UserinfoV2MeServiceResource($this, $this->serviceName, 'me', json_decode('{"methods": {"get": {"path": "userinfo/v2/me", "scopes": ["https://www.googleapis.com/auth/userinfo.email", "https://www.googleapis.com/auth/userinfo.profile"], "id": "oauth2.userinfo.v2.me.get", "httpMethod": "GET", "response": {"$ref": "Userinfo"}}}}', true));
  }
}
Cange to copy paste:
Code: [Select]
* @author Google, Inc.
 */
class Google_Oauth2Service extends Google_Service {
  public $userinfo;
  public $userinfo_v2_me;
  public $servicePath;
  public $version;
  public $serviceName;

  /**
   * Constructs the internal representation of the Oauth2 service.
   *
   * @param Google_Client $client
   */
  public function __construct(Google_Client $client) {
    $this->servicePath = '';
    $this->version = 'v2';
    $this->serviceName = 'oauth2';

    $client->addService($this->serviceName, $this->version);
    $this->userinfo = new Google_UserinfoServiceResource(
      $this,
      $this->serviceName,
      'userinfo',
      json_decode('{"methods": {"get": {"path": "oauth2/v2/userinfo", "scopes": ["https://www.googleapis.com/auth/userinfo.email", "https://www.googleapis.com/auth/userinfo.profile"], "id": "oauth2.userinfo.get", "httpMethod": "GET", "response": {"$ref": "Userinfo"}}}}', true)
    );
    $this->userinfo_v2_me = new Google_UserinfoV2MeServiceResource(
      $this,
      $this->serviceName,
      'me',
      json_decode('{"methods": {"get": {"path": "userinfo/v2/me", "scopes": ["https://www.googleapis.com/auth/userinfo.email", "https://www.googleapis.com/auth/userinfo.profile"], "id": "oauth2.userinfo.v2.me.get", "httpMethod": "GET", "response": {"$ref": "Userinfo"}}}}', true)
    );
  }
}

Next file: plugins/google_login/src/auth/Google_OAuth2.php

Code: [Select]
  /**
   * @param $service
   * @param string|null $code
   * @throws Google_AuthException
   * @return string
   */
  public function authenticate($service, $code = null) {
    if (!$code && isset($_GET['code'])) {
      $code = $_GET['code'];
    }

    if ($code) {
      // We got here from the redirect from a successful authorization grant, fetch the access token
      $request = Google_Client::$io->makeRequest(new Google_HttpRequest(self::OAUTH2_TOKEN_URI, 'POST', array(), array(
          'code' => $code,
          'grant_type' => 'authorization_code',
          'redirect_uri' => $this->redirectUri,
          'client_id' => $this->clientId,
          'client_secret' => $this->clientSecret
      )));
      if ($request->getResponseHttpCode() == 200) {
        $this->setAccessToken($request->getResponseBody());
        $this->token['created'] = time();
        return $this->getAccessToken();
      } else {
        $response = $request->getResponseBody();
        $decodedResponse = json_decode($response, true);
        if ($decodedResponse != null && $decodedResponse['error']) {
          $response = $decodedResponse['error'];
        }
        throw new Google_AuthException("Error fetching OAuth2 access token, message: '$response'", $request->getResponseHttpCode());
      }
    }

    $authUrl = $this->createAuthUrl($service['scope']);
    header('Location: ' . $authUrl);
    return true;
  }

Change to copy paste:
Code: [Select]
/**
 * @param $service
 * @param string|null $code
 * @throws Google_AuthException
 * @return string
 */
public function authenticate($service, $code = null) {
    if (!$code && isset($_GET['code'])) {
        $code = $_GET['code'];
    }

    if ($code) {
        // We got here from the redirect from a successful authorization grant, fetch the access token
        $request = Google_Client::$io->makeRequest(new Google_HttpRequest(
            self::OAUTH2_TOKEN_URI,
            'POST',
            array(),
            array(
                'code' => $code,
                'grant_type' => 'authorization_code',
                'redirect_uri' => $this->redirectUri,
                'client_id' => $this->clientId,
                'client_secret' => $this->clientSecret
            )
        ));

        if ($request->getResponseHttpCode() == 200) {
            $this->setAccessToken($request->getResponseBody());
            $this->token['created'] = time();
            return $this->getAccessToken();
        } else {
            $response = $request->getResponseBody();
            $decodedResponse = json_decode($response, true);
           
            if ($decodedResponse != null && isset($decodedResponse['error'])) {
                $response = $decodedResponse['error'];
               
                // Handle invalid_grant specifically
                if ($response === 'invalid_grant') {
                    // Clear any existing session and restart auth
                    if (session_status() === PHP_SESSION_ACTIVE) {
                        session_unset();
                        session_destroy();
                    }
                    $authUrl = $this->createAuthUrl($service['scope']);
                    header('Location: ' . $authUrl);
                    exit;
                }
            }
           
            throw new Google_AuthException(
                "Error fetching OAuth2 access token, message: '$response'",
                $request->getResponseHttpCode()
            );
        }
    }

    $authUrl = $this->createAuthUrl($service['scope']);
    header('Location: ' . $authUrl);
    return true;
}

All errors gone :-) on php 8+ ....i did not test on lower php versions
« Last Edit: March 26, 2025, 06:20:51 PM by mwindey »

*

MB Themes

Re: Google login API outdated
« Reply #5 on: March 27, 2025, 08:28:30 AM »
Thanks, looks to be fine ;)
  To get fast support, we need following details: Detail description, URL to reproduce problem, Screenshots