0

我正在编写一个 Java 应用程序来与 Google Cloud Storage 中的文件进行交互。我发现gcloud-java我正在努力工作。

查看他们的示例,似乎我应该能够在使用 登录后简单地运行它们gcloud,但它似乎不起作用。我正在尝试运行StorageExample,上面写着“如果未提供,将使用已登录的项目”,但尽管登录,无论我是否指定,我都无法访问我的项目。

$ gcloud auth login
Your browser has been opened to visit:

    https://accounts.google.com/o/oauth2/auth?....


Saved Application Default Credentials.

You are now logged in as [...].
Your current project is [...].  You can change this setting by running:
  $ gcloud config set project PROJECT_ID

$ java -cp GCloudDemo.jar com.google.gcloud.examples.storage.StorageExample list
Exception in thread "main" java.lang.IllegalArgumentException: A project ID is required for this service but could not be determined from the builder or the environment.  Please set a project ID using the builder.
        at com.google.common.base.Preconditions.checkArgument(Preconditions.java:122)
        at com.google.gcloud.ServiceOptions.<init>(ServiceOptions.java:317)
        ...
        at com.google.gcloud.examples.storage.StorageExample.main(StorageExample.java:566)

$ java -cp GCloudDemo.jar com.google.gcloud.examples.storage.StorageExample ... list
Exception in thread "main" com.google.gcloud.storage.StorageException: Login Required
        at com.google.gcloud.spi.DefaultStorageRpc.translate(DefaultStorageRpc.java:94)
        at com.google.gcloud.spi.DefaultStorageRpc.list(DefaultStorageRpc.java:148)
        ...
        at com.google.gcloud.examples.storage.StorageExample$ListAction.run(StorageExample.java:1)
        at com.google.gcloud.examples.storage.StorageExample.main(StorageExample.java:579)
Caused by: com.google.api.client.googleapis.json.GoogleJsonResponseException: 401 Unauthorized
{
  "code" : 401,
  "errors" : [ {
    "domain" : "global",
    "location" : "Authorization",
    "locationType" : "header",
    "message" : "Login Required",
    "reason" : "required"
  } ],
  "message" : "Login Required"
}
        at com.google.api.client.googleapis.json.GoogleJsonResponseException.from(GoogleJsonResponseException.java:145)
        ...
        at com.google.gcloud.spi.DefaultStorageRpc.list(DefaultStorageRpc.java:138)
        ... 10 more

很明显我的环境出了点问题,但我该怎么做才能修复它?我想更正我的环境而不是更改示例。

笔记:

  • 我在 Windows 10 上的 Cygwin 2.0.4 中运行
  • 我不是通过 Maven 构建的,但我认为这不是问题的根源
4

1 回答 1

1

原因gcloud auth login不起作用是因为 Java 不能很好地与 Cygwin 配合使用,特别是在绝对路径方面。

gcloud将您的设置和凭据存储在 Cygwin 主目录中~/.config/gcloud但 Java 将通过. 您可以通过环境变量指定应用程序应在何处查找此目录,如下所示(确保仅在调用时指定它;为您的 shell 设置它会依次中断):System.getProperty("user.home")CLOUDSDK_CONFIGjavagcloud

$ CLOUDSDK_CONFIG=$(cygpath -w ~/.config/gcloud) \
  java -cp GCloudDemo.jar com.google.gcloud.examples.storage.StorageExample list 
Exception in thread "main" com.google.gcloud.storage.StorageException: Login Required
    at ....

请注意,我们不必指定项目,但由于某种原因我们仍未登录。事实证明,演示应用程序在报告身份验证错误方面做得并不好。AuthCredentials.createApplicationDefaults()在方法开头添加显式调用会main()报告更有用的错误消息:

Exception in thread "main" java.io.IOException: The Application Default Credentials are
not available. They are available if running in Google Compute Engine. Otherwise, the
environment variable GOOGLE_APPLICATION_CREDENTIALS must be defined pointing to a
file defining the credentials. See https://developers.google.com/accounts/docs/application-default-credentials
for more information.

所以现在我们至少有GOOGLE_APPLICATION_CREDENTIALS环境变量可以继续了。但是为什么不gcloud为我们配置东西呢?看起来这实际上是库中的一个错误;它现在应该能够找到我们的登录令牌。

同时,我们也可以通过显式指定凭证路径来解决它:

$ CLOUDSDK_CONFIG=$(cygpath -w ~/.config/gcloud) \
  GOOGLE_APPLICATION_CREDENTIALS=$(cygpath -w ~/.config/gcloud)/application_default_credentials.json \
  java -cp GCloudDemo.jar com.google.gcloud.examples.storage.StorageExample list
Bucket{name=...}

成功!

于 2016-02-27T21:33:45.583 回答