HybridAuth with Google provider 在验证时随机 returns "invalid_request"
HybridAuth with Google provider randomly returns "invalid_request" when authenticating
我们使用 Google OAuth2 使用 HybridAuth 2.4.0 对我们的用户进行内部应用程序的身份验证,直到大约一周前我们开始看到越来越多的随机 "invalid_request" https://accounts.google.com/o/oauth2/token.
的回复
当我说"random"时,今天更有可能:不起作用然后系统地工作大约一分钟(连续多次认证会成功)再次停止工作(即:"invalid_request" 回复).
我们尝试升级到最新版本的 HybridAuth(2.8.0 和现在的 2.8.1),它没有解决遇到的问题。
还检查了服务器时间(NTP 同步且良好),尝试生成新的 Google API 秘密,创建新的 Google API 项目,没有更好的运气。
我们认为这是一个 environment/server 问题,因为我们有一个本地开发环境,OAuth2 身份验证始终有效。此外,当使用 Google API 游乐场并使用与 PROD 服务器相同的有效负载向 https://accounts.google.com/o/oauth2/token 发出请求时,它有效,我们将得到 "Bearer" access_token.
一些 HybridAuth 调试日志:
2016-12-06T13:34:56+00:00 -- Endpoint: call adapter [Google] loginFinish()
2016-12-06T13:34:56+00:00 -- Enter OAuth2Client::request( https://accounts.google.com/o/oauth2/token )
2016-12-06T13:34:56+00:00 -- OAuth2Client::request(). dump request params: -- a:5:{s:9:"client_id";s:72:"[obfuscated].apps.googleusercontent.com";s:13:"client_secret";s:24:"[obfuscated]";s:10:"grant_type";s:18:"authorization_code";s:12:"redirect_uri";s:55:"https://[obfuscated]/endpoint?hauth.done=Google";s:4:"code";s:45:"[obfuscated]";}
2016-12-06T13:34:58+00:00 -- OAuth2Client::request(). dump request info: -- a:26:{s:3:"url";s:42:"https://accounts.google.com/o/oauth2/token";s:12:"content_type";s:31:"application/json; charset=utf-8";s:9:"http_code";i:400;s:11:"header_size";i:428;s:12:"request_size";i:318;s:8:"filetime";i:-1;s:17:"ssl_verify_result";i:0;s:14:"redirect_count";i:0;s:10:"total_time";d:2.5824850000000001;s:15:"namelookup_time";d:2.0000000000000002E-5;s:12:"connect_time";d:0.14088800000000001;s:16:"pretransfer_time";d:0.426647;s:11:"size_upload";d:753;s:13:"size_download";d:33;s:14:"speed_download";d:12;s:12:"speed_upload";d:291;s:23:"download_content_length";d:-1;s:21:"upload_content_length";d:753;s:18:"starttransfer_time";d:2.427991;s:13:"redirect_time";d:0;s:12:"redirect_url";s:0:"";s:10:"primary_ip";s:14:"[obfuscated]";s:8:"certinfo";a:0:{}s:12:"primary_port";i:443;s:8:"local_ip";s:11:"[obfuscated]";s:10:"local_port";i:35617;}
2016-12-06T13:34:58+00:00 -- OAuth2Client::request(). dump request result: -- s:33:"{
"error" : "invalid_request"
}";
任何关于进一步调查方向的线索将不胜感激:)
已成功解决问题。看起来 Hybridauth 正在将一个数组传递到 POSTFIELDS
curl_setopt($ch, CURLOPT_POSTFIELDS, array(
'code='. urlencode($code),
'client_id=' . urlencode($clientID),
'client_secret=' . urlencode($clientSecret),
'redirect_uri=http%3A%2F%2Flocalhost%2Fexperiments%2FnewGALogin.php',
'grant_type=authorization_code'
));
当输入为数组时,生成的 Content-Type 将为 multipart/form-data,这不符合 OAuth 2.0 规范,服务器将忽略它。当输入是查询编码的字符串(例如使用 http_build_query 构建)时,Content-Type: 将是 application/x-www-form-urlencoded,这是规范所要求的。
请参阅 "Notes" 部分:http://php.net/manual/en/function.curl-setopt.php
因此,如果我们将其作为查询字符串传递:
curl_setopt($ch, CURLOPT_POSTFIELDS,
'code=' . urlencode($code) . '&' .
'client_id=' . urlencode($clientID) . '&' .
'client_secret=' . urlencode($clientSecret) . '&' .
'redirect_uri=http%3A%2F%2Flocalhost%2Fexperiments%2FnewGALogin.php' . '&' .
'grant_type=authorization_code'
);
我们不再看到这个问题。
希望对您有所帮助!
@Adzzz 回答正确(非常感谢)。修补 hybridAuth 的一些代码...
//file hybridauth/hybridauth/Hybrid/thirdparty/OAuth/OAuth2Client.php line 234
if( $type == "POST" ){
curl_setopt($ch, CURLOPT_POST, 1);
$paramsString="";
if($params){
foreach($params as $k=>$v){
$paramsString.=$k."=";
$paramsString.=$v."&";
}
curl_setopt( $ch, CURLOPT_POSTFIELDS, $paramsString );
}
//original code curl_setopt( $ch, CURLOPT_POSTFIELDS, $params );
}
我们使用 Google OAuth2 使用 HybridAuth 2.4.0 对我们的用户进行内部应用程序的身份验证,直到大约一周前我们开始看到越来越多的随机 "invalid_request" https://accounts.google.com/o/oauth2/token.
的回复当我说"random"时,今天更有可能:不起作用然后系统地工作大约一分钟(连续多次认证会成功)再次停止工作(即:"invalid_request" 回复).
我们尝试升级到最新版本的 HybridAuth(2.8.0 和现在的 2.8.1),它没有解决遇到的问题。
还检查了服务器时间(NTP 同步且良好),尝试生成新的 Google API 秘密,创建新的 Google API 项目,没有更好的运气。
我们认为这是一个 environment/server 问题,因为我们有一个本地开发环境,OAuth2 身份验证始终有效。此外,当使用 Google API 游乐场并使用与 PROD 服务器相同的有效负载向 https://accounts.google.com/o/oauth2/token 发出请求时,它有效,我们将得到 "Bearer" access_token.
一些 HybridAuth 调试日志:
2016-12-06T13:34:56+00:00 -- Endpoint: call adapter [Google] loginFinish()
2016-12-06T13:34:56+00:00 -- Enter OAuth2Client::request( https://accounts.google.com/o/oauth2/token )
2016-12-06T13:34:56+00:00 -- OAuth2Client::request(). dump request params: -- a:5:{s:9:"client_id";s:72:"[obfuscated].apps.googleusercontent.com";s:13:"client_secret";s:24:"[obfuscated]";s:10:"grant_type";s:18:"authorization_code";s:12:"redirect_uri";s:55:"https://[obfuscated]/endpoint?hauth.done=Google";s:4:"code";s:45:"[obfuscated]";}
2016-12-06T13:34:58+00:00 -- OAuth2Client::request(). dump request info: -- a:26:{s:3:"url";s:42:"https://accounts.google.com/o/oauth2/token";s:12:"content_type";s:31:"application/json; charset=utf-8";s:9:"http_code";i:400;s:11:"header_size";i:428;s:12:"request_size";i:318;s:8:"filetime";i:-1;s:17:"ssl_verify_result";i:0;s:14:"redirect_count";i:0;s:10:"total_time";d:2.5824850000000001;s:15:"namelookup_time";d:2.0000000000000002E-5;s:12:"connect_time";d:0.14088800000000001;s:16:"pretransfer_time";d:0.426647;s:11:"size_upload";d:753;s:13:"size_download";d:33;s:14:"speed_download";d:12;s:12:"speed_upload";d:291;s:23:"download_content_length";d:-1;s:21:"upload_content_length";d:753;s:18:"starttransfer_time";d:2.427991;s:13:"redirect_time";d:0;s:12:"redirect_url";s:0:"";s:10:"primary_ip";s:14:"[obfuscated]";s:8:"certinfo";a:0:{}s:12:"primary_port";i:443;s:8:"local_ip";s:11:"[obfuscated]";s:10:"local_port";i:35617;}
2016-12-06T13:34:58+00:00 -- OAuth2Client::request(). dump request result: -- s:33:"{
"error" : "invalid_request"
}";
任何关于进一步调查方向的线索将不胜感激:)
已成功解决问题。看起来 Hybridauth 正在将一个数组传递到 POSTFIELDS
curl_setopt($ch, CURLOPT_POSTFIELDS, array(
'code='. urlencode($code),
'client_id=' . urlencode($clientID),
'client_secret=' . urlencode($clientSecret),
'redirect_uri=http%3A%2F%2Flocalhost%2Fexperiments%2FnewGALogin.php',
'grant_type=authorization_code'
));
当输入为数组时,生成的 Content-Type 将为 multipart/form-data,这不符合 OAuth 2.0 规范,服务器将忽略它。当输入是查询编码的字符串(例如使用 http_build_query 构建)时,Content-Type: 将是 application/x-www-form-urlencoded,这是规范所要求的。
请参阅 "Notes" 部分:http://php.net/manual/en/function.curl-setopt.php
因此,如果我们将其作为查询字符串传递:
curl_setopt($ch, CURLOPT_POSTFIELDS,
'code=' . urlencode($code) . '&' .
'client_id=' . urlencode($clientID) . '&' .
'client_secret=' . urlencode($clientSecret) . '&' .
'redirect_uri=http%3A%2F%2Flocalhost%2Fexperiments%2FnewGALogin.php' . '&' .
'grant_type=authorization_code'
);
我们不再看到这个问题。
希望对您有所帮助!
@Adzzz 回答正确(非常感谢)。修补 hybridAuth 的一些代码...
//file hybridauth/hybridauth/Hybrid/thirdparty/OAuth/OAuth2Client.php line 234
if( $type == "POST" ){
curl_setopt($ch, CURLOPT_POST, 1);
$paramsString="";
if($params){
foreach($params as $k=>$v){
$paramsString.=$k."=";
$paramsString.=$v."&";
}
curl_setopt( $ch, CURLOPT_POSTFIELDS, $paramsString );
}
//original code curl_setopt( $ch, CURLOPT_POSTFIELDS, $params );
}