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.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data/CHANGELOG.md +46 -0
- data/CONTRIBUTING.md +18 -4
- data/README.md +886 -510
- data/bin/asession +27 -20
- data/examples/build_exec +65 -76
- data/examples/build_exec_rubyc +40 -0
- data/examples/get_proto_file.rb +7 -0
- data/lib/aspera/agent/alpha.rb +18 -24
- data/lib/aspera/agent/base.rb +2 -18
- data/lib/aspera/agent/connect.rb +34 -15
- data/lib/aspera/agent/direct.rb +44 -54
- data/lib/aspera/agent/httpgw.rb +2 -3
- data/lib/aspera/agent/node.rb +11 -21
- data/lib/aspera/agent/{trsdk.rb → transferd.rb} +27 -51
- data/lib/aspera/api/alee.rb +15 -0
- data/lib/aspera/api/aoc.rb +139 -105
- data/lib/aspera/api/ats.rb +1 -1
- data/lib/aspera/api/cos_node.rb +1 -1
- data/lib/aspera/api/httpgw.rb +15 -10
- data/lib/aspera/api/node.rb +70 -32
- data/lib/aspera/ascmd.rb +56 -48
- data/lib/aspera/ascp/installation.rb +166 -70
- data/lib/aspera/ascp/management.rb +30 -8
- data/lib/aspera/assert.rb +10 -5
- data/lib/aspera/cli/formatter.rb +166 -162
- data/lib/aspera/cli/hints.rb +2 -1
- data/lib/aspera/cli/info.rb +12 -10
- data/lib/aspera/cli/main.rb +28 -13
- data/lib/aspera/cli/manager.rb +7 -2
- data/lib/aspera/cli/plugin.rb +17 -31
- data/lib/aspera/cli/plugins/alee.rb +3 -3
- data/lib/aspera/cli/plugins/aoc.rb +246 -208
- data/lib/aspera/cli/plugins/ats.rb +16 -14
- data/lib/aspera/cli/plugins/config.rb +154 -94
- data/lib/aspera/cli/plugins/console.rb +3 -3
- data/lib/aspera/cli/plugins/cos.rb +1 -0
- data/lib/aspera/cli/plugins/faspex.rb +15 -23
- data/lib/aspera/cli/plugins/faspex5.rb +64 -50
- data/lib/aspera/cli/plugins/faspio.rb +2 -2
- data/lib/aspera/cli/plugins/httpgw.rb +1 -1
- data/lib/aspera/cli/plugins/node.rb +174 -109
- data/lib/aspera/cli/plugins/orchestrator.rb +14 -13
- data/lib/aspera/cli/plugins/preview.rb +8 -9
- data/lib/aspera/cli/plugins/server.rb +5 -9
- data/lib/aspera/cli/plugins/shares.rb +2 -2
- data/lib/aspera/cli/sync_actions.rb +2 -2
- data/lib/aspera/cli/transfer_agent.rb +12 -14
- data/lib/aspera/cli/transfer_progress.rb +37 -17
- data/lib/aspera/cli/version.rb +1 -1
- data/lib/aspera/command_line_builder.rb +4 -5
- data/lib/aspera/coverage.rb +13 -1
- data/lib/aspera/environment.rb +75 -25
- data/lib/aspera/faspex_gw.rb +2 -2
- data/lib/aspera/json_rpc.rb +1 -1
- data/lib/aspera/keychain/macos_security.rb +7 -12
- data/lib/aspera/log.rb +3 -4
- data/lib/aspera/node_simulator.rb +230 -112
- data/lib/aspera/oauth/base.rb +64 -83
- data/lib/aspera/oauth/factory.rb +52 -6
- data/lib/aspera/oauth/generic.rb +4 -8
- data/lib/aspera/oauth/jwt.rb +6 -3
- data/lib/aspera/oauth/url_json.rb +1 -2
- data/lib/aspera/oauth/web.rb +5 -2
- data/lib/aspera/persistency_action_once.rb +16 -8
- data/lib/aspera/persistency_folder.rb +20 -2
- data/lib/aspera/preview/generator.rb +1 -1
- data/lib/aspera/preview/utils.rb +11 -17
- data/lib/aspera/products/alpha.rb +30 -0
- data/lib/aspera/products/connect.rb +48 -0
- data/lib/aspera/products/other.rb +82 -0
- data/lib/aspera/products/transferd.rb +54 -0
- data/lib/aspera/rest.rb +116 -87
- data/lib/aspera/secret_hider.rb +2 -2
- data/lib/aspera/ssh.rb +31 -24
- data/lib/aspera/transfer/faux_file.rb +4 -4
- data/lib/aspera/transfer/parameters.rb +16 -17
- data/lib/aspera/transfer/spec.rb +12 -12
- data/lib/aspera/transfer/spec.yaml +22 -20
- data/lib/aspera/transfer/sync.rb +2 -10
- data/lib/aspera/transfer/uri.rb +3 -3
- data/lib/aspera/uri_reader.rb +1 -1
- data/lib/aspera/web_auth.rb +166 -17
- data/lib/aspera/web_server_simple.rb +4 -3
- data/lib/transferd_pb.rb +86 -0
- data/lib/transferd_services_pb.rb +84 -0
- data.tar.gz.sig +0 -0
- metadata +58 -22
- metadata.gz.sig +0 -0
- 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
|
-
#
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
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 =
|
86
|
-
raise "no
|
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 =
|
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(
|
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 =
|
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(
|
134
|
-
file_cert = File.join(
|
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)){"
|
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 #{
|
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
|
-
|
234
|
-
|
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(
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
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
|
248
|
-
|
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
|
-
|
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?(
|
333
|
+
if backup && !Dir.empty?(folder)
|
254
334
|
Log.log.warn('Previous install exists, renaming folder.')
|
255
|
-
File.rename(
|
335
|
+
File.rename(folder, "#{folder}.#{Time.now.strftime('%Y%m%d%H%M%S')}")
|
256
336
|
# TODO: delete old archives ?
|
257
337
|
end
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
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(
|
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 =
|
279
|
-
sdk_ascp_path = File.join(
|
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 =
|
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(
|
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
|
-
|
307
|
-
|
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
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
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,
|
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(
|
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),
|
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),
|
31
|
-
|
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
|
|