mirror of https://github.com/apache/jclouds.git
Merge pull request #1366 from dralves/gce-compute
gce - compute abstraction
This commit is contained in:
commit
009109381f
|
@ -1,16 +1,54 @@
|
||||||
Status:
|
jclouds Google Compute Engine Provider
|
||||||
|
======
|
||||||
|
|
||||||
All the private apis are implemented and tested.
|
|
||||||
Snapshots are disabled because they are also disabled in GCE.
|
|
||||||
Just missing jcloud.compute glue code (coming soon!)
|
|
||||||
|
|
||||||
How to run the live tests:
|
Authenticating into the instances:
|
||||||
|
--------
|
||||||
|
|
||||||
Pre-requisites:
|
User:
|
||||||
A Google Api account with Google Compute Engine enabled
|
If no user is provided in GoogleComputeTemplateOptions when launching an instance by defaul "jclouds" is used.
|
||||||
The access pk (provided by google in PKCS12 format) in pem format.
|
|
||||||
|
|
||||||
running all tests:
|
Credential:
|
||||||
|
|
||||||
|
GCE uses exclusively ssh keys to login into instances.
|
||||||
|
In order for an instance to be sshable a public key must be installed. Public keys are installed if they are present in the project or instance's metatada.
|
||||||
|
|
||||||
|
For an instance to be ssable one of the following must happen:
|
||||||
|
1 - the project's metadata has an adequately built "sshKeys" entry and a corresponding private key is provided in GoogleComputeTemplateOptions when createNodesInGroup is called.
|
||||||
|
2 - an instance of GoogleComputeTemplateOptions with an adequate public and private key is provided.
|
||||||
|
|
||||||
|
NOTE: if methods 2 is chosen the global project keys will not be installed in the instance.
|
||||||
|
|
||||||
|
Please refer to Google's documentation on how to form valid project wide ssh keys metadata entries.
|
||||||
|
|
||||||
|
FAQ:
|
||||||
|
--------
|
||||||
|
|
||||||
|
* Q. What is the identity for GCE?
|
||||||
|
|
||||||
|
A. the identity is the developer email which can be obtained from the admin GUI. Its usually something in the form: <my account id>@developer.gserviceaccount.com
|
||||||
|
|
||||||
|
* Q. What is the crendential for GCE
|
||||||
|
|
||||||
|
A. the credential is a private key, in pem format. It can be extracted from the p12 keystore that is obtained when creating a "Service Account" (in the GUI: Google apis console > Api Access > Create another client ID > "Service Account"
|
||||||
|
|
||||||
|
* Q. How to convert a p12 keystore into a pem format jclouds-gce can handle:
|
||||||
|
|
||||||
|
A.
|
||||||
|
|
||||||
|
1. Convert the p12 file into pem format (it will ask for the keystore password, which is usually "notasecret"):
|
||||||
|
openssl pkcs12 -in <my_keystore>.p12 -out <my_keystore>.pem -nodes
|
||||||
|
|
||||||
|
2. Extract only the pk and remove passphrase
|
||||||
|
openssl rsa -in <my_keystore>.pem -out <my_key>.pem
|
||||||
|
|
||||||
|
The last file (<my_key>.pem) should contain the pk that needs to be passed to google-compute as a string literal property named "google-compute.credential".
|
||||||
|
|
||||||
|
|
||||||
|
Running the live tests:
|
||||||
|
--------
|
||||||
|
|
||||||
mvn clean install -Plive -Dtest.google-compute.identity=<my account>@developer.gserviceaccount.com -Dtest.google-compute.credential=<path to pk.pem>
|
mvn clean install -Plive -Dtest.google-compute.identity=<my account>@developer.gserviceaccount.com -Dtest.google-compute.credential=<path to pk.pem>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -63,6 +63,13 @@
|
||||||
<artifactId>jclouds-compute</artifactId>
|
<artifactId>jclouds-compute</artifactId>
|
||||||
<version>${jclouds.version}</version>
|
<version>${jclouds.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jclouds</groupId>
|
||||||
|
<artifactId>jclouds-compute</artifactId>
|
||||||
|
<version>${jclouds.version}</version>
|
||||||
|
<type>test-jar</type>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.jclouds</groupId>
|
<groupId>org.jclouds</groupId>
|
||||||
<artifactId>jclouds-core</artifactId>
|
<artifactId>jclouds-core</artifactId>
|
||||||
|
@ -76,6 +83,12 @@
|
||||||
<version>${jclouds.version}</version>
|
<version>${jclouds.version}</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jclouds.driver</groupId>
|
||||||
|
<artifactId>jclouds-sshj</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>ch.qos.logback</groupId>
|
<groupId>ch.qos.logback</groupId>
|
||||||
<artifactId>logback-classic</artifactId>
|
<artifactId>logback-classic</artifactId>
|
||||||
|
|
|
@ -18,14 +18,12 @@
|
||||||
*/
|
*/
|
||||||
package org.jclouds.googlecompute;
|
package org.jclouds.googlecompute;
|
||||||
|
|
||||||
import static org.jclouds.Constants.PROPERTY_SESSION_INTERVAL;
|
import com.google.common.collect.ImmutableSet;
|
||||||
import static org.jclouds.oauth.v2.config.OAuthProperties.AUDIENCE;
|
import com.google.common.reflect.TypeToken;
|
||||||
import static org.jclouds.oauth.v2.config.OAuthProperties.SIGNATURE_OR_MAC_ALGORITHM;
|
import com.google.inject.Module;
|
||||||
|
|
||||||
import java.net.URI;
|
|
||||||
import java.util.Properties;
|
|
||||||
|
|
||||||
import org.jclouds.apis.ApiMetadata;
|
import org.jclouds.apis.ApiMetadata;
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
import org.jclouds.googlecompute.compute.config.GoogleComputeServiceContextModule;
|
||||||
import org.jclouds.googlecompute.config.GoogleComputeParserModule;
|
import org.jclouds.googlecompute.config.GoogleComputeParserModule;
|
||||||
import org.jclouds.googlecompute.config.GoogleComputeRestClientModule;
|
import org.jclouds.googlecompute.config.GoogleComputeRestClientModule;
|
||||||
import org.jclouds.googlecompute.config.OAuthModuleWithoutTypeAdapters;
|
import org.jclouds.googlecompute.config.OAuthModuleWithoutTypeAdapters;
|
||||||
|
@ -33,9 +31,14 @@ import org.jclouds.oauth.v2.config.OAuthAuthenticationModule;
|
||||||
import org.jclouds.rest.RestContext;
|
import org.jclouds.rest.RestContext;
|
||||||
import org.jclouds.rest.internal.BaseRestApiMetadata;
|
import org.jclouds.rest.internal.BaseRestApiMetadata;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableSet;
|
import java.net.URI;
|
||||||
import com.google.common.reflect.TypeToken;
|
import java.util.Properties;
|
||||||
import com.google.inject.Module;
|
|
||||||
|
import static org.jclouds.Constants.PROPERTY_SESSION_INTERVAL;
|
||||||
|
import static org.jclouds.compute.config.ComputeServiceProperties.TEMPLATE;
|
||||||
|
import static org.jclouds.oauth.v2.config.OAuthProperties.AUDIENCE;
|
||||||
|
import static org.jclouds.oauth.v2.config.OAuthProperties.SIGNATURE_OR_MAC_ALGORITHM;
|
||||||
|
import static org.jclouds.reflect.Reflection2.typeToken;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implementation of {@link ApiMetadata} for GoogleCompute v1beta13 API
|
* Implementation of {@link ApiMetadata} for GoogleCompute v1beta13 API
|
||||||
|
@ -44,9 +47,8 @@ import com.google.inject.Module;
|
||||||
*/
|
*/
|
||||||
public class GoogleComputeApiMetadata extends BaseRestApiMetadata {
|
public class GoogleComputeApiMetadata extends BaseRestApiMetadata {
|
||||||
|
|
||||||
public static final TypeToken<RestContext<GoogleComputeApi, GoogleComputeAsyncApi>> CONTEXT_TOKEN = new TypeToken<RestContext<GoogleComputeApi, GoogleComputeAsyncApi>>() {
|
public static final TypeToken<RestContext<GoogleComputeApi, GoogleComputeAsyncApi>> CONTEXT_TOKEN = new
|
||||||
private static final long serialVersionUID = 1L;
|
TypeToken<RestContext<GoogleComputeApi, GoogleComputeAsyncApi>>() {};
|
||||||
};
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Builder toBuilder() {
|
public Builder toBuilder() {
|
||||||
|
@ -67,6 +69,10 @@ public class GoogleComputeApiMetadata extends BaseRestApiMetadata {
|
||||||
properties.put(AUDIENCE, "https://accounts.google.com/o/oauth2/token");
|
properties.put(AUDIENCE, "https://accounts.google.com/o/oauth2/token");
|
||||||
properties.put(SIGNATURE_OR_MAC_ALGORITHM, "RS256");
|
properties.put(SIGNATURE_OR_MAC_ALGORITHM, "RS256");
|
||||||
properties.put(PROPERTY_SESSION_INTERVAL, 3600);
|
properties.put(PROPERTY_SESSION_INTERVAL, 3600);
|
||||||
|
properties.setProperty(TEMPLATE, "osFamily=GCEL,osVersionMatches=1[012].[01][04],locationId=us-central1-a," +
|
||||||
|
"loginUser=jclouds");
|
||||||
|
properties.put(GoogleComputeConstants.OPERATION_COMPLETE_INTERVAL, 500);
|
||||||
|
properties.put(GoogleComputeConstants.OPERATION_COMPLETE_TIMEOUT, 600000);
|
||||||
return properties;
|
return properties;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -82,11 +88,14 @@ public class GoogleComputeApiMetadata extends BaseRestApiMetadata {
|
||||||
.version("v1beta13")
|
.version("v1beta13")
|
||||||
.defaultEndpoint("https://www.googleapis.com/compute/v1beta13")
|
.defaultEndpoint("https://www.googleapis.com/compute/v1beta13")
|
||||||
.defaultProperties(GoogleComputeApiMetadata.defaultProperties())
|
.defaultProperties(GoogleComputeApiMetadata.defaultProperties())
|
||||||
|
.view(typeToken(ComputeServiceContext.class))
|
||||||
.defaultModules(ImmutableSet.<Class<? extends Module>>builder()
|
.defaultModules(ImmutableSet.<Class<? extends Module>>builder()
|
||||||
.add(GoogleComputeRestClientModule.class)
|
.add(GoogleComputeRestClientModule.class)
|
||||||
.add(GoogleComputeParserModule.class)
|
.add(GoogleComputeParserModule.class)
|
||||||
.add(OAuthAuthenticationModule.class)
|
.add(OAuthAuthenticationModule.class)
|
||||||
.add(OAuthModuleWithoutTypeAdapters.class).build());
|
.add(OAuthModuleWithoutTypeAdapters.class)
|
||||||
|
.add(GoogleComputeServiceContextModule.class)
|
||||||
|
.build());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,21 +19,40 @@
|
||||||
|
|
||||||
package org.jclouds.googlecompute;
|
package org.jclouds.googlecompute;
|
||||||
|
|
||||||
|
import com.google.common.annotations.Beta;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.domain.LocationBuilder;
|
||||||
|
import org.jclouds.domain.LocationScope;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author David Alves
|
* @author David Alves
|
||||||
*/
|
*/
|
||||||
public interface GoogleComputeConstants {
|
public interface GoogleComputeConstants {
|
||||||
|
|
||||||
|
public static final String GOOGLE_PROVIDER_NAME = "google-compute";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The name of the project that keeps public resources.
|
||||||
|
*/
|
||||||
|
public static final String GOOGLE_PROJECT = "google";
|
||||||
|
|
||||||
public static final String COMPUTE_SCOPE = "https://www.googleapis.com/auth/compute";
|
public static final String COMPUTE_SCOPE = "https://www.googleapis.com/auth/compute";
|
||||||
|
|
||||||
public static final String COMPUTE_READONLY_SCOPE = "https://www.googleapis.com/auth/compute.readonly";
|
public static final String COMPUTE_READONLY_SCOPE = "https://www.googleapis.com/auth/compute.readonly";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* TODO storage scopes should be moved to the GCS api
|
* The total time, in msecs, to wait for an operation to complete.
|
||||||
*/
|
*/
|
||||||
|
@Beta
|
||||||
|
public static final String OPERATION_COMPLETE_TIMEOUT = "jclouds.google-compute.operation-complete-timeout";
|
||||||
|
|
||||||
public static final String STORAGE_WRITEONLY_SCOPE = "https://www.googleapis.com/auth/devstorage.write_only";
|
/**
|
||||||
|
* The interval, in msecs, between calls to check whether an operation has completed.
|
||||||
|
*/
|
||||||
|
@Beta
|
||||||
|
public static final String OPERATION_COMPLETE_INTERVAL = "jclouds.google-compute.operation-complete-interval";
|
||||||
|
|
||||||
public static final String STORAGE_READONLY_SCOPE = "https://www.googleapis.com/auth/devstorage.read_only";
|
public static final Location GOOGLE_PROVIDER_LOCATION = new LocationBuilder().scope(LocationScope.PROVIDER).id
|
||||||
|
(GOOGLE_PROVIDER_NAME).description(GOOGLE_PROVIDER_NAME).build();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,183 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Optional;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.util.concurrent.ListeningExecutorService;
|
||||||
|
import org.jclouds.Constants;
|
||||||
|
import org.jclouds.collect.Memoized;
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
import org.jclouds.compute.callables.RunScriptOnNode;
|
||||||
|
import org.jclouds.compute.domain.Hardware;
|
||||||
|
import org.jclouds.compute.domain.Image;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.domain.TemplateBuilder;
|
||||||
|
import org.jclouds.compute.extensions.ImageExtension;
|
||||||
|
import org.jclouds.compute.functions.GroupNamingConvention;
|
||||||
|
import org.jclouds.compute.internal.BaseComputeService;
|
||||||
|
import org.jclouds.compute.internal.PersistNodeCredentials;
|
||||||
|
import org.jclouds.compute.options.TemplateOptions;
|
||||||
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
|
import org.jclouds.compute.strategy.CreateNodesInGroupThenAddToSet;
|
||||||
|
import org.jclouds.compute.strategy.DestroyNodeStrategy;
|
||||||
|
import org.jclouds.compute.strategy.GetImageStrategy;
|
||||||
|
import org.jclouds.compute.strategy.GetNodeMetadataStrategy;
|
||||||
|
import org.jclouds.compute.strategy.InitializeRunScriptOnNodeOrPlaceInBadMap;
|
||||||
|
import org.jclouds.compute.strategy.ListNodesStrategy;
|
||||||
|
import org.jclouds.compute.strategy.RebootNodeStrategy;
|
||||||
|
import org.jclouds.compute.strategy.ResumeNodeStrategy;
|
||||||
|
import org.jclouds.compute.strategy.SuspendNodeStrategy;
|
||||||
|
import org.jclouds.domain.Credentials;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeApi;
|
||||||
|
import org.jclouds.googlecompute.compute.options.GoogleComputeTemplateOptions;
|
||||||
|
import org.jclouds.googlecompute.config.UserProject;
|
||||||
|
import org.jclouds.googlecompute.domain.Operation;
|
||||||
|
import org.jclouds.http.HttpResponse;
|
||||||
|
import org.jclouds.scriptbuilder.functions.InitAdminAccess;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import javax.inject.Provider;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static java.util.concurrent.TimeUnit.MILLISECONDS;
|
||||||
|
import static org.jclouds.compute.config.ComputeServiceProperties.TIMEOUT_NODE_RUNNING;
|
||||||
|
import static org.jclouds.compute.config.ComputeServiceProperties.TIMEOUT_NODE_SUSPENDED;
|
||||||
|
import static org.jclouds.compute.config.ComputeServiceProperties.TIMEOUT_NODE_TERMINATED;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_INTERVAL;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_TIMEOUT;
|
||||||
|
import static org.jclouds.util.Predicates2.retry;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class GoogleComputeService extends BaseComputeService {
|
||||||
|
|
||||||
|
private final Function<Set<? extends NodeMetadata>, Set<String>> findOrphanedGroups;
|
||||||
|
private final GroupNamingConvention.Factory namingConvention;
|
||||||
|
private final GoogleComputeApi api;
|
||||||
|
private final Supplier<String> project;
|
||||||
|
private final Predicate<AtomicReference<Operation>> operationDonePredicate;
|
||||||
|
private final long operationCompleteCheckInterval;
|
||||||
|
private final long operationCompleteCheckTimeout;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
protected GoogleComputeService(ComputeServiceContext context,
|
||||||
|
Map<String, Credentials> credentialStore,
|
||||||
|
@Memoized Supplier<Set<? extends Image>> images,
|
||||||
|
@Memoized Supplier<Set<? extends Hardware>> hardwareProfiles,
|
||||||
|
@Memoized Supplier<Set<? extends Location>> locations,
|
||||||
|
ListNodesStrategy listNodesStrategy,
|
||||||
|
GetImageStrategy getImageStrategy,
|
||||||
|
GetNodeMetadataStrategy getNodeMetadataStrategy,
|
||||||
|
CreateNodesInGroupThenAddToSet runNodesAndAddToSetStrategy,
|
||||||
|
RebootNodeStrategy rebootNodeStrategy,
|
||||||
|
DestroyNodeStrategy destroyNodeStrategy,
|
||||||
|
ResumeNodeStrategy resumeNodeStrategy,
|
||||||
|
SuspendNodeStrategy suspendNodeStrategy,
|
||||||
|
Provider<TemplateBuilder> templateBuilderProvider,
|
||||||
|
@Named("DEFAULT") Provider<TemplateOptions> templateOptionsProvider,
|
||||||
|
@Named(TIMEOUT_NODE_RUNNING) Predicate<AtomicReference<NodeMetadata>> nodeRunning,
|
||||||
|
@Named(TIMEOUT_NODE_TERMINATED) Predicate<AtomicReference<NodeMetadata>>
|
||||||
|
nodeTerminated,
|
||||||
|
@Named(TIMEOUT_NODE_SUSPENDED)
|
||||||
|
Predicate<AtomicReference<NodeMetadata>> nodeSuspended,
|
||||||
|
InitializeRunScriptOnNodeOrPlaceInBadMap.Factory initScriptRunnerFactory,
|
||||||
|
InitAdminAccess initAdminAccess,
|
||||||
|
RunScriptOnNode.Factory runScriptOnNodeFactory,
|
||||||
|
PersistNodeCredentials persistNodeCredentials,
|
||||||
|
ComputeServiceConstants.Timeouts timeouts,
|
||||||
|
@Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
|
||||||
|
Optional<ImageExtension> imageExtension,
|
||||||
|
Function<Set<? extends NodeMetadata>, Set<String>> findOrphanedGroups,
|
||||||
|
GroupNamingConvention.Factory namingConvention,
|
||||||
|
GoogleComputeApi api,
|
||||||
|
@UserProject Supplier<String> project,
|
||||||
|
Predicate<AtomicReference<Operation>> operationDonePredicate,
|
||||||
|
@Named(OPERATION_COMPLETE_INTERVAL) Long operationCompleteCheckInterval,
|
||||||
|
@Named(OPERATION_COMPLETE_TIMEOUT) Long operationCompleteCheckTimeout) {
|
||||||
|
|
||||||
|
super(context, credentialStore, images, hardwareProfiles, locations, listNodesStrategy, getImageStrategy,
|
||||||
|
getNodeMetadataStrategy, runNodesAndAddToSetStrategy, rebootNodeStrategy, destroyNodeStrategy,
|
||||||
|
resumeNodeStrategy, suspendNodeStrategy, templateBuilderProvider, templateOptionsProvider, nodeRunning,
|
||||||
|
nodeTerminated, nodeSuspended, initScriptRunnerFactory, initAdminAccess, runScriptOnNodeFactory,
|
||||||
|
persistNodeCredentials, timeouts, userExecutor, imageExtension);
|
||||||
|
this.findOrphanedGroups = checkNotNull(findOrphanedGroups, "find orphaned groups function");
|
||||||
|
this.namingConvention = checkNotNull(namingConvention, "naming convention factory");
|
||||||
|
this.api = checkNotNull(api, "google compute api");
|
||||||
|
this.project = checkNotNull(project, "user project name");
|
||||||
|
this.operationDonePredicate = checkNotNull(operationDonePredicate, "operation completed predicate");
|
||||||
|
this.operationCompleteCheckInterval = checkNotNull(operationCompleteCheckInterval,
|
||||||
|
"operation completed check interval");
|
||||||
|
this.operationCompleteCheckTimeout = checkNotNull(operationCompleteCheckTimeout,
|
||||||
|
"operation completed check timeout");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected synchronized void cleanUpIncidentalResourcesOfDeadNodes(Set<? extends NodeMetadata> deadNodes) {
|
||||||
|
Set<String> orphanedGroups = findOrphanedGroups.apply(deadNodes);
|
||||||
|
for (String orphanedGroup : orphanedGroups) {
|
||||||
|
cleanUpNetworksAndFirewallsForGroup(orphanedGroup);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected void cleanUpNetworksAndFirewallsForGroup(String groupName) {
|
||||||
|
String resourceName = namingConvention.create().sharedNameForGroup(groupName);
|
||||||
|
AtomicReference<Operation> operation = new AtomicReference<Operation>(api.getFirewallApiForProject(project.get())
|
||||||
|
.delete(resourceName));
|
||||||
|
|
||||||
|
retry(operationDonePredicate, operationCompleteCheckTimeout, operationCompleteCheckInterval,
|
||||||
|
MILLISECONDS).apply(operation);
|
||||||
|
|
||||||
|
if (operation.get().getHttpError().isPresent()) {
|
||||||
|
HttpResponse response = operation.get().getHttpError().get();
|
||||||
|
logger.warn("delete orphaned firewall failed. Http Error Code: " + response.getStatusCode() +
|
||||||
|
" HttpError: " + response.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
operation = new AtomicReference<Operation>(api.getNetworkApiForProject(project.get()).delete(resourceName));
|
||||||
|
|
||||||
|
retry(operationDonePredicate, operationCompleteCheckTimeout, operationCompleteCheckInterval,
|
||||||
|
MILLISECONDS).apply(operation);
|
||||||
|
|
||||||
|
if (operation.get().getHttpError().isPresent()) {
|
||||||
|
HttpResponse response = operation.get().getHttpError().get();
|
||||||
|
logger.warn("delete orphaned network failed. Http Error Code: " + response.getStatusCode() +
|
||||||
|
" HttpError: " + response.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns template options, except of type {@link GoogleComputeTemplateOptions}.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions templateOptions() {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.templateOptions());
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,252 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Objects;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.common.util.concurrent.UncheckedTimeoutException;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import org.jclouds.compute.ComputeServiceAdapter;
|
||||||
|
import org.jclouds.compute.domain.Hardware;
|
||||||
|
import org.jclouds.compute.domain.Template;
|
||||||
|
import org.jclouds.compute.options.TemplateOptions;
|
||||||
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
|
import org.jclouds.domain.LoginCredentials;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeApi;
|
||||||
|
import org.jclouds.googlecompute.compute.options.GoogleComputeTemplateOptions;
|
||||||
|
import org.jclouds.googlecompute.config.UserProject;
|
||||||
|
import org.jclouds.googlecompute.domain.Image;
|
||||||
|
import org.jclouds.googlecompute.domain.Instance;
|
||||||
|
import org.jclouds.googlecompute.domain.InstanceTemplate;
|
||||||
|
import org.jclouds.googlecompute.domain.MachineType;
|
||||||
|
import org.jclouds.googlecompute.domain.Operation;
|
||||||
|
import org.jclouds.googlecompute.domain.Zone;
|
||||||
|
import org.jclouds.http.HttpResponse;
|
||||||
|
import org.jclouds.logging.Logger;
|
||||||
|
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static com.google.common.base.Preconditions.checkState;
|
||||||
|
import static java.util.concurrent.TimeUnit.MILLISECONDS;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.GOOGLE_PROJECT;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_INTERVAL;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_TIMEOUT;
|
||||||
|
import static org.jclouds.googlecompute.domain.Instance.NetworkInterface.AccessConfig.Type;
|
||||||
|
import static org.jclouds.util.Predicates2.retry;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class GoogleComputeServiceAdapter implements ComputeServiceAdapter<Instance, MachineType, Image, Zone> {
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
@Named(ComputeServiceConstants.COMPUTE_LOGGER)
|
||||||
|
protected Logger logger = Logger.NULL;
|
||||||
|
|
||||||
|
private final GoogleComputeApi api;
|
||||||
|
private final Supplier<String> userProject;
|
||||||
|
private final Function<TemplateOptions, ImmutableMap.Builder<String, String>> metatadaFromTemplateOptions;
|
||||||
|
private final Predicate<AtomicReference<Operation>> retryOperationDonePredicate;
|
||||||
|
private final long operationCompleteCheckInterval;
|
||||||
|
private final long operationCompleteCheckTimeout;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public GoogleComputeServiceAdapter(GoogleComputeApi api,
|
||||||
|
@UserProject Supplier<String> userProject,
|
||||||
|
Function<TemplateOptions,
|
||||||
|
ImmutableMap.Builder<String, String>> metatadaFromTemplateOptions,
|
||||||
|
Predicate<AtomicReference<Operation>> operationDonePredicate,
|
||||||
|
@Named(OPERATION_COMPLETE_INTERVAL) Long operationCompleteCheckInterval,
|
||||||
|
@Named(OPERATION_COMPLETE_TIMEOUT) Long operationCompleteCheckTimeout) {
|
||||||
|
this.api = checkNotNull(api, "google compute api");
|
||||||
|
this.userProject = checkNotNull(userProject, "user project name");
|
||||||
|
this.metatadaFromTemplateOptions = checkNotNull(metatadaFromTemplateOptions,
|
||||||
|
"metadata from template options function");
|
||||||
|
this.operationCompleteCheckInterval = checkNotNull(operationCompleteCheckInterval,
|
||||||
|
"operation completed check interval");
|
||||||
|
this.operationCompleteCheckTimeout = checkNotNull(operationCompleteCheckTimeout,
|
||||||
|
"operation completed check timeout");
|
||||||
|
this.retryOperationDonePredicate = retry(operationDonePredicate, operationCompleteCheckTimeout,
|
||||||
|
operationCompleteCheckInterval, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public NodeAndInitialCredentials<Instance> createNodeWithGroupEncodedIntoName(
|
||||||
|
final String group, final String name, Template template) {
|
||||||
|
|
||||||
|
checkNotNull(template, "template");
|
||||||
|
|
||||||
|
GoogleComputeTemplateOptions options = GoogleComputeTemplateOptions.class.cast(template.getOptions()).clone();
|
||||||
|
checkState(options.getNetwork().isPresent(), "network was not present in template options");
|
||||||
|
Hardware hardware = checkNotNull(template.getHardware(), "hardware must be set");
|
||||||
|
URI machineType = checkNotNull(hardware.getUri(), "hardware uri must be set");
|
||||||
|
|
||||||
|
InstanceTemplate instanceTemplate = InstanceTemplate.builder()
|
||||||
|
.forMachineType(machineType);
|
||||||
|
|
||||||
|
if (options.isEnableNat()) {
|
||||||
|
instanceTemplate.addNetworkInterface(options.getNetwork().get(), Type.ONE_TO_ONE_NAT);
|
||||||
|
} else {
|
||||||
|
instanceTemplate.addNetworkInterface(options.getNetwork().get());
|
||||||
|
}
|
||||||
|
|
||||||
|
LoginCredentials credentials = getFromImageAndOverrideIfRequired(template.getImage(), options);
|
||||||
|
|
||||||
|
ImmutableMap.Builder<String, String> metadataBuilder = metatadaFromTemplateOptions.apply(options);
|
||||||
|
instanceTemplate.metadata(metadataBuilder.build());
|
||||||
|
instanceTemplate.tags(options.getTags());
|
||||||
|
instanceTemplate.serviceAccounts(options.getServiceAccounts());
|
||||||
|
instanceTemplate.image(checkNotNull(template.getImage().getUri(), "image URI is null"));
|
||||||
|
|
||||||
|
Operation operation = api.getInstanceApiForProject(userProject.get())
|
||||||
|
.createInZone(name, instanceTemplate, template.getLocation().getId());
|
||||||
|
|
||||||
|
if (options.shouldBlockUntilRunning()) {
|
||||||
|
waitOperationDone(operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
// some times the newly created instances are not immediately returned
|
||||||
|
AtomicReference<Instance> instance = new AtomicReference<Instance>();
|
||||||
|
|
||||||
|
retry(new Predicate<AtomicReference<Instance>>() {
|
||||||
|
@Override
|
||||||
|
public boolean apply(AtomicReference<Instance> input) {
|
||||||
|
input.set(api.getInstanceApiForProject(userProject.get()).get(name));
|
||||||
|
return input.get() != null;
|
||||||
|
}
|
||||||
|
}, operationCompleteCheckTimeout, operationCompleteCheckInterval, MILLISECONDS).apply(instance);
|
||||||
|
|
||||||
|
return new NodeAndInitialCredentials<Instance>(instance.get(), name, credentials);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterable<MachineType> listHardwareProfiles() {
|
||||||
|
return api.getMachineTypeApiForProject(userProject.get()).list().concat();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterable<Image> listImages() {
|
||||||
|
return ImmutableSet.<Image>builder()
|
||||||
|
.addAll(api.getImageApiForProject(userProject.get()).list().concat())
|
||||||
|
.addAll(api.getImageApiForProject(GOOGLE_PROJECT).list().concat())
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Image getImage(String id) {
|
||||||
|
return Objects.firstNonNull(api.getImageApiForProject(userProject.get()).get(id),
|
||||||
|
api.getImageApiForProject(GOOGLE_PROJECT).get(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterable<Zone> listLocations() {
|
||||||
|
return api.getZoneApiForProject(userProject.get()).list().concat();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Instance getNode(String name) {
|
||||||
|
return api.getInstanceApiForProject(userProject.get()).get(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterable<Instance> listNodes() {
|
||||||
|
return api.getInstanceApiForProject(userProject.get()).list().concat();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroyNode(final String name) {
|
||||||
|
waitOperationDone(api.getInstanceApiForProject(userProject.get()).delete(name));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void rebootNode(String name) {
|
||||||
|
throw new UnsupportedOperationException("reboot is not supported by GCE");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resumeNode(String name) {
|
||||||
|
throw new UnsupportedOperationException("resume is not supported by GCE");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void suspendNode(String name) {
|
||||||
|
throw new UnsupportedOperationException("suspend is not supported by GCE");
|
||||||
|
}
|
||||||
|
|
||||||
|
private LoginCredentials getFromImageAndOverrideIfRequired(org.jclouds.compute.domain.Image image,
|
||||||
|
GoogleComputeTemplateOptions options) {
|
||||||
|
LoginCredentials defaultCredentials = image.getDefaultCredentials();
|
||||||
|
String[] keys = defaultCredentials.getPrivateKey().split(":");
|
||||||
|
String publicKey = keys[0];
|
||||||
|
String privateKey = keys[1];
|
||||||
|
|
||||||
|
LoginCredentials.Builder credentialsBuilder = defaultCredentials.toBuilder();
|
||||||
|
credentialsBuilder.privateKey(privateKey);
|
||||||
|
|
||||||
|
// LoginCredentials from image stores the public key along with the private key in the privateKey field
|
||||||
|
// @see GoogleComputePopulateDefaultLoginCredentialsForImageStrategy
|
||||||
|
// so if options doesn't have a public key set we set it from the default
|
||||||
|
if (options.getPublicKey() == null) {
|
||||||
|
options.authorizePublicKey(publicKey);
|
||||||
|
}
|
||||||
|
if (options.hasLoginPrivateKeyOption()) {
|
||||||
|
credentialsBuilder.privateKey(options.getPrivateKey());
|
||||||
|
}
|
||||||
|
if (options.getLoginUser() != null) {
|
||||||
|
credentialsBuilder.identity(options.getLoginUser());
|
||||||
|
}
|
||||||
|
if (options.hasLoginPasswordOption()) {
|
||||||
|
credentialsBuilder.password(options.getLoginPassword());
|
||||||
|
}
|
||||||
|
if (options.shouldAuthenticateSudo() != null) {
|
||||||
|
credentialsBuilder.authenticateSudo(options.shouldAuthenticateSudo());
|
||||||
|
}
|
||||||
|
LoginCredentials credentials = credentialsBuilder.build();
|
||||||
|
options.overrideLoginCredentials(credentials);
|
||||||
|
return credentials;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void waitOperationDone(Operation operation) {
|
||||||
|
AtomicReference<Operation> operationRef = new AtomicReference<Operation>(operation);
|
||||||
|
|
||||||
|
// wait for the operation to complete
|
||||||
|
if (!retryOperationDonePredicate.apply(operationRef)) {
|
||||||
|
throw new UncheckedTimeoutException("operation did not reach DONE state" + operationRef.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if the operation failed
|
||||||
|
if (operationRef.get().getHttpError().isPresent()) {
|
||||||
|
HttpResponse response = operationRef.get().getHttpError().get();
|
||||||
|
throw new IllegalStateException("operation failed. Http Error Code: " + response.getStatusCode() +
|
||||||
|
" HttpError: " + response.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,196 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.config;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Optional;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.inject.Injector;
|
||||||
|
import com.google.inject.Provides;
|
||||||
|
import com.google.inject.TypeLiteral;
|
||||||
|
import org.jclouds.collect.Memoized;
|
||||||
|
import org.jclouds.compute.ComputeService;
|
||||||
|
import org.jclouds.compute.ComputeServiceAdapter;
|
||||||
|
import org.jclouds.compute.config.ComputeServiceAdapterContextModule;
|
||||||
|
import org.jclouds.compute.domain.Hardware;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.extensions.ImageExtension;
|
||||||
|
import org.jclouds.compute.options.TemplateOptions;
|
||||||
|
import org.jclouds.compute.strategy.PopulateDefaultLoginCredentialsForImageStrategy;
|
||||||
|
import org.jclouds.compute.strategy.PrioritizeCredentialsFromTemplate;
|
||||||
|
import org.jclouds.compute.strategy.impl.CreateNodesWithGroupEncodedIntoNameThenAddToSet;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeApi;
|
||||||
|
import org.jclouds.googlecompute.compute.GoogleComputeService;
|
||||||
|
import org.jclouds.googlecompute.compute.GoogleComputeServiceAdapter;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.BuildInstanceMetadata;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.GoogleComputeImageToImage;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.InstanceToNodeMetadata;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.MachineTypeToHardware;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.OrphanedGroupsFromDeadNodes;
|
||||||
|
import org.jclouds.googlecompute.compute.functions.ZoneToLocation;
|
||||||
|
import org.jclouds.googlecompute.compute.options.GoogleComputeTemplateOptions;
|
||||||
|
import org.jclouds.googlecompute.compute.predicates.AllNodesInGroupTerminated;
|
||||||
|
import org.jclouds.googlecompute.compute.strategy.ApplyTemplateOptionsCreateNodesWithGroupEncodedIntoNameThenAddToSet;
|
||||||
|
import org.jclouds.googlecompute.compute.strategy.GoogleComputePopulateDefaultLoginCredentialsForImageStrategy;
|
||||||
|
import org.jclouds.googlecompute.compute.strategy.UseNodeCredentialsButOverrideFromTemplate;
|
||||||
|
import org.jclouds.googlecompute.config.UserProject;
|
||||||
|
import org.jclouds.googlecompute.domain.Image;
|
||||||
|
import org.jclouds.googlecompute.domain.Instance;
|
||||||
|
import org.jclouds.googlecompute.domain.MachineType;
|
||||||
|
import org.jclouds.googlecompute.domain.Zone;
|
||||||
|
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import static com.google.common.collect.Iterables.transform;
|
||||||
|
import static com.google.common.collect.Maps.uniqueIndex;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class GoogleComputeServiceContextModule
|
||||||
|
extends ComputeServiceAdapterContextModule<Instance, MachineType, Image, Zone> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void configure() {
|
||||||
|
super.configure();
|
||||||
|
|
||||||
|
bind(ComputeService.class).to(GoogleComputeService.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<ComputeServiceAdapter<Instance, MachineType, Image, Zone>>() {})
|
||||||
|
.to(GoogleComputeServiceAdapter.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<Instance, NodeMetadata>>() {})
|
||||||
|
.to(InstanceToNodeMetadata.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<MachineType, Hardware>>() {})
|
||||||
|
.to(MachineTypeToHardware.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<Image, org.jclouds.compute.domain.Image>>() {})
|
||||||
|
.to(GoogleComputeImageToImage.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<Zone, Location>>() {})
|
||||||
|
.to(ZoneToLocation.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<TemplateOptions, ImmutableMap.Builder<String, String>>>() {})
|
||||||
|
.to(BuildInstanceMetadata.class);
|
||||||
|
|
||||||
|
bind(PopulateDefaultLoginCredentialsForImageStrategy.class)
|
||||||
|
.to(GoogleComputePopulateDefaultLoginCredentialsForImageStrategy.class);
|
||||||
|
|
||||||
|
bind(CreateNodesWithGroupEncodedIntoNameThenAddToSet.class).to(
|
||||||
|
ApplyTemplateOptionsCreateNodesWithGroupEncodedIntoNameThenAddToSet.class);
|
||||||
|
|
||||||
|
bind(TemplateOptions.class).to(GoogleComputeTemplateOptions.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Function<Set<? extends NodeMetadata>, Set<String>>>() {})
|
||||||
|
.to(OrphanedGroupsFromDeadNodes.class);
|
||||||
|
|
||||||
|
bind(new TypeLiteral<Predicate<String>>() {}).to(AllNodesInGroupTerminated.class);
|
||||||
|
|
||||||
|
bind(PrioritizeCredentialsFromTemplate.class).to(UseNodeCredentialsButOverrideFromTemplate.class);
|
||||||
|
|
||||||
|
install(new LocationsFromComputeServiceAdapterModule<Instance, MachineType, Image, Zone>() {});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Provides
|
||||||
|
@Singleton
|
||||||
|
@Memoized
|
||||||
|
public Supplier<Map<URI, ? extends org.jclouds.compute.domain.Image>> provideImagesMap(
|
||||||
|
final Supplier<Set<? extends org.jclouds.compute.domain.Image>> images) {
|
||||||
|
return new Supplier<Map<URI, ? extends org.jclouds.compute.domain.Image>>() {
|
||||||
|
@Override
|
||||||
|
public Map<URI, ? extends org.jclouds.compute.domain.Image> get() {
|
||||||
|
return uniqueIndex(images.get(), new Function<org.jclouds.compute.domain.Image, URI>() {
|
||||||
|
@Override
|
||||||
|
public URI apply(org.jclouds.compute.domain.Image input) {
|
||||||
|
return input.getUri();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Provides
|
||||||
|
@Singleton
|
||||||
|
@Memoized
|
||||||
|
public Supplier<Map<URI, ? extends Hardware>> provideHardwaresMap(
|
||||||
|
final Supplier<Set<? extends Hardware>> hardwares) {
|
||||||
|
return new Supplier<Map<URI, ? extends Hardware>>() {
|
||||||
|
@Override
|
||||||
|
public Map<URI, ? extends Hardware> get() {
|
||||||
|
return uniqueIndex(hardwares.get(), new Function<Hardware, URI>() {
|
||||||
|
@Override
|
||||||
|
public URI apply(Hardware input) {
|
||||||
|
return input.getUri();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Provides
|
||||||
|
@Singleton
|
||||||
|
@Memoized
|
||||||
|
public Supplier<Map<URI, ? extends Location>> provideLocations(
|
||||||
|
final GoogleComputeApi api, final Function<Zone, Location> zoneToLocation,
|
||||||
|
final @UserProject Supplier<String> userProject) {
|
||||||
|
return new Supplier<Map<URI, ? extends Location>>() {
|
||||||
|
@Override
|
||||||
|
public Map<URI, ? extends Location> get() {
|
||||||
|
return uniqueIndex(transform(api.getZoneApiForProject(userProject.get()).list().concat(), zoneToLocation),
|
||||||
|
new Function<Location, URI>() {
|
||||||
|
@Override
|
||||||
|
public URI apply(Location input) {
|
||||||
|
return (URI) input.getMetadata().get("selfLink");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Optional<ImageExtension> provideImageExtension(Injector i) {
|
||||||
|
return Optional.absent();
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public static final Map<Instance.Status, NodeMetadata.Status> toPortableNodeStatus =
|
||||||
|
ImmutableMap.<Instance.Status, NodeMetadata.Status>builder()
|
||||||
|
.put(Instance.Status.PROVISIONING, NodeMetadata.Status.PENDING)
|
||||||
|
.put(Instance.Status.STAGING, NodeMetadata.Status.PENDING)
|
||||||
|
.put(Instance.Status.RUNNING, NodeMetadata.Status.RUNNING)
|
||||||
|
.put(Instance.Status.STOPPING, NodeMetadata.Status.PENDING)
|
||||||
|
.put(Instance.Status.STOPPED, NodeMetadata.Status.SUSPENDED)
|
||||||
|
.put(Instance.Status.TERMINATED, NodeMetadata.Status.TERMINATED).build();
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
protected Map<Instance.Status, NodeMetadata.Status> toPortableNodeStatus() {
|
||||||
|
return toPortableNodeStatus;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import org.jclouds.compute.options.TemplateOptions;
|
||||||
|
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static java.lang.String.format;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepares metadata from the provided TemplateOptions
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Singleton
|
||||||
|
public class BuildInstanceMetadata implements Function<TemplateOptions, ImmutableMap.Builder<String, String>> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ImmutableMap.Builder apply(TemplateOptions input) {
|
||||||
|
ImmutableMap.Builder<String, String> builder = ImmutableMap.builder();
|
||||||
|
if (input.getPublicKey() != null) {
|
||||||
|
builder.put("sshKeys", format("%s:%s %s@localhost", checkNotNull(input.getLoginUser(),
|
||||||
|
"loginUser cannot be null"), input.getPublicKey(), input.getLoginUser()));
|
||||||
|
}
|
||||||
|
builder.putAll(input.getUserMetadata());
|
||||||
|
return builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,83 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import org.jclouds.compute.domain.ImageBuilder;
|
||||||
|
import org.jclouds.compute.domain.OperatingSystem;
|
||||||
|
import org.jclouds.compute.domain.OsFamily;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeConstants;
|
||||||
|
import org.jclouds.googlecompute.domain.Image;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static com.google.common.base.Joiner.on;
|
||||||
|
import static com.google.common.collect.Iterables.getLast;
|
||||||
|
import static com.google.common.collect.Iterables.limit;
|
||||||
|
import static com.google.common.collect.Iterables.skip;
|
||||||
|
import static org.jclouds.compute.domain.Image.Status;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.GOOGLE_PROVIDER_LOCATION;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a google compute domain specific image to a generic Image object.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class GoogleComputeImageToImage implements Function<Image, org.jclouds.compute.domain.Image> {
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public org.jclouds.compute.domain.Image apply(Image image) {
|
||||||
|
ImageBuilder builder = new ImageBuilder()
|
||||||
|
.id(image.getName())
|
||||||
|
.name(image.getName())
|
||||||
|
.providerId(image.getId())
|
||||||
|
.description(image.getDescription().orNull())
|
||||||
|
.status(Status.AVAILABLE)
|
||||||
|
.location(GOOGLE_PROVIDER_LOCATION)
|
||||||
|
.uri(image.getSelfLink());
|
||||||
|
|
||||||
|
List<String> splits = Lists.newArrayList(image.getName().split("-"));
|
||||||
|
OperatingSystem.Builder osBuilder = defaultOperatingSystem(image);
|
||||||
|
if (splits == null || splits.size() == 0 || splits.size() < 3) {
|
||||||
|
return builder.operatingSystem(osBuilder.build()).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
OsFamily family = OsFamily.fromValue(splits.get(0));
|
||||||
|
if (family != OsFamily.UNRECOGNIZED) {
|
||||||
|
osBuilder.family(family);
|
||||||
|
}
|
||||||
|
|
||||||
|
String version = on(".").join(limit(skip(splits, 1), splits.size() - 2));
|
||||||
|
osBuilder.version(version);
|
||||||
|
|
||||||
|
builder.version(getLast(splits));
|
||||||
|
return builder.operatingSystem(osBuilder.build()).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private OperatingSystem.Builder defaultOperatingSystem(Image image) {
|
||||||
|
return OperatingSystem.builder()
|
||||||
|
.family(OsFamily.LINUX)
|
||||||
|
.is64Bit(true)
|
||||||
|
.description(image.getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,114 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.jclouds.collect.Memoized;
|
||||||
|
import org.jclouds.compute.domain.Hardware;
|
||||||
|
import org.jclouds.compute.domain.Image;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadataBuilder;
|
||||||
|
import org.jclouds.compute.functions.GroupNamingConvention;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.googlecompute.domain.Instance;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a google compute domain Instance into a generic NodeMetatada object.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class InstanceToNodeMetadata implements Function<Instance, NodeMetadata> {
|
||||||
|
|
||||||
|
private final Map<Instance.Status, NodeMetadata.Status> toPortableNodeStatus;
|
||||||
|
private final GroupNamingConvention nodeNamingConvention;
|
||||||
|
private final Supplier<Map<URI, ? extends Image>> images;
|
||||||
|
private final Supplier<Map<URI, ? extends Hardware>> hardwares;
|
||||||
|
private final Supplier<Map<URI, ? extends Location>> locations;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public InstanceToNodeMetadata(Map<Instance.Status, NodeMetadata.Status> toPortableNodeStatus,
|
||||||
|
GroupNamingConvention.Factory namingConvention,
|
||||||
|
@Memoized Supplier<Map<URI, ? extends Image>> images,
|
||||||
|
@Memoized Supplier<Map<URI, ? extends Hardware>> hardwares,
|
||||||
|
@Memoized Supplier<Map<URI, ? extends Location>> locations) {
|
||||||
|
this.toPortableNodeStatus = toPortableNodeStatus;
|
||||||
|
this.nodeNamingConvention = namingConvention.createWithoutPrefix();
|
||||||
|
this.images = images;
|
||||||
|
this.hardwares = hardwares;
|
||||||
|
this.locations = locations;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public NodeMetadata apply(Instance input) {
|
||||||
|
Map<URI, ? extends Image> imagesMap = images.get();
|
||||||
|
Image image = checkNotNull(imagesMap.get(checkNotNull(input.getImage(), "image")),
|
||||||
|
"no image for %s. images: %s", input.getImage(), imagesMap.values());
|
||||||
|
|
||||||
|
return new NodeMetadataBuilder()
|
||||||
|
.id(input.getName())
|
||||||
|
.name(input.getName())
|
||||||
|
.providerId(input.getId())
|
||||||
|
.hostname(input.getName())
|
||||||
|
.imageId(image.getId())
|
||||||
|
.location(checkNotNull(locations.get().get(input.getZone()), "location for %s", input.getZone()))
|
||||||
|
.hardware(checkNotNull(hardwares.get().get(input.getMachineType()), "hardware type for %s",
|
||||||
|
input.getMachineType().toString()))
|
||||||
|
.operatingSystem(image.getOperatingSystem())
|
||||||
|
.status(toPortableNodeStatus.get(input.getStatus()))
|
||||||
|
.tags(input.getTags())
|
||||||
|
.uri(input.getSelfLink())
|
||||||
|
.userMetadata(input.getMetadata())
|
||||||
|
.group(nodeNamingConvention.groupInUniqueNameOrNull(input.getName()))
|
||||||
|
.privateAddresses(collectPrivateAddresses(input))
|
||||||
|
.publicAddresses(collectPublicAddresses(input))
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<String> collectPrivateAddresses(Instance input) {
|
||||||
|
ImmutableSet.Builder<String> privateAddressesBuilder = ImmutableSet.builder();
|
||||||
|
for (Instance.NetworkInterface networkInterface : input.getNetworkInterfaces()) {
|
||||||
|
if (networkInterface.getNetworkIP().isPresent()) {
|
||||||
|
privateAddressesBuilder.add(networkInterface.getNetworkIP().get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return privateAddressesBuilder.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<String> collectPublicAddresses(Instance input) {
|
||||||
|
ImmutableSet.Builder<String> publicAddressesBuilder = ImmutableSet.builder();
|
||||||
|
for (Instance.NetworkInterface networkInterface : input.getNetworkInterfaces()) {
|
||||||
|
for (Instance.NetworkInterface.AccessConfig accessConfig : networkInterface.getAccessConfigs()) {
|
||||||
|
if (accessConfig.getNatIP().isPresent()) {
|
||||||
|
publicAddressesBuilder.add(accessConfig.getNatIP().get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return publicAddressesBuilder.build();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import org.jclouds.compute.domain.Hardware;
|
||||||
|
import org.jclouds.compute.domain.HardwareBuilder;
|
||||||
|
import org.jclouds.compute.domain.Processor;
|
||||||
|
import org.jclouds.compute.domain.Volume;
|
||||||
|
import org.jclouds.compute.domain.internal.VolumeImpl;
|
||||||
|
import org.jclouds.googlecompute.domain.MachineType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a google compute domain specific machine type to a generic Hardware object.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class MachineTypeToHardware implements Function<MachineType, Hardware> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Hardware apply(MachineType input) {
|
||||||
|
return new HardwareBuilder()
|
||||||
|
.id(input.getName())
|
||||||
|
.name(input.getName())
|
||||||
|
.hypervisor("kvm")
|
||||||
|
.processor(new Processor(input.getGuestCpus(), 1.0))
|
||||||
|
.providerId(input.getId())
|
||||||
|
.ram(input.getMemoryMb())
|
||||||
|
.uri(input.getSelfLink())
|
||||||
|
.volumes(collectVolumes(input))
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Iterable<Volume> collectVolumes(MachineType input) {
|
||||||
|
ImmutableSet.Builder<Volume> volumes = ImmutableSet.builder();
|
||||||
|
for (MachineType.EphemeralDisk disk : input.getEphemeralDisks()) {
|
||||||
|
volumes.add(new VolumeImpl(null, Volume.Type.LOCAL, new Integer(disk.getDiskGb()).floatValue(), null, true,
|
||||||
|
false));
|
||||||
|
}
|
||||||
|
return volumes.build();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Singleton
|
||||||
|
public class OrphanedGroupsFromDeadNodes implements Function<Set<? extends NodeMetadata>, Set<String>> {
|
||||||
|
|
||||||
|
private final Predicate<String> isOrphanedGroupPredicate;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public OrphanedGroupsFromDeadNodes(Predicate<String> isOrphanedGroupPredicate) {
|
||||||
|
this.isOrphanedGroupPredicate = isOrphanedGroupPredicate;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> apply(Set<? extends NodeMetadata> deadNodes) {
|
||||||
|
Set<String> groups = Sets.newLinkedHashSet();
|
||||||
|
for (NodeMetadata deadNode : deadNodes) {
|
||||||
|
groups.add(deadNode.getGroup());
|
||||||
|
}
|
||||||
|
Set<String> orphanedGroups = Sets.newLinkedHashSet();
|
||||||
|
for (String group : groups) {
|
||||||
|
if (isOrphanedGroupPredicate.apply(group)) {
|
||||||
|
orphanedGroups.add(group);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return orphanedGroups;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.domain.LocationBuilder;
|
||||||
|
import org.jclouds.domain.LocationScope;
|
||||||
|
import org.jclouds.googlecompute.domain.Zone;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.GOOGLE_PROVIDER_LOCATION;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms a google compute domain specific zone to a generic Zone object.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class ZoneToLocation implements Function<Zone, Location> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Location apply(Zone input) {
|
||||||
|
return new LocationBuilder()
|
||||||
|
.description(input.getDescription().orNull())
|
||||||
|
.metadata(ImmutableMap.of("selfLink", (Object) checkNotNull(input.getSelfLink(), "zone URI")))
|
||||||
|
.id(input.getName())
|
||||||
|
.scope(LocationScope.ZONE)
|
||||||
|
.parent(GOOGLE_PROVIDER_LOCATION)
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,280 @@
|
||||||
|
package org.jclouds.googlecompute.compute.options;
|
||||||
|
|
||||||
|
import com.google.common.base.Optional;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import org.jclouds.compute.options.TemplateOptions;
|
||||||
|
import org.jclouds.domain.LoginCredentials;
|
||||||
|
import org.jclouds.googlecompute.domain.Instance;
|
||||||
|
import org.jclouds.scriptbuilder.domain.Statement;
|
||||||
|
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import static com.google.common.base.Optional.fromNullable;
|
||||||
|
import static org.jclouds.googlecompute.domain.Instance.ServiceAccount;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Instance options specific to Google Compute Engine.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class GoogleComputeTemplateOptions extends TemplateOptions {
|
||||||
|
|
||||||
|
private Optional<URI> network = Optional.absent();
|
||||||
|
private Optional<String> networkName = Optional.absent();
|
||||||
|
private Set<Instance.ServiceAccount> serviceAccounts = Sets.newLinkedHashSet();
|
||||||
|
private boolean enableNat = true;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions clone() {
|
||||||
|
GoogleComputeTemplateOptions options = new GoogleComputeTemplateOptions();
|
||||||
|
copyTo(options);
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void copyTo(TemplateOptions to) {
|
||||||
|
super.copyTo(to);
|
||||||
|
if (to instanceof GoogleComputeTemplateOptions) {
|
||||||
|
GoogleComputeTemplateOptions eTo = GoogleComputeTemplateOptions.class.cast(to);
|
||||||
|
eTo.network(getNetwork().orNull());
|
||||||
|
eTo.network(getNetworkName().orNull());
|
||||||
|
eTo.serviceAccounts(getServiceAccounts());
|
||||||
|
eTo.enableNat(isEnableNat());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see #getNetworkName()
|
||||||
|
*/
|
||||||
|
public GoogleComputeTemplateOptions network(String networkName) {
|
||||||
|
this.networkName = fromNullable(networkName);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see #getNetwork()
|
||||||
|
*/
|
||||||
|
public GoogleComputeTemplateOptions network(URI network) {
|
||||||
|
this.network = fromNullable(network);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see #getServiceAccounts()
|
||||||
|
* @see ServiceAccount
|
||||||
|
*/
|
||||||
|
public GoogleComputeTemplateOptions addServiceAccount(ServiceAccount serviceAccout) {
|
||||||
|
this.serviceAccounts.add(serviceAccout);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see #getServiceAccounts()
|
||||||
|
* @see ServiceAccount
|
||||||
|
*/
|
||||||
|
public GoogleComputeTemplateOptions serviceAccounts(Set<ServiceAccount> serviceAccounts) {
|
||||||
|
this.serviceAccounts = Sets.newLinkedHashSet(serviceAccounts);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see #isEnableNat()
|
||||||
|
*/
|
||||||
|
public GoogleComputeTemplateOptions enableNat(boolean enableNat) {
|
||||||
|
this.enableNat = enableNat;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions blockOnPort(int port, int seconds) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.blockOnPort(port, seconds));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions inboundPorts(int... ports) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.inboundPorts(ports));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions authorizePublicKey(String publicKey) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.authorizePublicKey(publicKey));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions installPrivateKey(String privateKey) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.installPrivateKey(privateKey));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions blockUntilRunning(boolean blockUntilRunning) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.blockUntilRunning(blockUntilRunning));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions dontAuthorizePublicKey() {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.dontAuthorizePublicKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions nameTask(String name) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.nameTask(name));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions runAsRoot(boolean runAsRoot) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.runAsRoot(runAsRoot));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions runScript(Statement script) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.runScript(script));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions overrideLoginCredentials(LoginCredentials overridingCredentials) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.overrideLoginCredentials(overridingCredentials));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions overrideLoginPassword(String password) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.overrideLoginPassword(password));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions overrideLoginPrivateKey(String privateKey) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.overrideLoginPrivateKey(privateKey));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions overrideLoginUser(String loginUser) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.overrideLoginUser(loginUser));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions overrideAuthenticateSudo(boolean authenticateSudo) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.overrideAuthenticateSudo(authenticateSudo));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions userMetadata(Map<String, String> userMetadata) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.userMetadata(userMetadata));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions userMetadata(String key, String value) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.userMetadata(key, value));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions tags(Iterable<String> tags) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.tags(tags));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions wrapInInitScript(boolean wrapInInitScript) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.wrapInInitScript(wrapInInitScript));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions runScript(String script) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.runScript(script));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public GoogleComputeTemplateOptions blockOnComplete(boolean blockOnComplete) {
|
||||||
|
return GoogleComputeTemplateOptions.class.cast(super.blockOnComplete(blockOnComplete));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the ServiceAccounts to enable in the instances.
|
||||||
|
*/
|
||||||
|
public Set<Instance.ServiceAccount> getServiceAccounts() {
|
||||||
|
return serviceAccounts;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the URI of an existing network the instances will be attached to. If no network URI or network name are
|
||||||
|
* provided a new network will be created for the project.
|
||||||
|
*/
|
||||||
|
public Optional<URI> getNetwork() {
|
||||||
|
return network;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the name of an existing network the instances will be attached to, the network is assumed to belong to
|
||||||
|
* user's project. If no network URI network name are provided a new network will be created for the project.
|
||||||
|
*/
|
||||||
|
public Optional<String> getNetworkName() {
|
||||||
|
return networkName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return whether an AccessConfig with Type ONE_TO_ONE_NAT should be enabled in the instances. When true
|
||||||
|
* instances will have a NAT address that will be publicly accessible.
|
||||||
|
*/
|
||||||
|
public boolean isEnableNat() {
|
||||||
|
return enableNat;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,53 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.predicates;
|
||||||
|
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import org.jclouds.compute.ComputeService;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static com.google.common.collect.Iterables.all;
|
||||||
|
import static com.google.common.collect.Sets.filter;
|
||||||
|
import static org.jclouds.compute.predicates.NodePredicates.TERMINATED;
|
||||||
|
import static org.jclouds.compute.predicates.NodePredicates.all;
|
||||||
|
import static org.jclouds.compute.predicates.NodePredicates.inGroup;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Singleton
|
||||||
|
public class AllNodesInGroupTerminated implements Predicate<String> {
|
||||||
|
|
||||||
|
private final ComputeService computeService;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public AllNodesInGroupTerminated(ComputeService computeService) {
|
||||||
|
this.computeService = checkNotNull(computeService, "compute service");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(String groupName) {
|
||||||
|
return all(filter(computeService.listNodesDetailsMatching(all()), inGroup(groupName)), TERMINATED);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,200 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.strategy;
|
||||||
|
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.common.collect.Multimap;
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture;
|
||||||
|
import com.google.common.util.concurrent.ListeningExecutorService;
|
||||||
|
import org.jclouds.Constants;
|
||||||
|
import org.jclouds.compute.config.CustomizationResponse;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.domain.Template;
|
||||||
|
import org.jclouds.compute.functions.GroupNamingConvention;
|
||||||
|
import org.jclouds.compute.strategy.CreateNodeWithGroupEncodedIntoName;
|
||||||
|
import org.jclouds.compute.strategy.CustomizeNodeAndAddToGoodMapOrPutExceptionIntoBadMap;
|
||||||
|
import org.jclouds.compute.strategy.ListNodesStrategy;
|
||||||
|
import org.jclouds.compute.strategy.impl.CreateNodesWithGroupEncodedIntoNameThenAddToSet;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeApi;
|
||||||
|
import org.jclouds.googlecompute.compute.options.GoogleComputeTemplateOptions;
|
||||||
|
import org.jclouds.googlecompute.config.UserProject;
|
||||||
|
import org.jclouds.googlecompute.domain.Firewall;
|
||||||
|
import org.jclouds.googlecompute.domain.Network;
|
||||||
|
import org.jclouds.googlecompute.domain.Operation;
|
||||||
|
import org.jclouds.googlecompute.options.FirewallOptions;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static com.google.common.collect.ImmutableSet.of;
|
||||||
|
import static java.util.concurrent.TimeUnit.MILLISECONDS;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_INTERVAL;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.OPERATION_COMPLETE_TIMEOUT;
|
||||||
|
import static org.jclouds.util.Predicates2.retry;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class ApplyTemplateOptionsCreateNodesWithGroupEncodedIntoNameThenAddToSet extends
|
||||||
|
CreateNodesWithGroupEncodedIntoNameThenAddToSet {
|
||||||
|
|
||||||
|
public static final String EXTERIOR_RANGE = "0.0.0.0/0";
|
||||||
|
public static final String DEFAULT_INTERNAL_NETWORK_RANGE = "10.0.0.0/8";
|
||||||
|
|
||||||
|
private final GoogleComputeApi api;
|
||||||
|
private final Supplier<String> userProject;
|
||||||
|
private final Predicate<AtomicReference<Operation>> operationDonePredicate;
|
||||||
|
private final long operationCompleteCheckInterval;
|
||||||
|
private final long operationCompleteCheckTimeout;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
protected ApplyTemplateOptionsCreateNodesWithGroupEncodedIntoNameThenAddToSet(
|
||||||
|
CreateNodeWithGroupEncodedIntoName addNodeWithGroupStrategy,
|
||||||
|
ListNodesStrategy listNodesStrategy,
|
||||||
|
GroupNamingConvention.Factory namingConvention,
|
||||||
|
@Named(Constants.PROPERTY_USER_THREADS)
|
||||||
|
ListeningExecutorService userExecutor,
|
||||||
|
CustomizeNodeAndAddToGoodMapOrPutExceptionIntoBadMap.Factory
|
||||||
|
customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory,
|
||||||
|
GoogleComputeApi api,
|
||||||
|
@UserProject Supplier<String> userProject,
|
||||||
|
Predicate<AtomicReference<Operation>> operationDonePredicate,
|
||||||
|
@Named(OPERATION_COMPLETE_INTERVAL) Long operationCompleteCheckInterval,
|
||||||
|
@Named(OPERATION_COMPLETE_TIMEOUT) Long operationCompleteCheckTimeout) {
|
||||||
|
super(addNodeWithGroupStrategy, listNodesStrategy, namingConvention, userExecutor,
|
||||||
|
customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory);
|
||||||
|
|
||||||
|
this.api = checkNotNull(api, "google compute api");
|
||||||
|
this.userProject = checkNotNull(userProject, "user project name");
|
||||||
|
this.operationCompleteCheckInterval = checkNotNull(operationCompleteCheckInterval,
|
||||||
|
"operation completed check interval");
|
||||||
|
this.operationCompleteCheckTimeout = checkNotNull(operationCompleteCheckTimeout,
|
||||||
|
"operation completed check timeout");
|
||||||
|
this.operationDonePredicate = operationDonePredicate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<?, ListenableFuture<Void>> execute(String group, int count, Template template,
|
||||||
|
Set<NodeMetadata> goodNodes, Map<NodeMetadata, Exception> badNodes,
|
||||||
|
Multimap<NodeMetadata, CustomizationResponse> customizationResponses) {
|
||||||
|
|
||||||
|
String sharedResourceName = namingConvention.create().sharedNameForGroup(group);
|
||||||
|
Template mutableTemplate = template.clone();
|
||||||
|
GoogleComputeTemplateOptions templateOptions = GoogleComputeTemplateOptions.class.cast(mutableTemplate
|
||||||
|
.getOptions());
|
||||||
|
assert template.getOptions().equals(templateOptions) : "options didn't clone properly";
|
||||||
|
|
||||||
|
// get or create the network and create a firewall with the users configuration
|
||||||
|
Network network = getOrCreateNetwork(templateOptions, sharedResourceName);
|
||||||
|
createFirewall(templateOptions, network, sharedResourceName);
|
||||||
|
templateOptions.network(network.getSelfLink());
|
||||||
|
|
||||||
|
return super.execute(group, count, mutableTemplate, goodNodes, badNodes, customizationResponses);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Try and find a network either previously created by jclouds or user defined.
|
||||||
|
*/
|
||||||
|
private Network getOrCreateNetwork(GoogleComputeTemplateOptions templateOptions, String sharedResourceName) {
|
||||||
|
|
||||||
|
String networkName = templateOptions.getNetworkName().isPresent() ? templateOptions.getNetworkName().get() :
|
||||||
|
sharedResourceName;
|
||||||
|
|
||||||
|
// check if the network was previously created (cache???)
|
||||||
|
Network network = api.getNetworkApiForProject(userProject.get()).get(networkName);
|
||||||
|
|
||||||
|
if (network != null) {
|
||||||
|
return network;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (network == null && templateOptions.getNetwork().isPresent()) {
|
||||||
|
throw new IllegalStateException("user defined network does not exist: " + templateOptions.getNetwork().get());
|
||||||
|
}
|
||||||
|
|
||||||
|
AtomicReference<Operation> operation = new AtomicReference<Operation>(api.getNetworkApiForProject(userProject
|
||||||
|
.get()).createInIPv4Range(sharedResourceName, DEFAULT_INTERNAL_NETWORK_RANGE));
|
||||||
|
retry(operationDonePredicate, operationCompleteCheckTimeout, operationCompleteCheckInterval,
|
||||||
|
MILLISECONDS).apply(operation);
|
||||||
|
if (operation.get().getHttpError().isPresent()) {
|
||||||
|
throw new IllegalStateException("Could not create network, operation failed" + operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
return checkNotNull(api.getNetworkApiForProject(userProject.get()).get(sharedResourceName),
|
||||||
|
"no network with name %s was found", sharedResourceName);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tries to find if a firewall already exists for this group, if not it creates one.
|
||||||
|
*
|
||||||
|
* @see org.jclouds.googlecompute.features.FirewallAsyncApi#patch(String, org.jclouds.googlecompute.options.FirewallOptions)
|
||||||
|
*/
|
||||||
|
private void createFirewall(GoogleComputeTemplateOptions templateOptions, Network network,
|
||||||
|
String sharedResourceName) {
|
||||||
|
|
||||||
|
Firewall firewall = api.getFirewallApiForProject(userProject.get()).get(sharedResourceName);
|
||||||
|
|
||||||
|
if (firewall != null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImmutableSet.Builder<Firewall.Rule> rules = ImmutableSet.builder();
|
||||||
|
|
||||||
|
Firewall.Rule.Builder tcpRule = Firewall.Rule.builder();
|
||||||
|
tcpRule.IPProtocol(Firewall.Rule.IPProtocol.TCP);
|
||||||
|
Firewall.Rule.Builder udpRule = Firewall.Rule.builder();
|
||||||
|
udpRule.IPProtocol(Firewall.Rule.IPProtocol.UDP);
|
||||||
|
for (Integer port : templateOptions.getInboundPorts()) {
|
||||||
|
tcpRule.addPort(port);
|
||||||
|
udpRule.addPort(port);
|
||||||
|
}
|
||||||
|
rules.add(tcpRule.build());
|
||||||
|
rules.add(udpRule.build());
|
||||||
|
|
||||||
|
|
||||||
|
FirewallOptions options = new FirewallOptions()
|
||||||
|
.name(sharedResourceName)
|
||||||
|
.network(network.getSelfLink())
|
||||||
|
.sourceTags(templateOptions.getTags())
|
||||||
|
.allowedRules(rules.build())
|
||||||
|
.sourceRanges(of(DEFAULT_INTERNAL_NETWORK_RANGE, EXTERIOR_RANGE));
|
||||||
|
|
||||||
|
AtomicReference<Operation> operation = new AtomicReference<Operation>(api.getFirewallApiForProject(userProject
|
||||||
|
.get()).createInNetwork(
|
||||||
|
sharedResourceName,
|
||||||
|
network.getSelfLink(),
|
||||||
|
options));
|
||||||
|
|
||||||
|
retry(operationDonePredicate, operationCompleteCheckTimeout, operationCompleteCheckInterval,
|
||||||
|
MILLISECONDS).apply(operation);
|
||||||
|
|
||||||
|
if (operation.get().getHttpError().isPresent()) {
|
||||||
|
throw new IllegalStateException("Could not create firewall, operation failed" + operation.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,74 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.strategy;
|
||||||
|
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import org.jclouds.compute.domain.TemplateBuilderSpec;
|
||||||
|
import org.jclouds.compute.strategy.PopulateDefaultLoginCredentialsForImageStrategy;
|
||||||
|
import org.jclouds.domain.LoginCredentials;
|
||||||
|
import org.jclouds.ssh.internal.RsaSshKeyPairGenerator;
|
||||||
|
|
||||||
|
import javax.annotation.PostConstruct;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static org.jclouds.compute.config.ComputeServiceProperties.TEMPLATE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Singleton
|
||||||
|
public class GoogleComputePopulateDefaultLoginCredentialsForImageStrategy implements
|
||||||
|
PopulateDefaultLoginCredentialsForImageStrategy {
|
||||||
|
|
||||||
|
private final TemplateBuilderSpec templateBuilder;
|
||||||
|
private final RsaSshKeyPairGenerator keyPairGenerator;
|
||||||
|
private String compoundKey;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
GoogleComputePopulateDefaultLoginCredentialsForImageStrategy(@Named(TEMPLATE) String templateSpec,
|
||||||
|
RsaSshKeyPairGenerator keyPairGenerator)
|
||||||
|
throws NoSuchAlgorithmException {
|
||||||
|
this.templateBuilder = TemplateBuilderSpec.parse(checkNotNull(templateSpec, "template builder spec"));
|
||||||
|
checkNotNull(templateBuilder.getLoginUser(), "template builder spec must provide a loginUser");
|
||||||
|
this.keyPairGenerator = checkNotNull(keyPairGenerator, "keypair generator");
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostConstruct
|
||||||
|
private void generateKeys() {
|
||||||
|
Map<String, String> keys = keyPairGenerator.get();
|
||||||
|
// as we need to store both the pubk and the pk, store them separated by : (base64 does not contain that char)
|
||||||
|
compoundKey = String.format("%s:%s", checkNotNull(keys.get("public"), "public key cannot be null"),
|
||||||
|
checkNotNull(keys.get("private"), "private key cannot be null"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public LoginCredentials apply(Object image) {
|
||||||
|
return LoginCredentials.builder()
|
||||||
|
.authenticateSudo(templateBuilder.getAuthenticateSudo() != null ?
|
||||||
|
templateBuilder.getAuthenticateSudo() : false)
|
||||||
|
.privateKey(compoundKey)
|
||||||
|
.user(templateBuilder.getLoginUser()).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
package org.jclouds.googlecompute.compute.strategy;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import com.google.inject.Singleton;
|
||||||
|
import org.jclouds.compute.domain.Template;
|
||||||
|
import org.jclouds.compute.options.RunScriptOptions;
|
||||||
|
import org.jclouds.compute.strategy.PrioritizeCredentialsFromTemplate;
|
||||||
|
import org.jclouds.domain.LoginCredentials;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GCE needs the credentials to create the node so the node credentials already take the Image credentials into account,
|
||||||
|
* as such only overriding the TemplateOptions credentials is required.
|
||||||
|
*
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Singleton
|
||||||
|
public class UseNodeCredentialsButOverrideFromTemplate extends PrioritizeCredentialsFromTemplate {
|
||||||
|
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public UseNodeCredentialsButOverrideFromTemplate(
|
||||||
|
Function<Template, LoginCredentials> credentialsFromImageOrTemplateOptions) {
|
||||||
|
super(credentialsFromImageOrTemplateOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
public LoginCredentials apply(Template template, LoginCredentials fromNode) {
|
||||||
|
RunScriptOptions options = checkNotNull(template.getOptions(), "template options are required");
|
||||||
|
LoginCredentials.Builder builder = LoginCredentials.builder(fromNode);
|
||||||
|
if (options.getLoginUser() != null)
|
||||||
|
builder.user(template.getOptions().getLoginUser());
|
||||||
|
if (options.getLoginPassword() != null)
|
||||||
|
builder.password(options.getLoginPassword());
|
||||||
|
if (options.getLoginPrivateKey() != null)
|
||||||
|
builder.privateKey(options.getLoginPrivateKey());
|
||||||
|
if (options.shouldAuthenticateSudo() != null && options.shouldAuthenticateSudo())
|
||||||
|
builder.authenticateSudo(true);
|
||||||
|
return builder.build();
|
||||||
|
}
|
||||||
|
}
|
|
@ -60,6 +60,7 @@ import org.jclouds.http.annotation.ServerError;
|
||||||
import org.jclouds.json.config.GsonModule.DateAdapter;
|
import org.jclouds.json.config.GsonModule.DateAdapter;
|
||||||
import org.jclouds.json.config.GsonModule.Iso8601DateAdapter;
|
import org.jclouds.json.config.GsonModule.Iso8601DateAdapter;
|
||||||
import org.jclouds.location.Provider;
|
import org.jclouds.location.Provider;
|
||||||
|
import org.jclouds.location.config.LocationModule;
|
||||||
import org.jclouds.rest.ConfiguresRestClient;
|
import org.jclouds.rest.ConfiguresRestClient;
|
||||||
import org.jclouds.rest.config.RestClientModule;
|
import org.jclouds.rest.config.RestClientModule;
|
||||||
|
|
||||||
|
@ -164,4 +165,9 @@ public class GoogleComputeRestClientModule extends RestClientModule<GoogleComput
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// @Override
|
||||||
|
// protected void installLocations() {
|
||||||
|
// install(new LocationModule());
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
|
@ -555,7 +555,7 @@ public class Instance extends Resource {
|
||||||
|
|
||||||
private final String name;
|
private final String name;
|
||||||
private final URI network;
|
private final URI network;
|
||||||
private final String networkIP;
|
private final Optional<String> networkIP;
|
||||||
private final Set<AccessConfig> accessConfigs;
|
private final Set<AccessConfig> accessConfigs;
|
||||||
|
|
||||||
@ConstructorProperties({
|
@ConstructorProperties({
|
||||||
|
@ -565,7 +565,7 @@ public class Instance extends Resource {
|
||||||
Set<AccessConfig> accessConfigs) {
|
Set<AccessConfig> accessConfigs) {
|
||||||
this.name = checkNotNull(name, "name");
|
this.name = checkNotNull(name, "name");
|
||||||
this.network = checkNotNull(network, "network");
|
this.network = checkNotNull(network, "network");
|
||||||
this.networkIP = checkNotNull(networkIP, "networkIP");
|
this.networkIP = fromNullable(networkIP);
|
||||||
this.accessConfigs = accessConfigs == null ? ImmutableSet.<AccessConfig>of() : accessConfigs;
|
this.accessConfigs = accessConfigs == null ? ImmutableSet.<AccessConfig>of() : accessConfigs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -586,7 +586,7 @@ public class Instance extends Resource {
|
||||||
/**
|
/**
|
||||||
* @return An IPV4 internal network address to assign to this instance.
|
* @return An IPV4 internal network address to assign to this instance.
|
||||||
*/
|
*/
|
||||||
public String getNetworkIP() {
|
public Optional<String> getNetworkIP() {
|
||||||
return networkIP;
|
return networkIP;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -698,7 +698,7 @@ public class Instance extends Resource {
|
||||||
|
|
||||||
public Builder fromNetworkInterface(NetworkInterface in) {
|
public Builder fromNetworkInterface(NetworkInterface in) {
|
||||||
return this.network(in.getNetwork())
|
return this.network(in.getNetwork())
|
||||||
.networkIP(in.getNetworkIP())
|
.networkIP(in.getNetworkIP().orNull())
|
||||||
.accessConfigs(in.getAccessConfigs());
|
.accessConfigs(in.getAccessConfigs());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,415 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute;
|
||||||
|
|
||||||
|
import com.google.common.base.Throwables;
|
||||||
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import org.jclouds.compute.ComputeService;
|
||||||
|
import org.jclouds.compute.RunNodesException;
|
||||||
|
import org.jclouds.compute.domain.Template;
|
||||||
|
import org.jclouds.domain.Location;
|
||||||
|
import org.jclouds.googlecompute.compute.options.GoogleComputeTemplateOptions;
|
||||||
|
import org.jclouds.googlecompute.domain.Instance;
|
||||||
|
import org.jclouds.googlecompute.features.InstanceApiExpectTest;
|
||||||
|
import org.jclouds.googlecompute.internal.BaseGoogleComputeServiceExpectTest;
|
||||||
|
import org.jclouds.http.HttpRequest;
|
||||||
|
import org.jclouds.http.HttpResponse;
|
||||||
|
import org.jclouds.util.Strings2;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import javax.ws.rs.core.MediaType;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Properties;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import static com.google.common.collect.Iterables.getOnlyElement;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.COMPUTE_READONLY_SCOPE;
|
||||||
|
import static org.jclouds.googlecompute.GoogleComputeConstants.COMPUTE_SCOPE;
|
||||||
|
import static org.jclouds.googlecompute.features.FirewallApiExpectTest.GET_FIREWALL_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.ImageApiExpectTest.LIST_PROJECT_IMAGES_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.ImageApiExpectTest.LIST_PROJECT_IMAGES_RESPONSE;
|
||||||
|
import static org.jclouds.googlecompute.features.InstanceApiExpectTest.LIST_INSTANCES_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.InstanceApiExpectTest.LIST_INSTANCES_RESPONSE;
|
||||||
|
import static org.jclouds.googlecompute.features.MachineTypeApiExpectTest.LIST_MACHINE_TYPES_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.MachineTypeApiExpectTest.LIST_MACHINE_TYPES_RESPONSE;
|
||||||
|
import static org.jclouds.googlecompute.features.NetworkApiExpectTest.GET_NETWORK_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.OperationApiExpectTest.GET_OPERATION_REQUEST;
|
||||||
|
import static org.jclouds.googlecompute.features.OperationApiExpectTest.GET_OPERATION_RESPONSE;
|
||||||
|
import static org.jclouds.googlecompute.features.ZoneApiExpectTest.LIST_ZONES_REQ;
|
||||||
|
import static org.jclouds.googlecompute.features.ZoneApiExpectTest.LIST_ZONES_RESPONSE;
|
||||||
|
import static org.jclouds.util.Strings2.toStringAndClose;
|
||||||
|
import static org.testng.Assert.assertEquals;
|
||||||
|
import static org.testng.Assert.assertNotNull;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Test(groups = "unit")
|
||||||
|
public class GoogleComputeServiceExpectTest extends BaseGoogleComputeServiceExpectTest {
|
||||||
|
|
||||||
|
public static final HttpRequest LIST_GOOGLE_IMAGES_REQUEST = HttpRequest
|
||||||
|
.builder()
|
||||||
|
.method("GET")
|
||||||
|
.endpoint("https://www.googleapis.com/compute/v1beta13/projects/google/images")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
|
||||||
|
public static final HttpResponse LIST_GOOGLE_IMAGES_RESPONSE = HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(staticPayloadFromResource("/image_list_single_page.json")).build();
|
||||||
|
|
||||||
|
private HttpRequest INSERT_NETWORK_REQUEST = HttpRequest
|
||||||
|
.builder()
|
||||||
|
.method("POST")
|
||||||
|
.endpoint("https://www.googleapis.com/compute/v1beta13/projects/myproject/networks")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN)
|
||||||
|
.payload(payloadFromStringWithContentType("{\"name\":\"jclouds-test\",\"IPv4Range\":\"10.0.0.0/8\"}",
|
||||||
|
MediaType.APPLICATION_JSON))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
private HttpRequest INSERT_FIREWALL_REQUEST = HttpRequest
|
||||||
|
.builder()
|
||||||
|
.method("POST")
|
||||||
|
.endpoint("https://www.googleapis.com/compute/v1beta13/projects/myproject/firewalls")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN)
|
||||||
|
.payload(payloadFromStringWithContentType("{\"name\":\"jclouds-test\",\"network\":\"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/networks/jclouds-test\"," +
|
||||||
|
"\"sourceRanges\":[\"10.0.0.0/8\",\"0.0.0.0/0\"],\"allowed\":[{\"IPProtocol\":\"tcp\"," +
|
||||||
|
"\"ports\":[\"22\"]}," +
|
||||||
|
"{\"IPProtocol\":\"udp\",\"ports\":[\"22\"]}]}",
|
||||||
|
MediaType.APPLICATION_JSON))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
private HttpResponse GET_NETWORK_RESPONSE = HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(payloadFromStringWithContentType("{\n" +
|
||||||
|
" \"kind\": \"compute#network\",\n" +
|
||||||
|
" \"id\": \"13024414170909937976\",\n" +
|
||||||
|
" \"creationTimestamp\": \"2012-10-24T20:13:19.967\",\n" +
|
||||||
|
" \"selfLink\": \"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/networks/jclouds-test\",\n" +
|
||||||
|
" \"name\": \"jclouds-test\",\n" +
|
||||||
|
" \"description\": \"test network\",\n" +
|
||||||
|
" \"IPv4Range\": \"10.0.0.0/8\",\n" +
|
||||||
|
" \"gatewayIPv4\": \"10.0.0.1\"\n" +
|
||||||
|
"}", MediaType.APPLICATION_JSON)).build();
|
||||||
|
|
||||||
|
private HttpResponse SUCESSFULL_OPERATION_RESPONSE = HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(payloadFromResource("/operation.json")).build();
|
||||||
|
|
||||||
|
|
||||||
|
private HttpResponse getInstanceResponseForInstanceAndNetworkAndStatus(String instanceName, String networkName,
|
||||||
|
String status) throws
|
||||||
|
IOException {
|
||||||
|
return HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(payloadFromStringWithContentType(
|
||||||
|
replaceInstanceNameNetworkAndStatusOnResource("/instance_get.json",
|
||||||
|
instanceName, networkName, status),
|
||||||
|
"application/json")).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private HttpResponse getListInstancesResponseForSingleInstanceAndNetworkAndStatus(String instanceName,
|
||||||
|
String networkName,
|
||||||
|
String status) {
|
||||||
|
return HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(payloadFromStringWithContentType(
|
||||||
|
replaceInstanceNameNetworkAndStatusOnResource("/instance_list.json",
|
||||||
|
instanceName, networkName, status),
|
||||||
|
"application/json")).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String replaceInstanceNameNetworkAndStatusOnResource(String resourceName, String instanceName,
|
||||||
|
String networkName, String status) {
|
||||||
|
try {
|
||||||
|
return Strings2.toStringAndClose(this.getClass().getResourceAsStream(resourceName)).replace("test-0",
|
||||||
|
instanceName).replace("default", networkName).replace("RUNNING", status);
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private HttpRequest createInstanceRequestForInstance(String instanceName, String networkName, String publicKey) {
|
||||||
|
return HttpRequest
|
||||||
|
.builder()
|
||||||
|
.method("POST")
|
||||||
|
.endpoint("https://www.googleapis.com/compute/v1beta13/projects/myproject/instances")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN)
|
||||||
|
.payload(payloadFromStringWithContentType("{\"name\":\"" + instanceName + "\"," +
|
||||||
|
"\"machineType\":\"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/machineTypes/n1-standard-1\"," +
|
||||||
|
"\"zone\":\"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/zones/us-central1-a\"," +
|
||||||
|
"\"image\":\"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/google/images/gcel-12-04-v20121106\"," +
|
||||||
|
"\"tags\":[],\"serviceAccounts\":[]," +
|
||||||
|
"\"networkInterfaces\":[{\"network\":\"https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/networks/" + networkName + "\"," +
|
||||||
|
"\"accessConfigs\":[{\"type\":\"ONE_TO_ONE_NAT\"}]}]," +
|
||||||
|
"\"metadata\":{\"kind\":\"compute#metadata\",\"items\":[{\"key\":\"sshKeys\"," +
|
||||||
|
"\"value\":\"jclouds:" +
|
||||||
|
publicKey + " jclouds@localhost\"}]}}",
|
||||||
|
MediaType.APPLICATION_JSON)).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
private HttpRequest getInstanceRequestForInstance(String instanceName) {
|
||||||
|
return HttpRequest
|
||||||
|
.builder()
|
||||||
|
.method("GET")
|
||||||
|
.endpoint("https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/instances/" + instanceName)
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Properties setupProperties() {
|
||||||
|
Properties overrides = super.setupProperties();
|
||||||
|
overrides.put("google-compute.identity", "myproject");
|
||||||
|
try {
|
||||||
|
overrides.put("google-compute.credential", toStringAndClose(getClass().getResourceAsStream("/testpk.pem")));
|
||||||
|
} catch (IOException e) {
|
||||||
|
Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
return overrides;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(enabled = false)
|
||||||
|
public void testThrowsAuthorizationException() throws Exception {
|
||||||
|
|
||||||
|
Properties properties = new Properties();
|
||||||
|
properties.setProperty("oauth.identity", "MOMMA");
|
||||||
|
properties.setProperty("oauth.credential", "MiA");
|
||||||
|
|
||||||
|
ComputeService client = requestsSendResponses(ImmutableMap.<HttpRequest, HttpResponse>of(), createModule(),
|
||||||
|
properties);
|
||||||
|
Template template = client.templateBuilder().build();
|
||||||
|
Template toMatch = client.templateBuilder().imageId(template.getImage().getId()).build();
|
||||||
|
assertEquals(toMatch.getImage(), template.getImage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testTemplateMatch() throws Exception {
|
||||||
|
ImmutableMap<HttpRequest, HttpResponse> requestResponseMap = ImmutableMap.
|
||||||
|
<HttpRequest, HttpResponse>builder()
|
||||||
|
.put(requestForScopes(COMPUTE_READONLY_SCOPE), TOKEN_RESPONSE)
|
||||||
|
.put(LIST_ZONES_REQ, LIST_ZONES_RESPONSE)
|
||||||
|
.put(LIST_PROJECT_IMAGES_REQUEST, LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.put(LIST_GOOGLE_IMAGES_REQUEST, LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.put(LIST_MACHINE_TYPES_REQUEST, LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
ComputeService client = requestsSendResponses(requestResponseMap);
|
||||||
|
Template template = client.templateBuilder().build();
|
||||||
|
Template toMatch = client.templateBuilder().imageId(template.getImage().getId()).build();
|
||||||
|
assertEquals(toMatch.getImage(), template.getImage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNetworksAndFirewallDeletedWhenAllGroupNodesAreTerminated() throws IOException {
|
||||||
|
|
||||||
|
HttpRequest deleteNodeRequest = HttpRequest.builder()
|
||||||
|
.method("DELETE")
|
||||||
|
.endpoint("https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/instances/test-delete-networks")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
|
||||||
|
HttpRequest deleteFirewallRequest = HttpRequest.builder()
|
||||||
|
.method("DELETE")
|
||||||
|
.endpoint("https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/firewalls/jclouds-test-delete")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
|
||||||
|
HttpRequest deleteNetworkReqquest = HttpRequest.builder()
|
||||||
|
.method("DELETE")
|
||||||
|
.endpoint("https://www.googleapis" +
|
||||||
|
".com/compute/v1beta13/projects/myproject/networks/jclouds-test-delete")
|
||||||
|
.addHeader("Accept", "application/json")
|
||||||
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
|
||||||
|
List<HttpRequest> orderedRequests = ImmutableList.<HttpRequest>builder()
|
||||||
|
.add(requestForScopes(COMPUTE_READONLY_SCOPE))
|
||||||
|
.add(getInstanceRequestForInstance("test-delete-networks"))
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.add(requestForScopes(COMPUTE_SCOPE))
|
||||||
|
.add(deleteNodeRequest)
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.add(getInstanceRequestForInstance("test-delete-networks"))
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.add(LIST_INSTANCES_REQUEST)
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.add(deleteFirewallRequest)
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.add(deleteNetworkReqquest)
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
|
||||||
|
List<HttpResponse> orderedResponses = ImmutableList.<HttpResponse>builder()
|
||||||
|
.add(TOKEN_RESPONSE)
|
||||||
|
.add(getInstanceResponseForInstanceAndNetworkAndStatus("test-delete-networks", "test-network", Instance
|
||||||
|
.Status.RUNNING.name()))
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.add(TOKEN_RESPONSE)
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.add(getInstanceResponseForInstanceAndNetworkAndStatus("test-delete-networks", "test-network", Instance
|
||||||
|
.Status.TERMINATED.name()))
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.add(getListInstancesResponseForSingleInstanceAndNetworkAndStatus("test-delete-networks",
|
||||||
|
"test-network", Instance
|
||||||
|
.Status.TERMINATED.name()))
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
ComputeService client = orderedRequestsSendResponses(orderedRequests, orderedResponses);
|
||||||
|
client.destroyNode("test-delete-networks");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testListLocationsWhenResponseIs2xx() throws Exception {
|
||||||
|
|
||||||
|
ImmutableMap<HttpRequest, HttpResponse> requestResponseMap = ImmutableMap.
|
||||||
|
<HttpRequest, HttpResponse>builder()
|
||||||
|
.put(requestForScopes(COMPUTE_READONLY_SCOPE), TOKEN_RESPONSE)
|
||||||
|
.put(LIST_ZONES_REQ, LIST_ZONES_RESPONSE)
|
||||||
|
.put(LIST_INSTANCES_REQUEST, LIST_INSTANCES_RESPONSE)
|
||||||
|
.put(LIST_PROJECT_IMAGES_REQUEST, LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.put(LIST_GOOGLE_IMAGES_REQUEST, LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.put(LIST_MACHINE_TYPES_REQUEST, LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
ComputeService apiWhenServersExist = requestsSendResponses(requestResponseMap);
|
||||||
|
|
||||||
|
Set<? extends Location> locations = apiWhenServersExist.listAssignableLocations();
|
||||||
|
|
||||||
|
assertNotNull(locations);
|
||||||
|
assertEquals(locations.size(), 2);
|
||||||
|
assertEquals(locations.iterator().next().getId(), "us-central1-a");
|
||||||
|
|
||||||
|
assertNotNull(apiWhenServersExist.listNodes());
|
||||||
|
assertEquals(apiWhenServersExist.listNodes().size(), 1);
|
||||||
|
assertEquals(apiWhenServersExist.listNodes().iterator().next().getId(), "test-0");
|
||||||
|
assertEquals(apiWhenServersExist.listNodes().iterator().next().getName(), "test-0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(dependsOnMethods = "testListLocationsWhenResponseIs2xx")
|
||||||
|
public void testCreateNodeWhenNetworkNorFirewallExistDoesNotExist() throws RunNodesException, IOException {
|
||||||
|
|
||||||
|
|
||||||
|
String payload = Strings2.toStringAndClose(InstanceApiExpectTest.class.getResourceAsStream("/instance_get.json"));
|
||||||
|
payload = payload.replace("test-0", "test-1");
|
||||||
|
|
||||||
|
HttpResponse getInstanceResponse = HttpResponse.builder().statusCode(200)
|
||||||
|
.payload(payloadFromStringWithContentType(payload, "application/json")).build();
|
||||||
|
|
||||||
|
List<HttpRequest> orderedRequests = ImmutableList.<HttpRequest>builder()
|
||||||
|
.add(requestForScopes(COMPUTE_READONLY_SCOPE))
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.add(GET_NETWORK_REQUEST)
|
||||||
|
.add(requestForScopes(COMPUTE_SCOPE))
|
||||||
|
.add(INSERT_NETWORK_REQUEST)
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.add(GET_NETWORK_REQUEST)
|
||||||
|
.add(GET_FIREWALL_REQUEST)
|
||||||
|
.add(INSERT_FIREWALL_REQUEST)
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.add(LIST_INSTANCES_REQUEST)
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.add(createInstanceRequestForInstance("test-1", "jclouds-test", openSshKey))
|
||||||
|
.add(GET_OPERATION_REQUEST)
|
||||||
|
.add(getInstanceRequestForInstance("test-1"))
|
||||||
|
.add(LIST_PROJECT_IMAGES_REQUEST)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_REQUEST)
|
||||||
|
.add(LIST_ZONES_REQ)
|
||||||
|
.add(LIST_MACHINE_TYPES_REQUEST)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
List<HttpResponse> orderedResponses = ImmutableList.<HttpResponse>builder()
|
||||||
|
.add(TOKEN_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.add(HttpResponse.builder().statusCode(404).build())
|
||||||
|
.add(TOKEN_RESPONSE)
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.add(GET_NETWORK_RESPONSE)
|
||||||
|
.add(HttpResponse.builder().statusCode(404).build())
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.add(LIST_INSTANCES_RESPONSE)
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.add(SUCESSFULL_OPERATION_RESPONSE)
|
||||||
|
.add(GET_OPERATION_RESPONSE)
|
||||||
|
.add(getInstanceResponse)
|
||||||
|
.add(LIST_PROJECT_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_GOOGLE_IMAGES_RESPONSE)
|
||||||
|
.add(LIST_ZONES_RESPONSE)
|
||||||
|
.add(LIST_MACHINE_TYPES_RESPONSE)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
|
||||||
|
ComputeService computeService = orderedRequestsSendResponses(orderedRequests, orderedResponses);
|
||||||
|
|
||||||
|
GoogleComputeTemplateOptions options = computeService.templateOptions().as(GoogleComputeTemplateOptions.class);
|
||||||
|
|
||||||
|
getOnlyElement(computeService.createNodesInGroup("test", 1, options));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,96 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.inject.Module;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.internal.BaseComputeServiceLiveTest;
|
||||||
|
import org.jclouds.oauth.v2.OAuthTestUtils;
|
||||||
|
import org.jclouds.sshj.config.SshjSshClientModule;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import java.util.Properties;
|
||||||
|
|
||||||
|
import static org.testng.Assert.assertTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Test(groups = "live", singleThreaded = true)
|
||||||
|
public class GoogleComputeServiceLiveTest extends BaseComputeServiceLiveTest {
|
||||||
|
|
||||||
|
public GoogleComputeServiceLiveTest() {
|
||||||
|
provider = "google-compute";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Nodes may have additional metadata entries (particularly they may have an "sshKeys" entry)
|
||||||
|
*/
|
||||||
|
protected void checkUserMetadataInNodeEquals(NodeMetadata node, ImmutableMap<String, String> userMetadata) {
|
||||||
|
assertTrue(node.getUserMetadata().keySet().containsAll(userMetadata.keySet()));
|
||||||
|
}
|
||||||
|
|
||||||
|
// do not run until the auth exception problem is figured out.
|
||||||
|
@Test(enabled = false)
|
||||||
|
@Override
|
||||||
|
public void testCorrectAuthException() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
// reboot is not supported by GCE
|
||||||
|
@Test(enabled = true, dependsOnMethods = "testGet")
|
||||||
|
public void testReboot() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
// suspend/Resume is not supported by GCE
|
||||||
|
@Test(enabled = true, dependsOnMethods = "testReboot")
|
||||||
|
public void testSuspendResume() throws Exception {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(enabled = true, dependsOnMethods = "testSuspendResume")
|
||||||
|
@Override
|
||||||
|
public void testGetNodesWithDetails() throws Exception {
|
||||||
|
super.testGetNodesWithDetails();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(enabled = true, dependsOnMethods = "testSuspendResume")
|
||||||
|
@Override
|
||||||
|
public void testListNodes() throws Exception {
|
||||||
|
super.testListNodes();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(enabled = true, dependsOnMethods = {"testListNodes", "testGetNodesWithDetails"})
|
||||||
|
@Override
|
||||||
|
public void testDestroyNodes() {
|
||||||
|
super.testDestroyNodes();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Module getSshModule() {
|
||||||
|
return new SshjSshClientModule();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Properties setupProperties() {
|
||||||
|
Properties properties = super.setupProperties();
|
||||||
|
OAuthTestUtils.setCredentialFromPemFile(properties, "google-compute.credential");
|
||||||
|
return properties;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import org.jclouds.compute.domain.OsFamily;
|
||||||
|
import org.jclouds.googlecompute.domain.Image;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import java.net.URI;
|
||||||
|
|
||||||
|
import static org.testng.Assert.assertEquals;
|
||||||
|
import static org.testng.Assert.assertSame;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
@Test(groups = "unit")
|
||||||
|
public class GoogleComputeImageToImageTest {
|
||||||
|
|
||||||
|
Image.Builder imageBuilder = Image.builder()
|
||||||
|
.id("1234")
|
||||||
|
.selfLink(URI.create("http://test.com"))
|
||||||
|
.sourceType("RAW")
|
||||||
|
.description("")
|
||||||
|
.rawDisk(Image.RawDisk.builder().source("").containerType("TAR").build());
|
||||||
|
|
||||||
|
public void testArbitratyImageName() {
|
||||||
|
GoogleComputeImageToImage imageToImage = new GoogleComputeImageToImage();
|
||||||
|
Image image = imageBuilder.name("arbitratyname").build();
|
||||||
|
org.jclouds.compute.domain.Image transformed = imageToImage.apply(image);
|
||||||
|
assertEquals(transformed.getName(), image.getName());
|
||||||
|
assertEquals(transformed.getId(), image.getName());
|
||||||
|
assertEquals(transformed.getProviderId(), image.getId());
|
||||||
|
assertSame(transformed.getOperatingSystem().getFamily(), OsFamily.LINUX);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testWellFormedImageName() {
|
||||||
|
GoogleComputeImageToImage imageToImage = new GoogleComputeImageToImage();
|
||||||
|
Image image = imageBuilder.name("ubuntu-12-04-v123123").build();
|
||||||
|
org.jclouds.compute.domain.Image transformed = imageToImage.apply(image);
|
||||||
|
assertEquals(transformed.getName(), image.getName());
|
||||||
|
assertEquals(transformed.getId(), image.getName());
|
||||||
|
assertEquals(transformed.getProviderId(), image.getId());
|
||||||
|
assertSame(transformed.getOperatingSystem().getFamily(), OsFamily.UBUNTU);
|
||||||
|
assertEquals(transformed.getOperatingSystem().getVersion(), "12.04");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,141 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.compute.functions;
|
||||||
|
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.common.collect.Sets;
|
||||||
|
import org.easymock.EasyMock;
|
||||||
|
import org.jclouds.compute.ComputeService;
|
||||||
|
import org.jclouds.compute.domain.ComputeMetadata;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.domain.internal.NodeMetadataImpl;
|
||||||
|
import org.jclouds.googlecompute.compute.predicates.AllNodesInGroupTerminated;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import static org.easymock.EasyMock.createMock;
|
||||||
|
import static org.easymock.EasyMock.expect;
|
||||||
|
import static org.easymock.EasyMock.replay;
|
||||||
|
import static org.testng.Assert.assertSame;
|
||||||
|
import static org.testng.Assert.assertTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class OrphanedGroupsFromDeadNodesTest {
|
||||||
|
|
||||||
|
private static class IdAndGroupOnlyNodeMetadata extends NodeMetadataImpl {
|
||||||
|
|
||||||
|
public IdAndGroupOnlyNodeMetadata(String id, String group, Status status) {
|
||||||
|
super(null, null, id, null, null, ImmutableMap.<String, String>of(), ImmutableSet.<String>of(), group, null,
|
||||||
|
null, null, status, null, 0, ImmutableSet.<String>of(), ImmutableSet.<String>of(), null, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDetectsAllOrphanedGroupsWhenAllNodesTerminated() {
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup1 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("a", "1", NodeMetadata.Status.TERMINATED)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup2 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("b", "2", NodeMetadata.Status.TERMINATED)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> allDeadNodes = Sets.union(deadNodesGroup1, deadNodesGroup2);
|
||||||
|
|
||||||
|
ComputeService mock = createMock(ComputeService.class);
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) deadNodesGroup1).once();
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) deadNodesGroup2).once();
|
||||||
|
|
||||||
|
replay(mock);
|
||||||
|
|
||||||
|
OrphanedGroupsFromDeadNodes orphanedGroupsFromDeadNodes = new OrphanedGroupsFromDeadNodes(new
|
||||||
|
AllNodesInGroupTerminated(mock));
|
||||||
|
|
||||||
|
Set<String> orphanedGroups = orphanedGroupsFromDeadNodes.apply(allDeadNodes);
|
||||||
|
|
||||||
|
assertSame(orphanedGroups.size(), 2);
|
||||||
|
assertTrue(orphanedGroups.contains("1"));
|
||||||
|
assertTrue(orphanedGroups.contains("2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDetectsAllOrphanedGroupsWhenSomeNodesTerminatedAndOtherMissing() {
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup1 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("a", "1", NodeMetadata.Status.TERMINATED)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup2 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("b", "2", NodeMetadata.Status.TERMINATED)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> allDeadNodes = Sets.union(deadNodesGroup1, deadNodesGroup2);
|
||||||
|
|
||||||
|
ComputeService mock = createMock(ComputeService.class);
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) deadNodesGroup1).once();
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) ImmutableSet.of()).once();
|
||||||
|
|
||||||
|
replay(mock);
|
||||||
|
|
||||||
|
OrphanedGroupsFromDeadNodes orphanedGroupsFromDeadNodes = new OrphanedGroupsFromDeadNodes(new
|
||||||
|
AllNodesInGroupTerminated(mock));
|
||||||
|
|
||||||
|
Set<String> orphanedGroups = orphanedGroupsFromDeadNodes.apply(allDeadNodes);
|
||||||
|
|
||||||
|
assertSame(orphanedGroups.size(), 2);
|
||||||
|
assertTrue(orphanedGroups.contains("1"));
|
||||||
|
assertTrue(orphanedGroups.contains("2"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDetectsAllOrphanedGroupsWhenSomeNodesAreAlive() {
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup1 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("a", "1", NodeMetadata.Status.TERMINATED)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> deadNodesGroup2 = (Set) ImmutableSet.builder()
|
||||||
|
.add(new IdAndGroupOnlyNodeMetadata("b", "2", NodeMetadata.Status.RUNNING)).build();
|
||||||
|
|
||||||
|
Set<? extends NodeMetadata> allDeadNodes = Sets.union(deadNodesGroup1, deadNodesGroup2);
|
||||||
|
|
||||||
|
ComputeService mock = createMock(ComputeService.class);
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) deadNodesGroup1).once();
|
||||||
|
expect(mock.listNodesDetailsMatching(EasyMock.<Predicate<ComputeMetadata>>anyObject()))
|
||||||
|
.andReturn((Set) deadNodesGroup2).once();
|
||||||
|
|
||||||
|
replay(mock);
|
||||||
|
|
||||||
|
OrphanedGroupsFromDeadNodes orphanedGroupsFromDeadNodes = new OrphanedGroupsFromDeadNodes(new
|
||||||
|
AllNodesInGroupTerminated(mock));
|
||||||
|
|
||||||
|
Set<String> orphanedGroups = orphanedGroupsFromDeadNodes.apply(allDeadNodes);
|
||||||
|
|
||||||
|
assertSame(orphanedGroups.size(), 1);
|
||||||
|
assertTrue(orphanedGroups.contains("1"));
|
||||||
|
}
|
||||||
|
}
|
|
@ -93,7 +93,7 @@ public class NetworkApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
||||||
TOKEN_RESPONSE, insert,
|
TOKEN_RESPONSE, insert,
|
||||||
insertNetworkResponse).getNetworkApiForProject("myproject");
|
insertNetworkResponse).getNetworkApiForProject("myproject");
|
||||||
|
|
||||||
assertEquals(api.createInIPv4Range("test-network", "10.0.1.0/8"), new ParseOperationTest().expected());
|
assertEquals(api.createInIPv4Range("test-network", "10.0.0.0/8"), new ParseOperationTest().expected());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testDeleteNetworkResponseIs2xx() {
|
public void testDeleteNetworkResponseIs2xx() {
|
||||||
|
|
|
@ -37,12 +37,12 @@ import static org.testng.Assert.assertTrue;
|
||||||
@Test(groups = "unit")
|
@Test(groups = "unit")
|
||||||
public class ZoneApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
public class ZoneApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
||||||
|
|
||||||
public static final String ZONES_URL_PREFIX = "https://www.googleapis.com/compute/v1beta13/projects/google/zones";
|
public static final String ZONES_URL_PREFIX = "https://www.googleapis.com/compute/v1beta13/projects/myproject/zones";
|
||||||
|
|
||||||
public static final HttpRequest GET_ZONE_REQ = HttpRequest
|
public static final HttpRequest GET_ZONE_REQ = HttpRequest
|
||||||
.builder()
|
.builder()
|
||||||
.method("GET")
|
.method("GET")
|
||||||
.endpoint(ZONES_URL_PREFIX + "/us-central2-a")
|
.endpoint(ZONES_URL_PREFIX + "/us-central1-a")
|
||||||
.addHeader("Accept", "application/json")
|
.addHeader("Accept", "application/json")
|
||||||
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
.addHeader("Authorization", "Bearer " + TOKEN).build();
|
||||||
|
|
||||||
|
@ -64,9 +64,9 @@ public class ZoneApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
||||||
.payload(payloadFromResource("/zone_get.json")).build();
|
.payload(payloadFromResource("/zone_get.json")).build();
|
||||||
|
|
||||||
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
||||||
TOKEN_RESPONSE, GET_ZONE_REQ, operationResponse).getZoneApiForProject("google");
|
TOKEN_RESPONSE, GET_ZONE_REQ, operationResponse).getZoneApiForProject("myproject");
|
||||||
|
|
||||||
assertEquals(api.get("us-central2-a"),
|
assertEquals(api.get("us-central1-a"),
|
||||||
new ParseZoneTest().expected());
|
new ParseZoneTest().expected());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,15 +75,15 @@ public class ZoneApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
||||||
HttpResponse operationResponse = HttpResponse.builder().statusCode(404).build();
|
HttpResponse operationResponse = HttpResponse.builder().statusCode(404).build();
|
||||||
|
|
||||||
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
||||||
TOKEN_RESPONSE, GET_ZONE_REQ, operationResponse).getZoneApiForProject("google");
|
TOKEN_RESPONSE, GET_ZONE_REQ, operationResponse).getZoneApiForProject("myproject");
|
||||||
|
|
||||||
assertNull(api.get("us-central2-a"));
|
assertNull(api.get("us-central1-a"));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testListZoneNoOptionsResponseIs2xx() throws Exception {
|
public void testListZoneNoOptionsResponseIs2xx() throws Exception {
|
||||||
|
|
||||||
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
||||||
TOKEN_RESPONSE, LIST_ZONES_REQ, LIST_ZONES_RESPONSE).getZoneApiForProject("google");
|
TOKEN_RESPONSE, LIST_ZONES_REQ, LIST_ZONES_RESPONSE).getZoneApiForProject("myproject");
|
||||||
|
|
||||||
assertEquals(api.listFirstPage().toString(),
|
assertEquals(api.listFirstPage().toString(),
|
||||||
new ParseZoneListTest().expected().toString());
|
new ParseZoneListTest().expected().toString());
|
||||||
|
@ -94,7 +94,7 @@ public class ZoneApiExpectTest extends BaseGoogleComputeApiExpectTest {
|
||||||
HttpResponse operationResponse = HttpResponse.builder().statusCode(404).build();
|
HttpResponse operationResponse = HttpResponse.builder().statusCode(404).build();
|
||||||
|
|
||||||
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
ZoneApi api = requestsSendResponses(requestForScopes(COMPUTE_READONLY_SCOPE),
|
||||||
TOKEN_RESPONSE, LIST_ZONES_REQ, operationResponse).getZoneApiForProject("google");
|
TOKEN_RESPONSE, LIST_ZONES_REQ, operationResponse).getZoneApiForProject("myproject");
|
||||||
|
|
||||||
assertTrue(api.list().concat().isEmpty());
|
assertTrue(api.list().concat().isEmpty());
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,29 +20,49 @@ package org.jclouds.googlecompute.internal;
|
||||||
|
|
||||||
import com.google.common.base.Joiner;
|
import com.google.common.base.Joiner;
|
||||||
import com.google.common.base.Supplier;
|
import com.google.common.base.Supplier;
|
||||||
import com.google.common.base.Throwables;
|
|
||||||
import com.google.common.base.Ticker;
|
import com.google.common.base.Ticker;
|
||||||
import com.google.inject.Binder;
|
import com.google.inject.Binder;
|
||||||
import com.google.inject.Module;
|
import com.google.inject.Module;
|
||||||
import com.google.inject.TypeLiteral;
|
import com.google.inject.TypeLiteral;
|
||||||
import org.jclouds.collect.PagedIterable;
|
import org.jclouds.collect.PagedIterable;
|
||||||
import org.jclouds.collect.PagedIterables;
|
import org.jclouds.collect.PagedIterables;
|
||||||
|
import org.jclouds.crypto.Crypto;
|
||||||
import org.jclouds.googlecompute.domain.ListPage;
|
import org.jclouds.googlecompute.domain.ListPage;
|
||||||
import org.jclouds.http.HttpRequest;
|
import org.jclouds.http.HttpRequest;
|
||||||
import org.jclouds.http.HttpResponse;
|
import org.jclouds.http.HttpResponse;
|
||||||
import org.jclouds.io.Payload;
|
import org.jclouds.io.Payload;
|
||||||
import org.jclouds.oauth.v2.OAuthConstants;
|
import org.jclouds.oauth.v2.OAuthConstants;
|
||||||
import org.jclouds.rest.internal.BaseRestApiExpectTest;
|
import org.jclouds.rest.internal.BaseRestApiExpectTest;
|
||||||
|
import org.jclouds.ssh.SshKeys;
|
||||||
import org.jclouds.util.Strings2;
|
import org.jclouds.util.Strings2;
|
||||||
|
|
||||||
import javax.ws.rs.core.MediaType;
|
import javax.ws.rs.core.MediaType;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
|
import java.security.KeyFactory;
|
||||||
|
import java.security.KeyPair;
|
||||||
|
import java.security.KeyPairGenerator;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.security.PrivateKey;
|
||||||
|
import java.security.PublicKey;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
import java.security.interfaces.RSAPublicKey;
|
||||||
|
import java.security.spec.InvalidKeySpecException;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import static com.google.common.base.Charsets.UTF_8;
|
import static com.google.common.base.Charsets.UTF_8;
|
||||||
|
import static com.google.common.base.Throwables.propagate;
|
||||||
import static com.google.common.io.BaseEncoding.base64Url;
|
import static com.google.common.io.BaseEncoding.base64Url;
|
||||||
|
import static org.easymock.EasyMock.createMock;
|
||||||
|
import static org.easymock.EasyMock.expect;
|
||||||
|
import static org.easymock.EasyMock.expectLastCall;
|
||||||
|
import static org.easymock.EasyMock.replay;
|
||||||
|
import static org.jclouds.crypto.Pems.privateKeySpec;
|
||||||
|
import static org.jclouds.crypto.Pems.publicKeySpec;
|
||||||
|
import static org.jclouds.crypto.PemsTest.PRIVATE_KEY;
|
||||||
|
import static org.jclouds.crypto.PemsTest.PUBLIC_KEY;
|
||||||
|
import static org.jclouds.io.Payloads.newStringPayload;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Adrian Cole
|
* @author Adrian Cole
|
||||||
|
@ -67,6 +87,8 @@ public class BaseGoogleComputeExpectTest<T> extends BaseRestApiExpectTest<T> {
|
||||||
" \"expires_in\" : 3600\n" +
|
" \"expires_in\" : 3600\n" +
|
||||||
"}")).build();
|
"}")).build();
|
||||||
|
|
||||||
|
protected String openSshKey;
|
||||||
|
|
||||||
|
|
||||||
public BaseGoogleComputeExpectTest() {
|
public BaseGoogleComputeExpectTest() {
|
||||||
provider = "google-compute";
|
provider = "google-compute";
|
||||||
|
@ -74,15 +96,42 @@ public class BaseGoogleComputeExpectTest<T> extends BaseRestApiExpectTest<T> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Module createModule() {
|
protected Module createModule() {
|
||||||
|
|
||||||
|
|
||||||
return new Module() {
|
return new Module() {
|
||||||
@Override
|
@Override
|
||||||
public void configure(Binder binder) {
|
public void configure(Binder binder) {
|
||||||
|
// predictable time
|
||||||
binder.bind(Ticker.class).toInstance(new Ticker() {
|
binder.bind(Ticker.class).toInstance(new Ticker() {
|
||||||
@Override
|
@Override
|
||||||
public long read() {
|
public long read() {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
try {
|
||||||
|
KeyFactory keyfactory = KeyFactory.getInstance("RSA");
|
||||||
|
PrivateKey privateKey = keyfactory.generatePrivate(privateKeySpec(newStringPayload
|
||||||
|
(PRIVATE_KEY)));
|
||||||
|
PublicKey publicKey = keyfactory.generatePublic(publicKeySpec(newStringPayload(PUBLIC_KEY)));
|
||||||
|
KeyPair keyPair = new KeyPair(publicKey, privateKey);
|
||||||
|
openSshKey = SshKeys.encodeAsOpenSSH(RSAPublicKey.class.cast(publicKey));
|
||||||
|
final Crypto crypto = createMock(Crypto.class);
|
||||||
|
KeyPairGenerator rsaKeyPairGenerator = createMock(KeyPairGenerator.class);
|
||||||
|
final SecureRandom secureRandom = createMock(SecureRandom.class);
|
||||||
|
expect(crypto.rsaKeyPairGenerator()).andReturn(rsaKeyPairGenerator).anyTimes();
|
||||||
|
rsaKeyPairGenerator.initialize(2048, secureRandom);
|
||||||
|
expectLastCall().anyTimes();
|
||||||
|
expect(rsaKeyPairGenerator.genKeyPair()).andReturn(keyPair).anyTimes();
|
||||||
|
replay(crypto, rsaKeyPairGenerator, secureRandom);
|
||||||
|
binder.bind(Crypto.class).toInstance(crypto);
|
||||||
|
binder.bind(SecureRandom.class).toInstance(secureRandom);
|
||||||
|
} catch (NoSuchAlgorithmException e) {
|
||||||
|
propagate(e);
|
||||||
|
} catch (InvalidKeySpecException e) {
|
||||||
|
propagate(e);
|
||||||
|
} catch (IOException e) {
|
||||||
|
propagate(e);
|
||||||
|
}
|
||||||
// predictable node names
|
// predictable node names
|
||||||
final AtomicInteger suffix = new AtomicInteger();
|
final AtomicInteger suffix = new AtomicInteger();
|
||||||
binder.bind(new TypeLiteral<Supplier<String>>() {
|
binder.bind(new TypeLiteral<Supplier<String>>() {
|
||||||
|
@ -134,7 +183,7 @@ public class BaseGoogleComputeExpectTest<T> extends BaseRestApiExpectTest<T> {
|
||||||
return payloadFromString(Strings2.toStringAndClose(BaseGoogleComputeExpectTest.class.getResourceAsStream
|
return payloadFromString(Strings2.toStringAndClose(BaseGoogleComputeExpectTest.class.getResourceAsStream
|
||||||
(resource)));
|
(resource)));
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw Throwables.propagate(e);
|
throw propagate(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.internal;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.inject.Module;
|
||||||
|
import org.jclouds.apis.ApiMetadata;
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
import org.jclouds.googlecompute.GoogleComputeApiMetadata;
|
||||||
|
import org.jclouds.http.HttpRequest;
|
||||||
|
import org.jclouds.http.HttpResponse;
|
||||||
|
|
||||||
|
import java.util.Properties;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public abstract class BaseGoogleComputeServiceContextExpectTest<T> extends BaseGoogleComputeExpectTest<T> implements
|
||||||
|
Function<ComputeServiceContext, T> {
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T createClient(Function<HttpRequest, HttpResponse> fn, Module module, Properties props) {
|
||||||
|
return apply(createComputeServiceContext(fn, module, props));
|
||||||
|
}
|
||||||
|
|
||||||
|
private ComputeServiceContext createComputeServiceContext(Function<HttpRequest, HttpResponse> fn, Module module,
|
||||||
|
Properties props) {
|
||||||
|
return createInjector(fn, module, props).getInstance(ComputeServiceContext.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ApiMetadata createApiMetadata() {
|
||||||
|
return new GoogleComputeApiMetadata();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
/*
|
||||||
|
* Licensed to jclouds, Inc. (jclouds) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. jclouds 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.googlecompute.internal;
|
||||||
|
|
||||||
|
import org.jclouds.compute.ComputeService;
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author David Alves
|
||||||
|
*/
|
||||||
|
public class BaseGoogleComputeServiceExpectTest extends BaseGoogleComputeServiceContextExpectTest<ComputeService> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ComputeService apply(ComputeServiceContext input) {
|
||||||
|
return input.getComputeService();
|
||||||
|
}
|
||||||
|
}
|
|
@ -46,8 +46,8 @@ public class ParseNetworkTest extends BaseGoogleComputeParseTest<Network> {
|
||||||
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test"))
|
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test"))
|
||||||
.name("default")
|
.name("default")
|
||||||
.description("Default network for the project")
|
.description("Default network for the project")
|
||||||
.IPv4Range("10.240.0.0/16")
|
.IPv4Range("10.0.0.0/8")
|
||||||
.gatewayIPv4("10.240.0.1")
|
.gatewayIPv4("10.0.0.1")
|
||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,8 +47,8 @@ public class ParseZoneListTest extends BaseGoogleComputeParseTest<ListPage<Zone>
|
||||||
public ListPage<Zone> expected() {
|
public ListPage<Zone> expected() {
|
||||||
return ListPage.<Zone>builder()
|
return ListPage.<Zone>builder()
|
||||||
.kind(Resource.Kind.ZONE_LIST)
|
.kind(Resource.Kind.ZONE_LIST)
|
||||||
.id("projects/google/zones")
|
.id("projects/myproject/zones")
|
||||||
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/google/zones"))
|
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/myproject/zones"))
|
||||||
.items(ImmutableSet.of(
|
.items(ImmutableSet.of(
|
||||||
new ParseZoneTest().expected()
|
new ParseZoneTest().expected()
|
||||||
, Zone.builder()
|
, Zone.builder()
|
||||||
|
@ -56,7 +56,7 @@ public class ParseZoneListTest extends BaseGoogleComputeParseTest<ListPage<Zone>
|
||||||
.creationTimestamp(new SimpleDateFormatDateService().iso8601DateParse
|
.creationTimestamp(new SimpleDateFormatDateService().iso8601DateParse
|
||||||
("2012-10-24T20:13:19.271"))
|
("2012-10-24T20:13:19.271"))
|
||||||
.selfLink(URI.create("https://www.googleapis" +
|
.selfLink(URI.create("https://www.googleapis" +
|
||||||
".com/compute/v1beta13/projects/google/zones/us-east1-a"))
|
".com/compute/v1beta13/projects/myproject/zones/us-east1-a"))
|
||||||
.name("us-east1-a")
|
.name("us-east1-a")
|
||||||
.description("us-east1-a")
|
.description("us-east1-a")
|
||||||
.status(Zone.Status.UP)
|
.status(Zone.Status.UP)
|
||||||
|
|
|
@ -45,9 +45,9 @@ public class ParseZoneTest extends BaseGoogleComputeParseTest<Zone> {
|
||||||
return Zone.builder()
|
return Zone.builder()
|
||||||
.id("13020128040171887099")
|
.id("13020128040171887099")
|
||||||
.creationTimestamp(new SimpleDateFormatDateService().iso8601DateParse("2012-10-19T16:42:54.131"))
|
.creationTimestamp(new SimpleDateFormatDateService().iso8601DateParse("2012-10-19T16:42:54.131"))
|
||||||
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/google/zones/us-central2-a"))
|
.selfLink(URI.create("https://www.googleapis.com/compute/v1beta13/projects/myproject/zones/us-central1-a"))
|
||||||
.name("us-central2-a")
|
.name("us-central1-a")
|
||||||
.description("us-central2-a")
|
.description("us-central1-a")
|
||||||
.status(Zone.Status.DOWN)
|
.status(Zone.Status.DOWN)
|
||||||
.addMaintenanceWindow(Zone.MaintenanceWindow.builder()
|
.addMaintenanceWindow(Zone.MaintenanceWindow.builder()
|
||||||
.name("2012-11-10-planned-outage")
|
.name("2012-11-10-planned-outage")
|
||||||
|
|
|
@ -5,6 +5,6 @@
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test",
|
||||||
"name": "default",
|
"name": "default",
|
||||||
"description": "Default network for the project",
|
"description": "Default network for the project",
|
||||||
"IPv4Range": "10.240.0.0/16",
|
"IPv4Range": "10.0.0.0/8",
|
||||||
"gatewayIPv4": "10.240.0.1"
|
"gatewayIPv4": "10.0.0.1"
|
||||||
}
|
}
|
|
@ -1 +1 @@
|
||||||
{"name":"test-network","IPv4Range":"10.0.1.0/8"}
|
{"name":"test-network","IPv4Range":"10.0.0.0/8"}
|
|
@ -11,8 +11,8 @@
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/networks/jclouds-test",
|
||||||
"name": "default",
|
"name": "default",
|
||||||
"description": "Default network for the project",
|
"description": "Default network for the project",
|
||||||
"IPv4Range": "10.240.0.0/16",
|
"IPv4Range": "10.0.0.0/8",
|
||||||
"gatewayIPv4": "10.240.0.1"
|
"gatewayIPv4": "10.0.0.1"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
|
@ -2,9 +2,9 @@
|
||||||
"kind": "compute#zone",
|
"kind": "compute#zone",
|
||||||
"id": "13020128040171887099",
|
"id": "13020128040171887099",
|
||||||
"creationTimestamp": "2012-10-19T16:42:54.131",
|
"creationTimestamp": "2012-10-19T16:42:54.131",
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/google/zones/us-central2-a",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/zones/us-central1-a",
|
||||||
"name": "us-central2-a",
|
"name": "us-central1-a",
|
||||||
"description": "us-central2-a",
|
"description": "us-central1-a",
|
||||||
"status": "DOWN",
|
"status": "DOWN",
|
||||||
"maintenanceWindows": [
|
"maintenanceWindows": [
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
{
|
{
|
||||||
"kind": "compute#zoneList",
|
"kind": "compute#zoneList",
|
||||||
"id": "projects/google/zones",
|
"id": "projects/myproject/zones",
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/google/zones",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/zones",
|
||||||
"items": [
|
"items": [
|
||||||
{
|
{
|
||||||
"kind": "compute#zone",
|
"kind": "compute#zone",
|
||||||
"id": "13020128040171887099",
|
"id": "13020128040171887099",
|
||||||
"creationTimestamp": "2012-10-19T16:42:54.131",
|
"creationTimestamp": "2012-10-19T16:42:54.131",
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/google/zones/us-central2-a",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/zones/us-central1-a",
|
||||||
"name": "us-central2-a",
|
"name": "us-central1-a",
|
||||||
"description": "us-central2-a",
|
"description": "us-central1-a",
|
||||||
"status": "DOWN",
|
"status": "DOWN",
|
||||||
"maintenanceWindows": [
|
"maintenanceWindows": [
|
||||||
{
|
{
|
||||||
|
@ -24,7 +24,7 @@
|
||||||
"kind": "compute#zone",
|
"kind": "compute#zone",
|
||||||
"id": "13024414164050619686",
|
"id": "13024414164050619686",
|
||||||
"creationTimestamp": "2012-10-24T20:13:19.271",
|
"creationTimestamp": "2012-10-24T20:13:19.271",
|
||||||
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/google/zones/us-east1-a",
|
"selfLink": "https://www.googleapis.com/compute/v1beta13/projects/myproject/zones/us-east1-a",
|
||||||
"name": "us-east1-a",
|
"name": "us-east1-a",
|
||||||
"description": "us-east1-a",
|
"description": "us-east1-a",
|
||||||
"status": "UP",
|
"status": "UP",
|
||||||
|
|
Loading…
Reference in New Issue