I am writing a Java application to interact with files in Google Cloud Storage. I found gcloud-java
which I'm trying to get working.
Looking at their examples it seems I should be able to simply run them after having logged-in with gcloud
, but it doesn't seem to be working. I'm trying to run StorageExample
, which says "logged-in project will be used if not supplied", but despite logging in I cannot access my project whether I specify it or not.
$ 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
Clearly something is wrong with my environment, but what do I have to do to fix it? I'd like to correct my environment rather than make changes to the example.
Notes:
- I'm running in Cygwin 2.0.4 on Windows 10
- I'm not building via Maven, but I assume that isn't the source of the problem
The reason
gcloud auth login
isn't working is because Java doesn't play well with Cygwin, specifically with regards to absolute paths.gcloud
stores your settings and credentials in your Cygwin home directory,~/.config/gcloud
, but Java is going to look for them in your Windows home directory viaSystem.getProperty("user.home")
. You can specify where the application should look for this directory via theCLOUDSDK_CONFIG
environment variable, like so (be sure to only specify it when callingjava
; setting it for your shell will in turn breakgcloud
):Notice that that we didn't have to specify a project, but we're still not logged in for some reason. It turns out the demo app doesn't do a great job of reporting authentication errors. Adding an explicit call to
AuthCredentials.createApplicationDefaults()
at the start of themain()
method reports a much more helpful error message:So now we at least have the
GOOGLE_APPLICATION_CREDENTIALS
environment variable to go on. But why isn'tgcloud
configuring things for us? It looks like that's actually a bug in the library; it should now be able to find our login token.In the meantime, we can workaround it by explicitly specifying the credentials path as well:
Success!