winrm 2.2.3 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/.rubocop.yml +13 -1
- data/.travis.yml +10 -11
- data/Gemfile +2 -3
- data/README.md +1 -1
- data/Rakefile +3 -4
- data/appveyor.yml +1 -2
- data/bin/rwinrm +90 -97
- data/changelog.md +5 -0
- data/lib/winrm.rb +3 -5
- data/lib/winrm/connection.rb +84 -86
- data/lib/winrm/connection_opts.rb +90 -91
- data/lib/winrm/exceptions.rb +14 -2
- data/lib/winrm/http/response_handler.rb +127 -96
- data/lib/winrm/http/transport.rb +462 -427
- data/lib/winrm/http/transport_factory.rb +1 -5
- data/lib/winrm/output.rb +1 -2
- data/lib/winrm/psrp/fragment.rb +0 -2
- data/lib/winrm/psrp/message.rb +1 -3
- data/lib/winrm/psrp/message_data.rb +0 -2
- data/lib/winrm/psrp/message_data/base.rb +0 -2
- data/lib/winrm/psrp/message_data/error_record.rb +0 -2
- data/lib/winrm/psrp/message_data/pipeline_host_call.rb +0 -2
- data/lib/winrm/psrp/message_data/pipeline_output.rb +48 -54
- data/lib/winrm/psrp/message_data/pipeline_state.rb +0 -2
- data/lib/winrm/psrp/message_data/runspacepool_host_call.rb +0 -2
- data/lib/winrm/psrp/message_data/runspacepool_state.rb +0 -2
- data/lib/winrm/psrp/message_data/session_capability.rb +0 -2
- data/lib/winrm/psrp/message_defragmenter.rb +2 -2
- data/lib/winrm/psrp/message_factory.rb +2 -3
- data/lib/winrm/psrp/message_fragmenter.rb +1 -3
- data/lib/winrm/psrp/powershell_output_decoder.rb +0 -2
- data/lib/winrm/psrp/receive_response_reader.rb +3 -5
- data/lib/winrm/psrp/uuid.rb +1 -2
- data/lib/winrm/shells/base.rb +6 -4
- data/lib/winrm/shells/cmd.rb +63 -65
- data/lib/winrm/shells/power_shell.rb +207 -202
- data/lib/winrm/shells/retryable.rb +44 -45
- data/lib/winrm/shells/shell_factory.rb +0 -2
- data/lib/winrm/version.rb +1 -3
- data/lib/winrm/wsmv/base.rb +0 -2
- data/lib/winrm/wsmv/cleanup_command.rb +1 -2
- data/lib/winrm/wsmv/close_shell.rb +1 -2
- data/lib/winrm/wsmv/command.rb +2 -3
- data/lib/winrm/wsmv/command_output.rb +2 -3
- data/lib/winrm/wsmv/command_output_decoder.rb +1 -2
- data/lib/winrm/wsmv/configuration.rb +0 -2
- data/lib/winrm/wsmv/create_pipeline.rb +0 -2
- data/lib/winrm/wsmv/create_shell.rb +2 -6
- data/lib/winrm/wsmv/header.rb +213 -215
- data/lib/winrm/wsmv/init_runspace_pool.rb +96 -95
- data/lib/winrm/wsmv/iso8601_duration.rb +0 -2
- data/lib/winrm/wsmv/keep_alive.rb +0 -2
- data/lib/winrm/wsmv/receive_response_reader.rb +128 -126
- data/lib/winrm/wsmv/send_data.rb +0 -2
- data/lib/winrm/wsmv/soap.rb +0 -2
- data/lib/winrm/wsmv/wql_pull.rb +54 -56
- data/lib/winrm/wsmv/wql_query.rb +98 -99
- data/lib/winrm/wsmv/write_stdin.rb +0 -2
- data/tests/integration/auth_timeout_spec.rb +0 -1
- data/tests/integration/cmd_spec.rb +2 -3
- data/tests/integration/issue_59_spec.rb +0 -1
- data/tests/integration/powershell_spec.rb +4 -5
- data/tests/integration/spec_helper.rb +3 -6
- data/tests/integration/transport_spec.rb +0 -1
- data/tests/integration/wql_spec.rb +33 -34
- data/tests/matchers.rb +2 -3
- data/tests/spec/configuration_spec.rb +0 -1
- data/tests/spec/connection_spec.rb +0 -2
- data/tests/spec/exception_spec.rb +0 -1
- data/tests/spec/http/transport_factory_spec.rb +1 -3
- data/tests/spec/http/transport_spec.rb +0 -1
- data/tests/spec/output_spec.rb +4 -3
- data/tests/spec/psrp/fragment_spec.rb +0 -2
- data/tests/spec/psrp/message_data/base_spec.rb +0 -2
- data/tests/spec/psrp/message_data/error_record_spec.rb +0 -2
- data/tests/spec/psrp/message_data/pipeline_host_call_spec.rb +0 -2
- data/tests/spec/psrp/message_data/pipeline_output_spec.rb +0 -2
- data/tests/spec/psrp/message_data/pipeline_state_spec.rb +0 -2
- data/tests/spec/psrp/message_data/runspace_pool_host_call_spec.rb +0 -2
- data/tests/spec/psrp/message_data/runspacepool_state_spec.rb +0 -2
- data/tests/spec/psrp/message_data/session_capability_spec.rb +0 -2
- data/tests/spec/psrp/message_data_spec.rb +0 -2
- data/tests/spec/psrp/message_defragmenter_spec.rb +0 -2
- data/tests/spec/psrp/message_fragmenter_spec.rb +0 -2
- data/tests/spec/psrp/powershell_output_decoder_spec.rb +0 -2
- data/tests/spec/psrp/psrp_message_spec.rb +10 -7
- data/tests/spec/psrp/recieve_response_reader_spec.rb +0 -2
- data/tests/spec/psrp/uuid_spec.rb +2 -2
- data/tests/spec/response_handler_spec.rb +69 -61
- data/tests/spec/shells/base_spec.rb +7 -5
- data/tests/spec/shells/cmd_spec.rb +4 -4
- data/tests/spec/shells/powershell_spec.rb +221 -175
- data/tests/spec/spec_helper.rb +0 -1
- data/tests/spec/stubs/responses/get_omi_command_output_response.xml.erb +23 -0
- data/tests/spec/stubs/responses/get_omi_command_output_response_not_done.xml.erb +24 -0
- data/tests/spec/stubs/responses/get_omi_config_response.xml +45 -0
- data/tests/spec/stubs/responses/get_omi_powershell_keepalive_response.xml.erb +33 -0
- data/tests/spec/stubs/responses/open_shell_omi.xml +43 -0
- data/tests/spec/stubs/responses/soap_fault_omi.xml +31 -0
- data/tests/spec/wsmv/cleanup_command_spec.rb +0 -2
- data/tests/spec/wsmv/close_shell_spec.rb +0 -2
- data/tests/spec/wsmv/command_output_decoder_spec.rb +0 -2
- data/tests/spec/wsmv/command_output_spec.rb +1 -3
- data/tests/spec/wsmv/command_spec.rb +0 -2
- data/tests/spec/wsmv/configuration_spec.rb +0 -2
- data/tests/spec/wsmv/create_pipeline_spec.rb +2 -3
- data/tests/spec/wsmv/create_shell_spec.rb +6 -5
- data/tests/spec/wsmv/init_runspace_pool_spec.rb +38 -36
- data/tests/spec/wsmv/keep_alive_spec.rb +4 -4
- data/tests/spec/wsmv/receive_response_reader_spec.rb +124 -123
- data/tests/spec/wsmv/send_data_spec.rb +4 -4
- data/tests/spec/wsmv/wql_query_spec.rb +11 -13
- data/tests/spec/wsmv/write_stdin_spec.rb +0 -2
- data/winrm.gemspec +11 -12
- metadata +73 -67
data/lib/winrm/wsmv/send_data.rb
CHANGED
data/lib/winrm/wsmv/soap.rb
CHANGED
data/lib/winrm/wsmv/wql_pull.rb
CHANGED
@@ -1,56 +1,54 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
@
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
"#{NS_ENUM}:
|
49
|
-
"#{
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
end
|
56
|
-
end
|
1
|
+
require 'nori'
|
2
|
+
require_relative 'base'
|
3
|
+
|
4
|
+
module WinRM
|
5
|
+
module WSMV
|
6
|
+
# WSMV message to 'pull' rest of enumeration results from Windows via WQL
|
7
|
+
class WqlPull < Base
|
8
|
+
def initialize(session_opts, namespace, enumeration_context)
|
9
|
+
@session_opts = session_opts
|
10
|
+
@namespace = namespace
|
11
|
+
@enumeration_context = enumeration_context
|
12
|
+
end
|
13
|
+
|
14
|
+
def process_response(response)
|
15
|
+
parser = Nori.new(
|
16
|
+
parser: :rexml,
|
17
|
+
advanced_typecasting: false,
|
18
|
+
convert_tags_to: ->(tag) { tag.snakecase.to_sym },
|
19
|
+
strip_namespaces: true
|
20
|
+
)
|
21
|
+
parser.parse(response.to_s)[:envelope][:body]
|
22
|
+
end
|
23
|
+
|
24
|
+
protected
|
25
|
+
|
26
|
+
def create_header(header)
|
27
|
+
header << Gyoku.xml(wql_header)
|
28
|
+
end
|
29
|
+
|
30
|
+
def create_body(body)
|
31
|
+
body.tag!("#{NS_ENUM}:Pull") { |en| en << Gyoku.xml(wql_body) }
|
32
|
+
end
|
33
|
+
|
34
|
+
private
|
35
|
+
|
36
|
+
def wql_header
|
37
|
+
merge_headers(
|
38
|
+
shared_headers(@session_opts),
|
39
|
+
resource_uri_wmi(@namespace),
|
40
|
+
action_enumerate_pull
|
41
|
+
)
|
42
|
+
end
|
43
|
+
|
44
|
+
def wql_body
|
45
|
+
{
|
46
|
+
"#{NS_ENUM}:EnumerationContext" => @enumeration_context,
|
47
|
+
"#{NS_WSMAN_DMTF}:OptimizeEnumeration" => nil,
|
48
|
+
"#{NS_ENUM}:MaxElements" => '32000',
|
49
|
+
"#{NS_WSMAN_MSFT}:SessionId" => "uuid:#{@session_opts[:session_id]}"
|
50
|
+
}
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
data/lib/winrm/wsmv/wql_query.rb
CHANGED
@@ -1,99 +1,98 @@
|
|
1
|
-
#
|
2
|
-
#
|
3
|
-
#
|
4
|
-
#
|
5
|
-
#
|
6
|
-
#
|
7
|
-
#
|
8
|
-
#
|
9
|
-
#
|
10
|
-
#
|
11
|
-
#
|
12
|
-
#
|
13
|
-
#
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
@
|
26
|
-
@
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
"#{NS_WSMAN_DMTF}:
|
87
|
-
"#{NS_WSMAN_DMTF}:
|
88
|
-
"#{
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
end
|
1
|
+
# Copyright 2016 Shawn Neal <sneal@sneal.net>
|
2
|
+
#
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
+
# you may not use this file except in compliance with the License.
|
5
|
+
# You may obtain a copy of the License at
|
6
|
+
#
|
7
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
8
|
+
#
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
+
# See the License for the specific language governing permissions and
|
13
|
+
# limitations under the License.
|
14
|
+
|
15
|
+
require 'nori'
|
16
|
+
require_relative 'base'
|
17
|
+
|
18
|
+
module WinRM
|
19
|
+
module WSMV
|
20
|
+
# WSMV message to query Windows via WQL
|
21
|
+
class WqlQuery < Base
|
22
|
+
def initialize(transport, session_opts, wql, namespace = nil)
|
23
|
+
@session_opts = session_opts
|
24
|
+
@wql = wql
|
25
|
+
@namespace = namespace
|
26
|
+
@transport = transport
|
27
|
+
end
|
28
|
+
|
29
|
+
def process_response(response, &block)
|
30
|
+
parser = Nori.new(
|
31
|
+
parser: :rexml,
|
32
|
+
advanced_typecasting: false,
|
33
|
+
convert_tags_to: ->(tag) { tag.snakecase.to_sym },
|
34
|
+
strip_namespaces: true
|
35
|
+
)
|
36
|
+
@items = Hash.new { |h, k| h[k] = [] }
|
37
|
+
|
38
|
+
hresp = parser.parse(response.to_s)[:envelope][:body][:enumerate_response]
|
39
|
+
process_items hresp[:items], &block
|
40
|
+
|
41
|
+
# Perform WS-Enum PULL's until we have all the elements
|
42
|
+
enumeration_context = hresp[:enumeration_context]
|
43
|
+
until enumeration_context.nil?
|
44
|
+
query = WqlPull.new(@session_opts, @namespace, enumeration_context)
|
45
|
+
hresp = query.process_response(@transport.send_request(query.build))[:pull_response]
|
46
|
+
process_items hresp[:items], &block
|
47
|
+
enumeration_context = hresp[:enumeration_context]
|
48
|
+
end
|
49
|
+
|
50
|
+
@items
|
51
|
+
end
|
52
|
+
|
53
|
+
protected
|
54
|
+
|
55
|
+
def create_header(header)
|
56
|
+
header << Gyoku.xml(wql_header)
|
57
|
+
end
|
58
|
+
|
59
|
+
def create_body(body)
|
60
|
+
body.tag!("#{NS_ENUM}:Enumerate") { |en| en << Gyoku.xml(wql_body) }
|
61
|
+
end
|
62
|
+
|
63
|
+
private
|
64
|
+
|
65
|
+
def process_items(items, &block)
|
66
|
+
return if items.nil?
|
67
|
+
|
68
|
+
items.each_pair do |k, v|
|
69
|
+
# Normalize items so the type always has an array even if it's just a single item.
|
70
|
+
v_ary = v.is_a?(Array) ? v : [v]
|
71
|
+
if block
|
72
|
+
v_ary.each { |val| yield k, val }
|
73
|
+
else
|
74
|
+
@items[k] += v_ary
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
def wql_header
|
80
|
+
merge_headers(shared_headers(@session_opts), resource_uri_wmi(@namespace), action_enumerate)
|
81
|
+
end
|
82
|
+
|
83
|
+
def wql_body
|
84
|
+
{
|
85
|
+
"#{NS_WSMAN_DMTF}:OptimizeEnumeration" => nil,
|
86
|
+
"#{NS_WSMAN_DMTF}:MaxElements" => '32000',
|
87
|
+
"#{NS_WSMAN_DMTF}:Filter" => @wql,
|
88
|
+
"#{NS_WSMAN_MSFT}:SessionId" => "uuid:#{@session_opts[:session_id]}",
|
89
|
+
:attributes! => {
|
90
|
+
"#{NS_WSMAN_DMTF}:Filter" => {
|
91
|
+
'Dialect' => 'http://schemas.microsoft.com/wbem/wsman/1/WQL'
|
92
|
+
}
|
93
|
+
}
|
94
|
+
}
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
98
|
+
end
|
@@ -1,4 +1,3 @@
|
|
1
|
-
# encoding: UTF-8
|
2
1
|
require_relative 'spec_helper'
|
3
2
|
|
4
3
|
describe 'winrm client cmd' do
|
@@ -21,7 +20,7 @@ describe 'winrm client cmd' do
|
|
21
20
|
end
|
22
21
|
|
23
22
|
describe 'codepage' do
|
24
|
-
let(:options) {
|
23
|
+
let(:options) { {} }
|
25
24
|
let(:shell) { winrm_connection.shell(:cmd, options) }
|
26
25
|
|
27
26
|
after { shell.close }
|
@@ -98,7 +97,7 @@ describe 'winrm client cmd' do
|
|
98
97
|
end
|
99
98
|
|
100
99
|
describe 'ipconfig with /all argument' do
|
101
|
-
subject(:output) { @cmd_shell.run('ipconfig', %w
|
100
|
+
subject(:output) { @cmd_shell.run('ipconfig', %w[/all]) }
|
102
101
|
it { should have_exit_code 0 }
|
103
102
|
it { should have_stdout_match(/Windows IP Configuration/) }
|
104
103
|
it { should have_no_stderr }
|
@@ -1,4 +1,3 @@
|
|
1
|
-
# encoding: UTF-8
|
2
1
|
require_relative 'spec_helper'
|
3
2
|
|
4
3
|
describe 'winrm client powershell' do
|
@@ -75,10 +74,10 @@ describe 'winrm client powershell' do
|
|
75
74
|
|
76
75
|
describe 'capturing output from Write-Host and Write-Error' do
|
77
76
|
subject(:output) do
|
78
|
-
script = <<-
|
77
|
+
script = <<-EOS
|
79
78
|
Write-Host 'Hello'
|
80
79
|
$host.ui.WriteErrorLine(', world!')
|
81
|
-
|
80
|
+
EOS
|
82
81
|
|
83
82
|
@captured_stdout = ''
|
84
83
|
@captured_stderr = ''
|
@@ -105,10 +104,10 @@ describe 'winrm client powershell' do
|
|
105
104
|
|
106
105
|
describe 'capturing output from pipeline followed by Host' do
|
107
106
|
subject(:output) do
|
108
|
-
script = <<-
|
107
|
+
script = <<-EOS
|
109
108
|
Write-Output 'output'
|
110
109
|
$host.UI.Writeline('host')
|
111
|
-
|
110
|
+
EOS
|
112
111
|
|
113
112
|
@captured_stdout = ''
|
114
113
|
@captured_stderr = ''
|
@@ -1,4 +1,3 @@
|
|
1
|
-
# encoding: UTF-8
|
2
1
|
require 'rubygems'
|
3
2
|
require 'bundler/setup'
|
4
3
|
require 'winrm'
|
@@ -12,8 +11,8 @@ module ConnectionHelper
|
|
12
11
|
end
|
13
12
|
|
14
13
|
def connection_opts
|
15
|
-
@
|
16
|
-
cfg = symbolize_keys(YAML.
|
14
|
+
@connection_opts ||= begin
|
15
|
+
cfg = symbolize_keys(YAML.safe_load(File.read(winrm_config_path)))
|
17
16
|
merge_environment(cfg)
|
18
17
|
end
|
19
18
|
end
|
@@ -22,9 +21,7 @@ module ConnectionHelper
|
|
22
21
|
merge_config_option_from_environment(config, 'user')
|
23
22
|
merge_config_option_from_environment(config, 'password')
|
24
23
|
merge_config_option_from_environment(config, 'no_ssl_peer_verification')
|
25
|
-
if ENV['use_ssl_peer_fingerprint']
|
26
|
-
config[:ssl_peer_fingerprint] = ENV['winrm_cert']
|
27
|
-
end
|
24
|
+
config[:ssl_peer_fingerprint] = ENV['winrm_cert'] if ENV['use_ssl_peer_fingerprint']
|
28
25
|
config[:endpoint] = ENV['winrm_endpoint'] if ENV['winrm_endpoint']
|
29
26
|
config
|
30
27
|
end
|
@@ -1,34 +1,33 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
output
|
11
|
-
|
12
|
-
output_caption
|
13
|
-
expect(output_caption).to include('
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
output
|
19
|
-
|
20
|
-
process_count
|
21
|
-
expect(
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
expect(
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
end
|
1
|
+
require_relative 'spec_helper'
|
2
|
+
|
3
|
+
describe 'winrm client wql' do
|
4
|
+
before(:all) do
|
5
|
+
@winrm = winrm_connection
|
6
|
+
end
|
7
|
+
|
8
|
+
it 'should query Win32_OperatingSystem' do
|
9
|
+
output = @winrm.run_wql('select * from Win32_OperatingSystem')
|
10
|
+
expect(output).to_not be_empty
|
11
|
+
output_caption = output[:win32_operating_system][0][:caption]
|
12
|
+
expect(output_caption).to include('Microsoft')
|
13
|
+
expect(output_caption).to include('Windows')
|
14
|
+
end
|
15
|
+
|
16
|
+
it 'should query Win32_Process' do
|
17
|
+
output = @winrm.run_wql('select * from Win32_Process')
|
18
|
+
expect(output).to_not be_empty
|
19
|
+
process_count = output[:win32_process].count
|
20
|
+
expect(process_count).to be > 1
|
21
|
+
expect(output[:win32_process]).to all(include(:command_line))
|
22
|
+
end
|
23
|
+
|
24
|
+
it 'should query Win32_Process with block' do
|
25
|
+
count = 0
|
26
|
+
@winrm.run_wql('select * from Win32_Process') do |type, item|
|
27
|
+
expect(type).to eq(:win32_process)
|
28
|
+
expect(item).to include(:command_line)
|
29
|
+
count += 1
|
30
|
+
end
|
31
|
+
expect(count).to be > 1
|
32
|
+
end
|
33
|
+
end
|