Class: RubySMB::SMB1::Tree
- Inherits:
-
Object
- Object
- RubySMB::SMB1::Tree
- Includes:
- Rap::NetShareEnum
- Defined in:
- lib/ruby_smb/smb1/tree.rb
Overview
An SMB1 connected remote Tree, as returned by a [RubySMB::SMB1::Packet::TreeConnectRequest]
Constant Summary
Constants included from Rap::NetShareEnum
Rap::NetShareEnum::DATA_DESCRIPTOR_LEVEL_1, Rap::NetShareEnum::DEFAULT_RECEIVE_BUFFER_SIZE, Rap::NetShareEnum::OPCODE, Rap::NetShareEnum::PARAM_DESCRIPTOR, Rap::NetShareEnum::SHARE_TYPES, Rap::NetShareEnum::STYPE_SPECIAL, Rap::NetShareEnum::STYPE_TEMPORARY
Instance Attribute Summary collapse
- #client ⇒ RubySMB::Client
- #guest_permissions ⇒ RubySMB::SMB1::BitField::DirectoryAccessMask
- #id ⇒ Integer
- #permissions ⇒ RubySMB::SMB1::BitField::DirectoryAccessMask
- #share ⇒ String
Instance Method Summary collapse
-
#disconnect! ⇒ WindowsError::ErrorCode
Disconnects this Tree from the current session.
-
#initialize(client:, share:, response:) ⇒ Tree
constructor
A new instance of Tree.
-
#list(directory: '\\', pattern: '*', unicode: true, type: RubySMB::SMB1::Packet::Trans2::FindInformationLevel::FindFileFullDirectoryInfo) ⇒ Array
List
directoryon the remote share. -
#open_file(opts) ⇒ RubySMB::SMB1::File
Open a file on the remote share.
- #open_pipe(opts) ⇒ Object
-
#set_header_fields(request) ⇒ RubySMB::SMB::Packet
Sets a few preset header fields that will always be set the same way for Tree operations.
Methods included from Rap::NetShareEnum
Constructor Details
#initialize(client:, share:, response:) ⇒ Tree
Returns a new instance of Tree.
36 37 38 39 40 41 42 |
# File 'lib/ruby_smb/smb1/tree.rb', line 36 def initialize(client:, share:, response:) @client = client @share = share @id = response.smb_header.tid @guest_permissions = response.parameter_block.guest_access_rights @permissions = response.parameter_block.access_rights end |
Instance Attribute Details
#client ⇒ RubySMB::Client
14 15 16 |
# File 'lib/ruby_smb/smb1/tree.rb', line 14 def client @client end |
#guest_permissions ⇒ RubySMB::SMB1::BitField::DirectoryAccessMask
19 20 21 |
# File 'lib/ruby_smb/smb1/tree.rb', line 19 def @guest_permissions end |
#id ⇒ Integer
34 35 36 |
# File 'lib/ruby_smb/smb1/tree.rb', line 34 def id @id end |
#permissions ⇒ RubySMB::SMB1::BitField::DirectoryAccessMask
24 25 26 |
# File 'lib/ruby_smb/smb1/tree.rb', line 24 def @permissions end |
#share ⇒ String
29 30 31 |
# File 'lib/ruby_smb/smb1/tree.rb', line 29 def share @share end |
Instance Method Details
#disconnect! ⇒ WindowsError::ErrorCode
Disconnects this Tree from the current session
48 49 50 51 52 53 54 55 56 57 58 59 60 61 |
# File 'lib/ruby_smb/smb1/tree.rb', line 48 def disconnect! request = RubySMB::SMB1::Packet::TreeDisconnectRequest.new request = set_header_fields(request) raw_response = client.send_recv(request) response = RubySMB::SMB1::Packet::TreeDisconnectResponse.read(raw_response) unless response.valid? raise RubySMB::Error::InvalidPacket.new( expected_proto: RubySMB::SMB1::SMB_PROTOCOL_ID, expected_cmd: RubySMB::SMB1::Packet::TreeDisconnectResponse::COMMAND, packet: response ) end response.status_code end |
#list(directory: '\\', pattern: '*', unicode: true, type: RubySMB::SMB1::Packet::Trans2::FindInformationLevel::FindFileFullDirectoryInfo) ⇒ Array
List directory on the remote share.
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 |
# File 'lib/ruby_smb/smb1/tree.rb', line 108 def list(directory: '\\', pattern: '*', unicode: true, type: RubySMB::SMB1::Packet::Trans2::FindInformationLevel::FindFileFullDirectoryInfo) info_standard = (type == RubySMB::SMB1::Packet::Trans2::FindInformationLevel::FindInfoStandard) find_first_request = RubySMB::SMB1::Packet::Trans2::FindFirst2Request.new find_first_request = set_header_fields(find_first_request) find_first_request.smb_header.flags2.unicode = 1 if unicode && !info_standard search_path = directory.dup search_path << '\\' unless search_path.end_with?('\\') search_path << pattern search_path = '\\' + search_path unless search_path.start_with?('\\') # Set the search parameters t2_params = find_first_request.data_block.trans2_parameters t2_params.search_attributes.hidden = 1 t2_params.search_attributes.system = 1 t2_params.search_attributes.directory = 1 t2_params.flags.close_eos = 1 t2_params.flags.resume_keys = 0 t2_params.information_level = type::CLASS_LEVEL t2_params.filename = search_path t2_params.search_count = info_standard ? 255 : 10 find_first_request = set_find_params(find_first_request) raw_response = client.send_recv(find_first_request) response = RubySMB::SMB1::Packet::Trans2::FindFirst2Response.read(raw_response) unless response.valid? raise RubySMB::Error::InvalidPacket.new( expected_proto: RubySMB::SMB1::SMB_PROTOCOL_ID, expected_cmd: RubySMB::SMB1::Packet::Trans2::FindFirst2Response::COMMAND, packet: response ) end unless response.status_code == WindowsError::NTStatus::STATUS_SUCCESS raise RubySMB::Error::UnexpectedStatusCode, response.status_code end t2p_override, t2d_override = response.win9x_trans2_overrides(raw_response) results = if t2d_override response.results(type, unicode: unicode, buffer: t2d_override) else response.results(type, unicode: unicode) end effective_params = t2p_override || response.data_block.trans2_parameters eos = effective_params.eos sid = effective_params.sid last = results.last&.file_name while eos.zero? && last find_next_request = RubySMB::SMB1::Packet::Trans2::FindNext2Request.new find_next_request = set_header_fields(find_next_request) find_next_request.smb_header.flags2.unicode = 1 if unicode t2_params = find_next_request.data_block.trans2_parameters t2_params.sid = sid t2_params.flags.close_eos = 1 t2_params.flags.resume_keys = 0 t2_params.information_level = type::CLASS_LEVEL t2_params.filename = last t2_params.search_count = 10 find_next_request = set_find_params(find_next_request) raw_response = client.send_recv(find_next_request) response = RubySMB::SMB1::Packet::Trans2::FindNext2Response.read(raw_response) unless response.valid? raise RubySMB::Error::InvalidPacket.new( expected_proto: RubySMB::SMB1::SMB_PROTOCOL_ID, expected_cmd: RubySMB::SMB1::Packet::Trans2::FindNext2Response::COMMAND, packet: response ) end unless response.status_code == WindowsError::NTStatus::STATUS_SUCCESS raise RubySMB::Error::UnexpectedStatusCode, response.status_code end batch = response.results(type, unicode: unicode) break if batch.empty? results += batch eos = response.data_block.trans2_parameters.eos last = results.last.file_name end results end |
#open_file(opts) ⇒ RubySMB::SMB1::File
Open a file on the remote share.
88 89 90 91 92 93 94 |
# File 'lib/ruby_smb/smb1/tree.rb', line 88 def open_file(opts) # Make sure we don't modify the caller's hash options opts = opts.dup opts[:filename] = opts[:filename].dup opts[:filename] = opts[:filename][1..-1] if opts[:filename].start_with?('\\'.encode(opts[:filename].encoding)) _open(**opts) end |
#open_pipe(opts) ⇒ Object
63 64 65 66 67 68 69 |
# File 'lib/ruby_smb/smb1/tree.rb', line 63 def open_pipe(opts) # Make sure we don't modify the caller's hash options opts = opts.dup opts[:filename] = opts[:filename].dup opts[:filename].prepend('\\') unless opts[:filename].start_with?('\\'.encode(opts[:filename].encoding)) _open(**opts) end |
#set_header_fields(request) ⇒ RubySMB::SMB::Packet
Sets a few preset header fields that will always be set the same way for Tree operations. This is, the TreeID and Extended Attributes.
203 204 205 206 207 |
# File 'lib/ruby_smb/smb1/tree.rb', line 203 def set_header_fields(request) request.smb_header.tid = @id request.smb_header.flags2.eas = 1 request end |