From 1bdf5eda18896acafc2cc71dde7394a0f06edea6 Mon Sep 17 00:00:00 2001 From: Alex Yarmula Date: Wed, 7 Apr 2010 16:10:37 -0700 Subject: [PATCH] added resolve images functionality (currently supported are ec2 and gogrid). current implementations are only resolving the credentials/authentication --- .../jclouds/aws/ec2/EC2ContextBuilder.java | 5 + .../internal/EC2AuthenticationModule.java | 37 + .../RunningInstanceToNodeMetadata.java | 11 +- .../EC2AuthenticateImagesStrategy.java | 48 ++ .../compute/EC2ComputeServiceLiveTest.java | 13 +- .../compute/ComputeServiceContextBuilder.java | 32 +- .../compute/config/ResolvesImages.java | 43 ++ .../strategy/AuthenticateImagesStrategy.java | 41 ++ .../jclouds/gogrid/GoGridContextBuilder.java | 8 +- .../GoGridComputeServiceContextModule.java | 656 +++++++++--------- .../internal/GoGridAuthenticationModule.java | 61 ++ 11 files changed, 614 insertions(+), 341 deletions(-) create mode 100644 aws/core/src/main/java/org/jclouds/aws/ec2/compute/config/internal/EC2AuthenticationModule.java create mode 100644 aws/core/src/main/java/org/jclouds/aws/ec2/compute/strategy/EC2AuthenticateImagesStrategy.java create mode 100644 compute/src/main/java/org/jclouds/compute/config/ResolvesImages.java create mode 100644 compute/src/main/java/org/jclouds/compute/strategy/AuthenticateImagesStrategy.java create mode 100644 gogrid/src/main/java/org/jclouds/gogrid/config/internal/GoGridAuthenticationModule.java diff --git a/aws/core/src/main/java/org/jclouds/aws/ec2/EC2ContextBuilder.java b/aws/core/src/main/java/org/jclouds/aws/ec2/EC2ContextBuilder.java index d7e2c32fa9..b96b51c24d 100755 --- a/aws/core/src/main/java/org/jclouds/aws/ec2/EC2ContextBuilder.java +++ b/aws/core/src/main/java/org/jclouds/aws/ec2/EC2ContextBuilder.java @@ -22,6 +22,7 @@ import java.util.List; import java.util.Properties; import org.jclouds.aws.ec2.compute.config.EC2ComputeServiceContextModule; +import org.jclouds.aws.ec2.compute.config.internal.EC2AuthenticationModule; import org.jclouds.aws.ec2.config.EC2RestClientModule; import org.jclouds.compute.ComputeServiceContextBuilder; import org.jclouds.http.config.JavaUrlHttpCommandExecutorServiceModule; @@ -68,4 +69,8 @@ public class EC2ContextBuilder extends modules.add(new EC2RestClientModule()); } + @Override + protected void addImageResolutionModule() { + modules.add(new EC2AuthenticationModule()); + } } diff --git a/aws/core/src/main/java/org/jclouds/aws/ec2/compute/config/internal/EC2AuthenticationModule.java b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/config/internal/EC2AuthenticationModule.java new file mode 100644 index 0000000000..fc49b5e415 --- /dev/null +++ b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/config/internal/EC2AuthenticationModule.java @@ -0,0 +1,37 @@ +/** + * + * Copyright (C) 2010 Cloud Conscious, LLC. + * + * ==================================================================== + * Licensed 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.aws.ec2.compute.config.internal; + +import com.google.inject.AbstractModule; +import org.jclouds.aws.ec2.compute.strategy.EC2AuthenticateImagesStrategy; +import org.jclouds.compute.config.ResolvesImages; +import org.jclouds.compute.strategy.AuthenticateImagesStrategy; + +/** + * @author Oleksiy Yarmula + */ +@ResolvesImages +public class EC2AuthenticationModule extends AbstractModule { + + @Override + protected void configure() { + bind(AuthenticateImagesStrategy.class).to(EC2AuthenticateImagesStrategy.class); + } + +} diff --git a/aws/core/src/main/java/org/jclouds/aws/ec2/compute/functions/RunningInstanceToNodeMetadata.java b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/functions/RunningInstanceToNodeMetadata.java index 5525e49067..aa64b6e125 100644 --- a/aws/core/src/main/java/org/jclouds/aws/ec2/compute/functions/RunningInstanceToNodeMetadata.java +++ b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/functions/RunningInstanceToNodeMetadata.java @@ -18,6 +18,7 @@ import org.jclouds.aws.ec2.services.AMIClient; import org.jclouds.compute.domain.NodeMetadata; import org.jclouds.compute.domain.NodeState; import org.jclouds.compute.domain.internal.NodeMetadataImpl; +import org.jclouds.compute.strategy.AuthenticateImagesStrategy; import org.jclouds.domain.Credentials; import com.google.common.base.Function; @@ -34,12 +35,15 @@ public class RunningInstanceToNodeMetadata implements Function credentialsMap; + private final AuthenticateImagesStrategy authenticator; @Inject public RunningInstanceToNodeMetadata(AMIClient amiClient, - Map credentialsMap) { + Map credentialsMap, + AuthenticateImagesStrategy authenticator) { this.amiClient = amiClient; this.credentialsMap = credentialsMap; + this.authenticator = authenticator; } @Override @@ -58,10 +62,7 @@ public class RunningInstanceToNodeMetadata implements Function extra = ImmutableMap. of(); diff --git a/aws/core/src/main/java/org/jclouds/aws/ec2/compute/strategy/EC2AuthenticateImagesStrategy.java b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/strategy/EC2AuthenticateImagesStrategy.java new file mode 100644 index 0000000000..cba673e759 --- /dev/null +++ b/aws/core/src/main/java/org/jclouds/aws/ec2/compute/strategy/EC2AuthenticateImagesStrategy.java @@ -0,0 +1,48 @@ +/** + * + * Copyright (C) 2010 Cloud Conscious, LLC. + * + * ==================================================================== + * Licensed 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.aws.ec2.compute.strategy; + +import org.jclouds.aws.ec2.domain.Image; +import org.jclouds.compute.strategy.AuthenticateImagesStrategy; +import org.jclouds.domain.Credentials; + +import static com.google.common.base.Preconditions.checkArgument; +import static com.google.common.base.Preconditions.checkNotNull; + +/** + * @author Oleksiy Yarmula + */ +public class EC2AuthenticateImagesStrategy implements AuthenticateImagesStrategy { + + @Override + public Credentials execute(Object resourceToAuthenticate) { + checkNotNull(resourceToAuthenticate); + checkArgument(resourceToAuthenticate instanceof Image, "Resource must be an image (for EC2)"); + Image image = (Image) resourceToAuthenticate; + + Credentials credentials = null; + + // canonical/alestic images use the ubuntu user to login + if (image.getImageOwnerId().matches("063491364108|099720109477")) + credentials = new Credentials("ubuntu", ""); + else credentials = new Credentials("root", ""); + + return credentials; + } +} diff --git a/aws/core/src/test/java/org/jclouds/aws/ec2/compute/EC2ComputeServiceLiveTest.java b/aws/core/src/test/java/org/jclouds/aws/ec2/compute/EC2ComputeServiceLiveTest.java index 57676a17be..68e5c9e95e 100644 --- a/aws/core/src/test/java/org/jclouds/aws/ec2/compute/EC2ComputeServiceLiveTest.java +++ b/aws/core/src/test/java/org/jclouds/aws/ec2/compute/EC2ComputeServiceLiveTest.java @@ -21,14 +21,13 @@ package org.jclouds.aws.ec2.compute; import static org.testng.Assert.assertEquals; import org.jclouds.compute.BaseComputeServiceLiveTest; -import org.jclouds.compute.domain.Architecture; -import org.jclouds.compute.domain.OsFamily; -import org.jclouds.compute.domain.Template; +import org.jclouds.compute.domain.*; import org.jclouds.ssh.jsch.config.JschSshClientModule; import org.testng.annotations.BeforeClass; import org.testng.annotations.Test; import com.google.common.collect.Iterables; +import java.util.Map; /** * @@ -62,6 +61,14 @@ public class EC2ComputeServiceLiveTest extends BaseComputeServiceLiveTest { Architecture.X86_32).imageId("ami-bb709dd2").build(); } + @Test + public void testCredentialsMapping() { +// Template simpleTemplate = client.templateBuilder().smallest().build(); +// client.runNodesWithTag("ec2", 1, simpleTemplate); + Map map = client.getNodesWithTag("ec2"); + int a = 5; + } + @Override protected JschSshClientModule getSshModule() { return new JschSshClientModule(); diff --git a/compute/src/main/java/org/jclouds/compute/ComputeServiceContextBuilder.java b/compute/src/main/java/org/jclouds/compute/ComputeServiceContextBuilder.java index 20f4f5a2a6..1819fa1b8c 100644 --- a/compute/src/main/java/org/jclouds/compute/ComputeServiceContextBuilder.java +++ b/compute/src/main/java/org/jclouds/compute/ComputeServiceContextBuilder.java @@ -18,8 +18,13 @@ */ package org.jclouds.compute; +import java.util.List; import java.util.Properties; +import com.google.common.base.Predicate; +import com.google.common.collect.Iterables; +import com.google.inject.Injector; +import org.jclouds.compute.config.ResolvesImages; import org.jclouds.compute.internal.ComputeServiceContextImpl; import org.jclouds.rest.RestContextBuilder; @@ -43,12 +48,18 @@ public abstract class ComputeServiceContextBuilder extends RestContextBuil } - /** + @Override + public Injector buildInjector() { + addImageResolutionModuleIfNotPresent(); + return super.buildInjector(); + } + + /** * {@inheritDoc} */ @Override public ComputeServiceContextBuilder withModules(Module... modules) { - return (ComputeServiceContextBuilder) super.withModules(modules); + return (ComputeServiceContextBuilder) super.withModules(modules); } public ComputeServiceContext buildComputeServiceContext() { @@ -57,4 +68,21 @@ public abstract class ComputeServiceContextBuilder extends RestContextBuil Key.get(Types.newParameterizedType(ComputeServiceContextImpl.class, asyncClientType .getType(), syncClientType.getType()))); } + + protected void addImageResolutionModuleIfNotPresent() { + if (!Iterables.any(modules, new Predicate() { + public boolean apply(Module input) { + return input.getClass().isAnnotationPresent(ResolvesImages.class); + } + + })) { + addImageResolutionModule(); + } + } + + @SuppressWarnings({"UnusedDeclaration"}) + protected void addImageResolutionModule() { + // do nothing; + // this is to be overridden when needed + } } \ No newline at end of file diff --git a/compute/src/main/java/org/jclouds/compute/config/ResolvesImages.java b/compute/src/main/java/org/jclouds/compute/config/ResolvesImages.java new file mode 100644 index 0000000000..48719fa83e --- /dev/null +++ b/compute/src/main/java/org/jclouds/compute/config/ResolvesImages.java @@ -0,0 +1,43 @@ +/** + * + * Copyright (C) 2010 Cloud Conscious, LLC. + * + * ==================================================================== + * Licensed 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.compute.config; + +import java.lang.annotation.Retention; +import java.lang.annotation.Target; + +import static java.lang.annotation.ElementType.TYPE; +import static java.lang.annotation.RetentionPolicy.RUNTIME; + +/** + * Marks that the module can augment an images of + * a given cloud. + * + * Such a module binds a strategy to determine such things, + * as: + *
    + *
  • default username
  • + *
  • default password, specific to a cloud and image.
  • + *
+ * + * @author Oleksiy Yarmula + */ +@Retention(RUNTIME) +@Target(TYPE) +public @interface ResolvesImages { +} diff --git a/compute/src/main/java/org/jclouds/compute/strategy/AuthenticateImagesStrategy.java b/compute/src/main/java/org/jclouds/compute/strategy/AuthenticateImagesStrategy.java new file mode 100644 index 0000000000..d9ed74f01b --- /dev/null +++ b/compute/src/main/java/org/jclouds/compute/strategy/AuthenticateImagesStrategy.java @@ -0,0 +1,41 @@ +/** + * + * Copyright (C) 2010 Cloud Conscious, LLC. + * + * ==================================================================== + * Licensed 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.compute.strategy; + +import org.jclouds.domain.Credentials; + +/** + * @author Oleksiy Yarmula + */ +public interface AuthenticateImagesStrategy { + + /** + * Processes the resource to determine credentials. + * + * @param resourceToAuthenticate + * this can be any resource, such as an image, + * running server instance or other. It's the + * responsibility of an implementation to apply + * the cloud-specific logic. + * @return credentials object. Note: the key + * may not be set, but the account must be set + */ + Credentials execute(Object resourceToAuthenticate); + +} diff --git a/gogrid/src/main/java/org/jclouds/gogrid/GoGridContextBuilder.java b/gogrid/src/main/java/org/jclouds/gogrid/GoGridContextBuilder.java index 2bf3b8bc4e..9e48712e88 100644 --- a/gogrid/src/main/java/org/jclouds/gogrid/GoGridContextBuilder.java +++ b/gogrid/src/main/java/org/jclouds/gogrid/GoGridContextBuilder.java @@ -34,7 +34,7 @@ import org.jclouds.compute.ComputeServiceContextBuilder; import org.jclouds.compute.internal.ComputeServiceContextImpl; import org.jclouds.gogrid.config.GoGridComputeServiceContextModule; import org.jclouds.gogrid.config.GoGridRestClientModule; -import org.jclouds.gogrid.reference.GoGridConstants; +import org.jclouds.gogrid.config.internal.GoGridAuthenticationModule; import com.google.inject.Module; import com.google.inject.TypeLiteral; @@ -60,6 +60,11 @@ public class GoGridContextBuilder extends ComputeServiceContextBuilder>() { })); } - } diff --git a/gogrid/src/main/java/org/jclouds/gogrid/config/GoGridComputeServiceContextModule.java b/gogrid/src/main/java/org/jclouds/gogrid/config/GoGridComputeServiceContextModule.java index cb585296fb..534fd2c663 100644 --- a/gogrid/src/main/java/org/jclouds/gogrid/config/GoGridComputeServiceContextModule.java +++ b/gogrid/src/main/java/org/jclouds/gogrid/config/GoGridComputeServiceContextModule.java @@ -18,7 +18,7 @@ */ package org.jclouds.gogrid.config; -import static com.google.common.base.Preconditions.checkNotNull; +import static com.google.common.base.Preconditions.*; import static org.jclouds.compute.domain.OsFamily.CENTOS; import java.net.InetAddress; @@ -36,6 +36,7 @@ import javax.inject.Inject; import javax.inject.Named; import javax.inject.Singleton; +import com.google.inject.AbstractModule; import org.jclouds.compute.ComputeService; import org.jclouds.compute.ComputeServiceContext; import org.jclouds.compute.domain.Architecture; @@ -54,11 +55,7 @@ import org.jclouds.compute.internal.ComputeServiceContextImpl; import org.jclouds.compute.internal.TemplateBuilderImpl; import org.jclouds.compute.predicates.RunScriptRunning; import org.jclouds.compute.reference.ComputeServiceConstants; -import org.jclouds.compute.strategy.AddNodeWithTagStrategy; -import org.jclouds.compute.strategy.DestroyNodeStrategy; -import org.jclouds.compute.strategy.GetNodeMetadataStrategy; -import org.jclouds.compute.strategy.ListNodesStrategy; -import org.jclouds.compute.strategy.RebootNodeStrategy; +import org.jclouds.compute.strategy.*; import org.jclouds.domain.Credentials; import org.jclouds.domain.Location; import org.jclouds.domain.LocationScope; @@ -97,372 +94,373 @@ import com.google.inject.TypeLiteral; */ public class GoGridComputeServiceContextModule extends GoGridContextModule { - @Override - protected void configure() { - super.configure(); - bind(new TypeLiteral>() { - }).to(ServerToNodeMetadata.class); - bind(AddNodeWithTagStrategy.class).to(GoGridAddNodeWithTagStrategy.class); - bind(ListNodesStrategy.class).to(GoGridListNodesStrategy.class); - bind(GetNodeMetadataStrategy.class).to(GoGridGetNodeMetadataStrategy.class); - bind(RebootNodeStrategy.class).to(GoGridRebootNodeStrategy.class); - bind(DestroyNodeStrategy.class).to(GoGridDestroyNodeStrategy.class); - } + @Override + protected void configure() { + super.configure(); + bind(new TypeLiteral>() { + }).to(ServerToNodeMetadata.class); + bind(AddNodeWithTagStrategy.class).to(GoGridAddNodeWithTagStrategy.class); + bind(ListNodesStrategy.class).to(GoGridListNodesStrategy.class); + bind(GetNodeMetadataStrategy.class).to(GoGridGetNodeMetadataStrategy.class); + bind(RebootNodeStrategy.class).to(GoGridRebootNodeStrategy.class); + bind(DestroyNodeStrategy.class).to(GoGridDestroyNodeStrategy.class); + } - @Provides - TemplateBuilder provideTemplate(Map locations, - Map images, Map sizes, - Location defaultLocation) { - return new TemplateBuilderImpl(locations, images, sizes, defaultLocation).osFamily(CENTOS) - .imageNameMatches(".*w/ None.*"); - } + @Provides + TemplateBuilder provideTemplate(Map locations, + Map images, Map sizes, + Location defaultLocation) { + return new TemplateBuilderImpl(locations, images, sizes, defaultLocation).osFamily(CENTOS) + .imageNameMatches(".*w/ None.*"); + } - @Provides - @Named("NAMING_CONVENTION") - @Singleton - String provideNamingConvention() { - return "%s-%d"; - } + @Provides + @Named("NAMING_CONVENTION") + @Singleton + String provideNamingConvention() { + return "%s-%d"; + } - @Singleton - public static class GoGridAddNodeWithTagStrategy implements AddNodeWithTagStrategy { - private final GoGridClient client; - private final Function sizeToRam; - private final Function serverToNodeMetadata; - private RetryablePredicate serverLatestJobCompleted; - private RetryablePredicate serverLatestJobCompletedShort; + @Singleton + public static class GoGridAddNodeWithTagStrategy implements AddNodeWithTagStrategy { + private final GoGridClient client; + private final Function sizeToRam; + private final Function serverToNodeMetadata; + private RetryablePredicate serverLatestJobCompleted; + private RetryablePredicate serverLatestJobCompletedShort; - @Inject - protected GoGridAddNodeWithTagStrategy(GoGridClient client, - Function serverToNodeMetadata, Function sizeToRam) { - this.client = client; - this.serverToNodeMetadata = serverToNodeMetadata; - this.sizeToRam = sizeToRam; - this.serverLatestJobCompleted = new RetryablePredicate( - new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); - this.serverLatestJobCompletedShort = new RetryablePredicate( - new ServerLatestJobCompleted(client.getJobServices()), 60, 20, TimeUnit.SECONDS); - } + @Inject + protected GoGridAddNodeWithTagStrategy(GoGridClient client, + Function serverToNodeMetadata, Function sizeToRam) { + this.client = client; + this.serverToNodeMetadata = serverToNodeMetadata; + this.sizeToRam = sizeToRam; + this.serverLatestJobCompleted = new RetryablePredicate( + new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); + this.serverLatestJobCompletedShort = new RetryablePredicate( + new ServerLatestJobCompleted(client.getJobServices()), 60, 20, TimeUnit.SECONDS); + } - @Override - public NodeMetadata execute(String tag, String name, Template template) { - Server addedServer = null; - boolean notStarted = true; - int numOfRetries = 20; - // lock-free consumption of a shared resource: IP address pool - while (notStarted) { // TODO: replace with Predicate-based thread collision avoidance for - // simplicity - Set availableIps = client.getIpServices().getIpList( - new GetIpListOptions().onlyUnassigned().onlyWithType(IpType.PUBLIC)); - if (availableIps.size() == 0) - throw new RuntimeException("No public IPs available on this account."); - int ipIndex = new SecureRandom().nextInt(availableIps.size()); - Ip availableIp = Iterables.get(availableIps, ipIndex); - try { - addedServer = client.getServerServices().addServer(name, - checkNotNull(template.getImage().getId()), - sizeToRam.apply(template.getSize()), availableIp.getIp()); - notStarted = false; - } catch (Exception e) { - if (--numOfRetries == 0) - Throwables.propagate(e); - notStarted = true; + @Override + public NodeMetadata execute(String tag, String name, Template template) { + Server addedServer = null; + boolean notStarted = true; + int numOfRetries = 20; + // lock-free consumption of a shared resource: IP address pool + while (notStarted) { // TODO: replace with Predicate-based thread collision avoidance for + // simplicity + Set availableIps = client.getIpServices().getIpList( + new GetIpListOptions().onlyUnassigned().onlyWithType(IpType.PUBLIC)); + if (availableIps.size() == 0) + throw new RuntimeException("No public IPs available on this account."); + int ipIndex = new SecureRandom().nextInt(availableIps.size()); + Ip availableIp = Iterables.get(availableIps, ipIndex); + try { + addedServer = client.getServerServices().addServer(name, + checkNotNull(template.getImage().getId()), + sizeToRam.apply(template.getSize()), availableIp.getIp()); + notStarted = false; + } catch (Exception e) { + if (--numOfRetries == 0) + Throwables.propagate(e); + notStarted = true; + } } - } - serverLatestJobCompleted.apply(addedServer); + serverLatestJobCompleted.apply(addedServer); - client.getServerServices().power(addedServer.getName(), PowerCommand.START); - serverLatestJobCompletedShort.apply(addedServer); + client.getServerServices().power(addedServer.getName(), PowerCommand.START); + serverLatestJobCompletedShort.apply(addedServer); - addedServer = Iterables.getOnlyElement(client.getServerServices().getServersByName( - addedServer.getName())); - return serverToNodeMetadata.apply(addedServer); - } - } + addedServer = Iterables.getOnlyElement(client.getServerServices().getServersByName( + addedServer.getName())); + return serverToNodeMetadata.apply(addedServer); + } + } - @Singleton - public static class GoGridRebootNodeStrategy implements RebootNodeStrategy { - private final GoGridClient client; - private RetryablePredicate serverLatestJobCompleted; - private RetryablePredicate serverLatestJobCompletedShort; + @Singleton + public static class GoGridRebootNodeStrategy implements RebootNodeStrategy { + private final GoGridClient client; + private RetryablePredicate serverLatestJobCompleted; + private RetryablePredicate serverLatestJobCompletedShort; - @Inject - protected GoGridRebootNodeStrategy(GoGridClient client) { - this.client = client; - this.serverLatestJobCompleted = new RetryablePredicate( - new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); - this.serverLatestJobCompletedShort = new RetryablePredicate( - new ServerLatestJobCompleted(client.getJobServices()), 60, 20, TimeUnit.SECONDS); - } + @Inject + protected GoGridRebootNodeStrategy(GoGridClient client) { + this.client = client; + this.serverLatestJobCompleted = new RetryablePredicate( + new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); + this.serverLatestJobCompletedShort = new RetryablePredicate( + new ServerLatestJobCompleted(client.getJobServices()), 60, 20, TimeUnit.SECONDS); + } - @Override - public boolean execute(ComputeMetadata node) { - Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( - node.getName())); - client.getServerServices().power(server.getName(), PowerCommand.RESTART); - serverLatestJobCompleted.apply(server); - client.getServerServices().power(server.getName(), PowerCommand.START); - return serverLatestJobCompletedShort.apply(server); - } - } + @Override + public boolean execute(ComputeMetadata node) { + Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( + node.getName())); + client.getServerServices().power(server.getName(), PowerCommand.RESTART); + serverLatestJobCompleted.apply(server); + client.getServerServices().power(server.getName(), PowerCommand.START); + return serverLatestJobCompletedShort.apply(server); + } + } - @Singleton - public static class GoGridListNodesStrategy implements ListNodesStrategy { - private final GoGridClient client; - private final Function serverToNodeMetadata; + @Singleton + public static class GoGridListNodesStrategy implements ListNodesStrategy { + private final GoGridClient client; + private final Function serverToNodeMetadata; - @Inject - protected GoGridListNodesStrategy(GoGridClient client, - Function serverToNodeMetadata) { - this.client = client; - this.serverToNodeMetadata = serverToNodeMetadata; - } + @Inject + protected GoGridListNodesStrategy(GoGridClient client, + Function serverToNodeMetadata) { + this.client = client; + this.serverToNodeMetadata = serverToNodeMetadata; + } - @Override - public Iterable execute() { - return Iterables.transform(client.getServerServices().getServerList(), - serverToNodeMetadata); - } + @Override + public Iterable execute() { + return Iterables.transform(client.getServerServices().getServerList(), + serverToNodeMetadata); + } - } + } - @Singleton - public static class GoGridGetNodeMetadataStrategy implements GetNodeMetadataStrategy { - private final GoGridClient client; - private final Function serverToNodeMetadata; + @Singleton + public static class GoGridGetNodeMetadataStrategy implements GetNodeMetadataStrategy { + private final GoGridClient client; + private final Function serverToNodeMetadata; - @Inject - protected GoGridGetNodeMetadataStrategy(GoGridClient client, - Function serverToNodeMetadata) { - this.client = client; - this.serverToNodeMetadata = serverToNodeMetadata; - } + @Inject + protected GoGridGetNodeMetadataStrategy(GoGridClient client, + Function serverToNodeMetadata) { + this.client = client; + this.serverToNodeMetadata = serverToNodeMetadata; + } - @Override - public NodeMetadata execute(ComputeMetadata node) { - Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( - node.getName())); - return server == null ? null : serverToNodeMetadata.apply(server); - } - } + @Override + public NodeMetadata execute(ComputeMetadata node) { + Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( + node.getName())); + return server == null ? null : serverToNodeMetadata.apply(server); + } + } - @Singleton - public static class GoGridDestroyNodeStrategy implements DestroyNodeStrategy { - private final GoGridClient client; - private RetryablePredicate serverLatestJobCompleted; + @Singleton + public static class GoGridDestroyNodeStrategy implements DestroyNodeStrategy { + private final GoGridClient client; + private RetryablePredicate serverLatestJobCompleted; - @Inject - protected GoGridDestroyNodeStrategy(GoGridClient client) { - this.client = client; - this.serverLatestJobCompleted = new RetryablePredicate( - new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); - } + @Inject + protected GoGridDestroyNodeStrategy(GoGridClient client) { + this.client = client; + this.serverLatestJobCompleted = new RetryablePredicate( + new ServerLatestJobCompleted(client.getJobServices()), 800, 20, TimeUnit.SECONDS); + } - @Override - public boolean execute(ComputeMetadata node) { - Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( - node.getName())); - client.getServerServices().deleteByName(server.getName()); - return serverLatestJobCompleted.apply(server); - } + @Override + public boolean execute(ComputeMetadata node) { + Server server = Iterables.getOnlyElement(client.getServerServices().getServersByName( + node.getName())); + client.getServerServices().deleteByName(server.getName()); + return serverLatestJobCompleted.apply(server); + } - } + } - @Singleton - @Provides - Map provideServerToNodeState() { - return ImmutableMap. builder().put("On", NodeState.RUNNING).put( - "Starting", NodeState.PENDING).put("Off", NodeState.SUSPENDED).put("Saving", - NodeState.PENDING).put("Restarting", NodeState.PENDING).put("Stopping", - NodeState.PENDING).build(); - } + @Singleton + @Provides + Map provideServerToNodeState() { + return ImmutableMap. builder().put("On", NodeState.RUNNING).put( + "Starting", NodeState.PENDING).put("Off", NodeState.SUSPENDED).put("Saving", + NodeState.PENDING).put("Restarting", NodeState.PENDING).put("Stopping", + NodeState.PENDING).build(); + } - @Singleton - @Provides - Function provideStringIpToInetAddress() { - return new Function() { - @Override - public InetAddress apply(String from) { - try { - return InetAddress.getByName(from); - } catch (UnknownHostException e) { - // TODO: log the failure. - return null; + @Singleton + @Provides + Function provideStringIpToInetAddress() { + return new Function() { + @Override + public InetAddress apply(String from) { + try { + return InetAddress.getByName(from); + } catch (UnknownHostException e) { + // TODO: log the failure. + return null; + } } - } - }; - } + }; + } - /** - * Finds matches to required configurations. GoGrid's documentation only specifies how much RAM - * one can get with different instance types. The # of cores and disk sizes are purely empyrical - * and aren't guaranteed. However, these are the matches found: Ram: 512MB, CPU: 1 core, HDD: 28 - * GB Ram: 1GB, CPU: 1 core, HDD: 57 GB Ram: 2GB, CPU: 1 core, HDD: 113 GB Ram: 4GB, CPU: 3 - * cores, HDD: 233 GB Ram: 8GB, CPU: 6 cores, HDD: 462 GB (as of March 2010) - * - * @return matched size - */ - @Singleton - @Provides - Function provideSizeToRam() { - return new Function() { - @Override - public String apply(Size size) { - if (size.getRam() >= 8 * 1024 || size.getCores() >= 6 || size.getDisk() >= 450) - return "8GB"; - if (size.getRam() >= 4 * 1024 || size.getCores() >= 3 || size.getDisk() >= 230) - return "4GB"; - if (size.getRam() >= 2 * 1024 || size.getDisk() >= 110) - return "2GB"; - if (size.getRam() >= 1024 || size.getDisk() >= 55) - return "1GB"; - return "512MB"; /* smallest */ - } - }; - } + /** + * Finds matches to required configurations. GoGrid's documentation only specifies how much RAM + * one can get with different instance types. The # of cores and disk sizes are purely empyrical + * and aren't guaranteed. However, these are the matches found: Ram: 512MB, CPU: 1 core, HDD: 28 + * GB Ram: 1GB, CPU: 1 core, HDD: 57 GB Ram: 2GB, CPU: 1 core, HDD: 113 GB Ram: 4GB, CPU: 3 + * cores, HDD: 233 GB Ram: 8GB, CPU: 6 cores, HDD: 462 GB (as of March 2010) + * + * @return matched size + */ + @Singleton + @Provides + Function provideSizeToRam() { + return new Function() { + @Override + public String apply(Size size) { + if (size.getRam() >= 8 * 1024 || size.getCores() >= 6 || size.getDisk() >= 450) + return "8GB"; + if (size.getRam() >= 4 * 1024 || size.getCores() >= 3 || size.getDisk() >= 230) + return "4GB"; + if (size.getRam() >= 2 * 1024 || size.getDisk() >= 110) + return "2GB"; + if (size.getRam() >= 1024 || size.getDisk() >= 55) + return "1GB"; + return "512MB"; /* smallest */ + } + }; + } - @Singleton - private static class ServerToNodeMetadata implements Function { - private final Map serverStateToNodeState; - private final Function stringIpToInetAddress; - private final GoGridClient client; + @Singleton + private static class ServerToNodeMetadata implements Function { + private final Map serverStateToNodeState; + private final Function stringIpToInetAddress; + private final GoGridClient client; + private final AuthenticateImagesStrategy authenticator; - @SuppressWarnings("unused") - @Inject - ServerToNodeMetadata(Map serverStateToNodeState, - Function stringIpToInetAddress, GoGridClient client) { - this.serverStateToNodeState = serverStateToNodeState; - this.stringIpToInetAddress = stringIpToInetAddress; - this.client = client; - } + @SuppressWarnings("unused") + @Inject + ServerToNodeMetadata(Map serverStateToNodeState, + Function stringIpToInetAddress, GoGridClient client, + AuthenticateImagesStrategy authenticator) { + this.serverStateToNodeState = serverStateToNodeState; + this.stringIpToInetAddress = stringIpToInetAddress; + this.client = client; + this.authenticator = authenticator; + } - @Override - public NodeMetadata apply(Server from) { - String locationId = "Unavailable"; - String tag = CharMatcher.JAVA_LETTER.retainFrom(from.getName()); - Credentials creds = client.getServerServices().getServerCredentialsList().get( - from.getName()); - Set ipSet = ImmutableSet - .of(stringIpToInetAddress.apply(from.getIp().getIp())); - NodeState state = serverStateToNodeState.get(from.getState().getName()); + @Override + public NodeMetadata apply(Server from) { + String locationId = "Unavailable"; + String tag = CharMatcher.JAVA_LETTER.retainFrom(from.getName()); + Set ipSet = ImmutableSet + .of(stringIpToInetAddress.apply(from.getIp().getIp())); + NodeState state = serverStateToNodeState.get(from.getState().getName()); + Credentials creds = authenticator.execute(from); + return new NodeMetadataImpl(from.getId() + "", from.getName(), locationId, null, + ImmutableMap. of(), tag, state, ipSet, ImmutableList + . of(), ImmutableMap. of(), creds); + } + } - return new NodeMetadataImpl(from.getId() + "", from.getName(), locationId, null, - ImmutableMap. of(), tag, state, ipSet, ImmutableList - . of(), ImmutableMap. of(), creds); - } - } + @Provides + @Singleton + ComputeServiceContext provideContext(ComputeService computeService, + RestContext context) { + return new ComputeServiceContextImpl(computeService, context); + } - @Provides - @Singleton - ComputeServiceContext provideContext(ComputeService computeService, - RestContext context) { - return new ComputeServiceContextImpl(computeService, context); - } + @Provides + @Singleton + @Named("NOT_RUNNING") + protected Predicate runScriptRunning(RunScriptRunning stateRunning) { + return new RetryablePredicate(Predicates.not(stateRunning), 600, 3, + TimeUnit.SECONDS); + } - @Provides - @Singleton - @Named("NOT_RUNNING") - protected Predicate runScriptRunning(RunScriptRunning stateRunning) { - return new RetryablePredicate(Predicates.not(stateRunning), 600, 3, - TimeUnit.SECONDS); - } + @Provides + @Singleton + Location getDefaultLocation(Map locations) { + return locations.get("SANFRANCISCO"); + } - @Provides - @Singleton - Location getDefaultLocation(Map locations) { - return locations.get("SANFRANCISCO"); - } + @Provides + @Singleton + Map getDefaultLocations(GoGridClient sync, LogHolder holder, + Function indexer) { + final Set locations = Sets.newHashSet(); + holder.logger.debug(">> providing locations"); + locations.add(new LocationImpl(LocationScope.ZONE, "SANFRANCISCO", "San Francisco, CA", null, + true)); + holder.logger.debug("<< locations(%d)", locations.size()); + return Maps.uniqueIndex(locations, new Function() { - @Provides - @Singleton - Map getDefaultLocations(GoGridClient sync, LogHolder holder, - Function indexer) { - final Set locations = Sets.newHashSet(); - holder.logger.debug(">> providing locations"); - locations.add(new LocationImpl(LocationScope.ZONE, "SANFRANCISCO", "San Francisco, CA", null, - true)); - holder.logger.debug("<< locations(%d)", locations.size()); - return Maps.uniqueIndex(locations, new Function() { + @Override + public String apply(Location from) { + return from.getId(); + } + }); + } - @Override - public String apply(Location from) { - return from.getId(); - } - }); - } + @Provides + @Singleton + protected Function indexer() { + return new Function() { + @Override + public String apply(ComputeMetadata from) { + return from.getId(); + } + }; + } - @Provides - @Singleton - protected Function indexer() { - return new Function() { - @Override - public String apply(ComputeMetadata from) { - return from.getId(); - } - }; - } - - @Provides - @Singleton - protected Map provideSizes(GoGridClient sync, - Map images, LogHolder holder, - Function indexer) throws InterruptedException, + @Provides + @Singleton + protected Map provideSizes(GoGridClient sync, + Map images, LogHolder holder, + Function indexer) throws InterruptedException, TimeoutException, ExecutionException { - final Set sizes = Sets.newHashSet(); - holder.logger.debug(">> providing sizes"); + final Set sizes = Sets.newHashSet(); + holder.logger.debug(">> providing sizes"); - sizes.add(new SizeImpl("1", "1", null, null, ImmutableMap. of(), 1, 512, 28, - ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); - sizes.add(new SizeImpl("2", "2", null, null, ImmutableMap. of(), 1, 1024, 57, - ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); - sizes.add(new SizeImpl("3", "3", null, null, ImmutableMap. of(), 1, 2048, - 113, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); - sizes.add(new SizeImpl("4", "4", null, null, ImmutableMap. of(), 3, 4096, - 233, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); - sizes.add(new SizeImpl("5", "5", null, null, ImmutableMap. of(), 6, 8192, - 462, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); - holder.logger.debug("<< sizes(%d)", sizes.size()); - return Maps.uniqueIndex(sizes, indexer); - } + sizes.add(new SizeImpl("1", "1", null, null, ImmutableMap. of(), 1, 512, 28, + ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); + sizes.add(new SizeImpl("2", "2", null, null, ImmutableMap. of(), 1, 1024, 57, + ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); + sizes.add(new SizeImpl("3", "3", null, null, ImmutableMap. of(), 1, 2048, + 113, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); + sizes.add(new SizeImpl("4", "4", null, null, ImmutableMap. of(), 3, 4096, + 233, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); + sizes.add(new SizeImpl("5", "5", null, null, ImmutableMap. of(), 6, 8192, + 462, ImmutableSet. of(Architecture.X86_32, Architecture.X86_64))); + holder.logger.debug("<< sizes(%d)", sizes.size()); + return Maps.uniqueIndex(sizes, indexer); + } - private static class LogHolder { - @Resource - @Named(ComputeServiceConstants.COMPUTE_LOGGER) - protected Logger logger = Logger.NULL; - } + private static class LogHolder { + @Resource + @Named(ComputeServiceConstants.COMPUTE_LOGGER) + protected Logger logger = Logger.NULL; + } - public static final Pattern GOGRID_OS_NAME_PATTERN = Pattern.compile("([a-zA-Z]*)(.*)"); + public static final Pattern GOGRID_OS_NAME_PATTERN = Pattern.compile("([a-zA-Z]*)(.*)"); - @Provides - @Singleton - protected Map provideImages(final GoGridClient sync, LogHolder holder, - Function indexer, Location location) + @Provides + @Singleton + protected Map provideImages(final GoGridClient sync, LogHolder holder, + Function indexer, Location location) throws InterruptedException, ExecutionException, TimeoutException { - final Set images = Sets.newHashSet(); - holder.logger.debug(">> providing images"); - Set allImages = sync.getImageServices().getImageList(); - for (ServerImage from : allImages) { - OsFamily os = null; - Architecture arch = (from.getOs().getName().indexOf("64") == -1 && from.getDescription() - .indexOf("64") == -1) ? Architecture.X86_32 : Architecture.X86_64; - String osDescription; - String version = ""; + final Set images = Sets.newHashSet(); + holder.logger.debug(">> providing images"); + Set allImages = sync.getImageServices().getImageList(); + for (ServerImage from : allImages) { + OsFamily os = null; + Architecture arch = (from.getOs().getName().indexOf("64") == -1 && from.getDescription() + .indexOf("64") == -1) ? Architecture.X86_32 : Architecture.X86_64; + String osDescription; + String version = ""; - osDescription = from.getOs().getName(); + osDescription = from.getOs().getName(); - String matchedOs = GoGridUtils.parseStringByPatternAndGetNthMatchGroup(from.getOs() - .getName(), GOGRID_OS_NAME_PATTERN, 1); - try { - os = OsFamily.fromValue(matchedOs.toLowerCase()); - } catch (IllegalArgumentException e) { - holder.logger.debug("<< didn't match os(%s)", matchedOs); - } + String matchedOs = GoGridUtils.parseStringByPatternAndGetNthMatchGroup(from.getOs() + .getName(), GOGRID_OS_NAME_PATTERN, 1); + try { + os = OsFamily.fromValue(matchedOs.toLowerCase()); + } catch (IllegalArgumentException e) { + holder.logger.debug("<< didn't match os(%s)", matchedOs); + } - images.add(new ImageImpl(from.getId() + "", from.getFriendlyName(), location.getId(), - null, ImmutableMap. of(), from.getDescription(), version, os, - osDescription, arch)); - } - holder.logger.debug("<< images(%d)", images.size()); - return Maps.uniqueIndex(images, indexer); - } + images.add(new ImageImpl(from.getId() + "", from.getFriendlyName(), location.getId(), + null, ImmutableMap. of(), from.getDescription(), version, os, + osDescription, arch)); + } + holder.logger.debug("<< images(%d)", images.size()); + return Maps.uniqueIndex(images, indexer); + } } diff --git a/gogrid/src/main/java/org/jclouds/gogrid/config/internal/GoGridAuthenticationModule.java b/gogrid/src/main/java/org/jclouds/gogrid/config/internal/GoGridAuthenticationModule.java new file mode 100644 index 0000000000..377d68a96e --- /dev/null +++ b/gogrid/src/main/java/org/jclouds/gogrid/config/internal/GoGridAuthenticationModule.java @@ -0,0 +1,61 @@ +/** + * + * Copyright (C) 2010 Cloud Conscious, LLC. + * + * ==================================================================== + * Licensed 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.gogrid.config.internal; + +import com.google.inject.AbstractModule; +import org.jclouds.compute.config.ResolvesImages; +import org.jclouds.compute.strategy.AuthenticateImagesStrategy; +import org.jclouds.domain.Credentials; +import org.jclouds.gogrid.GoGridClient; +import org.jclouds.gogrid.domain.Server; + +import javax.inject.Inject; + +import static com.google.common.base.Preconditions.checkArgument; +import static com.google.common.base.Preconditions.checkNotNull; + +/** + * @author Oleksiy Yarmula + */ +@ResolvesImages +public class GoGridAuthenticationModule extends AbstractModule { + @Override + protected void configure() { + bind(AuthenticateImagesStrategy.class).to(GoGridAuthenticateImagesStrategy.class); + } + + public static class GoGridAuthenticateImagesStrategy implements AuthenticateImagesStrategy { + private final GoGridClient client; + + @Inject + protected GoGridAuthenticateImagesStrategy(GoGridClient client) { + this.client = client; + } + + @Override + public Credentials execute(Object resourceToAuthenticate) { + checkNotNull(resourceToAuthenticate); + checkArgument(resourceToAuthenticate instanceof Server, "Resource must be a server (for GoGrid)"); + Server server = (Server) resourceToAuthenticate; + return client.getServerServices().getServerCredentialsList().get( + server.getName()); + } + } + +}