mirror of https://github.com/apache/jclouds.git
Merge pull request #363 from andreaturli/development
issue 384: GuestAdditionsInstaller
This commit is contained in:
commit
df65c91a54
|
@ -27,7 +27,7 @@ package org.jclouds.virtualbox.config;
|
||||||
*/
|
*/
|
||||||
public interface VirtualBoxConstants {
|
public interface VirtualBoxConstants {
|
||||||
|
|
||||||
public static final String VIRTUALBOX_IMAGE_PREFIX = "jclouds#image#";
|
public static final String VIRTUALBOX_IMAGE_PREFIX = "jclouds-image-";
|
||||||
|
|
||||||
public static final String VIRTUALBOX_PRECONFIGURATION_URL = "jclouds.virtualbox.preconfigurationurl";
|
public static final String VIRTUALBOX_PRECONFIGURATION_URL = "jclouds.virtualbox.preconfigurationurl";
|
||||||
|
|
||||||
|
|
|
@ -24,6 +24,8 @@ import com.google.common.base.Splitter;
|
||||||
import com.google.common.base.Supplier;
|
import com.google.common.base.Supplier;
|
||||||
import com.google.common.cache.LoadingCache;
|
import com.google.common.cache.LoadingCache;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
|
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
import org.jclouds.compute.reference.ComputeServiceConstants;
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
import org.jclouds.logging.Logger;
|
import org.jclouds.logging.Logger;
|
||||||
import org.jclouds.ssh.SshClient;
|
import org.jclouds.ssh.SshClient;
|
||||||
|
@ -32,6 +34,7 @@ import org.jclouds.virtualbox.domain.ExecutionType;
|
||||||
import org.jclouds.virtualbox.domain.IsoSpec;
|
import org.jclouds.virtualbox.domain.IsoSpec;
|
||||||
import org.jclouds.virtualbox.domain.MasterSpec;
|
import org.jclouds.virtualbox.domain.MasterSpec;
|
||||||
import org.jclouds.virtualbox.domain.VmSpec;
|
import org.jclouds.virtualbox.domain.VmSpec;
|
||||||
|
import org.jclouds.virtualbox.predicates.GuestAdditionsInstaller;
|
||||||
import org.jclouds.virtualbox.util.MachineUtils;
|
import org.jclouds.virtualbox.util.MachineUtils;
|
||||||
import org.virtualbox_4_1.*;
|
import org.virtualbox_4_1.*;
|
||||||
|
|
||||||
|
@ -50,7 +53,8 @@ public class CreateAndInstallVm implements Function<MasterSpec, IMachine> {
|
||||||
@Resource
|
@Resource
|
||||||
@Named(ComputeServiceConstants.COMPUTE_LOGGER)
|
@Named(ComputeServiceConstants.COMPUTE_LOGGER)
|
||||||
protected Logger logger = Logger.NULL;
|
protected Logger logger = Logger.NULL;
|
||||||
|
|
||||||
|
private final ComputeServiceContext context;
|
||||||
private final Supplier<VirtualBoxManager> manager;
|
private final Supplier<VirtualBoxManager> manager;
|
||||||
private final CreateAndRegisterMachineFromIsoIfNotAlreadyExists createAndRegisterMachineFromIsoIfNotAlreadyExists;
|
private final CreateAndRegisterMachineFromIsoIfNotAlreadyExists createAndRegisterMachineFromIsoIfNotAlreadyExists;
|
||||||
|
|
||||||
|
@ -64,10 +68,11 @@ public class CreateAndInstallVm implements Function<MasterSpec, IMachine> {
|
||||||
private final MachineUtils machineUtils;
|
private final MachineUtils machineUtils;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public CreateAndInstallVm(Supplier<VirtualBoxManager> manager,
|
public CreateAndInstallVm(ComputeServiceContext context, Supplier<VirtualBoxManager> manager,
|
||||||
CreateAndRegisterMachineFromIsoIfNotAlreadyExists CreateAndRegisterMachineFromIsoIfNotAlreadyExists,
|
CreateAndRegisterMachineFromIsoIfNotAlreadyExists CreateAndRegisterMachineFromIsoIfNotAlreadyExists,
|
||||||
Predicate<SshClient> sshResponds, Function<IMachine, SshClient> sshClientForIMachine,
|
Predicate<SshClient> sshResponds, Function<IMachine, SshClient> sshClientForIMachine,
|
||||||
ExecutionType executionType, MachineUtils machineUtils, @Preconfiguration LoadingCache<IsoSpec, URI> preConfiguration) {
|
ExecutionType executionType, MachineUtils machineUtils, @Preconfiguration LoadingCache<IsoSpec, URI> preConfiguration) {
|
||||||
|
this.context = context;
|
||||||
this.manager = manager;
|
this.manager = manager;
|
||||||
this.createAndRegisterMachineFromIsoIfNotAlreadyExists = CreateAndRegisterMachineFromIsoIfNotAlreadyExists;
|
this.createAndRegisterMachineFromIsoIfNotAlreadyExists = CreateAndRegisterMachineFromIsoIfNotAlreadyExists;
|
||||||
this.sshResponds = sshResponds;
|
this.sshResponds = sshResponds;
|
||||||
|
@ -75,7 +80,6 @@ public class CreateAndInstallVm implements Function<MasterSpec, IMachine> {
|
||||||
this.executionType = executionType;
|
this.executionType = executionType;
|
||||||
this.machineUtils = machineUtils;
|
this.machineUtils = machineUtils;
|
||||||
this.preConfiguration = preConfiguration;
|
this.preConfiguration = preConfiguration;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -99,6 +103,9 @@ public class CreateAndInstallVm implements Function<MasterSpec, IMachine> {
|
||||||
logger.debug(">> awaiting installation to finish node(%s)", vmName);
|
logger.debug(">> awaiting installation to finish node(%s)", vmName);
|
||||||
|
|
||||||
checkState(sshResponds.apply(client), "timed out waiting for guest %s to be accessible via ssh", vmName);
|
checkState(sshResponds.apply(client), "timed out waiting for guest %s to be accessible via ssh", vmName);
|
||||||
|
|
||||||
|
logger.debug(">> awaiting installation of guest additions on vm: %s", vmName);
|
||||||
|
checkState(new GuestAdditionsInstaller(context).apply(vmName));
|
||||||
|
|
||||||
logger.debug("<< installation of image complete. Powering down node(%s)", vmName);
|
logger.debug("<< installation of image complete. Powering down node(%s)", vmName);
|
||||||
ensureMachineHasPowerDown(vmName);
|
ensureMachineHasPowerDown(vmName);
|
||||||
|
|
|
@ -35,6 +35,7 @@ import org.jclouds.compute.domain.Processor;
|
||||||
import org.jclouds.compute.reference.ComputeServiceConstants;
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
import org.jclouds.domain.LocationBuilder;
|
import org.jclouds.domain.LocationBuilder;
|
||||||
import org.jclouds.domain.LocationScope;
|
import org.jclouds.domain.LocationScope;
|
||||||
|
import org.jclouds.domain.LoginCredentials;
|
||||||
import org.jclouds.javax.annotation.Nullable;
|
import org.jclouds.javax.annotation.Nullable;
|
||||||
import org.jclouds.logging.Logger;
|
import org.jclouds.logging.Logger;
|
||||||
import org.virtualbox_4_1.IMachine;
|
import org.virtualbox_4_1.IMachine;
|
||||||
|
@ -104,9 +105,9 @@ public class IMachineToNodeMetadata implements Function<IMachine, NodeMetadata>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// nodeMetadataBuilder.imageId("");
|
LoginCredentials loginCredentials = new LoginCredentials("toor", "password", null, true);
|
||||||
// nodeMetadataBuilder.group("");
|
nodeMetadataBuilder.credentials(loginCredentials);
|
||||||
|
|
||||||
return nodeMetadataBuilder.build();
|
return nodeMetadataBuilder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,67 @@
|
||||||
|
package org.jclouds.virtualbox.predicates;
|
||||||
|
|
||||||
|
import static org.jclouds.compute.options.RunScriptOptions.Builder.runAsRoot;
|
||||||
|
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
import org.jclouds.compute.RunScriptData;
|
||||||
|
import org.jclouds.compute.domain.ExecResponse;
|
||||||
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
|
import org.jclouds.logging.Logger;
|
||||||
|
import org.jclouds.scriptbuilder.domain.StatementList;
|
||||||
|
import org.jclouds.scriptbuilder.domain.Statements;
|
||||||
|
import org.jclouds.virtualbox.statements.InstallGuestAdditions;
|
||||||
|
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Splitter;
|
||||||
|
import com.google.common.base.Throwables;
|
||||||
|
import com.google.common.collect.Iterables;
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
public class GuestAdditionsInstaller implements Predicate<String> {
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
@Named(ComputeServiceConstants.COMPUTE_LOGGER)
|
||||||
|
protected Logger logger = Logger.NULL;
|
||||||
|
|
||||||
|
private final ComputeServiceContext context;
|
||||||
|
private String vboxVersion;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public GuestAdditionsInstaller(ComputeServiceContext context) {
|
||||||
|
this.context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(String vmName) {
|
||||||
|
StatementList statementList = prepareStatementList();
|
||||||
|
|
||||||
|
ListenableFuture<ExecResponse> execFuture = context.getComputeService().submitScriptOnNode(vmName, statementList,
|
||||||
|
runAsRoot(true).wrapInInitScript(false));
|
||||||
|
ExecResponse execResponse = null;
|
||||||
|
try {
|
||||||
|
execResponse = execFuture.get();
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
Throwables.propagate(e);
|
||||||
|
} catch (ExecutionException e) {
|
||||||
|
Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
return execResponse == null ? false : execResponse.getExitCode() == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private StatementList prepareStatementList() {
|
||||||
|
vboxVersion = Iterables.get(Splitter.on('r').split(context.getProviderSpecificContext().getBuildVersion()), 0);
|
||||||
|
InstallGuestAdditions installGuestAdditions = new InstallGuestAdditions(vboxVersion);
|
||||||
|
StatementList statementList = new StatementList(Statements.exec(RunScriptData.aptInstallLazyUpgrade("curl")),
|
||||||
|
installGuestAdditions);
|
||||||
|
return statementList;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -34,13 +34,13 @@ import org.jclouds.scriptbuilder.domain.Statement;
|
||||||
import com.google.common.collect.ImmutableMultimap;
|
import com.google.common.collect.ImmutableMultimap;
|
||||||
|
|
||||||
public class InstallGuestAdditions implements Statement {
|
public class InstallGuestAdditions implements Statement {
|
||||||
|
|
||||||
private final String vboxVersion;
|
private final String vboxVersion;
|
||||||
private final String mountPoint;
|
private final String mountPoint;
|
||||||
|
|
||||||
public InstallGuestAdditions(String vboxVersion) {
|
public InstallGuestAdditions(String vboxVersion) {
|
||||||
this(vboxVersion, "/mnt");
|
this(vboxVersion, "/mnt");
|
||||||
}
|
}
|
||||||
|
|
||||||
public InstallGuestAdditions(String vboxVersion, String mountPoint) {
|
public InstallGuestAdditions(String vboxVersion, String mountPoint) {
|
||||||
this.vboxVersion = checkNotNull(vboxVersion, "vboxVersion");
|
this.vboxVersion = checkNotNull(vboxVersion, "vboxVersion");
|
||||||
|
@ -57,17 +57,18 @@ public class InstallGuestAdditions implements Statement {
|
||||||
checkNotNull(family, "family");
|
checkNotNull(family, "family");
|
||||||
if (family == OsFamily.WINDOWS)
|
if (family == OsFamily.WINDOWS)
|
||||||
throw new UnsupportedOperationException("windows not yet implemented");
|
throw new UnsupportedOperationException("windows not yet implemented");
|
||||||
|
|
||||||
String vboxGuestAdditionsIso = "VBoxGuestAdditions_" + vboxVersion + ".iso";
|
String vboxGuestAdditionsIso = "VBoxGuestAdditions_" + vboxVersion + ".iso";
|
||||||
ScriptBuilder scriptBuilder = new ScriptBuilder()
|
ScriptBuilder scriptBuilder = new ScriptBuilder()
|
||||||
.addStatement(
|
.addStatement(
|
||||||
new SaveHttpResponseTo("{tmp}", vboxGuestAdditionsIso, "GET", URI
|
new SaveHttpResponseTo("{tmp}{fs}", vboxGuestAdditionsIso, "GET", URI
|
||||||
.create("http://download.virtualbox.org/virtualbox/" + vboxVersion + "/"
|
.create("http://download.virtualbox.org/virtualbox/" + vboxVersion + "/"
|
||||||
+ vboxGuestAdditionsIso), ImmutableMultimap.<String, String> of()))
|
+ vboxGuestAdditionsIso), ImmutableMultimap.<String, String> of()))
|
||||||
.addStatement(exec(String.format("mount -o loop {tmp}{fs}%s %s", vboxGuestAdditionsIso, mountPoint)))
|
.addStatement(exec(String.format("mount -o loop {tmp}{fs}%s %s", vboxGuestAdditionsIso, mountPoint)))
|
||||||
.addStatement(call("installGuestAdditions"))
|
.addStatement(call("installModuleAssistantIfNeeded"))
|
||||||
.addStatement(exec(String.format("sh %s%s", mountPoint, "/VBoxLinuxAdditions.run")))
|
.addStatement(exec(String.format("%s%s", mountPoint, "/VBoxLinuxAdditions.run")))
|
||||||
.addStatement(exec(String.format("umount %s", mountPoint)));
|
.addStatement(exec(String.format("umount %s", mountPoint)));
|
||||||
|
|
||||||
return scriptBuilder.render(family);
|
return scriptBuilder.render(family);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,14 +1,10 @@
|
||||||
function installGuestAdditions {
|
function installGuestAdditions {
|
||||||
unset OSNAME;
|
unset OSNAME;
|
||||||
[ $# -eq 1 ] || {
|
|
||||||
abort "installGuestAdditions requires virtual machine name parameter"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
local OSNAME=`lsb_release -d -s | cut -d ' ' -f 1`; shift
|
local OSNAME=`lsb_release -d -s | cut -d ' ' -f 1`; shift
|
||||||
if [ $OSNAME = 'Ubuntu' ]
|
if [ $OSNAME = 'Ubuntu' ]
|
||||||
then
|
then
|
||||||
echo "OS Name is Ubuntu"
|
echo "OS is Ubuntu"
|
||||||
`apt-get install build-essential module-assistant && m-a prepare -i`
|
apt-get -f -y -qq --force-yes install build-essential module-assistant && m-a prepare -i
|
||||||
fi
|
fi
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
function installModuleAssistantIfNeeded {
|
||||||
|
unset OSNAME;
|
||||||
|
local OSNAME=`lsb_release -d -s | cut -d ' ' -f 1`; shift
|
||||||
|
if [ $OSNAME = 'Ubuntu' ]
|
||||||
|
then
|
||||||
|
echo "OS is Ubuntu"
|
||||||
|
apt-get -f -y -qq --force-yes install build-essential module-assistant;
|
||||||
|
m-a prepare -i
|
||||||
|
fi
|
||||||
|
}
|
|
@ -37,6 +37,6 @@ public class InstallGuestAdditionsTest {
|
||||||
public void testUnix() throws IOException {
|
public void testUnix() throws IOException {
|
||||||
InstallGuestAdditions statement = new InstallGuestAdditions("4.1.6");
|
InstallGuestAdditions statement = new InstallGuestAdditions("4.1.6");
|
||||||
assertEquals(statement.render(OsFamily.UNIX), CharStreams.toString(Resources.newReaderSupplier(Resources
|
assertEquals(statement.render(OsFamily.UNIX), CharStreams.toString(Resources.newReaderSupplier(Resources
|
||||||
.getResource("test_install_guest_additions." + ShellToken.SH.to(OsFamily.UNIX)), Charsets.UTF_8)));
|
.getResource("test_guest_additions_installer." + ShellToken.SH.to(OsFamily.UNIX)), Charsets.UTF_8)));
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -23,11 +23,16 @@ import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_IMAGE
|
||||||
import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_INSTALLATION_KEY_SEQUENCE;
|
import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_INSTALLATION_KEY_SEQUENCE;
|
||||||
import static org.testng.Assert.assertEquals;
|
import static org.testng.Assert.assertEquals;
|
||||||
|
|
||||||
import com.google.inject.Injector;
|
|
||||||
|
|
||||||
import org.jclouds.config.ValueOfConfigurationKeyOrNull;
|
import org.jclouds.config.ValueOfConfigurationKeyOrNull;
|
||||||
import org.jclouds.virtualbox.BaseVirtualBoxClientLiveTest;
|
import org.jclouds.virtualbox.BaseVirtualBoxClientLiveTest;
|
||||||
import org.jclouds.virtualbox.domain.*;
|
import org.jclouds.virtualbox.domain.HardDisk;
|
||||||
|
import org.jclouds.virtualbox.domain.IsoSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.MasterSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.NetworkSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.StorageController;
|
||||||
|
import org.jclouds.virtualbox.domain.VmSpec;
|
||||||
|
import org.jclouds.virtualbox.functions.CloneAndRegisterMachineFromIMachineIfNotAlreadyExists;
|
||||||
|
import org.jclouds.virtualbox.functions.CreateAndRegisterMachineFromIsoIfNotAlreadyExists;
|
||||||
import org.testng.annotations.BeforeClass;
|
import org.testng.annotations.BeforeClass;
|
||||||
import org.testng.annotations.Test;
|
import org.testng.annotations.Test;
|
||||||
import org.virtualbox_4_1.CleanupMode;
|
import org.virtualbox_4_1.CleanupMode;
|
||||||
|
@ -38,6 +43,7 @@ import org.virtualbox_4_1.StorageBus;
|
||||||
import com.google.common.base.CaseFormat;
|
import com.google.common.base.CaseFormat;
|
||||||
import com.google.common.base.Function;
|
import com.google.common.base.Function;
|
||||||
import com.google.common.collect.ImmutableSet;
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.inject.Injector;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Andrea Turli
|
* @author Andrea Turli
|
||||||
|
|
|
@ -76,7 +76,7 @@ public class CreateAndInstallVmLiveTest extends BaseVirtualBoxClientLiveTest {
|
||||||
.attachISO(0, 0, operatingSystemIso)
|
.attachISO(0, 0, operatingSystemIso)
|
||||||
.attachHardDisk(hardDisk)
|
.attachHardDisk(hardDisk)
|
||||||
.attachISO(1, 1, guestAdditionsIso).build();
|
.attachISO(1, 1, guestAdditionsIso).build();
|
||||||
vmSpecification = VmSpec.builder().id("jclouds#image#create-and-install-vm-test").name(vmName).memoryMB(512).osTypeId("")
|
vmSpecification = VmSpec.builder().id("jclouds-image-create-and-install-vm-test").name(vmName).memoryMB(512).osTypeId("")
|
||||||
.controller(ideController)
|
.controller(ideController)
|
||||||
.forceOverwrite(true)
|
.forceOverwrite(true)
|
||||||
.cleanUpMode(CleanupMode.Full).build();
|
.cleanUpMode(CleanupMode.Full).build();
|
||||||
|
|
|
@ -0,0 +1,143 @@
|
||||||
|
/**
|
||||||
|
* 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.virtualbox.predicates;
|
||||||
|
|
||||||
|
import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_IMAGE_PREFIX;
|
||||||
|
import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_INSTALLATION_KEY_SEQUENCE;
|
||||||
|
import static org.testng.Assert.assertTrue;
|
||||||
|
|
||||||
|
import org.jclouds.config.ValueOfConfigurationKeyOrNull;
|
||||||
|
import org.jclouds.virtualbox.BaseVirtualBoxClientLiveTest;
|
||||||
|
import org.jclouds.virtualbox.domain.HardDisk;
|
||||||
|
import org.jclouds.virtualbox.domain.IsoSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.MasterSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.NatAdapter;
|
||||||
|
import org.jclouds.virtualbox.domain.NetworkSpec;
|
||||||
|
import org.jclouds.virtualbox.domain.StorageController;
|
||||||
|
import org.jclouds.virtualbox.domain.VmSpec;
|
||||||
|
import org.jclouds.virtualbox.functions.CreateAndInstallVm;
|
||||||
|
import org.testng.annotations.BeforeClass;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
import org.virtualbox_4_1.CleanupMode;
|
||||||
|
import org.virtualbox_4_1.IMachine;
|
||||||
|
import org.virtualbox_4_1.IProgress;
|
||||||
|
import org.virtualbox_4_1.ISession;
|
||||||
|
import org.virtualbox_4_1.LockType;
|
||||||
|
import org.virtualbox_4_1.StorageBus;
|
||||||
|
|
||||||
|
import com.google.common.base.CaseFormat;
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Splitter;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.inject.Injector;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Andrea Turli
|
||||||
|
*/
|
||||||
|
@Test(groups = "live", singleThreaded = true, testName = "GuestAdditionsInstallerLiveTest")
|
||||||
|
public class GuestAdditionsInstallerLiveTest extends
|
||||||
|
BaseVirtualBoxClientLiveTest {
|
||||||
|
|
||||||
|
private MasterSpec sourceMachineSpec;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@BeforeClass(groups = "live")
|
||||||
|
public void setupClient() {
|
||||||
|
super.setupClient();
|
||||||
|
String sourceName = VIRTUALBOX_IMAGE_PREFIX
|
||||||
|
+ CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_HYPHEN, getClass()
|
||||||
|
.getSimpleName());
|
||||||
|
|
||||||
|
StorageController ideController = StorageController
|
||||||
|
.builder()
|
||||||
|
.name("IDE Controller")
|
||||||
|
.bus(StorageBus.IDE)
|
||||||
|
.attachISO(0, 0, operatingSystemIso)
|
||||||
|
.attachHardDisk(
|
||||||
|
HardDisk.builder().diskpath(adminDisk).controllerPort(0)
|
||||||
|
.deviceSlot(1).autoDelete(true).build())
|
||||||
|
.attachISO(1, 1, guestAdditionsIso).build();
|
||||||
|
|
||||||
|
VmSpec sourceVmSpec = VmSpec.builder().id(sourceName).name(sourceName)
|
||||||
|
.osTypeId("").memoryMB(512).cleanUpMode(CleanupMode.Full)
|
||||||
|
.controller(ideController).forceOverwrite(true).build();
|
||||||
|
|
||||||
|
Injector injector = context.utils().injector();
|
||||||
|
Function<String, String> configProperties = injector
|
||||||
|
.getInstance(ValueOfConfigurationKeyOrNull.class);
|
||||||
|
IsoSpec isoSpec = IsoSpec
|
||||||
|
.builder()
|
||||||
|
.sourcePath(operatingSystemIso)
|
||||||
|
.installationScript(
|
||||||
|
configProperties.apply(VIRTUALBOX_INSTALLATION_KEY_SEQUENCE)
|
||||||
|
.replace("HOSTNAME", sourceVmSpec.getVmName())).build();
|
||||||
|
|
||||||
|
NetworkSpec networkSpec = //NetworkSpec.builder().build();
|
||||||
|
NetworkSpec.builder()
|
||||||
|
.natNetworkAdapter(0, NatAdapter.builder().tcpRedirectRule("127.0.0.1", 2222, "", 22).build())
|
||||||
|
.build();
|
||||||
|
sourceMachineSpec = MasterSpec.builder().iso(isoSpec).vm(sourceVmSpec).network(networkSpec).build();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGuestAdditionsAreInstalled() throws Exception {
|
||||||
|
try {
|
||||||
|
IMachine machine = getVmWithGuestAdditionsInstalled();
|
||||||
|
assertTrue(machineUtils.lockSessionOnMachineAndApply(machine.getName(), LockType.Shared, new Function<ISession, Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean apply(ISession session) {
|
||||||
|
return session.getMachine().getGuestPropertyValue("/VirtualBox/GuestAdd/Version") != null;
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
} finally {
|
||||||
|
for (VmSpec spec : ImmutableSet.of(
|
||||||
|
sourceMachineSpec.getVmSpec())) {
|
||||||
|
ensureMachineHasPowerDown(spec.getVmName());
|
||||||
|
undoVm(spec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private IMachine getVmWithGuestAdditionsInstalled() {
|
||||||
|
try {
|
||||||
|
Injector injector = context.utils().injector();
|
||||||
|
return injector.getInstance(
|
||||||
|
CreateAndInstallVm.class).apply(
|
||||||
|
sourceMachineSpec);
|
||||||
|
} catch (IllegalStateException e) {
|
||||||
|
// already created
|
||||||
|
return manager.get().getVBox()
|
||||||
|
.findMachine(sourceMachineSpec.getVmSpec().getVmId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ensureMachineHasPowerDown(String vmName) {
|
||||||
|
machineUtils.lockSessionOnMachineAndApply(vmName, LockType.Shared, new Function<ISession, Void>() {
|
||||||
|
@Override
|
||||||
|
public Void apply(ISession session) {
|
||||||
|
IProgress powerDownProgress = session.getConsole().powerDown();
|
||||||
|
powerDownProgress.waitForCompletion(-1);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,92 @@
|
||||||
|
package org.jclouds.virtualbox.statements;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static org.jclouds.compute.options.RunScriptOptions.Builder.runAsRoot;
|
||||||
|
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
import javax.inject.Named;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
|
import org.jclouds.compute.ComputeServiceContext;
|
||||||
|
import org.jclouds.compute.RunScriptData;
|
||||||
|
import org.jclouds.compute.callables.RunScriptOnNode.Factory;
|
||||||
|
import org.jclouds.compute.domain.ExecResponse;
|
||||||
|
import org.jclouds.compute.domain.NodeMetadata;
|
||||||
|
import org.jclouds.compute.reference.ComputeServiceConstants;
|
||||||
|
import org.jclouds.logging.Logger;
|
||||||
|
import org.jclouds.scriptbuilder.domain.StatementList;
|
||||||
|
import org.jclouds.scriptbuilder.domain.Statements;
|
||||||
|
import org.jclouds.ssh.SshClient;
|
||||||
|
import org.jclouds.virtualbox.domain.ExecutionType;
|
||||||
|
import org.jclouds.virtualbox.functions.CreateAndRegisterMachineFromIsoIfNotAlreadyExists;
|
||||||
|
import org.jclouds.virtualbox.functions.IMachineToNodeMetadata;
|
||||||
|
import org.jclouds.virtualbox.functions.LaunchMachineIfNotAlreadyRunning;
|
||||||
|
import org.jclouds.virtualbox.util.MachineUtils;
|
||||||
|
import org.virtualbox_4_1.IMachine;
|
||||||
|
import org.virtualbox_4_1.VirtualBoxManager;
|
||||||
|
|
||||||
|
import com.google.common.base.Function;
|
||||||
|
import com.google.common.base.Predicate;
|
||||||
|
import com.google.common.base.Supplier;
|
||||||
|
import com.google.common.base.Throwables;
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture;
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
public class GuestAdditionsInstaller implements Function<String, IMachine> {
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
@Named(ComputeServiceConstants.COMPUTE_LOGGER)
|
||||||
|
protected Logger logger = Logger.NULL;
|
||||||
|
|
||||||
|
private final ComputeServiceContext context;
|
||||||
|
|
||||||
|
private final Supplier<VirtualBoxManager> manager;
|
||||||
|
private final ExecutionType executionType;
|
||||||
|
private final MachineUtils machineUtils;
|
||||||
|
// TODO remove this hardcoded value
|
||||||
|
private String vboxVersion = "4.1.6";
|
||||||
|
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
public GuestAdditionsInstaller(ComputeServiceContext context, Supplier<VirtualBoxManager> manager,
|
||||||
|
CreateAndRegisterMachineFromIsoIfNotAlreadyExists createAndRegisterMachineFromIsoIfNotAlreadyExists,
|
||||||
|
Predicate<SshClient> installGuestAdditionsViaSshResponds, Function<IMachine, SshClient> sshClientForIMachine,
|
||||||
|
ExecutionType executionType, MachineUtils machineUtils, Factory runScriptOnNodeFactory,
|
||||||
|
Supplier<NodeMetadata> guest) {
|
||||||
|
this.context = context;
|
||||||
|
this.manager = manager;
|
||||||
|
this.executionType = executionType;
|
||||||
|
this.machineUtils = machineUtils;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public IMachine apply(String vmName) {
|
||||||
|
IMachine vm = manager.get().getVBox().findMachine(vmName);
|
||||||
|
ensureMachineIsLaunched(vmName);
|
||||||
|
|
||||||
|
InstallGuestAdditions installGuestAdditions = new InstallGuestAdditions(vboxVersion);
|
||||||
|
StatementList statementList = new StatementList(Statements.exec(RunScriptData.aptInstallLazyUpgrade("curl")),
|
||||||
|
installGuestAdditions);
|
||||||
|
|
||||||
|
NodeMetadata vmMetadata = new IMachineToNodeMetadata().apply(vm);
|
||||||
|
|
||||||
|
ListenableFuture<ExecResponse> execFuture = context.getComputeService().submitScriptOnNode(vmMetadata.getId(), statementList,
|
||||||
|
runAsRoot(true).wrapInInitScript(false));
|
||||||
|
try {
|
||||||
|
execFuture.get();
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
Throwables.propagate(e);
|
||||||
|
} catch (ExecutionException e) {
|
||||||
|
Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
return vm;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ensureMachineIsLaunched(String vmName) {
|
||||||
|
machineUtils.applyForMachine(vmName, new LaunchMachineIfNotAlreadyRunning(manager.get(), executionType, ""));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,55 @@
|
||||||
|
/**
|
||||||
|
* 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.virtualbox.statements;
|
||||||
|
|
||||||
|
import static org.jclouds.virtualbox.config.VirtualBoxConstants.VIRTUALBOX_IMAGE_PREFIX;
|
||||||
|
|
||||||
|
import org.jclouds.virtualbox.BaseVirtualBoxClientLiveTest;
|
||||||
|
import org.testng.annotations.BeforeClass;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import com.google.common.base.CaseFormat;
|
||||||
|
import com.google.inject.Injector;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Andrea Turli
|
||||||
|
*/
|
||||||
|
@Test(groups = "live", singleThreaded = true, testName = "InstallGuestAdditionsTest")
|
||||||
|
public class InstallGuestAdditionsTest extends BaseVirtualBoxClientLiveTest {
|
||||||
|
|
||||||
|
private String vmName;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@BeforeClass(groups = "live")
|
||||||
|
public void setupClient() {
|
||||||
|
super.setupClient();
|
||||||
|
vmName = VIRTUALBOX_IMAGE_PREFIX
|
||||||
|
+ CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_HYPHEN, getClass()
|
||||||
|
.getSimpleName());
|
||||||
|
|
||||||
|
vmName = "jclouds-image-create-and-install-vm-live-test";
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testInstallGuestAdditionsOnTheMachine() throws Exception {
|
||||||
|
Injector injector = context.utils().injector();
|
||||||
|
injector.getInstance(GuestAdditionsInstaller.class).apply(vmName);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
#!/bin/bash
|
||||||
|
set +u
|
||||||
|
shopt -s xpg_echo
|
||||||
|
shopt -s expand_aliases
|
||||||
|
unset PATH JAVA_HOME LD_LIBRARY_PATH
|
||||||
|
function abort {
|
||||||
|
echo "aborting: $@" 1>&2
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
function installModuleAssistantIfNeeded {
|
||||||
|
unset OSNAME;
|
||||||
|
local OSNAME=`lsb_release -d -s | cut -d ' ' -f 1`; shift
|
||||||
|
if [ $OSNAME = 'Ubuntu' ]
|
||||||
|
then
|
||||||
|
echo "OS is Ubuntu"
|
||||||
|
apt-get -f -y -qq --force-yes install build-essential module-assistant && m-a prepare -i
|
||||||
|
fi
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
export PATH=/usr/ucb/bin:/bin:/sbin:/usr/bin:/usr/sbin
|
||||||
|
(mkdir -p /tmp/ && cd /tmp/ && [ ! -f VBoxGuestAdditions_4.1.6.iso ] && curl -q -s -S -L --connect-timeout 10 --max-time 600 --retry 20 -C - -X GET http://download.virtualbox.org/virtualbox/4.1.6/VBoxGuestAdditions_4.1.6.iso >VBoxGuestAdditions_4.1.6.iso)
|
||||||
|
mount -o loop /tmp/VBoxGuestAdditions_4.1.6.iso /mnt
|
||||||
|
installModuleAssistantIfNeeded || exit 1
|
||||||
|
/mnt/VBoxLinuxAdditions.run
|
||||||
|
umount /mnt
|
||||||
|
exit 0
|
|
@ -1,30 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
set +u
|
|
||||||
shopt -s xpg_echo
|
|
||||||
shopt -s expand_aliases
|
|
||||||
unset PATH JAVA_HOME LD_LIBRARY_PATH
|
|
||||||
function abort {
|
|
||||||
echo "aborting: $@" 1>&2
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
function installGuestAdditions {
|
|
||||||
unset OSNAME;
|
|
||||||
[ $# -eq 1 ] || {
|
|
||||||
abort "installGuestAdditions requires virtual machine name parameter"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
local OSNAME=`lsb_release -d -s | cut -d ' ' -f 1`; shift
|
|
||||||
if [ $OSNAME = 'Ubuntu' ]
|
|
||||||
then
|
|
||||||
echo "OS Name is Ubuntu"
|
|
||||||
`apt-get install build-essential module-assistant && m-a prepare -i`
|
|
||||||
fi
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
export PATH=/usr/ucb/bin:/bin:/sbin:/usr/bin:/usr/sbin
|
|
||||||
(mkdir -p /tmp && cd /tmp && [ ! -f VBoxGuestAdditions_4.1.6.iso ] && curl -q -s -S -L --connect-timeout 10 --max-time 600 --retry 20 -C - -X GET http://download.virtualbox.org/virtualbox/4.1.6/VBoxGuestAdditions_4.1.6.iso >VBoxGuestAdditions_4.1.6.iso)
|
|
||||||
mount -o loop /tmp/VBoxGuestAdditions_4.1.6.iso /mnt
|
|
||||||
installGuestAdditions || exit 1
|
|
||||||
sh /mnt/VBoxLinuxAdditions.run
|
|
||||||
umount /mnt
|
|
||||||
exit 0
|
|
|
@ -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.virtualbox.statements;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
import static org.jclouds.scriptbuilder.domain.Statements.call;
|
||||||
|
import static org.jclouds.scriptbuilder.domain.Statements.exec;
|
||||||
|
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.Collections;
|
||||||
|
|
||||||
|
import org.jclouds.scriptbuilder.ScriptBuilder;
|
||||||
|
import org.jclouds.scriptbuilder.domain.OsFamily;
|
||||||
|
import org.jclouds.scriptbuilder.domain.SaveHttpResponseTo;
|
||||||
|
import org.jclouds.scriptbuilder.domain.Statement;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableMultimap;
|
||||||
|
|
||||||
|
public class GuestAdditionsInstallation implements Statement {
|
||||||
|
|
||||||
|
private final String vboxVersion;
|
||||||
|
private final String mountPoint;
|
||||||
|
|
||||||
|
public GuestAdditionsInstallation(String vboxVersion) {
|
||||||
|
this(vboxVersion, "/mnt");
|
||||||
|
}
|
||||||
|
|
||||||
|
public GuestAdditionsInstallation(String vboxVersion, String mountPoint) {
|
||||||
|
this.vboxVersion = checkNotNull(vboxVersion, "vboxVersion");
|
||||||
|
this.mountPoint = checkNotNull(mountPoint, "mountPoint");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterable<String> functionDependencies(OsFamily family) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String render(OsFamily family) {
|
||||||
|
checkNotNull(family, "family");
|
||||||
|
if (family == OsFamily.WINDOWS)
|
||||||
|
throw new UnsupportedOperationException("windows not yet implemented");
|
||||||
|
|
||||||
|
String vboxGuestAdditionsIso = "VBoxGuestAdditions_" + vboxVersion + ".iso";
|
||||||
|
ScriptBuilder scriptBuilder = new ScriptBuilder()
|
||||||
|
.addStatement(
|
||||||
|
new SaveHttpResponseTo("{tmp}", vboxGuestAdditionsIso, "GET", URI
|
||||||
|
.create("http://download.virtualbox.org/virtualbox/" + vboxVersion + "/"
|
||||||
|
+ vboxGuestAdditionsIso), ImmutableMultimap.<String, String> of()))
|
||||||
|
.addStatement(exec(String.format("mount -o loop {tmp}{fs}%s %s", vboxGuestAdditionsIso, mountPoint)))
|
||||||
|
.addStatement(call("installGuestAdditions"))
|
||||||
|
.addStatement(exec(String.format("sh %s%s", mountPoint, "/VBoxLinuxAdditions.run")))
|
||||||
|
.addStatement(exec(String.format("umount %s", mountPoint)));
|
||||||
|
return scriptBuilder.render(family);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue