X-Git-Url: https://git.proxmox.com/?p=pve-common.git;a=blobdiff_plain;f=src%2FPVE%2FRESTHandler.pm;h=c303c389e4da1273122786a2e9274ccd00c4a0d6;hp=0a64158679cce96926802445403c12166769bf7d;hb=e97f807c388c10250f442b1f16c5315df2ffc2af;hpb=8d3eb3ce17c60f128359e249e3ea9d114c792823 diff --git a/src/PVE/RESTHandler.pm b/src/PVE/RESTHandler.pm index 0a64158..c303c38 100644 --- a/src/PVE/RESTHandler.pm +++ b/src/PVE/RESTHandler.pm @@ -58,7 +58,7 @@ sub api_clone_schema { } sub api_dump_full { - my ($tree, $index, $class, $prefix) = @_; + my ($tree, $index, $class, $prefix, $raw_dump) = @_; $prefix = '' if !$prefix; @@ -70,7 +70,7 @@ sub api_dump_full { $path =~ s/\/+$//; if ($info->{subclass}) { - api_dump_full($tree, $index, $info->{subclass}, $path); + api_dump_full($tree, $index, $info->{subclass}, $path, $raw_dump); } else { next if !$path; @@ -110,12 +110,15 @@ sub api_dump_full { $k eq "path"; my $d = $info->{$k}; - - if ($k eq 'parameters') { - $data->{$k} = api_clone_schema($d); - } else { - $data->{$k} = ref($d) ? clone($d) : $d; + if ($raw_dump) { + $data->{$k} = $d; + } else { + if ($k eq 'parameters') { + $data->{$k} = api_clone_schema($d); + } else { + $data->{$k} = ref($d) ? clone($d) : $d; + } } } $res->{info}->{$info->{method}} = $data; @@ -139,13 +142,46 @@ sub api_dump_cleanup_tree { } +# api_dump_remove_refs: prepare API tree for use with to_json($tree) +sub api_dump_remove_refs { + my ($tree) = @_; + + my $class = ref($tree); + return $tree if !$class; + + if ($class eq 'ARRAY') { + my $res = []; + foreach my $el (@$tree) { + push @$res, api_dump_remove_refs($el); + } + return $res; + } elsif ($class eq 'HASH') { + my $res = {}; + foreach my $k (keys %$tree) { + if (my $itemclass = ref($tree->{$k})) { + if ($itemclass eq 'CODE') { + next if $k eq 'completion'; + } + $res->{$k} = api_dump_remove_refs($tree->{$k}); + } else { + $res->{$k} = $tree->{$k}; + } + } + return $res; + } elsif ($class eq 'Regexp') { + return "$tree"; # return string representation + } else { + die "unknown class '$class'\n"; + } +} + sub api_dump { - my ($class, $prefix) = @_; + my ($class, $prefix, $raw_dump) = @_; my $tree = []; my $index = {}; - api_dump_full($tree, $index, $class); + api_dump_full($tree, $index, $class, $prefix, $raw_dump); api_dump_cleanup_tree($tree); return $tree; }; @@ -396,7 +432,7 @@ sub handle { # $style: 'config', 'config-sub', 'arg' or 'fixed' # $mapdef: parameter mapping ({ desc => XXX, func => sub {...} }) my $get_property_description = sub { - my ($name, $style, $phash, $format, $hidepw, $mapdef) = @_; + my ($name, $style, $phash, $format, $mapdef) = @_; my $res = ''; @@ -413,10 +449,6 @@ my $get_property_description = sub { my $type_text = PVE::JSONSchema::schema_get_type_text($phash, $style); - if ($hidepw && $name eq 'password') { - $type_text = ''; - } - if ($mapdef && $phash->{type} eq 'string') { $type_text = $mapdef->{desc}; } @@ -514,15 +546,19 @@ my $compute_param_mapping_hash = sub { return $res if !defined($mapping_array); foreach my $item (@$mapping_array) { - my ($name, $func, $desc); + my ($name, $func, $desc, $interactive); if (ref($item) eq 'ARRAY') { - ($name, $func, $desc) = @$item; + ($name, $func, $desc, $interactive) = @$item; + } elsif (ref($item) eq 'HASH') { + # just use the hash + $res->{$item->{name}} = $item; + next; } else { $name = $item; $func = sub { return PVE::Tools::file_get_contents($_[0]) }; } $desc //= ''; - $res->{$name} = { desc => $desc, func => $func }; + $res->{$name} = { desc => $desc, func => $func, interactive => $interactive }; } return $res; @@ -540,10 +576,9 @@ 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') -# $hidepw ... hide password option (use this if you provide a read passwork callback) -# $param_mapping_func ... mapping for string parameters to file path parameters +# $param_cb ... mapping for string parameters to file path parameters sub usage_str { - my ($self, $name, $prefix, $arg_param, $fixed_param, $format, $hidepw, $param_mapping_func) = @_; + my ($self, $name, $prefix, $arg_param, $fixed_param, $format, $param_cb) = @_; $format = 'long' if !$format; @@ -576,7 +611,7 @@ sub usage_str { foreach my $k (@$arg_param) { next if defined($fixed_param->{$k}); # just to be sure next if !$prop->{$k}; # just to be sure - $argdescr .= &$get_property_description($k, 'fixed', $prop->{$k}, $format, 0); + $argdescr .= $get_property_description->($k, 'fixed', $prop->{$k}, $format); } my $idx_param = {}; # -vlan\d+ -scsi\d+ @@ -588,7 +623,13 @@ sub usage_str { my $type_text = $prop->{$k}->{type} || 'string'; - next if $hidepw && ($k eq 'password') && !$prop->{$k}->{optional}; + my $param_map = {}; + + if (defined($param_cb)) { + my $mapping = $param_cb->($name); + $param_map = $compute_param_mapping_hash->($mapping); + next if $k eq 'password' && $param_map->{$k} && !$prop->{$k}->{optional}; + } my $base = $k; if ($k =~ m/^([a-z]+)(\d+)$/) { @@ -600,11 +641,8 @@ sub usage_str { } } - my $param_mapping_hash = $compute_param_mapping_hash->(&$param_mapping_func($name)) - if $param_mapping_func; - $opts .= &$get_property_description($base, 'arg', $prop->{$k}, $format, - $hidepw, $param_mapping_hash->{$k}); + $opts .= $get_property_description->($base, 'arg', $prop->{$k}, $format, $param_map->{$k}); if (!$prop->{$k}->{optional}) { $args .= " " if $args; @@ -667,7 +705,7 @@ sub dump_properties { } } - $raw .= &$get_property_description($base, $style, $phash, $format, 0); + $raw .= $get_property_description->($base, $style, $phash, $format); next if $style ne 'config'; @@ -688,9 +726,10 @@ sub dump_properties { } my $replace_file_names_with_contents = sub { - my ($param, $param_mapping_hash) = @_; + my ($param, $param_map) = @_; - while (my ($k, $d) = each %$param_mapping_hash) { + while (my ($k, $d) = each %$param_map) { + next if $d->{interactive}; # handled by the JSONSchema's get_options code $param->{$k} = $d->{func}->($param->{$k}) if defined($param->{$k}); } @@ -699,17 +738,18 @@ my $replace_file_names_with_contents = sub { }; sub cli_handler { - my ($self, $prefix, $name, $args, $arg_param, $fixed_param, $read_password_func, $param_mapping_func) = @_; + my ($self, $prefix, $name, $args, $arg_param, $fixed_param, $param_cb) = @_; my $info = $self->map_method_by_name($name); my $res; eval { - my $param = PVE::JSONSchema::get_options($info->{parameters}, $args, $arg_param, $fixed_param, $read_password_func); + 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); - if (defined($param_mapping_func)) { - my $param_mapping_hash = $compute_param_mapping_hash->(&$param_mapping_func($name)); - &$replace_file_names_with_contents($param, $param_mapping_hash); + if (defined($param_map)) { + $replace_file_names_with_contents->($param, $param_map); } $res = $self->handle($info, $param); @@ -719,7 +759,7 @@ 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', $read_password_func, $param_mapping_func); + $err->{usage} = $self->usage_str($name, $prefix, $arg_param, $fixed_param, 'short', $param_cb); die $err; }