我下载了Google Endpoints Tic Tac Toe示例 - Java中的服务器代码。
为了快速运行它,我从API定义中删除了clientIds,因此我可以很快看到它在API Explorer中工作:
@Api(name = "tictactoe", version = "v1")
public class ScoresV1
{
...
我可以使用API资源管理器运行以下方法,打开OAuth并使用我的gmail accoumnt进行身份验证。我可以在调试器中看到“user”对象设置为该帐户:
@ApiMethod(name = "scores.list")
@SuppressWarnings("unchecked")
public List<Score> list(@Nullable @Named("limit") String limit,
@Nullable @Named("order") String order, User user)
throws OAuthRequestException, IOException
{
...
然后我继续生成Android客户端(仍然删除了clientIds)。我还下载了official Android app来获取UI类和res文件(帐户选择器和游戏界面)。
这就是我被困住的地方。在真正的Android设备上运行时,TictactoeActivity中的以下代码引发了GoogleAuthIOException:
@Override
protected ScoreCollection doInBackground(Void... unused)
{
ScoreCollection scores = null;
try
{
scores = service.scores().list().execute();
}
catch (IOException e)
{
e.printStackTrace();
以下是例外:
02-21 16:38:47.051: D/TicTacToe(6151): com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAuthIOException
02-21 16:38:47.051: D/TicTacToe(6151): at com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAccountCredential$RequestHandler.intercept(GoogleAccountCredential.java:286)
02-21 16:38:47.051: D/TicTacToe(6151): at com.google.api.client.http.HttpRequest.execute(HttpRequest.java:858)
...
在源代码中再挖掘一下,我发现这个代码在GoogleAuthUtil.getToken()中抛出了根异常,这有点无益:
com.google.android.gms.auth.GoogleAuthException: Unknown
现在,我在ScoreV1.list()上的本地开发服务器上有一个断点,在使用API Explorer进行测试时遇到了这个断点,但Android客户端在没有命中服务器的情况下失败了。现在,在本地开发服务器上设置了“-a 0.0.0.0”以接受传入连接。我已经检查过URL也是正确的,我可以从手机上运行以下网址 Chrome浏览器(因为没有设置用户,它给了我一个Auth错误,但没关系):
http://192.168.2.23:8888/_ah/api/tictactoe/v1/score?limit=1&order=1
所以我知道网络正在运行,手机可以访问它。我还验证了AndroidManifest.xml具有以下权限集:
<uses-sdk android:minSdkVersion="14" android:targetSdkVersion="14"/>
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.USE_CREDENTIALS" />
以下是在TicTacToe.java中显示我的URL和路径的代码:
public static final String DEFAULT_ROOT_URL = "http://192.168.2.23:8888/_ah/api/";
/**
* The default encoded service path of the service. This is determined when
* the library is generated and normally should not be changed.
*
* @since 1.7
*/
public static final String DEFAULT_SERVICE_PATH = "tictactoe/v1/";
/**
* The default encoded base URL of the service. This is determined when the
* library is generated and normally should not be changed.
*/
public static final String DEFAULT_BASE_URL = DEFAULT_ROOT_URL
+ DEFAULT_SERVICE_PATH;
...
现在,让我感到困惑的是,即使我将DEFAULT_ROOT_URL更改为随机的内容,例如5.5.5.5,我也会得到完全相同的GoogleAuthIOException!当主机无法访问时,我希望得到一个不同的例外......
此外,我尝试将服务器部署到生产环境,并将URL重新指向[myapp] .appspot.com地址 - 结果相同。服务器日志从未注册过请求。
实际上,我现在真的被卡住了。请帮忙!万分感谢!!!
答案 0 :(得分:0)
该错误很可能是由于Android应用程序与Cloud Endpoints API之间未发生授权所致。
由于我无法完全访问您的实际Endpoint API注释和Android代码,因此我建议您仔细查看以下几点:
确保您已使用clientIds
和audiences
正确注释了您的Cloud Endpoints API。 Android Audience值与Web Client Id的值相同。
确保您已部署更新的API并重新生成Endpoints API源以包含在Android应用程序中。
最后,在您的Androids源代码中,您必须使用GoogleAccountCredential.usingAudience(...)构建GoogleAccountCredential的实例,然后在调用端点API时使用此凭据对象。对于例如端点构建器采用HTTP传输,GSON工厂和凭据。不要将最后一个参数(即Credential)留空。
答案 1 :(得分:0)
我也遇到了类似的GoogleAuthIOException异常:
com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAuthIOException
at com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAccountCredential$RequestHandler.intercept(GoogleAccountCredential.java:286)
at com.google.api.client.http.HttpRequest.execute(HttpRequest.java:859)
at com.google.api.client.googleapis.services.AbstractGoogleClientRequest.executeUnparsed(AbstractGoogleClientRequest.java:410)
at com.google.api.client.googleapis.services.AbstractGoogleClientRequest.executeUnparsed(AbstractGoogleClientRequest.java:343)
at com.google.api.client.googleapis.services.AbstractGoogleClientRequest.execute(AbstractGoogleClientRequest.java:460)
at com.example.MainActivity$ListPartiesTask.doInBackground(MainActivity.java:188)
at com.example.MainActivity$ListPartiesTask.doInBackground(MainActivity.java:178)
at android.os.AsyncTask$2.call(AsyncTask.java:287)
at java.util.concurrent.FutureTask.run(FutureTask.java:234)
at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:230)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1080)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:573)
at java.lang.Thread.run(Thread.java:841)
Caused by: com.google.android.gms.auth.GoogleAuthException: Unknown
at com.google.android.gms.auth.GoogleAuthUtil.zzb(Unknown Source)
at com.google.android.gms.auth.GoogleAuthUtil.zza(Unknown Source)
at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
at com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAccountCredential.getToken(GoogleAccountCredential.java:255)
at com.google.api.client.googleapis.extensions.android.gms.auth.GoogleAccountCredential$RequestHandler.intercept(GoogleAccountCredential.java:279)
... 12 more
原来我需要使用应用程序调试版本的sha1指纹创建一个客户端ID(我已经为该应用程序的签名版本创建了一个,但是如果你还没有做到这一点,那么你就是#39 ;我也需要这样做。)
您可以使用以下命令获取指纹(在Mac上,debug.keystore的地址在其他系统上可能会有所不同):
keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android
如果您需要获取释放密钥的指纹,可以执行以下操作:
keytool -list -v -keystore /PATH/TO/YOUR/release-key.jks -alias RELEASE_KEY_ALIAS