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
@@ -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 multi csv image].freeze
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
- FIELD_VALUE_HEADINGS = %i[key value].freeze
92
+ # column names for single object display in table
93
+ SINGLE_OBJECT_COLUMN_NAMES = %i[field value].freeze
105
94
 
106
- private_constant :DISPLAY_FORMATS, :DISPLAY_LEVELS, :CSV_RECORD_SEPARATOR, :CSV_FIELD_SEPARATOR, :FIELD_VALUE_HEADINGS
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
- # options are: format, output, display, fields, select, table_style, flat_hash, transpose_single
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.select{|k, _|selected_fields.include?(k)}}
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
- # this method displays a table
313
- # object_array: array of hash
314
- # fields: list of column names
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
- # display the table !
341
- display_message(:data, Terminal::Table.new(
342
- headings: fields,
343
- rows: final_table_rows,
344
- style: @options[:table_style]&.symbolize_keys))
345
- when :multi
346
- final_table_rows.each do |row|
347
- Log.log.debug{Log.dump(:row, row)}
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: FIELD_VALUE_HEADINGS,
350
- rows: fields.zip(row),
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
@@ -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
  ]
@@ -2,15 +2,17 @@
2
2
 
3
3
  module Aspera
4
4
  module Cli
5
- # name of command line tool, also used as foldername where config is stored
6
- PROGRAM_NAME = 'ascli'
7
- # name of the containing gem, same as in <gem name>.gemspec
8
- GEM_NAME = 'aspera-cli'
9
- DOC_URL = "https://www.rubydoc.info/gems/#{GEM_NAME}"
10
- GEM_URL = "https://rubygems.org/gems/#{GEM_NAME}"
11
- SRC_URL = 'https://github.com/IBM/aspera-cli'
12
- # set this to warn in advance when minimum required ruby version will increase
13
- # see also required_ruby_version in gemspec file
14
- RUBY_FUTURE_MINIMUM_VERSION = '3.0'
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
@@ -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(PROGRAM_NAME, @argv)
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 '#{PROGRAM_NAME}' instead of '#{current_prog_name}'") unless current_prog_name.eql?(PROGRAM_NAME)
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, gem: GEM_NAME, name: PROGRAM_NAME, help: DOC_URL, version: Cli::VERSION)
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}#{PROGRAM_NAME} -- a command line tool for Aspera Applications (v#{Cli::VERSION})
140
+ #{t}#{Info::CMD_NAME} -- a command line tool for Aspera Applications (v#{Cli::VERSION})
127
141
 
128
142
  SYNOPSIS
129
- #{t}#{PROGRAM_NAME} COMMANDS [OPTIONS] [ARGS]
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: #{PROGRAM_NAME} conf doc
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: #{PROGRAM_NAME}
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(PROGRAM_NAME)
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 = 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
@@ -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