152 lines
5.5 KiB
Ruby
152 lines
5.5 KiB
Ruby
require "tempfile"
|
|
|
|
require_relative "base"
|
|
|
|
module VagrantPlugins
|
|
module Ansible
|
|
module Provisioner
|
|
class Guest < Base
|
|
include Vagrant::Util
|
|
|
|
def initialize(machine, config)
|
|
super
|
|
@logger = Log4r::Logger.new("vagrant::provisioners::ansible_guest")
|
|
end
|
|
|
|
def provision
|
|
check_and_install_ansible
|
|
execute_ansible_galaxy_on_guest if config.galaxy_role_file
|
|
execute_ansible_playbook_on_guest
|
|
end
|
|
|
|
protected
|
|
|
|
#
|
|
# This handles verifying the Ansible installation, installing it if it was
|
|
# requested, and so on. This method will raise exceptions if things are wrong.
|
|
#
|
|
# Current limitations:
|
|
# - The installation of a specific Ansible version is not supported.
|
|
# Such feature is difficult to systematically provide via package repositories (apt, yum, ...).
|
|
# Installing via pip python packaging or directly from github source would be appropriate,
|
|
# but these approaches require more dependency burden.
|
|
# - There is no guarantee that the automated installation will replace
|
|
# a previous Ansible installation.
|
|
#
|
|
def check_and_install_ansible
|
|
@logger.info("Checking for Ansible installation...")
|
|
|
|
# If the guest cannot check if Ansible is installed,
|
|
# print a warning and try to continue without any installation attempt...
|
|
if !@machine.guest.capability?(:ansible_installed)
|
|
@machine.ui.warn(I18n.t("vagrant.provisioners.ansible.cannot_detect"))
|
|
return
|
|
end
|
|
|
|
# Try to install Ansible (if needed and requested)
|
|
if config.install &&
|
|
(config.version.to_s.to_sym == :latest ||
|
|
!@machine.guest.capability(:ansible_installed, config.version))
|
|
@machine.ui.detail I18n.t("vagrant.provisioners.ansible.installing")
|
|
@machine.guest.capability(:ansible_install)
|
|
end
|
|
|
|
# Check that ansible binaries are well installed on the guest,
|
|
@machine.communicate.execute(
|
|
"ansible-galaxy info --help && ansible-playbook --help",
|
|
:error_class => Ansible::Errors::AnsibleNotFoundOnGuest,
|
|
:error_key => :ansible_not_found_on_guest)
|
|
|
|
# Check if requested ansible version is available
|
|
if (!config.version.empty? &&
|
|
config.version.to_s.to_sym != :latest &&
|
|
!@machine.guest.capability(:ansible_installed, config.version))
|
|
raise Ansible::Errors::AnsibleVersionNotFoundOnGuest, required_version: config.version.to_s
|
|
end
|
|
end
|
|
|
|
def execute_ansible_galaxy_on_guest
|
|
command_values = {
|
|
:role_file => get_galaxy_role_file(config.provisioning_path),
|
|
:roles_path => get_galaxy_roles_path(config.provisioning_path)
|
|
}
|
|
|
|
remote_command = config.galaxy_command % command_values
|
|
|
|
execute_ansible_command_on_guest "galaxy", remote_command
|
|
end
|
|
|
|
def execute_ansible_playbook_on_guest
|
|
prepare_common_command_arguments
|
|
prepare_common_environment_variables
|
|
|
|
execute_ansible_command_on_guest "playbook", ansible_playbook_command_for_shell_execution
|
|
end
|
|
|
|
def execute_ansible_command_on_guest(name, command)
|
|
remote_command = "cd #{config.provisioning_path} && #{command}"
|
|
|
|
ui_running_ansible_command name, remote_command
|
|
|
|
result = execute_on_guest(remote_command)
|
|
raise Ansible::Errors::AnsibleCommandFailed if result != 0
|
|
end
|
|
|
|
def execute_on_guest(command)
|
|
@machine.communicate.execute(command, :error_check => false) do |type, data|
|
|
if [:stderr, :stdout].include?(type)
|
|
@machine.env.ui.info(data, :new_line => false, :prefix => false)
|
|
end
|
|
end
|
|
end
|
|
|
|
def ship_generated_inventory(inventory_content)
|
|
inventory_basedir = File.join(config.tmp_path, "inventory")
|
|
inventory_path = File.join(inventory_basedir, "vagrant_ansible_local_inventory")
|
|
|
|
create_and_chown_remote_folder(inventory_basedir)
|
|
@machine.communicate.sudo("rm -f #{inventory_path}", error_check: false)
|
|
|
|
Tempfile.open("vagrant-ansible-local-inventory-#{@machine.name}") do |f|
|
|
f.binmode
|
|
f.write(inventory_content)
|
|
f.fsync
|
|
f.close
|
|
@machine.communicate.upload(f.path, inventory_path)
|
|
end
|
|
|
|
return inventory_basedir
|
|
end
|
|
|
|
def generate_inventory_machines
|
|
machines = ""
|
|
|
|
# TODO: Instead, why not loop over active_machines and skip missing guests, like in Host?
|
|
machine.env.machine_names.each do |machine_name|
|
|
begin
|
|
@inventory_machines[machine_name] = machine_name
|
|
if @machine.name == machine_name
|
|
machines += "#{machine_name} ansible_connection=local\n"
|
|
else
|
|
machines += "#{machine_name}\n"
|
|
end
|
|
host_vars = get_inventory_host_vars_string(machine_name)
|
|
machines.sub!(/\n$/, " #{host_vars}\n") if host_vars
|
|
end
|
|
end
|
|
|
|
return machines
|
|
end
|
|
|
|
def create_and_chown_remote_folder(path)
|
|
@machine.communicate.tap do |comm|
|
|
comm.sudo("mkdir -p #{path}")
|
|
comm.sudo("chown -h #{@machine.ssh_info[:username]} #{path}")
|
|
end
|
|
end
|
|
|
|
end
|
|
end
|
|
end
|
|
end
|