require 'chef/provisioning/convergence_strategy/precreate_chef_objects' require 'pathname' class Chef module Provisioning class ConvergenceStrategy class InstallMsi < PrecreateChefObjects @@install_msi_cache = {} def initialize(convergence_options, config) super @install_msi_url = convergence_options[:install_msi_url] || 'https://www.chef.io/chef/install.msi' @install_msi_path = convergence_options[:install_msi_path] || "$env:TEMP\\#{File.basename(@install_msi_url)}" @chef_client_timeout = convergence_options.has_key?(:chef_client_timeout) ? convergence_options[:chef_client_timeout] : 120*60 # Default: 2 hours end attr_reader :install_msi_url attr_reader :install_msi_path def setup_convergence(action_handler, machine) if !convergence_options.has_key?(:client_rb_path) || !convergence_options.has_key?(:client_pem_path) system_drive = machine.execute_always('$env:SystemDrive').stdout.strip @convergence_options = Cheffish::MergedConfig.new(convergence_options, { :client_rb_path => "#{system_drive}\\chef\\client.rb", :client_pem_path => "#{system_drive}\\chef\\client.pem" }) end super # Check for existing chef client. version = machine.execute_always('chef-client -v') # Don't do install/upgrade if a chef client exists and # no chef version is defined by user configs or # the chef client's version already matches user config if version.exitstatus == 0 version = version.stdout.strip if !chef_version return elsif version =~ /Chef: #{chef_version}$/ Chef::Log.debug "Already installed chef version #{version}" return elsif version.include?(chef_version) Chef::Log.warn "Installed chef version #{version} contains desired version #{chef_version}. " + "If you see this message on consecutive chef runs tighten your desired version constraint to prevent " + "multiple convergence." end end # Install chef client # TODO ssh verification of install.msi before running arbtrary code would be nice? # TODO find a way to cache this on the host like with the Unix stuff. # Limiter is we don't know how to efficiently upload large files to # the remote machine with WMI. machine.execute(action_handler, "(New-Object System.Net.WebClient).DownloadFile(#{machine.escape(install_msi_url)}, #{machine.escape(install_msi_path)})") machine.execute(action_handler, "msiexec /qn /i #{machine.escape(install_msi_path)}") end def converge(action_handler, machine) super action_handler.open_stream(machine.node['name']) do |stdout| action_handler.open_stream(machine.node['name']) do |stderr| command_line = "chef-client" command_line << " -l #{config[:log_level].to_s}" if config[:log_level] machine.execute(action_handler, command_line, :stream_stdout => stdout, :stream_stderr => stderr, :timeout => @chef_client_timeout) end end end end end end end