X-Git-Url: https://git.proxmox.com/?p=pve-common.git;a=blobdiff_plain;f=src%2FPVE%2FRESTHandler.pm;h=49d43b94593c2f802b336271033d4df962024dea;hp=148b4d8654875e7d3d406b472ef1f44f778ff96f;hb=3b44baed32fa8630dd13191f01ba22547b3b07f7;hpb=4cbcd1386f225f42896fe790337d152718498693 diff --git a/src/PVE/RESTHandler.pm b/src/PVE/RESTHandler.pm index 148b4d8..49d43b9 100644 --- a/src/PVE/RESTHandler.pm +++ b/src/PVE/RESTHandler.pm @@ -17,8 +17,35 @@ my $method_path_lookup = {}; our $AUTOLOAD; # it's a package global +our $standard_output_options = { + 'output-format' => PVE::JSONSchema::get_standard_option('pve-output-format'), + noheader => { + description => "Do not show column headers (for 'text' format).", + type => 'boolean', + optional => 1, + default => 0, + }, + noborder => { + description => "Do not draw borders (for 'text' format).", + type => 'boolean', + optional => 1, + default => 0, + }, + quiet => { + description => "Suppress printing results.", + type => 'boolean', + optional => 1, + }, + 'human-readable' => { + description => "Call output rendering functions to produce human readable text.", + type => 'boolean', + optional => 1, + default => 1, + } +}; + sub api_clone_schema { - my ($schema) = @_; + my ($schema, $no_typetext) = @_; my $res = {}; my $ref = ref($schema); @@ -44,7 +71,7 @@ sub api_clone_schema { my $tmp = ref($pd) ? clone($pd) : $pd; # NOTE: add typetext property for more complex types, to # make the web api viewer code simpler - if (!(defined($tmp->{enum}) || defined($tmp->{pattern}))) { + if (!$no_typetext && !(defined($tmp->{enum}) || defined($tmp->{pattern}))) { my $typetext = PVE::JSONSchema::schema_get_type_text($tmp); if ($tmp->{type} && ($tmp->{type} ne $typetext)) { $tmp->{typetext} = $typetext; @@ -116,6 +143,8 @@ sub api_dump_full { } else { if ($k eq 'parameters') { $data->{$k} = api_clone_schema($d); + } elsif ($k eq 'returns') { + $data->{$k} = api_clone_schema($d, 1); } else { $data->{$k} = ref($d) ? clone($d) : $d; } @@ -576,15 +605,36 @@ my $compute_param_mapping_hash = sub { # 'short' ... command line only (text, one line) # 'full' ... text, include description # 'asciidoc' ... generate asciidoc for man pages (like 'full') -# $param_cb ... mapping for string parameters to file path parameters +# $param_cb ... mapping for string parameters to file path parameters +# $formatter_properties ... additional property definitions (passed to output formatter) sub getopt_usage { - my ($info, $prefix, $arg_param, $fixed_param, $format, $param_cb) = @_; + my ($info, $prefix, $arg_param, $fixed_param, $format, $param_cb, $formatter_properties) = @_; $format = 'long' if !$format; my $schema = $info->{parameters}; my $name = $info->{name}; - my $prop = $schema->{properties}; + my $prop = { %{$schema->{properties}} }; # copy + + my $has_output_format_option = $formatter_properties->{'output-format'} ? 1 : 0; + + if ($formatter_properties) { + foreach my $key (keys %$formatter_properties) { + if (!$standard_output_options->{$key}) { + $prop->{$key} = $formatter_properties->{$key}; + } + } + } + + # also remove $standard_output_options from $prop (pvesh, pveclient) + if ($prop->{'output-format'}) { + $has_output_format_option = 1; + foreach my $key (keys %$prop) { + if ($standard_output_options->{$key}) { + delete $prop->{$key}; + } + } + } my $out = ''; @@ -653,11 +703,15 @@ sub getopt_usage { if ($format eq 'asciidoc') { $out .= "*${prefix}*"; $out .= " `$args`" if $args; - $out .= $opts ? " `[OPTIONS]`\n" : "\n"; + $out .= " `[OPTIONS]`" if $opts; + $out .= " `[FORMAT_OPTIONS]`" if $has_output_format_option; + $out .= "\n"; } else { $out .= "USAGE: " if $format ne 'short'; $out .= "$prefix $args"; - $out .= $opts ? " [OPTIONS]\n" : "\n"; + $out .= " [OPTIONS]" if $opts; + $out .= " [FORMAT_OPTIONS]" if $has_output_format_option; + $out .= "\n"; } return $out if $format eq 'short'; @@ -680,11 +734,11 @@ sub getopt_usage { } sub usage_str { - my ($self, $name, $prefix, $arg_param, $fixed_param, $format, $param_cb) = @_; + my ($self, $name, $prefix, $arg_param, $fixed_param, $format, $param_cb, $formatter_properties) = @_; my $info = $self->map_method_by_name($name); - return getopt_usage($info, $prefix, $arg_param, $fixed_param, $format, $param_cb); + return getopt_usage($info, $prefix, $arg_param, $fixed_param, $format, $param_cb, $formatter_properties); } # generate docs from JSON schema properties @@ -745,33 +799,6 @@ my $replace_file_names_with_contents = sub { return $param; }; -our $standard_output_options = { - 'output-format' => PVE::JSONSchema::get_standard_option('pve-output-format'), - noheader => { - description => "Do not show column headers (for 'text' format).", - type => 'boolean', - optional => 1, - default => 1, - }, - noborder => { - description => "Do not draw borders (for 'text' format).", - type => 'boolean', - optional => 1, - default => 1, - }, - quiet => { - description => "Suppress printing results.", - type => 'boolean', - optional => 1, - }, - 'human-readable' => { - description => "Call output rendering functions to produce human readable text.", - type => 'boolean', - optional => 1, - default => 1, - } -}; - sub add_standard_output_properties { my ($propdef, $list) = @_; @@ -802,15 +829,24 @@ sub extract_standard_output_properties { } sub cli_handler { - my ($self, $prefix, $name, $args, $arg_param, $fixed_param, $param_cb) = @_; + my ($self, $prefix, $name, $args, $arg_param, $fixed_param, $param_cb, $formatter_properties) = @_; my $info = $self->map_method_by_name($name); - my $res; + my $fmt_param = {}; + eval { my $param_map = {}; $param_map = $compute_param_mapping_hash->($param_cb->($name)) if $param_cb; - my $param = PVE::JSONSchema::get_options($info->{parameters}, $args, $arg_param, $fixed_param, $param_map); + my $schema = { %{$info->{parameters}} }; # copy + $schema->{properties} = { %{$schema->{properties}}, %$formatter_properties } if $formatter_properties; + my $param = PVE::JSONSchema::get_options($schema, $args, $arg_param, $fixed_param, $param_map); + + if ($formatter_properties) { + foreach my $opt (keys %$formatter_properties) { + $fmt_param->{$opt} = delete $param->{$opt} if defined($param->{$opt}); + } + } if (defined($param_map)) { $replace_file_names_with_contents->($param, $param_map); @@ -823,12 +859,12 @@ sub cli_handler { die $err if !$ec || $ec ne "PVE::Exception" || !$err->is_param_exc(); - $err->{usage} = $self->usage_str($name, $prefix, $arg_param, $fixed_param, 'short', $param_cb); + $err->{usage} = $self->usage_str($name, $prefix, $arg_param, $fixed_param, 'short', $param_cb, $formatter_properties); die $err; } - return $res; + return wantarray ? ($res, $fmt_param) : $res; } # utility methods