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
data/lib/aspera/cli/formatter.rb
CHANGED
@@ -14,7 +14,6 @@ require 'pp'
|
|
14
14
|
|
15
15
|
module Aspera
|
16
16
|
module Cli
|
17
|
-
CONF_OVERVIEW_KEYS = %w[preset parameter value].freeze
|
18
17
|
# This class is used to transform a complex structure into a simple hash
|
19
18
|
class Flattener
|
20
19
|
def initialize(formatter)
|
@@ -30,17 +29,6 @@ module Aspera
|
|
30
29
|
return @result
|
31
30
|
end
|
32
31
|
|
33
|
-
# Special method for configuration overview
|
34
|
-
def config_over(something)
|
35
|
-
@result = []
|
36
|
-
something.each do |config, preset|
|
37
|
-
preset.each do |parameter, value|
|
38
|
-
@result.push(CONF_OVERVIEW_KEYS.zip([config, parameter, value]).to_h)
|
39
|
-
end
|
40
|
-
end
|
41
|
-
return @result
|
42
|
-
end
|
43
|
-
|
44
32
|
private
|
45
33
|
|
46
34
|
# Recursive function to flatten any type
|
@@ -98,12 +86,13 @@ module Aspera
|
|
98
86
|
CSV_RECORD_SEPARATOR = "\n"
|
99
87
|
CSV_FIELD_SEPARATOR = ','
|
100
88
|
# supported output formats
|
101
|
-
DISPLAY_FORMATS = %i[text nagios ruby json jsonpp yaml table
|
89
|
+
DISPLAY_FORMATS = %i[text nagios ruby json jsonpp yaml table csv image].freeze
|
102
90
|
# user output levels
|
103
91
|
DISPLAY_LEVELS = %i[info data error].freeze
|
104
|
-
|
92
|
+
# column names for single object display in table
|
93
|
+
SINGLE_OBJECT_COLUMN_NAMES = %i[field value].freeze
|
105
94
|
|
106
|
-
private_constant :
|
95
|
+
private_constant :FIELDS_LESS, :CSV_RECORD_SEPARATOR, :CSV_FIELD_SEPARATOR, :DISPLAY_FORMATS, :DISPLAY_LEVELS, :SINGLE_OBJECT_COLUMN_NAMES
|
107
96
|
# prefix to display error messages in user messages (terminal)
|
108
97
|
ERROR_FLASH = 'ERROR:'.bg_red.gray.blink.freeze
|
109
98
|
WARNING_FLASH = 'WARNING:'.bg_brown.black.blink.freeze
|
@@ -167,12 +156,42 @@ module Aspera
|
|
167
156
|
@spinner.spin
|
168
157
|
end
|
169
158
|
|
170
|
-
|
159
|
+
def long_operation_terminated
|
160
|
+
@spinner&.stop
|
161
|
+
@spinner = nil
|
162
|
+
end
|
163
|
+
|
164
|
+
def declare_options(options)
|
165
|
+
default_table_style = if Environment.instance.terminal_supports_unicode?
|
166
|
+
{border: :unicode_round}
|
167
|
+
else
|
168
|
+
{}
|
169
|
+
end
|
170
|
+
options.declare(:format, 'Output format', values: DISPLAY_FORMATS, handler: {o: self, m: :option_handler}, default: :table)
|
171
|
+
options.declare(:output, 'Destination for results', types: String, handler: {o: self, m: :option_handler})
|
172
|
+
options.declare(:display, 'Output only some information', values: DISPLAY_LEVELS, handler: {o: self, m: :option_handler}, default: :info)
|
173
|
+
options.declare(
|
174
|
+
:fields, "Comma separated list of: fields, or #{SpecialValues::ALL}, or #{SpecialValues::DEF}", handler: {o: self, m: :option_handler},
|
175
|
+
types: [String, Array, Regexp, Proc],
|
176
|
+
default: SpecialValues::DEF)
|
177
|
+
options.declare(:select, 'Select only some items in lists: column, value', types: [Hash, Proc], handler: {o: self, m: :option_handler})
|
178
|
+
options.declare(:table_style, 'Table display style', types: [Hash], handler: {o: self, m: :option_handler}, default: default_table_style)
|
179
|
+
options.declare(:flat_hash, '(Table) Display deep values as additional keys', values: :bool, handler: {o: self, m: :option_handler}, default: true)
|
180
|
+
options.declare(
|
181
|
+
:multi_single, '(Table) Control how object list is displayed as single table, or multiple objects', values: %i[no yes single],
|
182
|
+
handler: {o: self, m: :option_handler}, default: :no)
|
183
|
+
options.declare(:show_secrets, 'Show secrets on command output', values: :bool, handler: {o: self, m: :option_handler}, default: false)
|
184
|
+
options.declare(:image, 'Options for image display', types: Hash, handler: {o: self, m: :option_handler}, default: {})
|
185
|
+
end
|
186
|
+
|
187
|
+
# method accessed by option manager
|
188
|
+
# options are: format, output, display, fields, select, table_style, flat_hash, multi_single
|
171
189
|
def option_handler(option_symbol, operation, value=nil)
|
172
190
|
Aspera.assert_values(operation, %i[set get])
|
173
191
|
case operation
|
174
192
|
when :set
|
175
193
|
@options[option_symbol] = value
|
194
|
+
# special handling of some options
|
176
195
|
case option_symbol
|
177
196
|
when :output
|
178
197
|
$stdout = if value.eql?('-')
|
@@ -181,7 +200,9 @@ module Aspera
|
|
181
200
|
File.open(value, 'w')
|
182
201
|
end
|
183
202
|
when :image
|
203
|
+
# get list if key arguments of method
|
184
204
|
allowed_options = Preview::Terminal.method(:build).parameters.select{|i|i[0].eql?(:key)}.map{|i|i[1]}
|
205
|
+
# check that only supported options are given
|
185
206
|
unknown_options = value.keys.map(&:to_sym) - allowed_options
|
186
207
|
raise "Invalid parameter(s) for option image: #{unknown_options.join(', ')}, use #{allowed_options.join(', ')}" unless unknown_options.empty?
|
187
208
|
end
|
@@ -191,28 +212,6 @@ module Aspera
|
|
191
212
|
nil
|
192
213
|
end
|
193
214
|
|
194
|
-
def declare_options(options)
|
195
|
-
default_table_style = if Environment.instance.terminal_supports_unicode?
|
196
|
-
{border: :unicode_round}
|
197
|
-
else
|
198
|
-
{}
|
199
|
-
end
|
200
|
-
|
201
|
-
options.declare(:format, 'Output format', values: DISPLAY_FORMATS, handler: {o: self, m: :option_handler}, default: :table)
|
202
|
-
options.declare(:output, 'Destination for results', types: String, handler: {o: self, m: :option_handler})
|
203
|
-
options.declare(:display, 'Output only some information', values: DISPLAY_LEVELS, handler: {o: self, m: :option_handler}, default: :info)
|
204
|
-
options.declare(
|
205
|
-
:fields, "Comma separated list of: fields, or #{SpecialValues::ALL}, or #{SpecialValues::DEF}", handler: {o: self, m: :option_handler},
|
206
|
-
types: [String, Array, Regexp, Proc],
|
207
|
-
default: SpecialValues::DEF)
|
208
|
-
options.declare(:select, 'Select only some items in lists: column, value', types: [Hash, Proc], handler: {o: self, m: :option_handler})
|
209
|
-
options.declare(:table_style, 'Table display style', types: [Hash], handler: {o: self, m: :option_handler}, default: default_table_style)
|
210
|
-
options.declare(:flat_hash, 'Display deep values as additional keys', values: :bool, handler: {o: self, m: :option_handler}, default: true)
|
211
|
-
options.declare(:transpose_single, 'Single object fields output vertically', values: :bool, handler: {o: self, m: :option_handler}, default: true)
|
212
|
-
options.declare(:show_secrets, 'Show secrets on command output', values: :bool, handler: {o: self, m: :option_handler}, default: false)
|
213
|
-
options.declare(:image, 'Options for image display', types: Hash, handler: {o: self, m: :option_handler}, default: {})
|
214
|
-
end
|
215
|
-
|
216
215
|
# main output method
|
217
216
|
# data: for requested data, not displayed if level==error
|
218
217
|
# info: additional info, displayed if level==info
|
@@ -239,6 +238,116 @@ module Aspera
|
|
239
238
|
display_status(count_msg)
|
240
239
|
end
|
241
240
|
|
241
|
+
def hide_secrets(data)
|
242
|
+
SecretHider.deep_remove_secret(data) unless @options[:show_secrets] || @options[:display].eql?(:data)
|
243
|
+
end
|
244
|
+
|
245
|
+
# this method displays the results, especially the table format
|
246
|
+
# @param type [Symbol] type of data
|
247
|
+
# @param data [Object] data to display
|
248
|
+
# @param total [Integer] total number of items
|
249
|
+
# @param fields [Array<String>] list of fields to display
|
250
|
+
# @param name [String] name of the column to display
|
251
|
+
def display_results(type:, data: nil, total: nil, fields: nil, name: nil)
|
252
|
+
Log.log.debug{"display_results: #{type} class=#{data.class}"}
|
253
|
+
Log.log.trace1{"display_results:data=#{data}"}
|
254
|
+
Aspera.assert_type(type, Symbol){'result must have type'}
|
255
|
+
Aspera.assert(!data.nil? || %i[empty nothing].include?(type)){'result must have data'}
|
256
|
+
display_item_count(data.length, total) unless total.nil?
|
257
|
+
hide_secrets(data)
|
258
|
+
case @options[:format]
|
259
|
+
when :text
|
260
|
+
display_message(:data, data.to_s)
|
261
|
+
when :nagios
|
262
|
+
Nagios.process(data)
|
263
|
+
when :ruby
|
264
|
+
display_message(:data, PP.pp(filter_list_on_fields(data), +''))
|
265
|
+
when :json
|
266
|
+
display_message(:data, JSON.generate(filter_list_on_fields(data)))
|
267
|
+
when :jsonpp
|
268
|
+
display_message(:data, JSON.pretty_generate(filter_list_on_fields(data)))
|
269
|
+
when :yaml
|
270
|
+
display_message(:data, YAML.dump(filter_list_on_fields(data)))
|
271
|
+
when :image
|
272
|
+
# assume it is an url
|
273
|
+
url = data
|
274
|
+
case type
|
275
|
+
when :single_object, :object_list
|
276
|
+
url = [url] if type.eql?(:single_object)
|
277
|
+
raise 'image display requires a single result' unless url.length == 1
|
278
|
+
fields = compute_fields(url, fields)
|
279
|
+
raise 'select a field to display' unless fields.length == 1
|
280
|
+
url = url.first
|
281
|
+
raise 'no such field' unless url.key?(fields.first)
|
282
|
+
url = url[fields.first]
|
283
|
+
end
|
284
|
+
raise "not url: #{url.class} #{url}" unless url.is_a?(String)
|
285
|
+
display_message(:data, status_image(url))
|
286
|
+
when :table, :csv
|
287
|
+
case type
|
288
|
+
when :object_list, :single_object
|
289
|
+
obj_list = data
|
290
|
+
if type.eql?(:single_object)
|
291
|
+
obj_list = [obj_list]
|
292
|
+
@options[:multi_single] = :yes
|
293
|
+
end
|
294
|
+
Aspera.assert_type(obj_list, Array)
|
295
|
+
Aspera.assert(obj_list.all?(Hash)){"expecting Array of Hash: #{obj_list.inspect}"}
|
296
|
+
# :object_list is an array of hash tables, where key=colum name
|
297
|
+
obj_list = obj_list.map{|obj|Flattener.new(self).flatten(obj)} if @options[:flat_hash]
|
298
|
+
display_table(obj_list, compute_fields(obj_list, fields))
|
299
|
+
when :value_list
|
300
|
+
# :value_list is a simple array of values, name of column provided in the :name
|
301
|
+
display_table(data.map { |i| { name => i } }, [name])
|
302
|
+
when :empty # no table
|
303
|
+
display_message(:info, special_format('empty'))
|
304
|
+
return
|
305
|
+
when :nothing # no result expected
|
306
|
+
Log.log.debug('no result expected')
|
307
|
+
when :status # no table
|
308
|
+
# :status displays a simple message
|
309
|
+
display_message(:info, data)
|
310
|
+
when :text # no table
|
311
|
+
# :status displays a simple message
|
312
|
+
display_message(:data, data)
|
313
|
+
when :other_struct # no table
|
314
|
+
# :other_struct is any other type of structure
|
315
|
+
display_message(:data, PP.pp(data, +''))
|
316
|
+
else
|
317
|
+
raise "unknown data type: #{type}"
|
318
|
+
end
|
319
|
+
else
|
320
|
+
raise "not expected: #{@options[:format]}"
|
321
|
+
end
|
322
|
+
end
|
323
|
+
|
324
|
+
# @return text suitable to display an image from url
|
325
|
+
# @param blob [String] either a URL or image data
|
326
|
+
def status_image(blob)
|
327
|
+
# check if URL
|
328
|
+
begin
|
329
|
+
URI.parse(blob)
|
330
|
+
url = blob
|
331
|
+
unless Environment.instance.url_method.eql?(:text)
|
332
|
+
Environment.instance.open_uri(url)
|
333
|
+
return ''
|
334
|
+
end
|
335
|
+
blob = UriReader.read(url)
|
336
|
+
rescue URI::InvalidURIError
|
337
|
+
nil
|
338
|
+
end
|
339
|
+
# try base64
|
340
|
+
begin
|
341
|
+
blob = Base64.strict_decode64(blob)
|
342
|
+
rescue
|
343
|
+
nil
|
344
|
+
end
|
345
|
+
return Preview::Terminal.build(blob, **@options[:image].symbolize_keys)
|
346
|
+
end
|
347
|
+
#==========================================================================================
|
348
|
+
|
349
|
+
private
|
350
|
+
|
242
351
|
def all_fields(data)
|
243
352
|
data.each_with_object({}){|v, m|v.each_key{|c|m[c] = true}}.keys
|
244
353
|
end
|
@@ -295,7 +404,7 @@ module Aspera
|
|
295
404
|
return data if @options[:fields].eql?(SpecialValues::DEF)
|
296
405
|
selected_fields = compute_fields(data, @options[:fields])
|
297
406
|
return data.map{|i|i[selected_fields.first]} if selected_fields.length == 1
|
298
|
-
return data.map{|i|i.
|
407
|
+
return data.map{|i|i.slice(*selected_fields)}
|
299
408
|
end
|
300
409
|
|
301
410
|
# filter the list of items on the select option
|
@@ -309,9 +418,9 @@ module Aspera
|
|
309
418
|
end
|
310
419
|
end
|
311
420
|
|
312
|
-
#
|
313
|
-
# object_array
|
314
|
-
# fields
|
421
|
+
# displays a list of objects
|
422
|
+
# @param object_array [Array] array of hash
|
423
|
+
# @param fields [Array] list of column names
|
315
424
|
def display_table(object_array, fields)
|
316
425
|
Aspera.assert(!fields.nil?){'missing fields parameter'}
|
317
426
|
filter_columns_on_select(object_array)
|
@@ -325,29 +434,28 @@ module Aspera
|
|
325
434
|
display_message(:data, object_array.first[fields.first])
|
326
435
|
return
|
327
436
|
end
|
328
|
-
# Special case if only one row (it could be object_list or single_object)
|
329
|
-
if @options[:transpose_single] && object_array.length == 1
|
330
|
-
single = object_array.first
|
331
|
-
object_array = fields.map { |i| FIELD_VALUE_HEADINGS.zip([i, single[i]]).to_h }
|
332
|
-
fields = FIELD_VALUE_HEADINGS
|
333
|
-
end
|
334
437
|
Log.log.debug{Log.dump(:object_array, object_array)}
|
335
438
|
# convert data to string, and keep only display fields
|
336
439
|
final_table_rows = object_array.map { |r| fields.map { |c| r[c].to_s } }
|
440
|
+
# remove empty rows
|
441
|
+
final_table_rows.select!{|i| !(i.is_a?(Hash) && i.empty?)}
|
337
442
|
# here : fields : list of column names
|
338
443
|
case @options[:format]
|
339
444
|
when :table
|
340
|
-
|
341
|
-
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
445
|
+
if @options[:multi_single].eql?(:yes) ||
|
446
|
+
(@options[:multi_single].eql?(:single) && final_table_rows.length.eql?(1))
|
447
|
+
final_table_rows.each do |row|
|
448
|
+
Log.log.debug{Log.dump(:row, row)}
|
449
|
+
display_message(:data, Terminal::Table.new(
|
450
|
+
headings: SINGLE_OBJECT_COLUMN_NAMES,
|
451
|
+
rows: fields.zip(row),
|
452
|
+
style: @options[:table_style]&.symbolize_keys))
|
453
|
+
end
|
454
|
+
else
|
455
|
+
# display the table !
|
348
456
|
display_message(:data, Terminal::Table.new(
|
349
|
-
headings:
|
350
|
-
rows:
|
457
|
+
headings: fields,
|
458
|
+
rows: final_table_rows,
|
351
459
|
style: @options[:table_style]&.symbolize_keys))
|
352
460
|
end
|
353
461
|
when :csv
|
@@ -356,110 +464,6 @@ module Aspera
|
|
356
464
|
raise "not expected: #{@options[:format]}"
|
357
465
|
end
|
358
466
|
end
|
359
|
-
|
360
|
-
# @return text suitable to display an image from url
|
361
|
-
def status_image(blob)
|
362
|
-
begin
|
363
|
-
raise URI::InvalidURIError, 'not uri' if !(blob =~ /\A#{URI::DEFAULT_PARSER.make_regexp}\z/)
|
364
|
-
# it's a url
|
365
|
-
url = blob
|
366
|
-
unless Environment.instance.url_method.eql?(:text)
|
367
|
-
Environment.instance.open_uri(url)
|
368
|
-
return ''
|
369
|
-
end
|
370
|
-
# remote_image = Rest.new(base_url: url).read('')
|
371
|
-
# mime = remote_image[:http]['content-type']
|
372
|
-
# blob = remote_image[:http].body
|
373
|
-
# Log.log.warn("Image ? #{remote_image[:http]['content-type']}") unless mime.include?('image/')
|
374
|
-
blob = UriReader.read(url)
|
375
|
-
rescue URI::InvalidURIError
|
376
|
-
nil
|
377
|
-
end
|
378
|
-
# try base64
|
379
|
-
begin
|
380
|
-
blob = Base64.strict_decode64(blob)
|
381
|
-
rescue
|
382
|
-
nil
|
383
|
-
end
|
384
|
-
return Preview::Terminal.build(blob, **@options[:image].symbolize_keys)
|
385
|
-
end
|
386
|
-
|
387
|
-
# this method displays the results, especially the table format
|
388
|
-
# @param type [Symbol] type of data
|
389
|
-
# @param data [Object] data to display
|
390
|
-
# @param total [Integer] total number of items
|
391
|
-
# @param fields [Array<String>] list of fields to display
|
392
|
-
# @param name [String] name of the column to display
|
393
|
-
def display_results(type:, data: nil, total: nil, fields: nil, name: nil)
|
394
|
-
Log.log.debug{"display_results: #{type} class=#{data.class} data=#{data}"}
|
395
|
-
Aspera.assert_type(type, Symbol){'result must have type'}
|
396
|
-
Aspera.assert(!data.nil? || %i[empty nothing].include?(type)){'result must have data'}
|
397
|
-
display_item_count(data.length, total) unless total.nil?
|
398
|
-
SecretHider.deep_remove_secret(data) unless @options[:show_secrets] || @options[:display].eql?(:data)
|
399
|
-
case @options[:format]
|
400
|
-
when :text
|
401
|
-
display_message(:data, data.to_s)
|
402
|
-
when :nagios
|
403
|
-
Nagios.process(data)
|
404
|
-
when :ruby
|
405
|
-
display_message(:data, PP.pp(filter_list_on_fields(data), +''))
|
406
|
-
when :json
|
407
|
-
display_message(:data, JSON.generate(filter_list_on_fields(data)))
|
408
|
-
when :jsonpp
|
409
|
-
display_message(:data, JSON.pretty_generate(filter_list_on_fields(data)))
|
410
|
-
when :yaml
|
411
|
-
display_message(:data, YAML.dump(filter_list_on_fields(data)))
|
412
|
-
when :image
|
413
|
-
# assume it is an url
|
414
|
-
url = data
|
415
|
-
case type
|
416
|
-
when :single_object, :object_list
|
417
|
-
url = [url] if type.eql?(:single_object)
|
418
|
-
raise 'image display requires a single result' unless url.length == 1
|
419
|
-
fields = compute_fields(url, fields)
|
420
|
-
raise 'select a field to display' unless fields.length == 1
|
421
|
-
url = url.first
|
422
|
-
raise 'no such field' unless url.key?(fields.first)
|
423
|
-
url = url[fields.first]
|
424
|
-
end
|
425
|
-
raise "not url: #{url.class} #{url}" unless url.is_a?(String)
|
426
|
-
display_message(:data, status_image(url))
|
427
|
-
when :table, :csv, :multi
|
428
|
-
case type
|
429
|
-
when :config_over
|
430
|
-
display_table(Flattener.new(self).config_over(data), CONF_OVERVIEW_KEYS)
|
431
|
-
when :object_list, :single_object
|
432
|
-
obj_list = data
|
433
|
-
obj_list = [obj_list] if type.eql?(:single_object)
|
434
|
-
Aspera.assert_type(obj_list, Array)
|
435
|
-
Aspera.assert(obj_list.all?(Hash)){"expecting Array of Hash: #{obj_list.inspect}"}
|
436
|
-
# :object_list is an array of hash tables, where key=colum name
|
437
|
-
obj_list = obj_list.map{|obj|Flattener.new(self).flatten(obj)} if @options[:flat_hash]
|
438
|
-
display_table(obj_list, compute_fields(obj_list, fields))
|
439
|
-
when :value_list
|
440
|
-
# :value_list is a simple array of values, name of column provided in the :name
|
441
|
-
display_table(data.map { |i| { name => i } }, [name])
|
442
|
-
when :empty # no table
|
443
|
-
display_message(:info, special_format('empty'))
|
444
|
-
return
|
445
|
-
when :nothing # no result expected
|
446
|
-
Log.log.debug('no result expected')
|
447
|
-
when :status # no table
|
448
|
-
# :status displays a simple message
|
449
|
-
display_message(:info, data)
|
450
|
-
when :text # no table
|
451
|
-
# :status displays a simple message
|
452
|
-
display_message(:data, data)
|
453
|
-
when :other_struct # no table
|
454
|
-
# :other_struct is any other type of structure
|
455
|
-
display_message(:data, PP.pp(data, +''))
|
456
|
-
else
|
457
|
-
raise "unknown data type: #{type}"
|
458
|
-
end
|
459
|
-
else
|
460
|
-
raise "not expected: #{@options[:format]}"
|
461
|
-
end
|
462
|
-
end
|
463
467
|
end
|
464
468
|
end
|
465
469
|
end
|
data/lib/aspera/cli/hints.rb
CHANGED
@@ -4,6 +4,7 @@ require 'aspera/transfer/error'
|
|
4
4
|
require 'aspera/rest'
|
5
5
|
require 'aspera/log'
|
6
6
|
require 'aspera/assert'
|
7
|
+
require 'aspera/cli/info'
|
7
8
|
require 'net/ssh'
|
8
9
|
require 'openssl'
|
9
10
|
|
@@ -18,7 +19,7 @@ module Aspera
|
|
18
19
|
match: 'Remote host is not who we expected',
|
19
20
|
remediation: [
|
20
21
|
'For this specific error, refer to:',
|
21
|
-
"#{SRC_URL}#error-remote-host-is-not-who-we-expected",
|
22
|
+
"#{Info::SRC_URL}#error-remote-host-is-not-who-we-expected",
|
22
23
|
'Add this to arguments:',
|
23
24
|
%q{--ts=@json:'{"sshfp":null}'"}
|
24
25
|
]
|
data/lib/aspera/cli/info.rb
CHANGED
@@ -2,15 +2,17 @@
|
|
2
2
|
|
3
3
|
module Aspera
|
4
4
|
module Cli
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
5
|
+
module Info
|
6
|
+
# name of command line tool, also used as foldername where config is stored
|
7
|
+
CMD_NAME = 'ascli'
|
8
|
+
# name of the containing gem, same as in <gem name>.gemspec
|
9
|
+
GEM_NAME = 'aspera-cli'
|
10
|
+
DOC_URL = "https://www.rubydoc.info/gems/#{GEM_NAME}"
|
11
|
+
GEM_URL = "https://rubygems.org/gems/#{GEM_NAME}"
|
12
|
+
SRC_URL = 'https://github.com/IBM/aspera-cli'
|
13
|
+
# set this to warn in advance when minimum required ruby version will increase
|
14
|
+
# see also required_ruby_version in gemspec file
|
15
|
+
RUBY_FUTURE_MINIMUM_VERSION = '3.2'
|
16
|
+
end
|
15
17
|
end
|
16
18
|
end
|
data/lib/aspera/cli/main.rb
CHANGED
@@ -63,6 +63,18 @@ module Aspera
|
|
63
63
|
def result_image(blob, formatter:)
|
64
64
|
return Main.result_status(formatter.status_image(blob))
|
65
65
|
end
|
66
|
+
|
67
|
+
def result_single_object(data)
|
68
|
+
return {type: :single_object, data: data}
|
69
|
+
end
|
70
|
+
|
71
|
+
def result_object_list(data, fields: nil, total: nil)
|
72
|
+
return {type: :object_list, data: data, fields: fields, total: nil}
|
73
|
+
end
|
74
|
+
|
75
|
+
def result_value_list(data, name)
|
76
|
+
return {type: :value_list, data: data, name: name}
|
77
|
+
end
|
66
78
|
end
|
67
79
|
|
68
80
|
private
|
@@ -94,7 +106,7 @@ module Aspera
|
|
94
106
|
# create formatter, in case there is an exception, it is used to display.
|
95
107
|
@plug_init[:formatter] = Formatter.new
|
96
108
|
# create command line manager with arguments
|
97
|
-
@plug_init[:options] = Manager.new(
|
109
|
+
@plug_init[:options] = Manager.new(Info::CMD_NAME, @argv)
|
98
110
|
# formatter adds options
|
99
111
|
@plug_init[:formatter].declare_options(options)
|
100
112
|
ExtendedValue.instance.default_decoder = options.get_option(:struct_parser)
|
@@ -102,16 +114,18 @@ module Aspera
|
|
102
114
|
current_prog_name = File.basename($PROGRAM_NAME)
|
103
115
|
formatter.display_message(
|
104
116
|
:error,
|
105
|
-
"#{Formatter::WARNING_FLASH} Please use '#{
|
117
|
+
"#{Formatter::WARNING_FLASH} Please use '#{Info::CMD_NAME}' instead of '#{current_prog_name}'") unless current_prog_name.eql?(Info::CMD_NAME)
|
106
118
|
# declare and parse global options
|
107
119
|
declare_global_options
|
108
120
|
# the Config plugin adds the @preset parser, so declare before TransferAgent which may use it
|
109
|
-
@plug_init[:config] = Plugins::Config.new(**@plug_init,
|
121
|
+
@plug_init[:config] = Plugins::Config.new(**@plug_init, man_header: false)
|
110
122
|
@plug_init[:persistency] = @plug_init[:config].persistency
|
111
123
|
# data persistency
|
112
124
|
Aspera.assert(@plug_init[:persistency]){'missing persistency object'}
|
113
125
|
# the TransferAgent plugin may use the @preset parser
|
114
126
|
@plug_init[:config].transfer = @plug_init[:transfer] = TransferAgent.new(options, config)
|
127
|
+
# add commands for config plugin after all options have been added
|
128
|
+
@plug_init[:config].add_manual_header(false)
|
115
129
|
nil_keys = @plug_init.select{|_, value|value.nil?}.keys
|
116
130
|
Aspera.assert(nil_keys.empty?){"nil : #{nil_keys}"}
|
117
131
|
Log.log.debug('plugin env created'.red)
|
@@ -123,23 +137,23 @@ module Aspera
|
|
123
137
|
t = ' ' * 8
|
124
138
|
return <<~END_OF_BANNER
|
125
139
|
NAME
|
126
|
-
#{t}#{
|
140
|
+
#{t}#{Info::CMD_NAME} -- a command line tool for Aspera Applications (v#{Cli::VERSION})
|
127
141
|
|
128
142
|
SYNOPSIS
|
129
|
-
#{t}#{
|
143
|
+
#{t}#{Info::CMD_NAME} COMMANDS [OPTIONS] [ARGS]
|
130
144
|
|
131
145
|
DESCRIPTION
|
132
146
|
#{t}Use Aspera application to perform operations on command line.
|
133
|
-
#{t}Documentation and examples: #{GEM_URL}
|
134
|
-
#{t}execute: #{
|
135
|
-
#{t}or visit: #{DOC_URL}
|
136
|
-
#{t}source repo: #{SRC_URL}
|
147
|
+
#{t}Documentation and examples: #{Info::GEM_URL}
|
148
|
+
#{t}execute: #{Info::CMD_NAME} conf doc
|
149
|
+
#{t}or visit: #{Info::DOC_URL}
|
150
|
+
#{t}source repo: #{Info::SRC_URL}
|
137
151
|
|
138
152
|
ENVIRONMENT VARIABLES
|
139
153
|
#{t}Any option can be set as an environment variable, refer to the manual
|
140
154
|
|
141
155
|
COMMANDS
|
142
|
-
#{t}To list first level commands, execute: #{
|
156
|
+
#{t}To list first level commands, execute: #{Info::CMD_NAME}
|
143
157
|
#{t}Note that commands can be written shortened (provided it is unique).
|
144
158
|
|
145
159
|
OPTIONS
|
@@ -199,16 +213,17 @@ module Aspera
|
|
199
213
|
|
200
214
|
def exit_with_usage(include_all_plugins)
|
201
215
|
Log.log.debug{"exit_with_usage(#{include_all_plugins})".bg_red}
|
202
|
-
# display main plugin options
|
216
|
+
# display main plugin options (+config)
|
203
217
|
formatter.display_message(:error, options.parser)
|
204
218
|
if include_all_plugins
|
205
219
|
# list plugins that have a "require" field, i.e. all but main plugin
|
206
220
|
PluginFactory.instance.plugin_list.each do |plugin_name_sym|
|
221
|
+
# config was already included in the global options
|
207
222
|
next if plugin_name_sym.eql?(COMMAND_CONFIG)
|
208
223
|
# override main option parser with a brand new, to avoid having global options
|
209
224
|
plugin_env = @plug_init.clone
|
210
225
|
plugin_env[:only_manual] = true # force declaration of all options
|
211
|
-
plugin_env[:options] = Manager.new(
|
226
|
+
plugin_env[:options] = Manager.new(Info::CMD_NAME)
|
212
227
|
plugin_env[:options].parser.banner = '' # remove default banner
|
213
228
|
get_plugin_instance_with_options(plugin_name_sym, plugin_env)
|
214
229
|
# display generated help for plugin options
|
@@ -223,7 +238,7 @@ module Aspera
|
|
223
238
|
# early debug for parser
|
224
239
|
# Note: does not accept shortcuts
|
225
240
|
def early_debug_setup
|
226
|
-
Log.instance.program_name =
|
241
|
+
Log.instance.program_name = Info::CMD_NAME
|
227
242
|
@argv.each do |arg|
|
228
243
|
case arg
|
229
244
|
when '--' then break
|
data/lib/aspera/cli/manager.rb
CHANGED
@@ -105,13 +105,14 @@ module Aspera
|
|
105
105
|
# @param descr [String] description for help
|
106
106
|
# @param to_check [Object] value to check
|
107
107
|
# @param type_list [NilClass, Class, Array[Class]] accepted value type(s)
|
108
|
+
# @param check_array [bool] set to true if it is a list of values to check
|
108
109
|
def validate_type(what, descr, to_check, type_list, check_array: false)
|
109
110
|
return nil if type_list.nil?
|
110
111
|
Aspera.assert(type_list.is_a?(Array) && type_list.all?(Class)){'types must be a Class Array'}
|
111
112
|
value_list = check_array ? to_check : [to_check]
|
112
113
|
value_list.each do |value|
|
113
114
|
raise Cli::BadArgument,
|
114
|
-
"#{what.to_s.capitalize} #{descr} is a #{value.class} but must be #{type_list.length > 1 ? 'one of ' : ''}#{type_list.map(&:name).join(',')}" unless
|
115
|
+
"#{what.to_s.capitalize} #{descr} is a #{value.class} but must be #{type_list.length > 1 ? 'one of ' : ''}#{type_list.map(&:name).join(',')}" unless
|
115
116
|
type_list.any?{|t|value.is_a?(t)}
|
116
117
|
end
|
117
118
|
end
|
@@ -386,6 +387,11 @@ module Aspera
|
|
386
387
|
end
|
387
388
|
end
|
388
389
|
|
390
|
+
# allows a plugin to add an argument as next argument to process
|
391
|
+
def unshift_next_argument(argument)
|
392
|
+
@unprocessed_cmd_line_arguments.unshift(argument)
|
393
|
+
end
|
394
|
+
|
389
395
|
# check if there were unprocessed values to generate error
|
390
396
|
def command_or_arg_empty?
|
391
397
|
return @unprocessed_cmd_line_arguments.empty?
|
@@ -495,7 +501,6 @@ module Aspera
|
|
495
501
|
self.class.multi_choice_assert(false, message, accept_list)
|
496
502
|
end
|
497
503
|
end
|
498
|
-
result = nil
|
499
504
|
sensitive = option && @declared_options[descr.to_sym].is_a?(Hash) && @declared_options[descr.to_sym][:sensitive]
|
500
505
|
default_prompt = "#{what}: #{descr}"
|
501
506
|
# ask interactively
|