345 lines
14 KiB
Ruby
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
|