Merge pull request #11694 from soapy1/host-cap-build-iso

Build iso for Darwin host
This commit is contained in:
Sophia Castellarin 2020-07-07 10:29:52 -05:00 committed by GitHub
commit 02ef62dcff
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 227 additions and 0 deletions

View File

@ -432,6 +432,10 @@ module Vagrant
error_key(:host_explicit_not_detected)
end
class ISOBuildFailed < VagrantError
error_key(:iso_build_failed)
end
class LinuxMountFailed < VagrantError
error_key(:linux_mount_failed)
end

View File

@ -0,0 +1,19 @@
require 'pathname'
module Vagrant
module Util
class Directory
# Check if directory has any new updates
#
# @param [Pathname, String] Path to directory
# @param [Time] time to compare to eg. has any file in dir_path
# changed since this time
# @return [Boolean]
def self.directory_changed?(dir_path, threshold_time)
Dir.glob(Pathname.new(dir_path).join("**", "*")).any? do |path|
Pathname.new(path).mtime > threshold_time
end
end
end
end
end

View File

@ -0,0 +1,78 @@
require "tempfile"
require 'fileutils'
require 'pathname'
require "vagrant/util/subprocess"
require "vagrant/util/map_command_options"
require "vagrant/util/directory"
module VagrantPlugins
module HostDarwin
module Cap
class FsISO
@@logger = Log4r::Logger.new("vagrant::host::darwin::fs_iso")
BUILD_ISO_CMD = "hdiutil".freeze
# Check that the host has the ability to generate ISOs
#
# @param [Vagrant::Environment] env
# @return [Boolean]
def self.isofs_available(env)
!!Vagrant::Util::Which.which(BUILD_ISO_CMD)
end
# Generate an ISO file of the given source directory
#
# @param [Vagrant::Environment] env
# @param [String] source_directory Contents of ISO
# @param [Map] extra arguments to pass to the iso building command
# :file_destination (string) location to store ISO
# :volume_id (String) to set the volume name
# @return [Pathname] ISO location
# @note If file_destination exists, source_directory will be checked
# for recent modifications and a new ISO will be generated if requried.
def self.create_iso(env, source_directory, **extra_opts)
file_destination = extra_opts[:file_destination]
source_directory = Pathname.new(source_directory)
if file_destination.nil?
@@logger.info("No file destination specified, creating temp location")
tmpfile = Tempfile.new(["vagrant", ".iso"])
file_destination = Pathname.new(tmpfile.path)
tmpfile.delete
else
file_destination = Pathname.new(file_destination.to_s)
# If the file destination path is a folder, target the output to a randomly named
# file in that dir
if file_destination.extname != ".iso"
file_destination = file_destination.join("#{SecureRandom.hex(3)}_vagrant.iso")
end
end
# Ensure destination directory is available
FileUtils.mkdir_p(file_destination.dirname)
# If the destrination does not exist or there have been changes in the source directory since the last build, then build
if !file_destination.exist? || Vagrant::Util::Directory.directory_changed?(source_directory, file_destination.mtime)
@@logger.info("Building ISO from source #{source_directory}")
iso_command = [BUILD_ISO_CMD, "makehybrid"]
iso_command << "-hfs"
iso_command << "-iso"
iso_command << "-joliet"
iso_command << "-ov"
iso_command.concat(["-default-volume-name", extra_opts[:volume_id]]) if extra_opts[:volume_id]
iso_command << "-o"
iso_command << file_destination.to_s
iso_command << source_directory.to_s
result = Vagrant::Util::Subprocess.execute(*iso_command)
if result.exit_code != 0
raise Vagrant::Errors::ISOBuildFailed, cmd: iso_command.join(" "), stdout: result.stdout, stderr: result.stderr
end
end
@@logger.info("ISO available at #{file_destination}")
file_destination
end
end
end
end
end

View File

@ -11,6 +11,16 @@ module VagrantPlugins
Host
end
host_capability("darwin", "isofs_available") do
require_relative "cap/fs_iso"
Cap::FsISO
end
host_capability("darwin", "create_iso") do
require_relative "cap/fs_iso"
Cap::FsISO
end
host_capability("darwin", "provider_install_virtualbox") do
require_relative "cap/provider_install_virtualbox"
Cap::ProviderInstallVirtualBox

View File

@ -942,6 +942,18 @@ en:
("%{value}") could not be found. Please verify that the plugin is
installed which implements this host and that the value you used
for `config.vagrant.host` is correct.
iso_build_failed: |-
Failed to build iso image. The following command returned an error:
%{cmd}
Stdout from the command:
%{stdout}
Stderr from the command:
%{stderr}
hyperv_virtualbox_error: |-
Hyper-V and VirtualBox cannot be used together and will result in a
system crash. Vagrant will now exit. Please disable Hyper-V if you wish

View File

@ -0,0 +1,82 @@
require "pathname"
require_relative "../../../../base"
require_relative "../../../../../../plugins/hosts/darwin/cap/fs_iso"
describe VagrantPlugins::HostDarwin::Cap::FsISO do
include_context "unit"
let(:subject){ VagrantPlugins::HostDarwin::Cap::FsISO }
let(:env) { double("env") }
describe ".isofs_available" do
it "finds iso building utility when available" do
expect(Vagrant::Util::Which).to receive(:which).and_return(true)
expect(subject.isofs_available(env)).to eq(true)
end
it "does not find iso building utility when not available" do
expect(Vagrant::Util::Which).to receive(:which).and_return(false)
expect(subject.isofs_available(env)).to eq(false)
end
end
describe ".create_iso" do
let(:file_destination) { "/woo/out.iso" }
before do
allow(file_destination).to receive(:nil?).and_return(false)
end
it "builds an iso" do
expect(Vagrant::Util::Subprocess).to receive(:execute).with(
"hdiutil", "makehybrid", "-hfs", "-iso", "-joliet", "-ov", "-o", /.iso/, /\/foo\/src/
).and_return(double(exit_code: 0))
expect(FileUtils).to receive(:mkdir_p).with(Pathname.new(file_destination).dirname)
output = subject.create_iso(env, "/foo/src", file_destination: file_destination)
expect(output.to_s).to eq("/woo/out.iso")
end
it "builds an iso with volume_id" do
expect(Vagrant::Util::Subprocess).to receive(:execute).with(
"hdiutil", "makehybrid", "-hfs", "-iso", "-joliet", "-ov", "-default-volume-name", "cidata", "-o", /.iso/, /\/foo\/src/
).and_return(double(exit_code: 0))
expect(FileUtils).to receive(:mkdir_p).with(Pathname.new(file_destination).dirname)
output = subject.create_iso(env, "/foo/src", file_destination: file_destination, volume_id: "cidata")
expect(output.to_s).to eq("/woo/out.iso")
end
it "builds an iso given a file destination without an extension" do
expect(Vagrant::Util::Subprocess).to receive(:execute).with(
"hdiutil", "makehybrid", "-hfs", "-iso", "-joliet", "-ov", "-o", /.iso/, /\/foo\/src/
).and_return(double(exit_code: 0))
expect(FileUtils).to receive(:mkdir_p).with(Pathname.new("/woo/out_dir"))
output = subject.create_iso(env, "/foo/src", file_destination: "/woo/out_dir")
expect(output.to_s).to match(/\/woo\/out_dir\/[\w]{6}_vagrant.iso/)
end
it "builds an iso when no file destination is given" do
allow(Tempfile).to receive(:new).and_return(file_destination)
allow(file_destination).to receive(:path).and_return(file_destination)
allow(file_destination).to receive(:delete)
expect(Vagrant::Util::Subprocess).to receive(:execute).with(
"hdiutil", "makehybrid", "-hfs", "-iso", "-joliet", "-ov", "-o", /.iso/, /\/foo\/src/
).and_return(double(exit_code: 0))
# Should create a directory wherever Tempfile creates files by default
expect(FileUtils).to receive(:mkdir_p).with(Pathname.new(file_destination).dirname)
output = subject.create_iso(env, "/foo/src")
expect(output.to_s).to eq(file_destination)
end
it "raises an error if iso build failed" do
allow(Vagrant::Util::Subprocess).to receive(:execute).with(any_args).and_return(double(stdout: "nope", stderr: "nope", exit_code: 1))
expect(FileUtils).to receive(:mkdir_p).with(Pathname.new(file_destination).dirname)
expect{ subject.create_iso(env, "/foo/src", file_destination: file_destination) }.to raise_error(Vagrant::Errors::ISOBuildFailed)
end
end
end

View File

@ -0,0 +1,22 @@
require File.expand_path("../../../base", __FILE__)
require "vagrant/util/directory"
require "time"
describe Vagrant::Util::Directory do
include_context "unit"
let(:subject){ Vagrant::Util::Directory }
describe ".directory_changed?" do
it "should return false if the threshold time is larger the all mtimes" do
t = Time.new("3008", "09", "09")
expect(subject.directory_changed?(Dir.getwd, t)).to eq(false)
end
it "should return true if the threshold time is less than any mtimes" do
t = Time.new("1990", "06", "06")
expect(subject.directory_changed?(Dir.getwd, t)).to eq(true)
end
end
end