aspera-cli 4.19.0 → 4.21.1

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.
Files changed (91) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/CHANGELOG.md +46 -0
  4. data/CONTRIBUTING.md +18 -4
  5. data/README.md +886 -510
  6. data/bin/asession +27 -20
  7. data/examples/build_exec +65 -76
  8. data/examples/build_exec_rubyc +40 -0
  9. data/examples/get_proto_file.rb +7 -0
  10. data/lib/aspera/agent/alpha.rb +18 -24
  11. data/lib/aspera/agent/base.rb +2 -18
  12. data/lib/aspera/agent/connect.rb +34 -15
  13. data/lib/aspera/agent/direct.rb +44 -54
  14. data/lib/aspera/agent/httpgw.rb +2 -3
  15. data/lib/aspera/agent/node.rb +11 -21
  16. data/lib/aspera/agent/{trsdk.rb → transferd.rb} +27 -51
  17. data/lib/aspera/api/alee.rb +15 -0
  18. data/lib/aspera/api/aoc.rb +139 -105
  19. data/lib/aspera/api/ats.rb +1 -1
  20. data/lib/aspera/api/cos_node.rb +1 -1
  21. data/lib/aspera/api/httpgw.rb +15 -10
  22. data/lib/aspera/api/node.rb +70 -32
  23. data/lib/aspera/ascmd.rb +56 -48
  24. data/lib/aspera/ascp/installation.rb +166 -70
  25. data/lib/aspera/ascp/management.rb +30 -8
  26. data/lib/aspera/assert.rb +10 -5
  27. data/lib/aspera/cli/formatter.rb +166 -162
  28. data/lib/aspera/cli/hints.rb +2 -1
  29. data/lib/aspera/cli/info.rb +12 -10
  30. data/lib/aspera/cli/main.rb +28 -13
  31. data/lib/aspera/cli/manager.rb +7 -2
  32. data/lib/aspera/cli/plugin.rb +17 -31
  33. data/lib/aspera/cli/plugins/alee.rb +3 -3
  34. data/lib/aspera/cli/plugins/aoc.rb +246 -208
  35. data/lib/aspera/cli/plugins/ats.rb +16 -14
  36. data/lib/aspera/cli/plugins/config.rb +154 -94
  37. data/lib/aspera/cli/plugins/console.rb +3 -3
  38. data/lib/aspera/cli/plugins/cos.rb +1 -0
  39. data/lib/aspera/cli/plugins/faspex.rb +15 -23
  40. data/lib/aspera/cli/plugins/faspex5.rb +64 -50
  41. data/lib/aspera/cli/plugins/faspio.rb +2 -2
  42. data/lib/aspera/cli/plugins/httpgw.rb +1 -1
  43. data/lib/aspera/cli/plugins/node.rb +174 -109
  44. data/lib/aspera/cli/plugins/orchestrator.rb +14 -13
  45. data/lib/aspera/cli/plugins/preview.rb +8 -9
  46. data/lib/aspera/cli/plugins/server.rb +5 -9
  47. data/lib/aspera/cli/plugins/shares.rb +2 -2
  48. data/lib/aspera/cli/sync_actions.rb +2 -2
  49. data/lib/aspera/cli/transfer_agent.rb +12 -14
  50. data/lib/aspera/cli/transfer_progress.rb +37 -17
  51. data/lib/aspera/cli/version.rb +1 -1
  52. data/lib/aspera/command_line_builder.rb +4 -5
  53. data/lib/aspera/coverage.rb +13 -1
  54. data/lib/aspera/environment.rb +75 -25
  55. data/lib/aspera/faspex_gw.rb +2 -2
  56. data/lib/aspera/json_rpc.rb +1 -1
  57. data/lib/aspera/keychain/macos_security.rb +7 -12
  58. data/lib/aspera/log.rb +3 -4
  59. data/lib/aspera/node_simulator.rb +230 -112
  60. data/lib/aspera/oauth/base.rb +64 -83
  61. data/lib/aspera/oauth/factory.rb +52 -6
  62. data/lib/aspera/oauth/generic.rb +4 -8
  63. data/lib/aspera/oauth/jwt.rb +6 -3
  64. data/lib/aspera/oauth/url_json.rb +1 -2
  65. data/lib/aspera/oauth/web.rb +5 -2
  66. data/lib/aspera/persistency_action_once.rb +16 -8
  67. data/lib/aspera/persistency_folder.rb +20 -2
  68. data/lib/aspera/preview/generator.rb +1 -1
  69. data/lib/aspera/preview/utils.rb +11 -17
  70. data/lib/aspera/products/alpha.rb +30 -0
  71. data/lib/aspera/products/connect.rb +48 -0
  72. data/lib/aspera/products/other.rb +82 -0
  73. data/lib/aspera/products/transferd.rb +54 -0
  74. data/lib/aspera/rest.rb +116 -87
  75. data/lib/aspera/secret_hider.rb +2 -2
  76. data/lib/aspera/ssh.rb +31 -24
  77. data/lib/aspera/transfer/faux_file.rb +4 -4
  78. data/lib/aspera/transfer/parameters.rb +16 -17
  79. data/lib/aspera/transfer/spec.rb +12 -12
  80. data/lib/aspera/transfer/spec.yaml +22 -20
  81. data/lib/aspera/transfer/sync.rb +2 -10
  82. data/lib/aspera/transfer/uri.rb +3 -3
  83. data/lib/aspera/uri_reader.rb +1 -1
  84. data/lib/aspera/web_auth.rb +166 -17
  85. data/lib/aspera/web_server_simple.rb +4 -3
  86. data/lib/transferd_pb.rb +86 -0
  87. data/lib/transferd_services_pb.rb +84 -0
  88. data.tar.gz.sig +0 -0
  89. metadata +58 -22
  90. metadata.gz.sig +0 -0
  91. data/lib/aspera/ascp/products.rb +0 -156
@@ -3,24 +3,30 @@
3
3
  # cspell:ignore protobuf ckpt
4
4
  require 'aspera/environment'
5
5
  require 'aspera/data_repository'
6
- require 'aspera/ascp/products'
7
6
  require 'aspera/log'
7
+ require 'aspera/rest'
8
+ require 'aspera/uri_reader'
8
9
  require 'aspera/assert'
9
10
  require 'aspera/web_server_simple'
11
+ require 'aspera/cli/info'
12
+ require 'aspera/cli/version'
13
+ require 'aspera/products/alpha'
14
+ require 'aspera/products/connect'
15
+ require 'aspera/products/transferd'
16
+ require 'aspera/products/other'
10
17
  require 'English'
11
18
  require 'singleton'
12
19
  require 'xmlsimple'
13
- require 'zlib'
14
20
  require 'base64'
15
21
  require 'fileutils'
16
22
  require 'openssl'
17
-
23
+ require 'yaml'
18
24
  module Aspera
19
25
  module Ascp
20
26
  # Singleton that tells where to find ascp and other local resources (keys..) , using the "path(:name)" method.
21
27
  # It is used by object : AgentDirect to find necessary resources
22
28
  # By default it takes the first Aspera product found
23
- # but the user can specify ascp location by calling:
29
+ # The user can specify ascp location by calling:
24
30
  # Installation.instance.use_ascp_from_product(product_name)
25
31
  # or
26
32
  # Installation.instance.ascp_path=""
@@ -28,7 +34,7 @@ module Aspera
28
34
  include Singleton
29
35
  # protobuf generated files from sdk
30
36
  EXT_RUBY_PROTOBUF = '_pb.rb'
31
- RB_SDK_FOLDER = 'lib'
37
+ RB_SDK_SUBFOLDER = 'lib'
32
38
  DEFAULT_ASPERA_CONF = <<~END_OF_CONFIG_FILE
33
39
  <?xml version='1.0' encoding='UTF-8'?>
34
40
  <CONF version="2">
@@ -43,11 +49,38 @@ module Aspera
43
49
  # all ascp files (in SDK)
44
50
  EXE_FILES = %i[ascp ascp4 async].freeze
45
51
  FILES = %i[transferd ssh_private_dsa ssh_private_rsa aspera_license aspera_conf fallback_certificate fallback_private_key].unshift(*EXE_FILES).freeze
46
- private_constant :EXT_RUBY_PROTOBUF, :RB_SDK_FOLDER, :DEFAULT_ASPERA_CONF, :FILES
52
+ TRANSFER_SDK_LOCATION_URL = 'https://ibm.biz/sdk_location'
53
+ FILE_SCHEME_PREFIX = 'file:///'
54
+ SDK_ARCHIVE_FOLDERS = ['/bin/', '/aspera/'].freeze
55
+ # filename for ascp with optional extension (Windows)
56
+ private_constant :EXT_RUBY_PROTOBUF, :RB_SDK_SUBFOLDER, :DEFAULT_ASPERA_CONF, :FILES, :TRANSFER_SDK_LOCATION_URL, :FILE_SCHEME_PREFIX
47
57
  # options for SSH client private key
48
58
  CLIENT_SSH_KEY_OPTIONS = %i{dsa_rsa rsa per_client}.freeze
59
+
60
+ class << self
61
+ def transfer_sdk_location_url
62
+ ENV.fetch('ASCLI_TRANSFER_SDK_LOCATION_URL', TRANSFER_SDK_LOCATION_URL)
63
+ end
64
+
65
+ # Loads YAML from cloud with locations of SDK archives for all platforms
66
+ # @return location structure
67
+ def sdk_locations
68
+ location_url = transfer_sdk_location_url
69
+ transferd_locations = UriReader.read(location_url)
70
+ Log.log.debug{"Retrieving SDK locations from #{location_url}"}
71
+ begin
72
+ return YAML.load(transferd_locations)
73
+ rescue Psych::SyntaxError
74
+ raise "Error when parsing yaml data from: #{location_url}"
75
+ end
76
+ end
77
+ end
78
+
49
79
  # set ascp executable path
50
80
  def ascp_path=(v)
81
+ Aspera.assert_type(v, String)
82
+ Aspera.assert(!v.empty?) {'ascp path cannot be empty: check your config file'}
83
+ Aspera.assert(File.exist?(v)) {"No such file: [#{v}]"}
51
84
  @path_to_ascp = v
52
85
  end
53
86
 
@@ -55,37 +88,19 @@ module Aspera
55
88
  path(:ascp)
56
89
  end
57
90
 
58
- def sdk_ruby_folder
59
- ruby_pb_folder = File.join(sdk_folder, RB_SDK_FOLDER)
60
- FileUtils.mkdir_p(ruby_pb_folder)
61
- return ruby_pb_folder
62
- end
63
-
64
- # location of SDK files
91
+ # Compatibility
65
92
  def sdk_folder=(v)
66
- Log.log.debug{"sdk_folder=#{v}"}
67
- @sdk_dir = v
68
- sdk_folder
69
- end
70
-
71
- # backward compatibility in sample program
72
- alias_method :folder=, :sdk_folder=
73
-
74
- # @return the path to folder where SDK is installed
75
- def sdk_folder
76
- raise 'SDK path was ot initialized' if @sdk_dir.nil?
77
- FileUtils.mkdir_p(@sdk_dir)
78
- @sdk_dir
93
+ Products::Transferd.sdk_directory = v
79
94
  end
80
95
 
81
96
  # find ascp in named product (use value : FIRST_FOUND='FIRST' to just use first one)
82
- # or select one from Products.installed_products()
97
+ # or select one from installed_products()
83
98
  def use_ascp_from_product(product_name)
84
99
  if product_name.eql?(FIRST_FOUND)
85
- pl = Products.installed_products.first
86
- raise "no FASP installation found\nPlease check manual on how to install FASP." if pl.nil?
100
+ pl = installed_products.first
101
+ raise "no Aspera transfer module or SDK found.\nRefer to the manual or install SDK with command:\nascli conf ascp install" if pl.nil?
87
102
  else
88
- pl = Products.installed_products.find{|i|i[:name].eql?(product_name)}
103
+ pl = installed_products.find{|i|i[:name].eql?(product_name)}
89
104
  raise "no such product installed: #{product_name}" if pl.nil?
90
105
  end
91
106
  self.ascp_path = pl[:ascp_path]
@@ -98,6 +113,8 @@ module Aspera
98
113
  m[v.to_s] =
99
114
  begin
100
115
  path(v)
116
+ rescue Errno::ENOENT => e
117
+ e.message.gsub(/.*assertion failed: /, '').gsub(/\): .*/, ')')
101
118
  rescue => e
102
119
  e.message
103
120
  end
@@ -105,7 +122,7 @@ module Aspera
105
122
  end
106
123
 
107
124
  def check_or_create_sdk_file(filename, force: false, &block)
108
- return Environment.write_file_restricted(File.join(sdk_folder, filename), force: force, mode: 0o644, &block)
125
+ return Environment.write_file_restricted(File.join(Products::Transferd.sdk_directory, filename), force: force, mode: 0o644, &block)
109
126
  end
110
127
 
111
128
  # get path of one resource file of currently activated product
@@ -120,7 +137,7 @@ module Aspera
120
137
  file = @path_to_ascp.gsub('ascp', k.to_s)
121
138
  when :transferd
122
139
  file_is_optional = true
123
- file = transferd_filepath
140
+ file = Products::Transferd.transferd_path
124
141
  when :ssh_private_dsa, :ssh_private_rsa
125
142
  # assume last 3 letters are type
126
143
  type = k.to_s[-3..-1].to_sym
@@ -130,8 +147,8 @@ module Aspera
130
147
  when :aspera_conf
131
148
  file = check_or_create_sdk_file('aspera.conf') {DEFAULT_ASPERA_CONF}
132
149
  when :fallback_certificate, :fallback_private_key
133
- file_key = File.join(sdk_folder, 'aspera_fallback_cert_private_key.pem')
134
- file_cert = File.join(sdk_folder, 'aspera_fallback_cert.pem')
150
+ file_key = File.join(Products::Transferd.sdk_directory, 'aspera_fallback_cert_private_key.pem')
151
+ file_cert = File.join(Products::Transferd.sdk_directory, 'aspera_fallback_cert.pem')
135
152
  if !File.exist?(file_key) || !File.exist?(file_cert)
136
153
  require 'openssl'
137
154
  # create new self signed certificate for http fallback
@@ -145,7 +162,7 @@ module Aspera
145
162
  else Aspera.error_unexpected_value(k)
146
163
  end
147
164
  return nil if file_is_optional && !File.exist?(file)
148
- Aspera.assert(File.exist?(file)){"no such file: #{file}"}
165
+ Aspera.assert(File.exist?(file), exception_class: Errno::ENOENT){"#{k} not found (#{file})"}
149
166
  return file
150
167
  end
151
168
 
@@ -177,7 +194,7 @@ module Aspera
177
194
  return nil unless File.exist?(exe_path)
178
195
  exe_version = nil
179
196
  cmd_out = %x("#{exe_path}" #{vers_arg})
180
- raise "An error occurred when testing #{ascp_filename}: #{cmd_out}" unless $CHILD_STATUS == 0
197
+ raise "An error occurred when testing #{exe_path}: #{cmd_out}" unless $CHILD_STATUS == 0
181
198
  # get version from ascp, only after full extract, as windows requires DLLs (SSL/TLS/etc...)
182
199
  m = cmd_out.match(/ version ([0-9.]+)/)
183
200
  exe_version = m[1].gsub(/\.$/, '') unless m.nil?
@@ -229,67 +246,127 @@ module Aspera
229
246
  return data
230
247
  end
231
248
 
249
+ # information for `ascp info`
232
250
  def ascp_info
233
- files = file_paths
234
- return files.merge(ascp_pvcl_info).merge(ascp_ssl_info)
251
+ ascp_data = file_paths
252
+ ascp_data.merge!(ascp_pvcl_info)
253
+ ascp_data['sdk_locations'] = self.class.transfer_sdk_location_url
254
+ ascp_data.merge!(ascp_ssl_info)
255
+ return ascp_data
256
+ end
257
+
258
+ # @return the url for download of SDK archive for the given platform and version
259
+ def sdk_url_for_platform(platform: nil, version: nil)
260
+ locations = self.class.sdk_locations
261
+ platform = Environment.architecture if platform.nil?
262
+ locations = locations.select{|l|l['platform'].eql?(platform)}
263
+ raise "No SDK for platform: #{platform}" if locations.empty?
264
+ version = locations.max_by { |entry| Gem::Version.new(entry['version']) }['version'] if version.nil?
265
+ info = locations.select{|entry| entry['version'].eql?(version)}
266
+ raise "No such version: #{version} for #{platform}" if info.empty?
267
+ return info.first['url']
268
+ end
269
+
270
+ # @param &block called with entry information
271
+ def extract_archive_files(sdk_archive_path)
272
+ raise 'missing block' unless block_given?
273
+ case sdk_archive_path
274
+ # Windows and Mac use zip
275
+ when /\.zip$/
276
+ require 'zip'
277
+ # extract files from archive
278
+ Zip::File.open(sdk_archive_path) do |zip_file|
279
+ zip_file.each do |entry|
280
+ next if entry.name.end_with?('/')
281
+ yield(entry.name, entry.get_input_stream, nil)
282
+ end
283
+ end
284
+ # Other Unixes use tar.gz
285
+ when /\.tar\.gz/
286
+ require 'zlib'
287
+ require 'rubygems/package'
288
+ Zlib::GzipReader.open(sdk_archive_path) do |gzip|
289
+ Gem::Package::TarReader.new(gzip) do |tar|
290
+ tar.each do |entry|
291
+ next if entry.directory?
292
+ yield(entry.full_name, entry, entry.symlink? ? entry.header.linkname : nil)
293
+ end
294
+ end
295
+ end
296
+ else
297
+ raise "unknown archive extension: #{sdk_archive_path}"
298
+ end
235
299
  end
236
300
 
237
301
  # download aspera SDK or use local file
238
302
  # extracts ascp binary for current system architecture
303
+ # @param url [String] URL to SDK archive, or SpecialValues::DEF
304
+ # @param folder [String] destination
305
+ # @param backup [Bool]
306
+ # @param with_exe [Bool]
307
+ # @param &block [Proc] a lambda that receives a file path from archive and tells detination sub folder, or nil to not extract
239
308
  # @return ascp version (from execution)
240
- def install_sdk(sdk_url)
241
- # SDK is organized by architecture, check this first, in case architecture is not supported
242
- arch_filter = "#{Environment.architecture}/"
243
- require 'zip'
244
- sdk_zip_path = File.join(Dir.tmpdir, 'sdk.zip')
245
- if sdk_url.start_with?('file:')
309
+ def install_sdk(url: nil, version: nil, folder: nil, backup: true, with_exe: true, &block)
310
+ url = sdk_url_for_platform(version: version) if url.nil? || url.eql?('DEF')
311
+ folder = Products::Transferd.sdk_directory if folder.nil?
312
+ subfolder_lambda = block
313
+ if subfolder_lambda.nil?
314
+ subfolder_lambda = ->(name) do
315
+ if SDK_ARCHIVE_FOLDERS.any?{|i|name.include?(i)}
316
+ '/'
317
+ elsif name.end_with?(EXT_RUBY_PROTOBUF)
318
+ RB_SDK_SUBFOLDER
319
+ end
320
+ end
321
+ end
322
+ if url.start_with?('file:')
246
323
  # require specific file scheme: the path part is "relative", or absolute if there are 4 slash
247
- raise 'use format: file:///<path>' unless sdk_url.start_with?('file:///')
248
- sdk_zip_path = sdk_url.gsub(%r{^file:///}, '')
324
+ raise 'use format: file:///<path>' unless url.start_with?(FILE_SCHEME_PREFIX)
325
+ sdk_archive_path = url[FILE_SCHEME_PREFIX.length..-1]
326
+ delete_archive = false
249
327
  else
250
- Aspera::Rest.new(base_url: sdk_url, redirect_max: 3).call(operation: 'GET', save_to_file: sdk_zip_path)
328
+ sdk_archive_path = File.join(Dir.tmpdir, File.basename(url))
329
+ Aspera::Rest.new(base_url: url, redirect_max: 3).call(operation: 'GET', save_to_file: sdk_archive_path)
330
+ delete_archive = true
251
331
  end
252
332
  # rename old install
253
- if !Dir.empty?(sdk_folder)
333
+ if backup && !Dir.empty?(folder)
254
334
  Log.log.warn('Previous install exists, renaming folder.')
255
- File.rename(sdk_folder, "#{sdk_folder}.#{Time.now.strftime('%Y%m%d%H%M%S')}")
335
+ File.rename(folder, "#{folder}.#{Time.now.strftime('%Y%m%d%H%M%S')}")
256
336
  # TODO: delete old archives ?
257
337
  end
258
- # extract files from archive
259
- Zip::File.open(sdk_zip_path) do |zip_file|
260
- zip_file.each do |entry|
261
- # skip folder entries
262
- next if entry.name.end_with?('/')
263
- dest_folder = nil
264
- # binaries
265
- dest_folder = sdk_folder if entry.name.include?(arch_filter)
266
- # ruby adapters
267
- dest_folder = sdk_ruby_folder if entry.name.end_with?(EXT_RUBY_PROTOBUF)
268
- next if dest_folder.nil?
269
- File.open(File.join(dest_folder, File.basename(entry.name)), 'wb') do |output_stream|
270
- IO.copy_stream(entry.get_input_stream, output_stream)
271
- end
338
+ extract_archive_files(sdk_archive_path) do |entry_name, entry_stream, link_target|
339
+ subfolder = subfolder_lambda.call(entry_name)
340
+ next if subfolder.nil?
341
+ dest_folder = File.join(folder, subfolder)
342
+ FileUtils.mkdir_p(dest_folder)
343
+ new_file = File.join(dest_folder, File.basename(entry_name))
344
+ if link_target.nil?
345
+ File.open(new_file, 'wb') { |output_stream|IO.copy_stream(entry_stream, output_stream)}
346
+ else
347
+ File.symlink(link_target, new_file)
272
348
  end
273
349
  end
274
- File.unlink(sdk_zip_path) rescue nil # Windows may give error
350
+ File.unlink(sdk_archive_path) rescue nil if delete_archive # Windows may give error
351
+ return unless with_exe
275
352
  # ensure license file are generated so that ascp invocation for version works
276
353
  path(:aspera_license)
277
354
  path(:aspera_conf)
278
- sdk_ascp_file = Products.ascp_filename
279
- sdk_ascp_path = File.join(sdk_folder, sdk_ascp_file)
355
+ sdk_ascp_file = Environment.exe_file('ascp')
356
+ sdk_ascp_path = File.join(folder, sdk_ascp_file)
280
357
  raise "No #{sdk_ascp_file} found in SDK archive" unless File.exist?(sdk_ascp_path)
281
358
  EXE_FILES.each do |exe_sym|
282
359
  exe_path = sdk_ascp_path.gsub('ascp', exe_sym.to_s)
283
360
  Environment.restrict_file_access(exe_path, mode: 0o755) if File.exist?(exe_path)
284
361
  end
285
362
  sdk_ascp_version = get_ascp_version(sdk_ascp_path)
286
- sdk_daemon_path = transferd_filepath
363
+ sdk_daemon_path = Products::Transferd.transferd_path
287
364
  Log.log.warn{"No #{sdk_daemon_path} in SDK archive"} unless File.exist?(sdk_daemon_path)
288
365
  Environment.restrict_file_access(sdk_daemon_path, mode: 0o755) if File.exist?(sdk_daemon_path)
289
366
  transferd_version = get_exe_version(sdk_daemon_path, 'version')
290
367
  sdk_name = 'IBM Aspera Transfer SDK'
291
368
  sdk_version = transferd_version || sdk_ascp_version
292
- File.write(File.join(sdk_folder, Products::INFO_META_FILE), "<product><name>#{sdk_name}</name><version>#{sdk_version}</version></product>")
369
+ File.write(File.join(folder, Products::Other::INFO_META_FILE), "<product><name>#{sdk_name}</name><version>#{sdk_version}</version></product>")
293
370
  return sdk_name, sdk_version
294
371
  end
295
372
 
@@ -301,10 +378,29 @@ module Aspera
301
378
  def initialize
302
379
  @path_to_ascp = nil
303
380
  @sdk_dir = nil
381
+ @found_products = nil
304
382
  end
305
383
 
306
- def transferd_filepath
307
- return File.join(sdk_folder, 'asperatransferd' + Environment.exe_extension) # cspell:disable-line
384
+ public
385
+
386
+ # @return the list of installed products in format of product_locations_on_current_os
387
+ def installed_products
388
+ if @found_products.nil?
389
+ # :expected M app name is taken from the manifest if present, else defaults to this value
390
+ # :app_root M main folder for the application
391
+ # :log_root O location of log files (Linux uses syslog)
392
+ # :run_root O only for Connect Client, location of http port file
393
+ # :sub_bin O subfolder with executables, default : bin
394
+ scan_locations = Products::Transferd.locations.concat(
395
+ Products::Alpha.locations,
396
+ Products::Connect.locations,
397
+ Products::Other::LOCATION_ON_THIS_OS
398
+ )
399
+ # .each {|item| item.deep_do {|h, _k, _v, _m|h.freeze}}.freeze
400
+ # search installed products: with ascp
401
+ @found_products = Products::Other.find(scan_locations)
402
+ end
403
+ return @found_products
308
404
  end
309
405
  end
310
406
  end
@@ -1,5 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require 'aspera/assert'
4
+
3
5
  module Aspera
4
6
  module Ascp
5
7
  # processing of ascp management port events
@@ -196,18 +198,39 @@ module Aspera
196
198
  BOOLEAN_FIELDS = %w[Encryption Remote RateLock MinRateLock PolicyLock FilesEncrypt FilesDecrypt VLinkLocalEnabled VLinkRemoteEnabled
197
199
  MoveRange Keepalive TestLogin UseProxy Precalc RTTAutocorrect].freeze
198
200
  BOOLEAN_TRUE = 'Yes'
201
+
202
+ private_constant :OPERATIONS, :PARAMETERS, :MGT_HEADER, :MGT_FRAME_SEPARATOR, :INTEGER_FIELDS, :BOOLEAN_FIELDS, :BOOLEAN_TRUE
199
203
  # cspell: enable
200
204
 
201
205
  class << self
202
206
  # translates mgt port event into (enhanced) typed event
203
207
  def enhanced_event_format(event)
204
208
  return event.keys.each_with_object({}) do |e, h|
205
- new_name = e.capital_to_snake.gsub(/(usec)$/, '_\1').downcase
206
- value = event[e]
207
- value = value.to_i if INTEGER_FIELDS.include?(e)
208
- value = value.eql?(BOOLEAN_TRUE) if BOOLEAN_FIELDS.include?(e)
209
- h[new_name] = value
210
- end
209
+ new_name =
210
+ case e
211
+ when 'Elapsedusec' then 'elapsed_usec'
212
+ when 'Bytescont' then 'bytes_cont'
213
+ else e.capital_to_snake
214
+ end
215
+ h[new_name] =
216
+ if INTEGER_FIELDS.include?(e) then event[e].to_i
217
+ elsif BOOLEAN_FIELDS.include?(e) then event[e].eql?(BOOLEAN_TRUE)
218
+ else
219
+ event[e]
220
+ end
221
+ end
222
+ end
223
+
224
+ # build command to send on management port
225
+ # @param data [Hash] {'type'=>'START','source'=>_path_,'destination'=>_path_}
226
+ def command_to_stream(data)
227
+ # TODO: translate enhanced to capitalized ?
228
+ data
229
+ .keys
230
+ .map{|k|"#{k.capitalize}: #{data[k]}"}
231
+ .unshift(MGT_HEADER)
232
+ .push('', '')
233
+ .join("\n")
211
234
  end
212
235
  end
213
236
 
@@ -237,8 +260,7 @@ module Aspera
237
260
  @last_event = @event_build
238
261
  @event_build = nil
239
262
  return @last_event
240
- else
241
- raise "mgt port: unexpected line: [#{line}]"
263
+ else Aspera.error_unexpected_value(line){'mgt port'}
242
264
  end
243
265
  return nil
244
266
  end
data/lib/aspera/assert.rb CHANGED
@@ -8,13 +8,13 @@ module Aspera
8
8
  end
9
9
  class << self
10
10
  # the block is executed in the context of the Aspera module
11
- def assert(assertion, info = nil, level: 2, exception_class: AssertError)
11
+ def assert(assertion, info = nil, exception_class: AssertError)
12
12
  raise InternalError, 'bad assert: both info and block given' unless info.nil? || !block_given?
13
13
  return if assertion
14
14
  message = 'assertion failed'
15
15
  info = yield if block_given?
16
16
  message = "#{message}: #{info}" if info
17
- message = "#{message}: #{caller(level..level).first}"
17
+ message = "#{message}: #{caller.find{|call|!call.start_with?(__FILE__)}}"
18
18
  raise exception_class, message
19
19
  end
20
20
 
@@ -22,13 +22,18 @@ module Aspera
22
22
  # @param value [Object] the value to check
23
23
  # @param type [Class] the expected type
24
24
  def assert_type(value, type, exception_class: AssertError)
25
- assert(value.is_a?(type), level: 3, exception_class: exception_class){"#{block_given? ? "#{yield}: " : nil}expecting #{type}, but have #{value.inspect}"}
25
+ assert(value.is_a?(type), exception_class: exception_class){"#{block_given? ? "#{yield}: " : nil}expecting #{type}, but have #{value.inspect}"}
26
26
  end
27
27
 
28
28
  # assert that value is one of the given values
29
+ # @param value value to check
30
+ # @param values accepted values
31
+ # @param exception_class exception in case of no match
29
32
  def assert_values(value, values, exception_class: AssertError)
30
- assert(values.include?(value), level: 3, exception_class: exception_class) do
31
- "#{block_given? ? "#{yield}: " : nil}expecting one of #{values.inspect}, but have #{value.inspect}"
33
+ assert(values.include?(value), exception_class: exception_class) do
34
+ val_list = values.inspect
35
+ val_list = "one of #{val_list}" if values.is_a?(Array)
36
+ "#{block_given? ? "#{yield}: " : nil}expecting #{val_list}, but have #{value.inspect}"
32
37
  end
33
38
  end
34
39