From eed7b859ca65ea4ca803cfb20a675a5b1f06d247 Mon Sep 17 00:00:00 2001 From: Chris Roberts Date: Mon, 13 Mar 2017 13:53:31 -0700 Subject: [PATCH] Fix provision action for provisioners set to never. This updates the behavior of the provision action to never run a provisioner that is specified to "never" run unless it has been explicitly requested. Also adds test coverage to the provision action. --- lib/vagrant/action/builtin/provision.rb | 15 +- .../vagrant/action/builtin/provision_test.rb | 208 ++++++++++++++++++ 2 files changed, 218 insertions(+), 5 deletions(-) create mode 100644 test/unit/vagrant/action/builtin/provision_test.rb diff --git a/lib/vagrant/action/builtin/provision.rb b/lib/vagrant/action/builtin/provision.rb index 02868c735..6c26121b7 100644 --- a/lib/vagrant/action/builtin/provision.rb +++ b/lib/vagrant/action/builtin/provision.rb @@ -102,12 +102,17 @@ module Vagrant type_map = provisioner_type_map(env) provisioner_instances(env).each do |p, options| type_name = type_map[p] - next if env[:provision_types] && \ - !env[:provision_types].include?(type_name) && \ - !env[:provision_types].include?(options[:name]) - # Don't run if sentinel is around and we're not always running - next if !provision_enabled && options[:run] != :always + if options[:run] == :never + next if env[:provision_types].nil? || !env[:provision_types].include?(options[:name]) + else + next if env[:provision_types] && \ + !env[:provision_types].include?(type_name) && \ + !env[:provision_types].include?(options[:name]) + + # Don't run if sentinel is around and we're not always running + next if !provision_enabled && options[:run] != :always + end name = type_name if options[:name] diff --git a/test/unit/vagrant/action/builtin/provision_test.rb b/test/unit/vagrant/action/builtin/provision_test.rb new file mode 100644 index 000000000..031b87f1d --- /dev/null +++ b/test/unit/vagrant/action/builtin/provision_test.rb @@ -0,0 +1,208 @@ +require File.expand_path("../../../../base", __FILE__) + +require Vagrant.source_root.join("plugins/kernel_v2/config/vm") + +describe Vagrant::Action::Builtin::Provision do + include_context "unit" + + let(:app) { lambda { |env| } } + let(:env) { + { machine: machine, ui: ui, hook: hook, provision_ignore_sentinel: false } + } + let(:hook){ double("hook") } + + let(:machine) do + double("machine").tap do |machine| + allow(machine).to receive(:id).and_return('machine-id') + allow(machine).to receive(:data_dir).and_return(data_dir) + allow(machine).to receive(:config).and_return(machine_config) + allow(machine).to receive(:env).and_return(machine_env) + end + end + + let(:machine_config) do + double("machine_config").tap do |config| + config.stub(vm: vm_config) + end + end + + let(:data_dir){ temporary_dir } + + let(:machine_env) do + isolated_environment.tap do |i_env| + allow(i_env).to receive(:data_dir).and_return(data_dir) + allow(i_env).to receive(:lock).and_yield + end + end + + let(:vm_config) do + double("machine_vm_config").tap do |config| + allow(config).to receive(:provisioners).and_return([]) + end + end + + let(:ui) do + double("ui").tap do |result| + allow(result).to receive(:info) + end + end + + let(:instance){ described_class.new(app, env) } + + describe "#call" do + context "with no provisioners defined" do + it "should process empty set of provisioners" do + expect(instance.call(env)).to eq([]) + end + + context "with provisioning disabled" do + before{ env[:provision_enabled] = false } + after{ env.delete(:provision_enabled) } + + it "should not process any provisioners" do + expect(instance.call(env)).to be_nil + end + end + end + + context "with single provisioner defined" do + let(:provisioner) do + prov = VagrantPlugins::Kernel_V2::VagrantConfigProvisioner.new("spec-test", :shell) + prov.config = provisioner_config + prov + end + let(:provisioner_config){ {} } + + before{ expect(vm_config).to receive(:provisioners).and_return([provisioner]) } + + it "should call the defined provisioner" do + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + context "with provisioning disabled" do + before{ env[:provision_enabled] = false } + after{ env.delete(:provision_enabled) } + + it "should not process any provisioners" do + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + expect(instance.call(env)).to be_nil + end + end + + context "with provisioner configured to run once" do + before{ provisioner.run = :once } + + it "should run if machine is not provisioned" do + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if machine is provisioned" do + File.open(File.join(data_dir.to_s, "action_provision"), "w") do |file| + file.write("1.5:machine-id") + end + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if provision types are set and provisioner is not included" do + env[:provision_types] = ["other-provisioner", "other-test"] + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner name" do + env[:provision_types] = ["spec-test"] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner type" do + env[:provision_types] = [:shell] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + end + + context "with provisioner configured to run always" do + before{ provisioner.run = :always } + + it "should run if machine is not provisioned" do + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if machine is provisioned" do + File.open(File.join(data_dir.to_s, "action_provision"), "w") do |file| + file.write("1.5:machine-id") + end + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if provision types are set and provisioner is not included" do + env[:provision_types] = ["other-provisioner", "other-test"] + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner name" do + env[:provision_types] = ["spec-test"] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner type" do + env[:provision_types] = [:shell] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + end + + context "with provisioner configured to never run" do + before{ provisioner.run = :never } + + it "should not run if machine is not provisioned" do + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if machine is provisioned" do + File.open(File.join(data_dir.to_s, "action_provision"), "w") do |file| + file.write("1.5:machine-id") + end + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if provision types are set and provisioner is not included" do + env[:provision_types] = ["other-provisioner", "other-test"] + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner name" do + env[:provision_types] = ["spec-test"] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should run if provision types are set and include provisioner name and machine is provisioned" do + File.open(File.join(data_dir.to_s, "action_provision"), "w") do |file| + file.write("1.5:machine-id") + end + env[:provision_types] = ["spec-test"] + expect(hook).to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + + it "should not run if provision types are set and include provisioner type" do + env[:provision_types] = [:shell] + expect(hook).not_to receive(:call).with(:provisioner_run, anything) + instance.call(env) + end + end + end + end +end