
org.jclouds.googlecomputeengine.config.GoogleComputeEngineHttpApiModule Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jclouds-shaded Show documentation
Show all versions of jclouds-shaded Show documentation
Provides a shaded jclouds with relocated guava and guice
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jclouds.googlecomputeengine.config;
import static shaded.com.google.common.base.Suppliers.compose;
import static java.util.concurrent.TimeUnit.SECONDS;
import static javax.ws.rs.core.MediaType.APPLICATION_JSON;
import static org.jclouds.Constants.PROPERTY_SESSION_INTERVAL;
import static org.jclouds.googlecloud.config.GoogleCloudProperties.PROJECT_NAME;
import static org.jclouds.rest.config.BinderUtils.bindHttpApi;
import java.net.URI;
import java.util.concurrent.atomic.AtomicReference;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import org.jclouds.domain.Credentials;
import org.jclouds.googlecloud.config.CurrentProject;
import org.jclouds.googlecomputeengine.GoogleComputeEngineApi;
import org.jclouds.googlecomputeengine.domain.Project;
import org.jclouds.googlecomputeengine.handlers.GoogleComputeEngineErrorHandler;
import org.jclouds.http.HttpErrorHandler;
import org.jclouds.http.annotation.ClientError;
import org.jclouds.http.annotation.Redirection;
import org.jclouds.http.annotation.ServerError;
import org.jclouds.location.Provider;
import org.jclouds.oauth.v2.config.OAuthScopes;
import org.jclouds.oauth.v2.config.OAuthScopes.ReadOrWriteScopes;
import org.jclouds.oauth.v2.filters.OAuthFilter;
import org.jclouds.providers.ProviderMetadata;
import org.jclouds.rest.AuthorizationException;
import org.jclouds.rest.ConfiguresHttpApi;
import org.jclouds.rest.annotations.RequestFilters;
import org.jclouds.rest.annotations.SkipEncoding;
import org.jclouds.rest.config.HttpApiModule;
import org.jclouds.rest.suppliers.MemoizedRetryOnTimeOutButNotOnAuthorizationExceptionSupplier;
import shaded.com.google.common.base.Function;
import shaded.com.google.common.base.Strings;
import shaded.com.google.common.base.Supplier;
import shaded.com.google.common.base.Suppliers;
import shaded.com.google.inject.Provides;
@ConfiguresHttpApi
public final class GoogleComputeEngineHttpApiModule extends HttpApiModule {
@Override protected void configure() {
super.configure();
bindHttpApi(binder(), UseApiToResolveProjectName.GetProject.class);
bind(OAuthScopes.class).toInstance(ReadOrWriteScopes.create( //
"https://www.googleapis.com/auth/compute.readonly", //
"https://www.googleapis.com/auth/compute" //
));
}
@Override protected void bindErrorHandlers() {
bind(HttpErrorHandler.class).annotatedWith(Redirection.class).to(GoogleComputeEngineErrorHandler.class);
bind(HttpErrorHandler.class).annotatedWith(ClientError.class).to(GoogleComputeEngineErrorHandler.class);
bind(HttpErrorHandler.class).annotatedWith(ServerError.class).to(GoogleComputeEngineErrorHandler.class);
}
@Provides @Singleton @CurrentProject Supplier project(@Named(PROJECT_NAME) final String projectName,
@Provider Supplier defaultEndpoint, final UseApiToResolveProjectName useApiToResolveProjectName,
@Provider final Supplier creds, AtomicReference authException,
@Named(PROPERTY_SESSION_INTERVAL) long seconds) {
// Try to avoid a runtime lookup by accepting a project name supplied in context overrides.
if (Strings.emptyToNull(projectName) != null) {
return Suppliers.memoizeWithExpiration(Suppliers.compose(new Function() {
@Override public URI apply(URI input) {
return URI.create(String.format("%s/projects/%s", input, projectName));
}
}, defaultEndpoint), seconds, SECONDS);
}
// If the project name wasn't explicitly supplied, then we lookup via api.
// This supplier must be defensive against any auth exception.
return MemoizedRetryOnTimeOutButNotOnAuthorizationExceptionSupplier
.create(authException, compose(useApiToResolveProjectName, creds), seconds, SECONDS);
}
/**
* Parse the project ID from the identity, use it to lookup the project name, return the project-scoped uri.
*
* Why are we looking up the project name? We already have the project ID!
* Documentation suggests that the
* project name is interchangeable with the project ID, which we already have. However, in practice, using the
* project ID leads to problems in POST requests.
*
* For example, inserting an instance using the project ID in the instances url, but the project name in
* the machineType url results in an error of
* {@code Cross-project references for this resource type are not allowed}.
*
* Similar errors occur in POST requests to other resources including at least forwardingRules, images,
* targetPools.
*/
static final class UseApiToResolveProjectName implements Function {
@SkipEncoding({ '/', '=' })
@RequestFilters(OAuthFilter.class)
@Consumes(APPLICATION_JSON)
interface GetProject {
@Named("Projects:get")
@GET
@Path("/projects/{projectNumber}") Project get(@PathParam("projectNumber") String projectNumber);
}
private final GetProject api;
private final Supplier defaultEndpoint;
@Inject
UseApiToResolveProjectName(GetProject api, @Provider Supplier defaultEndpoint, ProviderMetadata metadata) {
this.api = api;
this.defaultEndpoint = defaultEndpoint;
}
@Override public URI apply(Credentials in) {
String projectNumber = CurrentProject.ClientEmail.toProjectNumber(in.identity);
return URI.create(defaultEndpoint.get() + "/projects/" + api.get(projectNumber).name());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy