mirror of
git://git.proxmox.com/git/pve-common.git
synced 2025-01-05 17:17:36 +03:00
section config: implement array support
enables section configs in the style of: ---- type: id property value property value2 property value3 ---- can be combined with property strings the provided create and update schema just pass through the array type to the api, so the api call must always contain the complete array also adds a test case for such array fields Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
This commit is contained in:
parent
9a3af02069
commit
69d9edcc75
@ -254,7 +254,15 @@ sub check_value {
|
|||||||
|
|
||||||
if (!$skipSchemaCheck) {
|
if (!$skipSchemaCheck) {
|
||||||
my $errors = {};
|
my $errors = {};
|
||||||
PVE::JSONSchema::check_prop($value, $schema, '', $errors);
|
|
||||||
|
my $checkschema = $schema;
|
||||||
|
|
||||||
|
if ($ct eq 'array') {
|
||||||
|
die "no item schema for array" if !defined($schema->{items});
|
||||||
|
$checkschema = $schema->{items};
|
||||||
|
}
|
||||||
|
|
||||||
|
PVE::JSONSchema::check_prop($value, $checkschema, '', $errors);
|
||||||
if (scalar(keys %$errors)) {
|
if (scalar(keys %$errors)) {
|
||||||
die "$errors->{$key}\n" if $errors->{$key};
|
die "$errors->{$key}\n" if $errors->{$key};
|
||||||
die "$errors->{_root}\n" if $errors->{_root};
|
die "$errors->{_root}\n" if $errors->{_root};
|
||||||
@ -311,6 +319,15 @@ sub parse_config {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
my $is_array = sub {
|
||||||
|
my ($type, $key) = @_;
|
||||||
|
|
||||||
|
my $schema = $pdata->{propertyList}->{$key};
|
||||||
|
die "unknown property type\n" if !$schema;
|
||||||
|
|
||||||
|
return $schema->{type} eq 'array';
|
||||||
|
};
|
||||||
|
|
||||||
my $errors = [];
|
my $errors = [];
|
||||||
while (@lines) {
|
while (@lines) {
|
||||||
my $line = $nextline->();
|
my $line = $nextline->();
|
||||||
@ -352,11 +369,19 @@ sub parse_config {
|
|||||||
my ($k, $v) = ($1, $3);
|
my ($k, $v) = ($1, $3);
|
||||||
|
|
||||||
eval {
|
eval {
|
||||||
|
if ($is_array->($type, $k)) {
|
||||||
|
if (!$unknown) {
|
||||||
|
$v = $plugin->check_value($type, $k, $v, $sectionId);
|
||||||
|
}
|
||||||
|
$config->{$k} = [] if !defined($config->{$k});
|
||||||
|
push $config->{$k}->@*, $v;
|
||||||
|
} else {
|
||||||
die "duplicate attribute\n" if defined($config->{$k});
|
die "duplicate attribute\n" if defined($config->{$k});
|
||||||
if (!$unknown) {
|
if (!$unknown) {
|
||||||
$v = $plugin->check_value($type, $k, $v, $sectionId);
|
$v = $plugin->check_value($type, $k, $v, $sectionId);
|
||||||
}
|
}
|
||||||
$config->{$k} = $v;
|
$config->{$k} = $v;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
if (my $err = $@) {
|
if (my $err = $@) {
|
||||||
warn "$errprefix (section '$sectionId') - unable to parse value of '$k': $err";
|
warn "$errprefix (section '$sectionId') - unable to parse value of '$k': $err";
|
||||||
@ -448,6 +473,13 @@ my $format_config_line = sub {
|
|||||||
if ($ct eq 'boolean') {
|
if ($ct eq 'boolean') {
|
||||||
return "\t$key " . ($value ? 1 : 0) . "\n"
|
return "\t$key " . ($value ? 1 : 0) . "\n"
|
||||||
if defined($value);
|
if defined($value);
|
||||||
|
} elsif ($ct eq 'array') {
|
||||||
|
die "property '$key' is not an array" if ref($value) ne 'ARRAY';
|
||||||
|
my $result = '';
|
||||||
|
for my $line ($value->@*) {
|
||||||
|
$result .= "\t$key $line\n" if $value ne '';
|
||||||
|
}
|
||||||
|
return $result;
|
||||||
} else {
|
} else {
|
||||||
return "\t$key $value\n" if "$value" ne '';
|
return "\t$key $value\n" if "$value" ne '';
|
||||||
}
|
}
|
||||||
|
@ -105,6 +105,25 @@ sub properties {
|
|||||||
minimum => 3,
|
minimum => 3,
|
||||||
maximum => 9,
|
maximum => 9,
|
||||||
},
|
},
|
||||||
|
arrayfield => {
|
||||||
|
description => "Array Field with property string",
|
||||||
|
type => 'array',
|
||||||
|
items => {
|
||||||
|
type => 'string',
|
||||||
|
description => 'a property string',
|
||||||
|
format => {
|
||||||
|
subfield1 => {
|
||||||
|
type => 'string',
|
||||||
|
description => 'first subfield'
|
||||||
|
},
|
||||||
|
subfield2 => {
|
||||||
|
type => 'integer',
|
||||||
|
minimum => 0,
|
||||||
|
optional => 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -113,6 +132,7 @@ sub options {
|
|||||||
common => { optional => 1 },
|
common => { optional => 1 },
|
||||||
field2 => {},
|
field2 => {},
|
||||||
another => {},
|
another => {},
|
||||||
|
arrayfield => { optional => 1 },
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -190,6 +210,10 @@ my $with_unknown_data = {
|
|||||||
type => 'two',
|
type => 'two',
|
||||||
field2 => 5,
|
field2 => 5,
|
||||||
another => 'even more text',
|
another => 'even more text',
|
||||||
|
arrayfield => [
|
||||||
|
'subfield1=test,subfield2=2',
|
||||||
|
'subfield1=test2',
|
||||||
|
],
|
||||||
},
|
},
|
||||||
invalid => {
|
invalid => {
|
||||||
type => 'bad',
|
type => 'bad',
|
||||||
@ -214,6 +238,8 @@ bad: invalid
|
|||||||
two: t3
|
two: t3
|
||||||
field2 5
|
field2 5
|
||||||
another even more text
|
another even more text
|
||||||
|
arrayfield subfield1=test,subfield2=2
|
||||||
|
arrayfield subfield1=test2
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
Conf->expect_fail('unknown-forbidden', $with_unknown_data, $with_unknown_text);
|
Conf->expect_fail('unknown-forbidden', $with_unknown_data, $with_unknown_text);
|
||||||
|
Loading…
Reference in New Issue
Block a user