All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jclouds.compute.strategy.impl.CreateNodesWithGroupEncodedIntoNameThenAddToSet Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.jclouds.compute.strategy.impl;

import static com.google.common.base.MoreObjects.toStringHelper;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.collect.Iterables.any;
import static com.google.common.collect.Maps.newLinkedHashMap;
import static com.google.common.collect.Sets.newLinkedHashSet;
import static org.jclouds.compute.util.ComputeServiceUtils.formatStatus;

import java.util.Map;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicReference;

import javax.annotation.Resource;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;

import com.google.common.base.MoreObjects;
import org.jclouds.Constants;
import org.jclouds.compute.config.CustomizationResponse;
import org.jclouds.compute.domain.ComputeMetadata;
import org.jclouds.compute.domain.NodeMetadata;
import org.jclouds.compute.domain.Template;
import org.jclouds.compute.functions.GroupNamingConvention;
import org.jclouds.compute.reference.ComputeServiceConstants;
import org.jclouds.compute.strategy.CreateNodeWithGroupEncodedIntoName;
import org.jclouds.compute.strategy.CreateNodesInGroupThenAddToSet;
import org.jclouds.compute.strategy.CustomizeNodeAndAddToGoodMapOrPutExceptionIntoBadMap;
import org.jclouds.compute.strategy.ListNodesStrategy;
import org.jclouds.logging.Logger;

import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Multimap;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;

/**
 * creates futures that correlate to
 */
@Singleton
public class CreateNodesWithGroupEncodedIntoNameThenAddToSet implements CreateNodesInGroupThenAddToSet {

   protected class AddNode implements Callable> {
      private final String name;
      private final String group;
      private final Template template;

      public AddNode(String name, String group, Template template) {
         this.name = checkNotNull(name, "name");
         this.group = checkNotNull(group, "group");
         this.template = checkNotNull(template, "template");
      }

      @Override
      public AtomicReference call() throws Exception {
         NodeMetadata node = null;
         logger.debug(">> adding node location(%s) name(%s) image(%s) hardware(%s)", template.getLocation().getId(),
               name, MoreObjects.firstNonNull(template.getImage().getProviderId(), template.getImage().getId()),
               MoreObjects.firstNonNull(template.getHardware().getProviderId(), template.getHardware().getId()));
         node = addNodeWithGroupStrategy.createNodeWithGroupEncodedIntoName(group, name, template);
         logger.debug("<< %s node(%s)", formatStatus(node), node.getId());
         return new AtomicReference(node);
      }

      public String toString() {
         return toStringHelper(this).add("name", name).add("group", group).add("template", template).toString();
      }

   }

   @Resource
   @Named(ComputeServiceConstants.COMPUTE_LOGGER)
   protected Logger logger = Logger.NULL;
   protected final CreateNodeWithGroupEncodedIntoName addNodeWithGroupStrategy;
   protected final ListNodesStrategy listNodesStrategy;
   protected final GroupNamingConvention.Factory namingConvention;
   protected final ListeningExecutorService userExecutor;
   protected final CustomizeNodeAndAddToGoodMapOrPutExceptionIntoBadMap.Factory customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory;

   @Inject
   protected CreateNodesWithGroupEncodedIntoNameThenAddToSet(
            CreateNodeWithGroupEncodedIntoName addNodeWithGroupStrategy,
            ListNodesStrategy listNodesStrategy,
            GroupNamingConvention.Factory namingConvention,
            @Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
            CustomizeNodeAndAddToGoodMapOrPutExceptionIntoBadMap.Factory customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory) {
      this.addNodeWithGroupStrategy = addNodeWithGroupStrategy;
      this.listNodesStrategy = listNodesStrategy;
      this.namingConvention = namingConvention;
      this.userExecutor = userExecutor;
      this.customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory = customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory;
   }

   /**
    * This implementation gets a list of acceptable node names to encode the group into, then it
    * simultaneously runs the nodes and applies options to them.
    */
   @Override
   public Map> execute(String group, int count, Template template, Set goodNodes,
            Map badNodes, Multimap customizationResponses) {
      Map> responses = newLinkedHashMap();
      for (String name : getNextNames(group, template, count)) {
         responses.put(name, Futures.transform(createNodeInGroupWithNameAndTemplate(group, name, template),
                  customizeNodeAndAddToGoodMapOrPutExceptionIntoBadMapFactory.create(template.getOptions(), goodNodes,
                           badNodes, customizationResponses), userExecutor));
      }
      return responses;
   }

   /**
    * This calls logic necessary to create a node and convert it from its provider-specific object
    * to the jclouds {@link NodeMetadata} object. This call directly precedes customization, such as
    * executing scripts.
    * 
    * 

The outcome of this operation does not imply the node is {@link Status#RUNNING * running}. If you want to insert logic after the node is created, yet before an attempt to * customize the node, then append your behaviour to this method. * * ex. to attach an ip address post-creation * *
    * @Override
    * protected ListenableFuture<AtomicReference<NodeMetadata>> createNodeInGroupWithNameAndTemplate(String group, String name,
    *          Template template) {
    * 
    *    ListenableFuture<AtomicReference<NodeMetadata>> future = super.addNodeIntoGroupWithNameAndTemplate(group, name, template);
    *    return Futures.compose(future, new Function<AtomicReference<NodeMetadata>, AtomicReference<NodeMetadata>>() {
    * 
    *       @Override
    *       public AtomicReference<NodeMetadata> apply(AtomicReference<NodeMetadata> input) {
    *          NodeMetadata node = input.get();
    *          // allocate and attach an ip
    *          input.set(NodeMetadataBuilder.fromNodeMetadata(node).publicAddresses(ImmutableSet.of(ip.getIp())).build());
    *          return input;
    *       }
    * 
    *    }, executor);
    * }
    * 
* * @param group group the node belongs to * @param name generated name of the node * @param template user-specified template * @return node that is created, yet not necessarily in {@link Status#RUNNING} */ protected ListenableFuture> createNodeInGroupWithNameAndTemplate(String group, String name, Template template) { return userExecutor.submit(new AddNode(name, group, template)); } /** * Find the next node names that can be used. If the nodeNames template option is not specified * or is empty, these will be derived from the group and the template. We will pre-allocate a * specified quantity, and attempt to verify that there is no name conflict with the current * service. If the nodeNames option is specified, names from that will be used instead, without * any check for name conflicts. * If there are insufficient names in nodeNames, subsequent names will be generated in the * default format. * * @param group * @param count * @param template * @return */ protected Set getNextNames(final String group, final Template template, int count) { Set names = newLinkedHashSet(); Set nodeNames = template.getOptions().getNodeNames(); if (nodeNames.size() >= count) { return ImmutableSet.copyOf(Iterables.limit(nodeNames, count)); } else { names.addAll(nodeNames); } Iterable currentNodes = listNodesStrategy.listNodes(); int maxTries = 100; int currentTries = 0; while (names.size() < count && currentTries++ < maxTries) { final String name = namingConvention.createWithoutPrefix().uniqueNameForGroup(group); if (!any(currentNodes, new Predicate() { @Override public boolean apply(ComputeMetadata input) { return name.equals(input.getName()); } })) { names.add(name); } } return names; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy