vagrant/test/unit/plugins/synced_folders/smb/synced_folder_test.rb

345 lines
14 KiB
Ruby

require_relative "../../../base"
require Vagrant.source_root.join("plugins/synced_folders/smb/synced_folder")
describe VagrantPlugins::SyncedFolderSMB::SyncedFolder do
include_context "unit"
let(:iso_env) do
env = isolated_environment
env.vagrantfile("")
env.create_vagrant_env
end
let(:guest){ double("guest") }
let(:host){ double("host") }
let(:machine) { iso_env.machine(iso_env.machine_names[0], :dummy) }
let(:host_caps){ [] }
let(:guest_caps){ [] }
let(:folders){ {"/first/path" => {}, "/second/path" => {}} }
let(:options){ {} }
before do
allow(machine.env).to receive(:host).and_return(host)
allow(machine).to receive(:guest).and_return(guest)
allow(machine).to receive(:ssh_info).and_return(username: 'sshuser')
allow(guest).to receive(:name).and_return("guest_name")
allow(host).to receive(:capability?).and_return(false)
host_caps.each do |cap|
allow(host).to receive(:capability?).with(cap).and_return(true)
allow(host).to receive(:capability).with(cap, any_args).and_return(true)
end
allow(guest).to receive(:capability?).and_return(false)
guest_caps.each do |cap|
allow(guest).to receive(:capability?).with(cap).and_return(true)
allow(guest).to receive(:capability).with(cap, any_args).and_return(true)
end
end
describe "#usable?" do
context "without supporting capabilities" do
it "is not usable" do
expect(subject.usable?(machine)).to be(false)
end
it "raises exception when raise_error enabled" do
expect{subject.usable?(machine, true)}.to raise_error(
VagrantPlugins::SyncedFolderSMB::Errors::SMBNotSupported)
end
end
context "with smb not installed" do
let(:host_caps){ [:smb_installed] }
it "is not usable" do
expect(host).to receive(:capability).with(:smb_installed).and_return(false)
expect(subject.usable?(machine)).to be(false)
end
end
context "with smb installed" do
let(:host_caps){ [:smb_installed] }
it "is usable" do
expect(subject.usable?(machine)).to be(true)
end
end
end
describe "#prepare" do
let(:host_caps){ [:smb_start, :smb_prepare] }
context "with username credentials provided" do
let(:folders){ {'/first/path' => {smb_username: 'smbuser'}} }
it "should prompt for credentials" do
expect(machine.env.ui).to receive(:ask).with(/name/, any_args).and_return('username').at_least(1)
expect(machine.env.ui).to receive(:ask).with(/word/, any_args).and_return('password').at_least(1)
subject.prepare(machine, folders, options)
end
it "should set credential information into all folder options and override username" do
expect(machine.env.ui).to receive(:ask).with(/name/, any_args).and_return('username').at_least(1)
expect(machine.env.ui).to receive(:ask).with(/word/, any_args).and_return('password').at_least(1)
subject.prepare(machine, folders, options)
expect(folders['/first/path'][:smb_username]).to eq('username')
expect(folders['/first/path'][:smb_password]).to eq('password')
end
it "will use configured default with no input" do
expect(machine.env.ui).to receive(:ask).with(/name/, any_args).and_return('').at_least(1)
expect(machine.env.ui).to receive(:ask).with(/word/, any_args).and_return('password').at_least(1)
subject.prepare(machine, folders, options)
expect(folders['/first/path'][:smb_username]).to eq('smbuser')
expect(folders['/first/path'][:smb_password]).to eq('password')
end
end
context "without credentials provided" do
before do
expect(machine.env.ui).to receive(:ask).with(/name/, any_args).and_return('username').at_least(1)
expect(machine.env.ui).to receive(:ask).with(/word/, any_args).and_return('password').at_least(1)
end
it "should prompt for credentials" do
subject.prepare(machine, folders, options)
end
it "should set credential information into all folder options" do
subject.prepare(machine, folders, options)
expect(folders['/first/path'][:smb_username]).to eq('username')
expect(folders['/first/path'][:smb_password]).to eq('password')
expect(folders['/second/path'][:smb_username]).to eq('username')
expect(folders['/second/path'][:smb_password]).to eq('password')
end
it "should start the SMB service if capability is available" do
expect(host).to receive(:capability).with(:smb_start, any_args)
subject.prepare(machine, folders, options)
end
context "with host smb_validate_password capability" do
let(:host_caps){ [:smb_start, :smb_prepare, :smb_validate_password] }
it "should validate the password" do
expect(host).to receive(:capability).with(:smb_validate_password, 'username', 'password').and_return(true)
subject.prepare(machine, folders, options)
end
it "should retry when validation fails" do
expect(host).to receive(:capability).with(:smb_validate_password, 'username', 'password').and_return(false)
expect(host).to receive(:capability).with(:smb_validate_password, 'username', 'password').and_return(true)
subject.prepare(machine, folders, options)
end
it "should raise an error if it exceeds the maximum number of retries" do
expect(host).to receive(:capability).with(:smb_validate_password, 'username', 'password').and_return(false).
exactly(VagrantPlugins::SyncedFolderSMB::SyncedFolder::CREDENTIAL_RETRY_MAX).times
expect{ subject.prepare(machine, folders, options) }.to raise_error(VagrantPlugins::SyncedFolderSMB::Errors::CredentialsRequestError)
end
end
end
context "with credentials provided" do
context "in single share entry" do
let(:folders){ {'/first/path' => {}, '/second/path' => {smb_username: 'smbuser', smb_password: 'smbpass'}} }
it "should not prompt for credentials" do
expect(machine.env.ui).not_to receive(:ask)
subject.prepare(machine, folders, options)
end
it "should add existing credentials to folder options without" do
subject.prepare(machine, folders, options)
expect(folders['/first/path'][:smb_username]).to eq('smbuser')
expect(folders['/first/path'][:smb_password]).to eq('smbpass')
end
end
context "in both entries" do
let(:folders){ {'/first/path' => {smb_username: 'user', smb_password: 'pass'},
'/second/path' => {smb_username: 'smbuser', smb_password: 'smbpass'}} }
it "should not modify existing credentials" do
subject.prepare(machine, folders, options)
expect(folders['/first/path'][:smb_username]).to eq('user')
expect(folders['/first/path'][:smb_password]).to eq('pass')
expect(folders['/second/path'][:smb_username]).to eq('smbuser')
expect(folders['/second/path'][:smb_password]).to eq('smbpass')
end
it "should register passwords with scrubber" do
expect(Vagrant::Util::CredentialScrubber).to receive(:sensitive).with('pass')
expect(Vagrant::Util::CredentialScrubber).to receive(:sensitive).with('smbpass')
subject.prepare(machine, folders, options)
end
end
end
end
describe "#enable" do
it "fails when guest does not support capability" do
expect{
subject.enable(machine, folders, options)
}.to raise_error(Vagrant::Errors::GuestCapabilityNotFound)
end
context "with guest capability supported" do
let(:guest_caps){ [:mount_smb_shared_folder, :choose_addressable_ip_addr] }
let(:host_caps){ [:configured_ip_addresses] }
it "should attempt to install smb on guest" do
expect(guest).to receive(:capability?).with(:smb_install).and_return(true)
expect(guest).to receive(:capability).with(:smb_install, any_args)
subject.enable(machine, folders, options)
end
it "should request host IP addresses" do
expect(host).to receive(:capability).with(:configured_ip_addresses)
subject.enable(machine, folders, options)
end
it "should determine guest accessible address" do
expect(guest).to receive(:capability).with(:choose_addressable_ip_addr, any_args)
subject.enable(machine, folders, options)
end
it "should error if no guest accessible address is available" do
expect(guest).to receive(:capability).with(:choose_addressable_ip_addr, any_args).and_return(nil)
expect{ subject.enable(machine, folders, options) }.to raise_error(
VagrantPlugins::SyncedFolderSMB::Errors::NoHostIPAddr)
end
it "should default owner and group to ssh username" do
subject.enable(machine, folders, options)
expect(folders["/first/path"][:owner]).to eq("sshuser")
expect(folders["/first/path"][:group]).to eq("sshuser")
expect(folders["/second/path"][:owner]).to eq("sshuser")
expect(folders["/second/path"][:group]).to eq("sshuser")
end
it "should set the host address in folder options" do
expect(guest).to receive(:capability).with(:choose_addressable_ip_addr, any_args).and_return("ADDR")
subject.enable(machine, folders, options)
expect(folders["/first/path"][:smb_host]).to eq("ADDR")
expect(folders["/second/path"][:smb_host]).to eq("ADDR")
end
it "should scrub folder configuration" do
expect(subject).to receive(:clean_folder_configuration).at_least(:once)
subject.enable(machine, folders, options)
end
context "with smb_host option set" do
let(:folders){ {"/first/path" => {smb_host: "ADDR"}, "/second/path" => {}} }
it "should not update the value" do
expect(guest).to receive(:capability).with(:choose_addressable_ip_addr, any_args).and_return("OTHER")
subject.enable(machine, folders, options)
expect(folders["/first/path"][:smb_host]).to eq("ADDR")
expect(folders["/second/path"][:smb_host]).to eq("OTHER")
end
end
context "with owner and group set" do
let(:folders){ {"/first/path" => {owner: "smbowner"}, "/second/path" => {group: "smbgroup"}} }
it "should not update set owner or group" do
subject.enable(machine, folders, options)
expect(folders["/first/path"][:owner]).to eq("smbowner")
expect(folders["/first/path"][:group]).to eq("sshuser")
expect(folders["/second/path"][:owner]).to eq("sshuser")
expect(folders["/second/path"][:group]).to eq("smbgroup")
end
end
context "with smb_username and smb_password set" do
let(:folders){ {
"/first/path" => {owner: "smbowner", smb_username: "user", smb_password: "pass"},
"/second/path" => {group: "smbgroup", smb_username: "user", smb_password: "pass"}
} }
it "should retain non password configuration options" do
subject.enable(machine, folders, options)
folder1 = folders["/first/path"]
folder2 = folders["/second/path"]
expect(folder1.key?(:owner)).to be_truthy
expect(folder1.key?(:smb_username)).to be_truthy
expect(folder2.key?(:group)).to be_truthy
expect(folder2.key?(:smb_username)).to be_truthy
end
it "should remove the smb_password option when set" do
subject.enable(machine, folders, options)
expect(folders["/first/path"].key?(:smb_password)).to be_falsey
expect(folders["/second/path"].key?(:smb_password)).to be_falsey
end
end
end
end
describe "#disable" do
it "should scrub folder configuration" do
expect(subject).to receive(:clean_folder_configuration).at_least(:once)
subject.disable(machine, folders, options)
end
context "with smb_username and smb_password set" do
let(:folders){ {
"/first/path" => {owner: "smbowner", smb_username: "user", smb_password: "pass"},
"/second/path" => {group: "smbgroup", smb_username: "user", smb_password: "pass"}
} }
it "should retain non password configuration options" do
subject.disable(machine, folders, options)
folder1 = folders["/first/path"]
folder2 = folders["/second/path"]
expect(folder1.key?(:owner)).to be_truthy
expect(folder1.key?(:smb_username)).to be_truthy
expect(folder2.key?(:group)).to be_truthy
expect(folder2.key?(:smb_username)).to be_truthy
end
it "should remove the smb_password option when set" do
subject.disable(machine, folders, options)
expect(folders["/first/path"].key?(:smb_password)).to be_falsey
expect(folders["/second/path"].key?(:smb_password)).to be_falsey
end
end
end
describe "#cleanup" do
context "without supporting capability" do
it "does nothing" do
subject.cleanup(machine, options)
end
end
context "with supporting capability" do
let(:host_caps){ [:smb_cleanup] }
it "runs cleanup" do
expect(host).to receive(:capability).with(:smb_cleanup, any_args)
subject.cleanup(machine, options)
end
end
end
describe "#clean_folder_configuration" do
it "should remove smb_password if defined" do
data = {smb_password: "password"}
subject.send(:clean_folder_configuration, data)
expect(data.key?(:smb_password)).to be_falsey
end
it "should not error if non-hash value provided" do
expect { subject.send(:clean_folder_configuration, nil) }.
not_to raise_error
end
end
end