require 'spec_helper' describe Ridley::HostConnector::WinRM do subject { connector } let(:connector) { described_class.new } let(:host) { 'fake.riotgames.com' } let(:command_uploader_double) { double('command_uploader', :cleanup => nil, :upload => nil, :command => nil) } let(:options) do { server_url: double('server_url'), validator_path: fixtures_path.join('my-fake.pem'), validator_client: double('validator_client'), encrypted_data_bag_secret: 'encrypted_data_bag_secret', winrm: Hash.new, chef_version: double('chef_version') } end before do described_class::CommandUploader.stub(:new).and_return(command_uploader_double) ::WinRM::WinRMWebService.stub(:new).and_return(double(:set_timeout => nil, :run_cmd => {exitcode: 0})) end describe "#run" do subject(:result) { connector.run(host, command, options) } let(:command) { "dir" } let(:command_uploader_stub) { double('CommandUploader', cleanup: true) } let(:stdout) { "stdout" } let(:stderr) { nil } let(:winrm_stub) { double } before do described_class::CommandUploader.stub(:new).and_return(command_uploader_stub) connector.stub(:winrm).and_return(winrm_stub) winrm_stub.stub(:run_cmd).and_yield(stdout, stderr).and_return(exitcode: 0) end context "when the command is forced" do let(:command_uploader) { double(upload: nil, command: command, cleanup: true) } let(:command) { "a forced-to-upload command" } let(:options) do { force_batch_file: true } end before do Ridley::HostConnector::WinRM::CommandUploader.stub(:new).and_return(command_uploader) end it "uploads the command" do expect(winrm_stub).to receive(:run_cmd).with("a forced-to-upload command") result end end context "when the command is too long" do let(:command_uploader) { double(upload: nil, command: command, cleanup: true) } let(:command) { "a" * 2048 } before do Ridley::HostConnector::WinRM::CommandUploader.stub(:new).and_return(command_uploader) end it "uploads the command" do expect(winrm_stub).to receive(:run_cmd).with("a" * 2048) result end end context "when the exit_code is 0" do it "returns a non-error HostConnector::Response" do expect(result).to be_a(Ridley::HostConnector::Response) expect(result).to_not be_error end it "sets the response's stdout message" do expect(result.stdout).to eq("stdout") end end context "when the exit_code is not 0" do let(:stderr) { "stderr" } before do winrm_stub.stub(:run_cmd).and_yield(stdout, stderr).and_return(exitcode: 1) end it "returns an error HostConnector::Response with an error" do expect(result).to be_a(Ridley::HostConnector::Response) expect(result).to be_error end it "sets the response's stderr message" do expect(result.stderr).to eq("stderr") end end context "when a WinRM::WinRMHTTPTransportError error is raised" do let(:stderr) { "error" } before { winrm_stub.stub(:run_cmd).and_yield(stdout, stderr).and_raise(::WinRM::WinRMHTTPTransportError) } it "returns an error HostConnector::Response with an error" do expect(result).to be_a(Ridley::HostConnector::Response) expect(result).to be_error end it "sets the response's stderr message to the exception's message" do expect(result.stderr).to eql("WinRM::WinRMHTTPTransportError") end end context "when the force_batch_file is true" do let(:options_with_force_batch_file) { options.merge(force_batch_file: true) } subject(:result) { connector.run(host, command, options_with_force_batch_file) } it "should always use a command uploader when a force_batch_file is set to true" do command_uploader_stub.should_receive(:upload).with(command) command_uploader_stub.should_receive(:command).and_return(command) result end end end describe "#bootstrap" do let(:bootstrap_context) { Ridley::BootstrapContext::Windows.new(options) } it "sends a #run message to self to bootstrap a node" do connector.should_receive(:run).with(host, anything, options) connector.bootstrap(host, options) end it "filters the whole command" do expect(Ridley::Logging.logger).to receive(:filter_param).with(bootstrap_context.boot_command) connector.bootstrap(host, options) end end describe "#chef_client" do subject(:chef_client) { connector.chef_client(host, options) } it "receives a command to run chef-client" do connector.should_receive(:run).with(host, "chef-client -l warn", options) chef_client end end describe "#put_secret" do subject(:put_secret) { connector.put_secret(host, secret, options) } let(:encrypted_data_bag_secret_path) { fixtures_path.join("encrypted_data_bag_secret").to_s } let(:secret) { File.read(encrypted_data_bag_secret_path).chomp } before do Ridley::HostConnector::WinRM::CommandUploader.stub(:new).and_return(double(:cleanup => nil)) end it "receives a command to copy the secret" do connector.should_receive(:run).with(host, "echo #{secret} > C:\\chef\\encrypted_data_bag_secret", options ) put_secret end it "filters the secret" do expect(Ridley::Logging.logger).to receive(:filter_param).with(secret) put_secret end end describe "#ruby_script" do subject(:ruby_script) { connector.ruby_script(host, command_lines, options) } let(:command_lines) { ["puts 'hello'", "puts 'there'"] } it "receives a ruby call with the command" do connector.should_receive(:run).with(host, "#{described_class::EMBEDDED_RUBY_PATH} -e \"puts 'hello';puts 'there'\"", options.merge(force_batch_file: true) ) ruby_script end end describe "#connector_port_open?" do context "when no winrm options are specified" do it "should connect using the default winrm port" do subject.should_receive(:port_open?).with(host, Ridley::HostConnector::WinRM::DEFAULT_PORT, nil) subject.connector_port_open?(host, {}) end end context "when winrm options are given" do it "should use the winrm port" do subject.should_receive(:port_open?).with(host, 1234, nil) subject.connector_port_open?(host, winrm: {port: 1234}) end end end end