mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-08 22:15:21 +00:00
virsh-completer: Separate comma list construction into a function
There are more arguments than 'shutdown --mode' that accept a list of strings separated by commas. 'nodedev-list --cap' is one of them. To avoid duplicating code, let's separate interesting bits of virshDomainShutdownModeCompleter() into a function that can then be reused. Signed-off-by: Michal Privoznik <mprivozn@redhat.com> Reviewed-by: Ján Tomko <jtomko@redhat.com> Reviewed-by: Erik Skultety <eskultet@redhat.com>
This commit is contained in:
parent
7283439837
commit
9ef216ce2d
@ -69,6 +69,79 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* virshCommaStringListComplete:
|
||||||
|
* @input: user input so far
|
||||||
|
* @options: ALL options available for argument
|
||||||
|
*
|
||||||
|
* Some arguments to our commands accept the following form:
|
||||||
|
*
|
||||||
|
* virsh command --arg str1,str2,str3
|
||||||
|
*
|
||||||
|
* This does not play nicely with our completer funtions, because
|
||||||
|
* they have to return strings prepended with user's input. For
|
||||||
|
* instance:
|
||||||
|
*
|
||||||
|
* str1,str2,str3,strA
|
||||||
|
* str1,str2,str3,strB
|
||||||
|
* str1,str2,str3,strC
|
||||||
|
*
|
||||||
|
* This helper function takes care of that. In this specific case
|
||||||
|
* it would be called as follows:
|
||||||
|
*
|
||||||
|
* virshCommaStringListComplete("str1,str2,str3",
|
||||||
|
* {"strA", "strB", "strC", NULL});
|
||||||
|
*
|
||||||
|
* Returns: string list of completions on success,
|
||||||
|
* NULL otherwise.
|
||||||
|
*/
|
||||||
|
static char **
|
||||||
|
virshCommaStringListComplete(const char *input,
|
||||||
|
const char **options)
|
||||||
|
{
|
||||||
|
const size_t optionsLen = virStringListLength(options);
|
||||||
|
VIR_AUTOFREE(char *) inputCopy = NULL;
|
||||||
|
VIR_AUTOSTRINGLIST inputList = NULL;
|
||||||
|
VIR_AUTOSTRINGLIST ret = NULL;
|
||||||
|
size_t nret = 0;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
if (STREQ_NULLABLE(input, " "))
|
||||||
|
input = NULL;
|
||||||
|
|
||||||
|
if (input) {
|
||||||
|
char *comma = NULL;
|
||||||
|
|
||||||
|
if (VIR_STRDUP(inputCopy, input) < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if ((comma = strrchr(inputCopy, ',')))
|
||||||
|
*comma = '\0';
|
||||||
|
else
|
||||||
|
VIR_FREE(inputCopy);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inputCopy && !(inputList = virStringSplit(inputCopy, ",", 0)))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (VIR_ALLOC_N(ret, optionsLen + 1) < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
for (i = 0; i < optionsLen; i++) {
|
||||||
|
if (virStringListHasString((const char **)inputList, options[i]))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if ((inputCopy && virAsprintf(&ret[nret], "%s,%s", inputCopy, options[i]) < 0) ||
|
||||||
|
(!inputCopy && VIR_STRDUP(ret[nret], options[i]) < 0))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
nret++;
|
||||||
|
}
|
||||||
|
|
||||||
|
VIR_RETURN_PTR(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
char **
|
char **
|
||||||
virshDomainNameCompleter(vshControl *ctl,
|
virshDomainNameCompleter(vshControl *ctl,
|
||||||
const vshCmd *cmd ATTRIBUTE_UNUSED,
|
const vshCmd *cmd ATTRIBUTE_UNUSED,
|
||||||
@ -993,52 +1066,13 @@ virshDomainShutdownModeCompleter(vshControl *ctl,
|
|||||||
const vshCmd *cmd,
|
const vshCmd *cmd,
|
||||||
unsigned int flags)
|
unsigned int flags)
|
||||||
{
|
{
|
||||||
const char *modes[] = {"acpi", "agent", "initctl", "signal", "paravirt"};
|
const char *modes[] = {"acpi", "agent", "initctl", "signal", "paravirt", NULL};
|
||||||
size_t i;
|
const char *mode = NULL;
|
||||||
char **ret = NULL;
|
|
||||||
size_t ntmp = 0;
|
|
||||||
VIR_AUTOSTRINGLIST tmp = NULL;
|
|
||||||
const char *modeConst = NULL;
|
|
||||||
VIR_AUTOFREE(char *) mode = NULL;
|
|
||||||
VIR_AUTOSTRINGLIST modesSpecified = NULL;
|
|
||||||
|
|
||||||
virCheckFlags(0, NULL);
|
virCheckFlags(0, NULL);
|
||||||
|
|
||||||
if (vshCommandOptStringQuiet(ctl, cmd, "mode", &modeConst) < 0)
|
if (vshCommandOptStringQuiet(ctl, cmd, "mode", &mode) < 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
if (STREQ_NULLABLE(modeConst, " "))
|
return virshCommaStringListComplete(mode, modes);
|
||||||
modeConst = NULL;
|
|
||||||
|
|
||||||
if (modeConst) {
|
|
||||||
char *modeTmp = NULL;
|
|
||||||
|
|
||||||
if (VIR_STRDUP(mode, modeConst) < 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if ((modeTmp = strrchr(mode, ',')))
|
|
||||||
*modeTmp = '\0';
|
|
||||||
else
|
|
||||||
VIR_FREE(mode);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mode && !(modesSpecified = virStringSplit(mode, ",", 0)))
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if (VIR_ALLOC_N(tmp, ARRAY_CARDINALITY(modes) + 1) < 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_CARDINALITY(modes); i++) {
|
|
||||||
if (virStringListHasString((const char **)modesSpecified, modes[i]))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if ((mode && virAsprintf(&tmp[ntmp], "%s,%s", mode, modes[i]) < 0) ||
|
|
||||||
(!mode && VIR_STRDUP(tmp[ntmp], modes[i]) < 0))
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
ntmp++;
|
|
||||||
}
|
|
||||||
|
|
||||||
VIR_STEAL_PTR(ret, tmp);
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user