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

org.jclouds.azurecompute.arm.compute.extensions.AzureComputeImageExtension Maven / Gradle / Ivy

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

import static shaded.com.google.common.base.Functions.compose;
import static shaded.com.google.common.base.Preconditions.checkArgument;
import static shaded.com.google.common.base.Preconditions.checkState;
import static org.jclouds.azurecompute.arm.compute.domain.ResourceGroupAndName.fromSlashEncoded;
import static org.jclouds.azurecompute.arm.config.AzureComputeProperties.TIMEOUT_RESOURCE_DELETED;
import static org.jclouds.compute.config.ComputeServiceProperties.TIMEOUT_NODE_SUSPENDED;

import java.net.URI;
import java.util.concurrent.Callable;

import javax.annotation.Resource;

import org.jclouds.Constants;
import org.jclouds.azurecompute.arm.AzureComputeApi;
import org.jclouds.azurecompute.arm.compute.config.AzurePredicatesModule.ImageAvailablePredicateFactory;
import org.jclouds.azurecompute.arm.compute.config.AzurePredicatesModule.VirtualMachineInStatePredicateFactory;
import org.jclouds.azurecompute.arm.compute.domain.ResourceGroupAndName;
import org.jclouds.azurecompute.arm.compute.functions.CustomImageToVMImage;
import org.jclouds.azurecompute.arm.domain.IdReference;
import org.jclouds.azurecompute.arm.domain.ImageProperties;
import org.jclouds.azurecompute.arm.domain.VMImage;
import org.jclouds.azurecompute.arm.domain.VirtualMachine;
import org.jclouds.compute.domain.CloneImageTemplate;
import org.jclouds.compute.domain.Image;
import org.jclouds.compute.domain.ImageTemplate;
import org.jclouds.compute.domain.ImageTemplateBuilder;
import org.jclouds.compute.extensions.ImageExtension;
import org.jclouds.compute.reference.ComputeServiceConstants;
import org.jclouds.logging.Logger;

import shaded.com.google.common.base.Function;
import shaded.com.google.common.base.Predicate;
import shaded.com.google.common.util.concurrent.ListenableFuture;
import shaded.com.google.common.util.concurrent.ListeningExecutorService;
import shaded.com.google.inject.Inject;
import shaded.com.google.inject.name.Named;

public class AzureComputeImageExtension implements ImageExtension {
   public static final String CUSTOM_IMAGE_OFFER = "custom";

   @Resource
   @Named(ComputeServiceConstants.COMPUTE_LOGGER)
   protected Logger logger = Logger.NULL;

   private final AzureComputeApi api;
   private final ListeningExecutorService userExecutor;
   private final ImageAvailablePredicateFactory imageAvailablePredicate;
   private final VirtualMachineInStatePredicateFactory nodeSuspendedPredicate;
   private final Function vmImageToImage;
   private final Predicate resourceDeleted;
   private final CustomImageToVMImage customImagetoVmImage;

   @Inject
   AzureComputeImageExtension(AzureComputeApi api, ImageAvailablePredicateFactory imageAvailablePredicate,
         @Named(TIMEOUT_NODE_SUSPENDED) VirtualMachineInStatePredicateFactory nodeSuspendedPredicate,
         @Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
         Function vmImageToImage, @Named(TIMEOUT_RESOURCE_DELETED) Predicate resourceDeleted,
         CustomImageToVMImage customImagetoVmImage) {
      this.api = api;
      this.imageAvailablePredicate = imageAvailablePredicate;
      this.nodeSuspendedPredicate = nodeSuspendedPredicate;
      this.userExecutor = userExecutor;
      this.vmImageToImage = vmImageToImage;
      this.resourceDeleted = resourceDeleted;
      this.customImagetoVmImage = customImagetoVmImage;
   }

   @Override
   public ImageTemplate buildImageTemplateFromNode(String name, String id) {
      return new ImageTemplateBuilder.CloneImageTemplateBuilder().nodeId(id).name(name.toLowerCase()).build();
   }

   @Override
   public ListenableFuture createImage(ImageTemplate template) {
      final CloneImageTemplate cloneTemplate = (CloneImageTemplate) template;
      final ResourceGroupAndName resourceGroupAndName = fromSlashEncoded(cloneTemplate.getSourceNodeId());
      final String resourceGroupName = resourceGroupAndName.resourceGroup();
      final String vmName = resourceGroupAndName.name();

      final VirtualMachine vm = api.getVirtualMachineApi(resourceGroupName).get(vmName);
      final IdReference vmIdRef = IdReference.create(vm.id());

      logger.debug(">> stopping node %s...", cloneTemplate.getSourceNodeId());
      api.getVirtualMachineApi(resourceGroupName).stop(vmName);
      checkState(nodeSuspendedPredicate.create(resourceGroupName).apply(vmName),
            "Node %s was not suspended within the configured time limit", cloneTemplate.getSourceNodeId());

      return userExecutor.submit(new Callable() {
         @Override
         public Image call() throws Exception {
            logger.debug(">> generalizing virtal machine %s...", vmName);

            api.getVirtualMachineApi(resourceGroupName).generalize(vmName);

            org.jclouds.azurecompute.arm.domain.Image imageFromVM = api.getVirtualMachineImageApi(resourceGroupName)
                  .createOrUpdate(cloneTemplate.getName(), vm.location(),
                        ImageProperties.builder().sourceVirtualMachine(vmIdRef).build());

            checkState(imageAvailablePredicate.create(resourceGroupName).apply(imageFromVM.name()),
                  "Image for node %s was not created within the configured time limit", cloneTemplate.getName());

            return compose(vmImageToImage, customImagetoVmImage).apply(imageFromVM);
         }
      });
   }

   @Override
   public boolean deleteImage(String id) {
      VMImage image = VMImage.decodeFieldsFromUniqueId(id);
      checkArgument(image.custom(), "Only custom images can be deleted");

      logger.debug(">> deleting image %s", id);
      URI uri = api.getVirtualMachineImageApi(image.resourceGroup()).delete(image.name());
      return resourceDeleted.apply(uri);
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy