mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-24 21:45:21 +00:00
18fd4899f3
Thing about vshReadlineInit() is - it's called multiple times. The first time from vshInit(), when @ctl was filled only partially (most notably, before any argv parsing is done, hence ctl->imode is set to false). The second time after argv parsing, from virshInit() -> vshInitReload(). In here, ctl->imode might have changed and thus vshReadlineInit() can't exit early - it needs to set up stuff for interactive mode (history basically). To allow vshReadlineInit() to be called again, vshReadlineDeinit() must set @autoCompleteOpaque to NULL. Fixes: cab1e71f0161fd24c5d6ff4c379d3a242ea8c2d9 Resolves: https://issues.redhat.com/browse/RHEL-53560 Signed-off-by: Michal Privoznik <mprivozn@redhat.com> Reviewed-by: Jiri Denemark <jdenemar@redhat.com>
3573 lines
92 KiB
C
3573 lines
92 KiB
C
/*
|
|
* vsh.c: common data to be used by clients to exercise the libvirt API
|
|
*
|
|
* Copyright (C) 2005-2019 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include "vsh.h"
|
|
|
|
#include <assert.h>
|
|
#include <stdarg.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
#include <signal.h>
|
|
|
|
#if WITH_READLINE
|
|
/* In order to have proper rl_message declaration with older
|
|
* versions of readline, we have to declare this. See 9ea3424a178
|
|
* for more info. */
|
|
# define HAVE_STDARG_H
|
|
# include <readline/readline.h>
|
|
# include <readline/history.h>
|
|
#endif
|
|
|
|
#include "internal.h"
|
|
#include "virbuffer.h"
|
|
#include "viralloc.h"
|
|
#include "virfile.h"
|
|
#include "virthread.h"
|
|
#include "vircommand.h"
|
|
#include "virstring.h"
|
|
#include "virutil.h"
|
|
|
|
#ifdef WITH_READLINE
|
|
/* For autocompletion */
|
|
vshControl *autoCompleteOpaque;
|
|
#endif
|
|
|
|
/* NOTE: It would be much nicer to have these two as part of vshControl
|
|
* structure, unfortunately readline doesn't support passing opaque data
|
|
* and only relies on static data accessible from the user-side callback
|
|
*/
|
|
const vshCmdGrp *cmdGroups;
|
|
|
|
|
|
double
|
|
vshPrettyCapacity(unsigned long long val, const char **unit)
|
|
{
|
|
double limit = 1024;
|
|
|
|
if (val < limit) {
|
|
*unit = "B";
|
|
return val;
|
|
}
|
|
limit *= 1024;
|
|
if (val < limit) {
|
|
*unit = "KiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
limit *= 1024;
|
|
if (val < limit) {
|
|
*unit = "MiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
limit *= 1024;
|
|
if (val < limit) {
|
|
*unit = "GiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
limit *= 1024;
|
|
if (val < limit) {
|
|
*unit = "TiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
limit *= 1024;
|
|
if (val < limit) {
|
|
*unit = "PiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
limit *= 1024;
|
|
*unit = "EiB";
|
|
return val / (limit / 1024);
|
|
}
|
|
|
|
|
|
int
|
|
vshNameSorter(const void *a, const void *b)
|
|
{
|
|
const char **sa = (const char**)a;
|
|
const char **sb = (const char**)b;
|
|
|
|
return vshStrcasecmp(*sa, *sb);
|
|
}
|
|
|
|
|
|
/*
|
|
* Convert the strings separated by ',' into array. The returned
|
|
* array is a NULL terminated string list. The caller has to free
|
|
* the array using g_strfreev or a similar method.
|
|
*
|
|
* Returns the length of the filled array on success, or -1
|
|
* on error.
|
|
*/
|
|
int
|
|
vshStringToArray(const char *str,
|
|
char ***array)
|
|
{
|
|
g_auto(GStrv) tmp = NULL;
|
|
GStrv n;
|
|
size_t ntoks = 0;
|
|
bool concat = false;
|
|
|
|
tmp = g_strsplit(str, ",", 0);
|
|
|
|
*array = g_new0(char *, g_strv_length(tmp) + 1);
|
|
(*array)[ntoks++] = g_strdup(tmp[0]);
|
|
|
|
/* undo splitting of comma escape (',,') by concatenating back on empty strings */
|
|
for (n = tmp + 1; n[0]; n++) {
|
|
if (concat) {
|
|
g_autofree char *old = (*array)[ntoks - 1];
|
|
|
|
(*array)[ntoks - 1] = g_strconcat(old, ",", n[0], NULL);
|
|
concat = false;
|
|
continue;
|
|
}
|
|
|
|
if (strlen(n[0]) == 0) {
|
|
concat = true;
|
|
} else {
|
|
(*array)[ntoks++] = g_strdup(n[0]);
|
|
}
|
|
}
|
|
|
|
/* corner case of ending with a single comma */
|
|
if (concat)
|
|
(*array)[ntoks++] = g_strdup("");
|
|
|
|
return ntoks;
|
|
}
|
|
|
|
virErrorPtr last_error;
|
|
|
|
/*
|
|
* Quieten libvirt until we're done with the command.
|
|
*/
|
|
void
|
|
vshErrorHandler(void *opaque G_GNUC_UNUSED,
|
|
virErrorPtr error G_GNUC_UNUSED)
|
|
{
|
|
virFreeError(last_error);
|
|
last_error = virSaveLastError();
|
|
}
|
|
|
|
/* Store a libvirt error that is from a helper API that doesn't raise errors
|
|
* so it doesn't get overwritten */
|
|
void
|
|
vshSaveLibvirtError(void)
|
|
{
|
|
virFreeError(last_error);
|
|
last_error = virSaveLastError();
|
|
}
|
|
|
|
|
|
/* Store libvirt error from helper API but don't overwrite existing errors */
|
|
void
|
|
vshSaveLibvirtHelperError(void)
|
|
{
|
|
if (last_error)
|
|
return;
|
|
|
|
if (virGetLastErrorCode() == VIR_ERR_OK)
|
|
return;
|
|
|
|
vshSaveLibvirtError();
|
|
}
|
|
|
|
|
|
/*
|
|
* Reset libvirt error on graceful fallback paths
|
|
*/
|
|
void
|
|
vshResetLibvirtError(void)
|
|
{
|
|
g_clear_pointer(&last_error, virFreeError);
|
|
virResetLastError();
|
|
}
|
|
|
|
/*
|
|
* Report an error when a command finishes. This is better than before
|
|
* (when correct operation would report errors), but it has some
|
|
* problems: we lose the smarter formatting of virDefaultErrorFunc(),
|
|
* and it can become harder to debug problems, if errors get reported
|
|
* twice during one command. This case shouldn't really happen anyway,
|
|
* and it's IMHO a bug that libvirt does that sometimes.
|
|
*/
|
|
void
|
|
vshReportError(vshControl *ctl)
|
|
{
|
|
if (last_error == NULL) {
|
|
/* Calling directly into libvirt util functions won't trigger the
|
|
* error callback (which sets last_error), so check it ourselves.
|
|
*
|
|
* If the returned error has CODE_OK, this most likely means that
|
|
* no error was ever raised, so just ignore */
|
|
last_error = virSaveLastError();
|
|
if (!last_error || last_error->code == VIR_ERR_OK)
|
|
goto out;
|
|
}
|
|
|
|
if (last_error->code == VIR_ERR_OK) {
|
|
vshError(ctl, "%s", _("unknown error"));
|
|
goto out;
|
|
}
|
|
|
|
vshError(ctl, "%s", last_error->message);
|
|
|
|
out:
|
|
vshResetLibvirtError();
|
|
}
|
|
|
|
/*
|
|
* Detection of disconnections and automatic reconnection support
|
|
*/
|
|
static int disconnected; /* we may have been disconnected */
|
|
|
|
|
|
/* vshCmddefSearch:
|
|
* @cmdname: name of command to find
|
|
*
|
|
* Looks for @cmdname in the global list of command definitions @cmdGroups and
|
|
* returns pointer to the definition struct if the command exists.
|
|
*/
|
|
static const vshCmdDef *
|
|
vshCmddefSearch(const char *cmdname)
|
|
{
|
|
const vshCmdGrp *g;
|
|
const vshCmdDef *c;
|
|
|
|
for (g = cmdGroups; g->name; g++) {
|
|
for (c = g->commands; c->name; c++) {
|
|
if (STREQ(c->name, cmdname))
|
|
return c;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Check if the internal command definitions are correct.
|
|
* None of the errors are to be marked as translatable. */
|
|
static int
|
|
vshCmddefCheckInternals(vshControl *ctl,
|
|
const vshCmdDef *cmd,
|
|
bool missingCompleters)
|
|
{
|
|
size_t i;
|
|
bool seenOptionalOption = false;
|
|
const char *seenOptionalPositionalOption = NULL;
|
|
g_auto(virBuffer) complbuf = VIR_BUFFER_INITIALIZER;
|
|
|
|
/* in order to perform the validation resolve the alias first */
|
|
if (cmd->alias) {
|
|
const vshCmdDef *alias;
|
|
|
|
if (!(alias = vshCmddefSearch(cmd->alias))) {
|
|
vshError(ctl, "command alias '%s' is pointing to a non-existent command '%s'",
|
|
cmd->name, cmd->alias);
|
|
return -1;
|
|
}
|
|
|
|
if (alias->alias) {
|
|
vshError(ctl, "command alias '%s' is pointing to another command alias '%s'",
|
|
cmd->name, cmd->alias);
|
|
return -1;
|
|
}
|
|
|
|
if (cmd->handler) {
|
|
vshError(ctl, "command '%s' has handler set", cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (cmd->opts) {
|
|
vshError(ctl, "command '%s' has options set", cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (cmd->info) {
|
|
vshError(ctl, "command '%s' has info set", cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (cmd->flags != 0) {
|
|
vshError(ctl, "command '%s' has multiple flags set", cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
/* we don't need to continue as the real command will be checked separately */
|
|
return 0;
|
|
}
|
|
|
|
/* Each command has to provide a non-empty help string. */
|
|
if (!cmd->info || !cmd->info->help || !*cmd->info->help) {
|
|
vshError(ctl, "command '%s' lacks help", cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (!cmd->opts)
|
|
return 0;
|
|
|
|
for (i = 0; cmd->opts[i].name; i++) {
|
|
const vshCmdOptDef *opt = &cmd->opts[i];
|
|
|
|
if (missingCompleters && !opt->completer) {
|
|
switch (opt->type) {
|
|
case VSH_OT_STRING:
|
|
case VSH_OT_ARGV:
|
|
virBufferStrcat(&complbuf, opt->name, ", ", NULL);
|
|
break;
|
|
|
|
case VSH_OT_BOOL:
|
|
/* only name is completed */
|
|
case VSH_OT_INT:
|
|
/* no point in completing numbers */
|
|
case VSH_OT_ALIAS:
|
|
/* alias is handled in the referenced command */
|
|
case VSH_OT_NONE:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* allow at most one optional positional option */
|
|
if (opt->positional && !opt->required) {
|
|
if (seenOptionalPositionalOption) {
|
|
vshError(ctl, "multiple optional positional arguments (%s, %s) of command '%s' are not allowed",
|
|
seenOptionalPositionalOption, opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
seenOptionalPositionalOption = opt->name;
|
|
}
|
|
|
|
/* all optional positional arguments must be defined after the required ones */
|
|
if (seenOptionalPositionalOption && opt->positional && opt->required) {
|
|
vshError(ctl, "required positional argument '%s' declared after an optional positional argument '%s' of command '%s'",
|
|
opt->name, seenOptionalPositionalOption, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
/* Mandate no completer flags if no completer is specified */
|
|
if (opt->completer_flags != 0 && !opt->completer) {
|
|
vshError(ctl, "completer_flags of argument '%s' of command '%s' must be 0 if no completer is used",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (opt->unwanted_positional && opt->positional) {
|
|
vshError(ctl, "unwanted_positional flag of argument '%s' of command '%s' must not be used together with positional",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
switch (opt->type) {
|
|
case VSH_OT_NONE:
|
|
vshError(ctl, "invalid type 'NONE' of option '%s' of command '%s'",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
|
|
case VSH_OT_BOOL:
|
|
if (opt->completer) {
|
|
vshError(ctl, "bool parameter '%s' of command '%s' has completer set",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (opt->positional || opt->unwanted_positional) {
|
|
vshError(ctl, "boolean parameter '%s' of command '%s' must not be positional",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
if (opt->required) {
|
|
vshError(ctl, "parameter '%s' of command '%s' misused 'required' flag",
|
|
opt->name, cmd->name);
|
|
return -1; /* bool can't be mandatory */
|
|
}
|
|
|
|
break;
|
|
|
|
case VSH_OT_ALIAS: {
|
|
size_t j;
|
|
g_autofree char *name = NULL;
|
|
char *p;
|
|
|
|
if (opt->required ||
|
|
opt->positional ||
|
|
opt->unwanted_positional ||
|
|
opt->completer ||
|
|
!opt->help) {
|
|
vshError(ctl, "parameter '%s' of command '%s' has incorrect alias option",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
if ((p = strchr(opt->help, '=')))
|
|
name = g_strndup(opt->help, p - opt->help);
|
|
else
|
|
name = g_strdup(opt->help);
|
|
for (j = i + 1; cmd->opts[j].name; j++) {
|
|
if (STREQ(name, cmd->opts[j].name) &&
|
|
cmd->opts[j].type != VSH_OT_ALIAS)
|
|
break;
|
|
}
|
|
if (p) {
|
|
/* If alias comes with value, replacement must not be bool */
|
|
if (cmd->opts[j].type == VSH_OT_BOOL) {
|
|
vshError(ctl, "alias '%s' of command '%s' has mismatched alias type",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
}
|
|
if (!cmd->opts[j].name) {
|
|
vshError(ctl, "alias '%s' of command '%s' has missing alias option",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_ARGV:
|
|
if (cmd->opts[i + 1].name) {
|
|
vshError(ctl, "parameter '%s' of command '%s' must be listed last",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_INT:
|
|
case VSH_OT_STRING:
|
|
if (opt->positional && seenOptionalOption) {
|
|
vshError(ctl, "parameter '%s' of command '%s' must be listed before optional parameters",
|
|
opt->name, cmd->name);
|
|
return -1;
|
|
}
|
|
|
|
seenOptionalOption = !opt->required;
|
|
break;
|
|
}
|
|
}
|
|
|
|
virBufferTrim(&complbuf, ", ");
|
|
|
|
if (missingCompleters && virBufferUse(&complbuf) > 0)
|
|
vshPrintExtra(ctl, "%s: %s\n", cmd->name, virBufferCurrentContent(&complbuf));
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static vshCmdOpt *
|
|
vshCmdGetOption(vshControl *ctl,
|
|
vshCmd *cmd,
|
|
const char *name,
|
|
char **optstr,
|
|
bool report)
|
|
{
|
|
g_autofree char *alias = NULL;
|
|
vshCmdOpt *n;
|
|
|
|
for (n = cmd->opts; n && n->def; n++) {
|
|
if (STRNEQ(n->def->name, name))
|
|
continue;
|
|
|
|
if (n->def->type == VSH_OT_ALIAS) {
|
|
char *value;
|
|
|
|
/* Two types of replacements:
|
|
opt->help = "string": straight replacement of name
|
|
opt->help = "string=value": treat boolean flag as
|
|
alias of option and its default value */
|
|
alias = g_strdup(n->def->help);
|
|
name = alias;
|
|
if ((value = strchr(name, '='))) {
|
|
*value = '\0';
|
|
if (*optstr) {
|
|
if (report)
|
|
vshError(ctl, _("invalid '=' after option --%1$s"),
|
|
n->def->name);
|
|
return NULL;
|
|
}
|
|
*optstr = g_strdup(value + 1);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (n->present && n->def->type != VSH_OT_ARGV) {
|
|
if (report)
|
|
vshError(ctl, _("option --%1$s already seen"), name);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
/* The 'help' command ignores extra options */
|
|
if (STRNEQ(cmd->def->name, "help") && report) {
|
|
vshError(ctl, _("command '%1$s' doesn't support option --%2$s"),
|
|
cmd->def->name, name);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
vshCmdOptAssign(vshControl *ctl,
|
|
vshCmd *cmd,
|
|
vshCmdOpt *opt,
|
|
const char *val,
|
|
bool report)
|
|
{
|
|
cmd->lastopt = opt;
|
|
|
|
opt->present = true;
|
|
|
|
switch (opt->def->type) {
|
|
case VSH_OT_BOOL:
|
|
/* nothing to do */
|
|
if (report) {
|
|
vshDebug(ctl, VSH_ERR_INFO, "%s: %s(bool)\n",
|
|
cmd->def->name, opt->def->name);
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_STRING:
|
|
case VSH_OT_INT:
|
|
if (report) {
|
|
vshDebug(ctl, VSH_ERR_INFO, "%s: %s(optdata): %s\n",
|
|
cmd->def->name, opt->def->name, NULLSTR(val));
|
|
}
|
|
|
|
opt->data = g_strdup(val);
|
|
break;
|
|
|
|
case VSH_OT_ARGV:
|
|
if (report) {
|
|
vshDebug(ctl, VSH_ERR_INFO, "%s: %s(argv: %zu): %s\n",
|
|
cmd->def->name, opt->def->name, opt->nargv, NULLSTR(val));
|
|
}
|
|
|
|
VIR_EXPAND_N(opt->argv, opt->nargv, 2);
|
|
/* VIR_EXPAND_N updates count */
|
|
opt->nargv--;
|
|
opt->argv[opt->nargv - 1] = g_strdup(val);
|
|
/* for completers to work properly we need to also remember the last
|
|
* field in 'data' */
|
|
g_clear_pointer(&opt->data, g_free);
|
|
opt->data = g_strdup(val);
|
|
break;
|
|
|
|
case VSH_OT_NONE:
|
|
case VSH_OT_ALIAS:
|
|
/* impossible code path */
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCmdGetNextPositionalOpt:
|
|
* @cmd: command structure
|
|
*
|
|
* Get next unpopulated positional argument definition.
|
|
*/
|
|
static vshCmdOpt *
|
|
vshCmdGetNextPositionalOpt(const vshCmd *cmd)
|
|
{
|
|
vshCmdOpt *n;
|
|
|
|
for (n = cmd->opts; n && n->def; n++) {
|
|
/* Consider only "positional" options. Tests ensure that boolean options
|
|
* don't set these. */
|
|
if (!(n->def->positional || n->def->unwanted_positional))
|
|
continue;
|
|
|
|
/* 'VSH_OT_ARGV' positionals must allow multiple arguments */
|
|
if (n->present &&
|
|
n->def->type != VSH_OT_ARGV)
|
|
continue;
|
|
|
|
return n;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
* Checks for required options
|
|
*/
|
|
static int
|
|
vshCommandCheckOpts(vshControl *ctl,
|
|
const vshCmd *cmd)
|
|
{
|
|
vshCmdOpt *n;
|
|
|
|
for (n = cmd->opts; n && n->def; n++) {
|
|
if (!n->present && n->def->required) {
|
|
if (n->def->positional) {
|
|
vshError(ctl,
|
|
_("command '%1$s' requires <%2$s> option"),
|
|
cmd->def->name, n->def->name);
|
|
} else {
|
|
vshError(ctl,
|
|
_("command '%1$s' requires --%2$s option"),
|
|
cmd->def->name, n->def->name);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static const vshCmdGrp *
|
|
vshCmdGrpSearch(const char *grpname)
|
|
{
|
|
const vshCmdGrp *g;
|
|
|
|
for (g = cmdGroups; g->name; g++) {
|
|
if (STREQ(g->name, grpname) || STREQ(g->keyword, grpname))
|
|
return g;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static bool
|
|
vshCmdGrpHelp(vshControl *ctl, const vshCmdGrp *grp)
|
|
{
|
|
const vshCmdDef *cmd = NULL;
|
|
|
|
vshPrint(ctl, _(" %1$s (help keyword '%2$s'):\n"), grp->name,
|
|
grp->keyword);
|
|
|
|
for (cmd = grp->commands; cmd->name; cmd++) {
|
|
if (cmd->alias ||
|
|
cmd->flags & VSH_CMD_FLAG_HIDDEN)
|
|
continue;
|
|
vshPrint(ctl, " %-30s %s\n", cmd->name, _(cmd->info->help));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
vshCmddefHelp(const vshCmdDef *def)
|
|
{
|
|
fputs(_(" NAME\n"), stdout);
|
|
fprintf(stdout, " %s - %s\n", def->name, _(def->info->help));
|
|
|
|
fputs(_("\n SYNOPSIS\n"), stdout);
|
|
fprintf(stdout, " %s", def->name);
|
|
if (def->opts) {
|
|
const vshCmdOptDef *opt;
|
|
for (opt = def->opts; opt->name; opt++) {
|
|
|
|
switch (opt->type) {
|
|
case VSH_OT_BOOL:
|
|
fprintf(stdout, " [--%s]", opt->name);
|
|
break;
|
|
|
|
case VSH_OT_STRING:
|
|
case VSH_OT_INT:
|
|
if (opt->required) {
|
|
fprintf(stdout, " ");
|
|
} else {
|
|
fprintf(stdout, " [");
|
|
}
|
|
|
|
if (opt->positional) {
|
|
fprintf(stdout, "<%s>", opt->name);
|
|
} else {
|
|
if (opt->type == VSH_OT_INT) {
|
|
fprintf(stdout, _("--%1$s <number>"), opt->name);
|
|
} else {
|
|
fprintf(stdout, _("--%1$s <string>"), opt->name);
|
|
}
|
|
}
|
|
|
|
if (!opt->required)
|
|
fprintf(stdout, "]");
|
|
break;
|
|
|
|
case VSH_OT_ARGV:
|
|
if (opt->positional) {
|
|
if (opt->required) {
|
|
fprintf(stdout, " <%s>...", opt->name);
|
|
} else {
|
|
fprintf(stdout, " [<%s>]...", opt->name);
|
|
}
|
|
} else {
|
|
if (opt->required) {
|
|
fprintf(stdout, _(" --%1$s <string>..."), opt->name);
|
|
} else {
|
|
fprintf(stdout, _(" [--%1$s <string>]..."), opt->name);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_ALIAS:
|
|
case VSH_OT_NONE:
|
|
/* aliases are intentionally undocumented */
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
fputc('\n', stdout);
|
|
|
|
if (def->info->desc && *def->info->desc) {
|
|
/* Print the description only if it's not empty. */
|
|
fputs(_("\n DESCRIPTION\n"), stdout);
|
|
fprintf(stdout, " %s\n", _(def->info->desc));
|
|
}
|
|
|
|
if (def->opts && def->opts->name) {
|
|
const vshCmdOptDef *opt;
|
|
fputs(_("\n OPTIONS\n"), stdout);
|
|
for (opt = def->opts; opt->name; opt++) {
|
|
g_autofree char *optstr = NULL;
|
|
|
|
switch (opt->type) {
|
|
case VSH_OT_BOOL:
|
|
optstr = g_strdup_printf("--%s", opt->name);
|
|
break;
|
|
|
|
case VSH_OT_INT:
|
|
if (opt->positional) {
|
|
optstr = g_strdup_printf(_("[--%1$s] <number>"), opt->name);
|
|
} else {
|
|
optstr = g_strdup_printf(_("--%1$s <number>"), opt->name);
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_STRING:
|
|
if (opt->positional) {
|
|
optstr = g_strdup_printf(_("[--%1$s] <string>"), opt->name);
|
|
} else {
|
|
optstr = g_strdup_printf(_("--%1$s <string>"), opt->name);
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_ARGV:
|
|
if (opt->positional) {
|
|
optstr = g_strdup_printf(_("[--%1$s] <string>..."), opt->name);
|
|
} else {
|
|
optstr = g_strdup_printf(_("--%1$s <string>..."), opt->name);
|
|
}
|
|
break;
|
|
|
|
case VSH_OT_ALIAS:
|
|
case VSH_OT_NONE:
|
|
continue;
|
|
}
|
|
|
|
fprintf(stdout, " %-15s %s\n", optstr, _(opt->help));
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* ---------------
|
|
* Utils for work with runtime commands data
|
|
* ---------------
|
|
*/
|
|
static void
|
|
vshCommandFree(vshCmd *cmd)
|
|
{
|
|
vshCmd *c = cmd;
|
|
|
|
while (c) {
|
|
vshCmd *tmp = c;
|
|
vshCmdOpt *n;
|
|
|
|
c = c->next;
|
|
|
|
for (n = tmp->opts; n && n->def; n++) {
|
|
g_free(n->data);
|
|
g_strfreev(n->argv);
|
|
g_free(n->argvstr);
|
|
}
|
|
|
|
g_free(tmp->opts);
|
|
|
|
g_free(tmp);
|
|
}
|
|
}
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(vshCmd, vshCommandFree);
|
|
|
|
/**
|
|
* vshCommandOpt:
|
|
* @cmd: parsed command line to search
|
|
* @name: option name to search for
|
|
* @opt: result of the search
|
|
* @needData: true if option must be non-boolean
|
|
*
|
|
* Look up an option passed to CMD by NAME. Returns 1 with *OPT set
|
|
* to the option if found, 0 with *OPT set to NULL if the name is
|
|
* valid and the option is not required, -1 with *OPT set to NULL if
|
|
* the option is required but not present, and assert if NAME is not
|
|
* valid (which indicates a programming error) unless cmd->skipChecks
|
|
* is set. No error messages are issued if a value is returned.
|
|
*/
|
|
static int
|
|
vshCommandOpt(const vshCmd *cmd,
|
|
const char *name,
|
|
vshCmdOpt **opt,
|
|
bool needData)
|
|
{
|
|
vshCmdOpt *n;
|
|
|
|
*opt = NULL;
|
|
|
|
for (n = cmd->opts; n && n->def; n++) {
|
|
if (STRNEQ(name, n->def->name))
|
|
continue;
|
|
|
|
if (!cmd->skipChecks)
|
|
assert(!needData || n->def->type != VSH_OT_BOOL);
|
|
|
|
if (n->present) {
|
|
*opt = n;
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (!cmd->skipChecks)
|
|
assert(false);
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCommandOptInt:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Convert option to int.
|
|
* On error, a message is displayed.
|
|
*
|
|
* Return value:
|
|
* >0 if option found and valid (@value updated)
|
|
* 0 if option not found and not required (@value untouched)
|
|
* <0 in all other cases (@value untouched)
|
|
*/
|
|
int
|
|
vshCommandOptInt(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, int *value)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if ((ret = virStrToLong_i(arg->data, NULL, 10, value)) < 0)
|
|
vshError(ctl,
|
|
_("Numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
else
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
vshCommandOptUIntInternal(vshControl *ctl,
|
|
const vshCmd *cmd,
|
|
const char *name,
|
|
unsigned int *value,
|
|
bool wrap)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if (wrap)
|
|
ret = virStrToLong_ui(arg->data, NULL, 10, value);
|
|
else
|
|
ret = virStrToLong_uip(arg->data, NULL, 10, value);
|
|
if (ret < 0)
|
|
vshError(ctl,
|
|
_("Numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
else
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptUInt:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Convert option to unsigned int, reject negative numbers
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptUInt(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned int *value)
|
|
{
|
|
return vshCommandOptUIntInternal(ctl, cmd, name, value, false);
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptUIntWrap:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Convert option to unsigned int, wraps negative numbers to positive
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptUIntWrap(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned int *value)
|
|
{
|
|
return vshCommandOptUIntInternal(ctl, cmd, name, value, true);
|
|
}
|
|
|
|
static int
|
|
vshCommandOptULInternal(vshControl *ctl,
|
|
const vshCmd *cmd,
|
|
const char *name,
|
|
unsigned long *value,
|
|
bool wrap)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if (wrap)
|
|
ret = virStrToLong_ul(arg->data, NULL, 10, value);
|
|
else
|
|
ret = virStrToLong_ulp(arg->data, NULL, 10, value);
|
|
if (ret < 0)
|
|
vshError(ctl,
|
|
_("Numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
else
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* vshCommandOptUL:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Convert option to unsigned long
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptUL(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned long *value)
|
|
{
|
|
return vshCommandOptULInternal(ctl, cmd, name, value, false);
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptULWrap:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Convert option to unsigned long, wraps negative numbers to positive
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptULWrap(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned long *value)
|
|
{
|
|
return vshCommandOptULInternal(ctl, cmd, name, value, true);
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptStringQuiet:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Returns option as STRING. On error -1 is returned but no error is set.
|
|
* Return value:
|
|
* >0 if option found and valid (@value updated)
|
|
* 0 if option not found and not required (@value untouched)
|
|
* <0 in all other cases (@value untouched)
|
|
*/
|
|
int
|
|
vshCommandOptStringQuiet(vshControl *ctl G_GNUC_UNUSED, const vshCmd *cmd,
|
|
const char *name, const char **value)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if (!arg->def->allowEmpty && *arg->data == '\0')
|
|
return -1;
|
|
*value = arg->data;
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptString:
|
|
* @ctl virtshell control structure
|
|
* @cmd command structure
|
|
* @name option name
|
|
* @value result (updated to NULL or the option argument)
|
|
*
|
|
* Gets a option argument as string.
|
|
*
|
|
* Returns 0 on success or when the option is not present and not
|
|
* required, *value is set to the option argument. On error -1 is
|
|
* returned and error message printed.
|
|
*/
|
|
int
|
|
vshCommandOptString(vshControl *ctl,
|
|
const vshCmd *cmd,
|
|
const char *name,
|
|
const char **value)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
const char *error = NULL;
|
|
|
|
/* clear out the value */
|
|
*value = NULL;
|
|
|
|
ret = vshCommandOpt(cmd, name, &arg, true);
|
|
/* option is not required and not present */
|
|
if (ret == 0)
|
|
return 0;
|
|
/* this should not be propagated here, just to be sure */
|
|
if (ret == -1)
|
|
error = N_("Mandatory option not present");
|
|
else if (arg && *arg->data == '\0' && !arg->def->allowEmpty)
|
|
error = N_("Option argument is empty");
|
|
|
|
if (error) {
|
|
if (!cmd->skipChecks)
|
|
vshError(ctl, _("Failed to get option '%1$s': %2$s"), name, _(error));
|
|
return -1;
|
|
}
|
|
|
|
*value = arg->data;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptLongLong:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Returns option as long long
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptLongLong(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, long long *value)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if ((ret = virStrToLong_ll(arg->data, NULL, 10, value)) < 0)
|
|
vshError(ctl,
|
|
_("Numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
else
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
vshCommandOptULongLongInternal(vshControl *ctl,
|
|
const vshCmd *cmd,
|
|
const char *name,
|
|
unsigned long long *value,
|
|
bool wrap)
|
|
{
|
|
vshCmdOpt *arg;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if (wrap)
|
|
ret = virStrToLong_ull(arg->data, NULL, 10, value);
|
|
else
|
|
ret = virStrToLong_ullp(arg->data, NULL, 10, value);
|
|
if (ret < 0)
|
|
vshError(ctl,
|
|
_("Numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
else
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptULongLong:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Returns option as long long, rejects negative numbers
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptULongLong(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned long long *value)
|
|
{
|
|
return vshCommandOptULongLongInternal(ctl, cmd, name, value, false);
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptULongLongWrap:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
*
|
|
* Returns option as long long, wraps negative numbers to positive
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptULongLongWrap(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned long long *value)
|
|
{
|
|
return vshCommandOptULongLongInternal(ctl, cmd, name, value, true);
|
|
}
|
|
|
|
/**
|
|
* vshCommandOptScaledInt:
|
|
* @ctl virtshell control structure
|
|
* @cmd command reference
|
|
* @name option name
|
|
* @value result
|
|
* @scale default of 1 or 1024, if no suffix is present
|
|
* @max maximum value permitted
|
|
*
|
|
* Returns option as long long, scaled according to suffix
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptScaledInt(vshControl *ctl, const vshCmd *cmd,
|
|
const char *name, unsigned long long *value,
|
|
int scale, unsigned long long max)
|
|
{
|
|
vshCmdOpt *arg;
|
|
char *end;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, name, &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
if (virStrToLong_ullp(arg->data, &end, 10, value) < 0 ||
|
|
virScaleInteger(value, end, scale, max) < 0) {
|
|
vshError(ctl,
|
|
_("Scaled numeric value '%1$s' for <%2$s> option is malformed or out of range"),
|
|
arg->data, name);
|
|
return -1;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCommandOptBool:
|
|
* @cmd command reference
|
|
* @name option name
|
|
*
|
|
* Returns true/false if the option exists. Note that this does NOT
|
|
* validate whether the option is actually boolean, or even whether
|
|
* name is legal; so that this can be used to probe whether a data
|
|
* option is present without actually using that data.
|
|
*/
|
|
bool
|
|
vshCommandOptBool(const vshCmd *cmd, const char *name)
|
|
{
|
|
vshCmdOpt *dummy;
|
|
|
|
return vshCommandOpt(cmd, name, &dummy, false) == 1;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCommandOptArgv:
|
|
* @cmd: command reference
|
|
* @name: name of argument
|
|
*
|
|
* Returns a NULL terminated list of strings of values passed as argument of
|
|
* ARGV argument named @name. The returned string list is owned by @cmd and
|
|
* caller must not free or modify it.
|
|
*/
|
|
const char **
|
|
vshCommandOptArgv(const vshCmd *cmd,
|
|
const char *name)
|
|
{
|
|
vshCmdOpt *opt;
|
|
|
|
if (vshCommandOpt(cmd, name, &opt, true) != 1)
|
|
return NULL;
|
|
|
|
return (const char **) opt->argv;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCommandOptArgvString:
|
|
* @cmd: command reference
|
|
* @name: name of argument
|
|
*
|
|
* Returns a string containing all values passed as ARGV argument @name
|
|
* delimited/concatenated by adding spaces.
|
|
*/
|
|
const char *
|
|
vshCommandOptArgvString(const vshCmd *cmd,
|
|
const char *name)
|
|
{
|
|
vshCmdOpt *opt;
|
|
|
|
if (vshCommandOpt(cmd, name, &opt, true) != 1)
|
|
return NULL;
|
|
|
|
if (!opt->argvstr)
|
|
opt->argvstr = g_strjoinv(" ", opt->argv);
|
|
|
|
return opt->argvstr;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshBlockJobOptionBandwidth:
|
|
* @ctl: virsh control data
|
|
* @cmd: virsh command description
|
|
* @bytes: return bandwidth in bytes/s instead of MiB/s
|
|
* @bandwidth: return value
|
|
*
|
|
* Extracts the value of --bandwidth either as a wrap-able number without scale
|
|
* or as a scaled integer. The returned value is checked to fit into a unsigned
|
|
* long data type. This is a legacy compatibility function and it should not
|
|
* be used for things other the block job APIs.
|
|
*
|
|
* Returns 0 on success, -1 on error.
|
|
*/
|
|
int
|
|
vshBlockJobOptionBandwidth(vshControl *ctl,
|
|
const vshCmd *cmd,
|
|
bool bytes,
|
|
unsigned long *bandwidth)
|
|
{
|
|
vshCmdOpt *arg;
|
|
char *end;
|
|
unsigned long long bw;
|
|
int ret;
|
|
|
|
if ((ret = vshCommandOpt(cmd, "bandwidth", &arg, true)) <= 0)
|
|
return ret;
|
|
|
|
/* due to historical reasons we declare to parse negative numbers and wrap
|
|
* them to the unsigned data type. */
|
|
if (virStrToLong_ul(arg->data, NULL, 10, bandwidth) < 0) {
|
|
/* try to parse the number as scaled size in this case we don't accept
|
|
* wrapping since it would be ridiculous. In case of a 32 bit host,
|
|
* limit the value to ULONG_MAX */
|
|
if (virStrToLong_ullp(arg->data, &end, 10, &bw) < 0 ||
|
|
virScaleInteger(&bw, end, 1, ULONG_MAX) < 0) {
|
|
vshError(ctl,
|
|
_("Scaled numeric value '%1$s' for <--bandwidth> option is malformed or out of range"),
|
|
arg->data);
|
|
return -1;
|
|
}
|
|
|
|
if (!bytes)
|
|
bw >>= 20;
|
|
|
|
*bandwidth = bw;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Executes command(s) and returns return code from last command
|
|
*/
|
|
bool
|
|
vshCommandRun(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
const vshClientHooks *hooks = ctl->hooks;
|
|
bool ret = true;
|
|
|
|
while (cmd) {
|
|
gint64 before, after;
|
|
bool enable_timing = ctl->timing;
|
|
|
|
before = g_get_real_time();
|
|
|
|
if ((cmd->def->flags & VSH_CMD_FLAG_NOCONNECT) ||
|
|
(hooks && hooks->connHandler && hooks->connHandler(ctl))) {
|
|
ret = cmd->def->handler(ctl, cmd);
|
|
} else {
|
|
/* connection is not usable, return error */
|
|
ret = false;
|
|
}
|
|
|
|
after = g_get_real_time();
|
|
|
|
/* try to automatically catch disconnections */
|
|
if (!ret &&
|
|
((last_error != NULL) &&
|
|
(((last_error->code == VIR_ERR_SYSTEM_ERROR) &&
|
|
(last_error->domain == VIR_FROM_REMOTE)) ||
|
|
(last_error->code == VIR_ERR_RPC) ||
|
|
(last_error->code == VIR_ERR_NO_CONNECT) ||
|
|
(last_error->code == VIR_ERR_INVALID_CONN))))
|
|
disconnected++;
|
|
|
|
if (!ret)
|
|
vshReportError(ctl);
|
|
|
|
if (STREQ(cmd->def->name, "quit") ||
|
|
STREQ(cmd->def->name, "exit")) /* hack ... */
|
|
return ret;
|
|
|
|
if (enable_timing) {
|
|
double diff_ms = (after - before) / 1000.0;
|
|
|
|
vshPrint(ctl, _("\n(Time: %1$.3f ms)\n\n"), diff_ms);
|
|
} else {
|
|
vshPrintExtra(ctl, "\n");
|
|
}
|
|
cmd = cmd->next;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/* ---------------
|
|
* Command parsing
|
|
* ---------------
|
|
*/
|
|
|
|
typedef enum {
|
|
VSH_TK_ERROR, /* Failed to parse a token */
|
|
VSH_TK_ARG, /* Arbitrary argument, might be option or empty */
|
|
VSH_TK_SUBCMD_END, /* Separation between commands */
|
|
VSH_TK_END /* No more commands */
|
|
} vshCommandToken;
|
|
|
|
typedef struct _vshCommandParser vshCommandParser;
|
|
struct _vshCommandParser {
|
|
vshCommandToken(*getNextArg)(vshControl *, vshCommandParser *,
|
|
char **, bool);
|
|
/* vshCommandStringGetArg() */
|
|
char *pos;
|
|
size_t point;
|
|
/* vshCommandArgvGetArg() */
|
|
char **arg_pos;
|
|
char **arg_end;
|
|
};
|
|
|
|
|
|
static vshCmd *
|
|
vshCmdNewHelp(const char *name)
|
|
{
|
|
vshCmd *c = g_new0(vshCmd, 1);
|
|
|
|
c->def = vshCmddefSearch("help");
|
|
|
|
c->opts = g_new0(vshCmdOpt, 2);
|
|
c->opts->def = c->def->opts;
|
|
c->opts->data = g_strdup(name);
|
|
c->opts->present = true;
|
|
|
|
return c;
|
|
}
|
|
|
|
|
|
static vshCmd *
|
|
vshCmdNew(vshControl *ctl,
|
|
const char *cmdname,
|
|
bool report)
|
|
{
|
|
g_autoptr(vshCmd) c = g_new0(vshCmd, 1);
|
|
const vshCmdOptDef *optdef;
|
|
vshCmdOpt *opt;
|
|
size_t nopts = 0;
|
|
|
|
if (!(c->def = vshCmddefSearch(cmdname))) {
|
|
if (report)
|
|
vshError(ctl, _("unknown command: '%1$s'"), cmdname);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/* resolve command alias */
|
|
if (c->def->alias) {
|
|
if (!(c->def = vshCmddefSearch(c->def->alias))) {
|
|
/* dead code: self-test ensures that the alias exists thus no error reported here */
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/* Find number of arguments */
|
|
for (optdef = c->def->opts; optdef && optdef->name; optdef++)
|
|
nopts++;
|
|
|
|
c->opts = g_new0(vshCmdOpt, nopts + 1);
|
|
opt = c->opts;
|
|
|
|
/* populate links to definitions */
|
|
for (optdef = c->def->opts; optdef && optdef->name; optdef++) {
|
|
opt->def = optdef;
|
|
opt++;
|
|
}
|
|
|
|
return g_steal_pointer(&c);
|
|
}
|
|
|
|
|
|
static int
|
|
vshCmdOptAssignPositional(vshControl *ctl,
|
|
vshCmd *cmd,
|
|
const char *val,
|
|
bool report)
|
|
{
|
|
vshCmdOpt *opt;
|
|
|
|
if (!(opt = vshCmdGetNextPositionalOpt(cmd))) {
|
|
/* ignore spurious arguments for 'help' command */
|
|
if (STREQ(cmd->def->name, "help"))
|
|
return 0;
|
|
|
|
if (report)
|
|
vshError(ctl, _("unexpected data '%1$s'"), val);
|
|
|
|
return -1;
|
|
}
|
|
|
|
vshCmdOptAssign(ctl, cmd, opt, val, report);
|
|
return 0;
|
|
}
|
|
|
|
|
|
typedef enum {
|
|
VSH_CMD_PARSER_STATE_START,
|
|
VSH_CMD_PARSER_STATE_COMMENT,
|
|
VSH_CMD_PARSER_STATE_COMMAND,
|
|
VSH_CMD_PARSER_STATE_ASSIGN_OPT,
|
|
VSH_CMD_PARSER_STATE_POSITIONAL_ONLY,
|
|
} vshCommandParserState;
|
|
|
|
static bool
|
|
vshCommandParse(vshControl *ctl,
|
|
vshCommandParser *parser,
|
|
vshCmd **partial)
|
|
{
|
|
g_autoptr(vshCmd) cmds = NULL; /* linked list of all parsed commands in this session */
|
|
vshCmd *cmds_last = NULL;
|
|
g_autoptr(vshCmd) cmd = NULL; /* currently parsed command */
|
|
vshCommandParserState state = VSH_CMD_PARSER_STATE_START;
|
|
vshCmdOpt *opt = NULL;
|
|
g_autofree char *optionvalue = NULL;
|
|
bool report = !partial;
|
|
bool ret = false;
|
|
|
|
if (partial) {
|
|
g_clear_pointer(partial, vshCommandFree);
|
|
} else {
|
|
g_clear_pointer(&ctl->cmd, vshCommandFree);
|
|
}
|
|
|
|
while (1) {
|
|
/* previous iteration might have already gotten a value. Store it as the
|
|
* token in this iteration */
|
|
g_autofree char *tkdata = g_steal_pointer(&optionvalue);
|
|
|
|
/* If we have a value already or the option to fill is a boolean we
|
|
* don't want to fetch a new token */
|
|
if (!(tkdata ||
|
|
(opt && opt->def->type == VSH_OT_BOOL))) {
|
|
vshCommandToken tk;
|
|
|
|
tk = parser->getNextArg(ctl, parser, &tkdata, report);
|
|
|
|
switch (tk) {
|
|
case VSH_TK_ARG:
|
|
/* will be handled below */
|
|
break;
|
|
|
|
case VSH_TK_ERROR:
|
|
goto out;
|
|
|
|
case VSH_TK_END:
|
|
case VSH_TK_SUBCMD_END:
|
|
/* The last argument name expects a value, but it's missing */
|
|
if (opt) {
|
|
if (partial) {
|
|
/* for completion to work we need to also store the
|
|
* last token into the last 'opt' */
|
|
vshCmdOptAssign(ctl, cmd, opt, tkdata, report);
|
|
} else {
|
|
if (opt->def->type == VSH_OT_INT)
|
|
vshError(ctl, _("expected syntax: --%1$s <number>"),
|
|
opt->def->name);
|
|
else
|
|
vshError(ctl, _("expected syntax: --%1$s <string>"),
|
|
opt->def->name);
|
|
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* command parsed -- allocate new struct for the command */
|
|
if (cmd) {
|
|
/* if we encountered --help, replace parsed command with 'help <cmdname>' */
|
|
if (cmd->helpOptionSeen) {
|
|
vshCmd *helpcmd = vshCmdNewHelp(cmd->def->name);
|
|
|
|
vshCommandFree(cmd);
|
|
cmd = helpcmd;
|
|
}
|
|
|
|
if (!partial &&
|
|
vshCommandCheckOpts(ctl, cmd) < 0)
|
|
goto out;
|
|
|
|
if (!cmds)
|
|
cmds = cmd;
|
|
if (cmds_last)
|
|
cmds_last->next = cmd;
|
|
cmds_last = g_steal_pointer(&cmd);
|
|
}
|
|
|
|
|
|
/* everything parsed */
|
|
if (tk == VSH_TK_END) {
|
|
ret = true;
|
|
goto out;
|
|
}
|
|
|
|
/* after processing the command we need to start over again to
|
|
* fetch another token */
|
|
state = VSH_CMD_PARSER_STATE_START;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* at this point we know that @tkdata is an argument */
|
|
switch (state) {
|
|
case VSH_CMD_PARSER_STATE_START:
|
|
if (*tkdata == '#') {
|
|
state = VSH_CMD_PARSER_STATE_COMMENT;
|
|
} else {
|
|
state = VSH_CMD_PARSER_STATE_COMMAND;
|
|
|
|
if (!(cmd = vshCmdNew(ctl, tkdata, !partial)))
|
|
goto out;
|
|
}
|
|
|
|
break;
|
|
|
|
case VSH_CMD_PARSER_STATE_COMMENT:
|
|
/* continue eating tokens until end of line or end of input */
|
|
state = VSH_CMD_PARSER_STATE_COMMENT;
|
|
break;
|
|
|
|
case VSH_CMD_PARSER_STATE_COMMAND: {
|
|
/* parsing individual options for the command. There are following options:
|
|
* --option
|
|
* --option value
|
|
* --option=value
|
|
* --aliasoptionwithvalue (value is part of the alias definition)
|
|
* value
|
|
* -- (terminate accepting '--option', fill only positional args)
|
|
*/
|
|
const char *optionname = tkdata + 2;
|
|
char *sep;
|
|
|
|
if (!STRPREFIX(tkdata, "--")) {
|
|
if (vshCmdOptAssignPositional(ctl, cmd, tkdata, report) < 0)
|
|
goto out;
|
|
break;
|
|
}
|
|
|
|
if (STREQ(tkdata, "--")) {
|
|
state = VSH_CMD_PARSER_STATE_POSITIONAL_ONLY;
|
|
break;
|
|
}
|
|
|
|
if ((sep = strchr(optionname, '='))) {
|
|
*(sep++) = '\0';
|
|
|
|
/* 'optionvalue' has lifetime until next iteration */
|
|
optionvalue = g_strdup(sep);
|
|
}
|
|
|
|
/* lookup the option. Note that vshCmdGetOption also resolves aliases
|
|
* and thus the value possibly contained in the alias */
|
|
if (STREQ(optionname, "help")) {
|
|
cmd->helpOptionSeen = true;
|
|
g_clear_pointer(&optionvalue, g_free);
|
|
} else if (!(opt = vshCmdGetOption(ctl, cmd, optionname, &optionvalue, report))) {
|
|
if (STRNEQ(cmd->def->name, "help"))
|
|
goto out;
|
|
|
|
/* ignore spurious arguments for 'help' command */
|
|
g_clear_pointer(&optionvalue, g_free);
|
|
state = VSH_CMD_PARSER_STATE_COMMAND;
|
|
} else {
|
|
state = VSH_CMD_PARSER_STATE_ASSIGN_OPT;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case VSH_CMD_PARSER_STATE_ASSIGN_OPT:
|
|
/* Parameter for a boolean was passed via --boolopt=val */
|
|
if (tkdata && opt->def->type == VSH_OT_BOOL) {
|
|
if (report)
|
|
vshError(ctl, _("invalid '=' after option --%1$s"),
|
|
opt->def->name);
|
|
goto out;
|
|
}
|
|
|
|
vshCmdOptAssign(ctl, cmd, opt, tkdata, report);
|
|
opt = NULL;
|
|
state = VSH_CMD_PARSER_STATE_COMMAND;
|
|
break;
|
|
|
|
case VSH_CMD_PARSER_STATE_POSITIONAL_ONLY:
|
|
state = VSH_CMD_PARSER_STATE_POSITIONAL_ONLY;
|
|
|
|
if (vshCmdOptAssignPositional(ctl, cmd, tkdata, report) < 0)
|
|
goto out;
|
|
break;
|
|
}
|
|
}
|
|
|
|
out:
|
|
|
|
if (partial) {
|
|
/* When parsing a command for command completion, the last processed
|
|
* command or the one being currently parsed */
|
|
if (cmd) {
|
|
*partial = g_steal_pointer(&cmd);
|
|
} else if (cmds == cmds_last) {
|
|
*partial = g_steal_pointer(&cmds);
|
|
} else {
|
|
/* break the last command out of the linked list and let the rest be freed */
|
|
vshCmd *nc;
|
|
|
|
for (nc = cmds; nc; nc = nc->next) {
|
|
if (nc->next == cmds_last) {
|
|
nc->next = NULL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
*partial = cmds_last;
|
|
}
|
|
} else {
|
|
/* for normal command parsing use the whole parsed command list, but
|
|
* only on success */
|
|
if (ret == true) {
|
|
ctl->cmd = g_steal_pointer(&cmds);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/* --------------------
|
|
* Command argv parsing
|
|
* --------------------
|
|
*/
|
|
|
|
static vshCommandToken ATTRIBUTE_NONNULL(2) ATTRIBUTE_NONNULL(3)
|
|
vshCommandArgvGetArg(vshControl *ctl G_GNUC_UNUSED,
|
|
vshCommandParser *parser,
|
|
char **res,
|
|
bool report G_GNUC_UNUSED)
|
|
{
|
|
if (parser->arg_pos == parser->arg_end) {
|
|
*res = NULL;
|
|
return VSH_TK_END;
|
|
}
|
|
|
|
*res = g_strdup(*parser->arg_pos);
|
|
parser->arg_pos++;
|
|
return VSH_TK_ARG;
|
|
}
|
|
|
|
bool
|
|
vshCommandArgvParse(vshControl *ctl, int nargs, char **argv)
|
|
{
|
|
vshCommandParser parser = { 0 };
|
|
|
|
if (nargs <= 0)
|
|
return false;
|
|
|
|
parser.arg_pos = argv;
|
|
parser.arg_end = argv + nargs;
|
|
parser.getNextArg = vshCommandArgvGetArg;
|
|
return vshCommandParse(ctl, &parser, NULL);
|
|
}
|
|
|
|
/* ----------------------
|
|
* Command string parsing
|
|
* ----------------------
|
|
*/
|
|
|
|
static vshCommandToken ATTRIBUTE_NONNULL(2) ATTRIBUTE_NONNULL(3)
|
|
vshCommandStringGetArg(vshControl *ctl, vshCommandParser *parser, char **res,
|
|
bool report)
|
|
{
|
|
bool single_quote = false;
|
|
bool double_quote = false;
|
|
char *p = parser->pos;
|
|
char *q = g_strdup(p);
|
|
|
|
*res = q;
|
|
|
|
while (*p == ' ' || *p == '\t' || (*p == '\\' && p[1] == '\n'))
|
|
p += 1 + (*p == '\\');
|
|
|
|
if (*p == '\0')
|
|
return VSH_TK_END;
|
|
if (*p == ';' || *p == '\n') {
|
|
parser->pos = ++p; /* = \0 or begin of next command */
|
|
return VSH_TK_SUBCMD_END;
|
|
}
|
|
if (*p == '#') { /* Argument starting with # is comment to end of line */
|
|
while (*p && *p != '\n')
|
|
p++;
|
|
parser->pos = p + !!*p;
|
|
return VSH_TK_SUBCMD_END;
|
|
}
|
|
|
|
while (*p) {
|
|
/* end of token is blank space or ';' */
|
|
if (!double_quote && !single_quote &&
|
|
(*p == ' ' || *p == '\t' || *p == ';' || *p == '\n'))
|
|
break;
|
|
|
|
if (!double_quote && *p == '\'') { /* single quote */
|
|
single_quote = !single_quote;
|
|
p++;
|
|
continue;
|
|
} else if (!single_quote && *p == '\\') { /* escape */
|
|
/*
|
|
* The same as in shell, a \ in "" is an escaper,
|
|
* but a \ in '' is not an escaper.
|
|
*/
|
|
p++;
|
|
if (*p == '\0') {
|
|
if (report)
|
|
vshError(ctl, "%s", _("dangling \\"));
|
|
return VSH_TK_ERROR;
|
|
} else if (*p == '\n') {
|
|
/* Elide backslash-newline entirely */
|
|
p++;
|
|
continue;
|
|
}
|
|
} else if (!single_quote && *p == '"') { /* double quote */
|
|
double_quote = !double_quote;
|
|
p++;
|
|
continue;
|
|
}
|
|
|
|
*q++ = *p++;
|
|
}
|
|
|
|
if (double_quote) {
|
|
/* We have seen a double quote, but not it's companion
|
|
* ending. It's valid though, in case when we're called
|
|
* from completer (report = false), but it's not valid
|
|
* when parsing real command (report= true). */
|
|
if (report) {
|
|
vshError(ctl, "%s", _("missing \""));
|
|
return VSH_TK_ERROR;
|
|
}
|
|
}
|
|
|
|
*q = '\0';
|
|
parser->pos = p;
|
|
return VSH_TK_ARG;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshCommandStringParse:
|
|
* @ctl virsh control structure
|
|
* @cmdstr: string to parse
|
|
* @partial: store partially parsed command here
|
|
*
|
|
* Parse given string @cmdstr as a command and store it under
|
|
* @ctl->cmd. For readline completion, if @partial is not NULL on
|
|
* the input then errors in parsing are ignored (because user is
|
|
* still in progress of writing the command string) and partially
|
|
* parsed command is stored at *@partial (caller has to free it
|
|
* afterwards).
|
|
*/
|
|
bool
|
|
vshCommandStringParse(vshControl *ctl,
|
|
char *cmdstr,
|
|
vshCmd **partial)
|
|
{
|
|
vshCommandParser parser = { 0 };
|
|
|
|
if (cmdstr == NULL || *cmdstr == '\0')
|
|
return false;
|
|
|
|
parser.pos = cmdstr;
|
|
parser.getNextArg = vshCommandStringGetArg;
|
|
return vshCommandParse(ctl, &parser, partial);
|
|
}
|
|
|
|
/**
|
|
* virshCommandOptTimeoutToMs:
|
|
* @ctl virsh control structure
|
|
* @cmd command reference
|
|
* @timeout result
|
|
*
|
|
* Parse an optional --timeout parameter in seconds, but store the
|
|
* value of the timeout in milliseconds.
|
|
* See vshCommandOptInt()
|
|
*/
|
|
int
|
|
vshCommandOptTimeoutToMs(vshControl *ctl, const vshCmd *cmd, int *timeout)
|
|
{
|
|
int ret;
|
|
unsigned int utimeout;
|
|
|
|
if ((ret = vshCommandOptUInt(ctl, cmd, "timeout", &utimeout)) <= 0)
|
|
return ret;
|
|
|
|
/* Ensure that the timeout is not zero and that we can convert
|
|
* it from seconds to milliseconds without overflowing. */
|
|
if (utimeout == 0 || utimeout > INT_MAX / 1000) {
|
|
vshError(ctl,
|
|
_("Numeric value '%1$u' for <%2$s> option is malformed or out of range"),
|
|
utimeout,
|
|
"timeout");
|
|
ret = -1;
|
|
} else {
|
|
*timeout = ((int) utimeout) * 1000;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/* ---------------
|
|
* Misc utils
|
|
* ---------------
|
|
*/
|
|
|
|
/* Return a non-NULL string representation of a typed parameter; exit on
|
|
* unknown type. */
|
|
char *
|
|
vshGetTypedParamValue(vshControl *ctl, virTypedParameterPtr item)
|
|
{
|
|
switch (item->type) {
|
|
case VIR_TYPED_PARAM_INT:
|
|
return g_strdup_printf("%d", item->value.i);
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_UINT:
|
|
return g_strdup_printf("%u", item->value.ui);
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_LLONG:
|
|
return g_strdup_printf("%lld", item->value.l);
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_ULLONG:
|
|
return g_strdup_printf("%llu", item->value.ul);
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_DOUBLE:
|
|
return g_strdup_printf("%f", item->value.d);
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_BOOLEAN:
|
|
return g_strdup(item->value.b ? _("yes") : _("no"));
|
|
break;
|
|
|
|
case VIR_TYPED_PARAM_STRING:
|
|
return g_strdup(item->value.s);
|
|
break;
|
|
|
|
default:
|
|
vshError(ctl, _("unimplemented parameter type %1$d"), item->type);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
|
|
void
|
|
vshDebug(vshControl *ctl, int level, const char *format, ...)
|
|
{
|
|
va_list ap;
|
|
g_autofree char *str = NULL;
|
|
|
|
/* Aligning log levels to that of libvirt.
|
|
* Traces with levels >= user-specified-level
|
|
* gets logged into file
|
|
*/
|
|
if (level < ctl->debug)
|
|
return;
|
|
|
|
va_start(ap, format);
|
|
vshOutputLogFile(ctl, level, format, ap);
|
|
va_end(ap);
|
|
|
|
va_start(ap, format);
|
|
str = g_strdup_vprintf(format, ap);
|
|
va_end(ap);
|
|
fputs(str, stdout);
|
|
fflush(stdout);
|
|
}
|
|
|
|
|
|
void
|
|
vshPrintVa(vshControl *ctl G_GNUC_UNUSED,
|
|
const char *format,
|
|
va_list ap)
|
|
{
|
|
g_autofree char *str = NULL;
|
|
|
|
str = g_strdup_vprintf(format, ap);
|
|
fputs(str, stdout);
|
|
fflush(stdout);
|
|
}
|
|
|
|
|
|
void
|
|
vshPrintExtra(vshControl *ctl,
|
|
const char *format,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
|
|
if (ctl && ctl->quiet)
|
|
return;
|
|
|
|
va_start(ap, format);
|
|
vshPrintVa(ctl, format, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
|
|
void
|
|
vshPrint(vshControl *ctl,
|
|
const char *format,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vshPrintVa(ctl, format, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
|
|
bool
|
|
vshTTYIsInterruptCharacter(vshControl *ctl G_GNUC_UNUSED,
|
|
const char chr G_GNUC_UNUSED)
|
|
{
|
|
#ifndef WIN32
|
|
if (ctl->istty &&
|
|
ctl->termattr.c_cc[VINTR] == chr)
|
|
return true;
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool
|
|
vshTTYAvailable(vshControl *ctl)
|
|
{
|
|
return ctl->istty;
|
|
}
|
|
|
|
|
|
int
|
|
vshTTYDisableInterrupt(vshControl *ctl G_GNUC_UNUSED)
|
|
{
|
|
#ifndef WIN32
|
|
struct termios termset = ctl->termattr;
|
|
|
|
if (!ctl->istty)
|
|
return -1;
|
|
|
|
/* check if we need to set the terminal */
|
|
if (termset.c_cc[VINTR] == _POSIX_VDISABLE)
|
|
return 0;
|
|
|
|
termset.c_cc[VINTR] = _POSIX_VDISABLE;
|
|
termset.c_lflag &= ~ICANON;
|
|
|
|
if (tcsetattr(STDIN_FILENO, TCSANOW, &termset) < 0)
|
|
return -1;
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
vshTTYRestore(vshControl *ctl G_GNUC_UNUSED)
|
|
{
|
|
#ifndef WIN32
|
|
if (!ctl->istty)
|
|
return 0;
|
|
|
|
if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &ctl->termattr) < 0)
|
|
return -1;
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
vshTTYMakeRaw(vshControl *ctl G_GNUC_UNUSED,
|
|
bool report_errors G_GNUC_UNUSED)
|
|
{
|
|
#ifndef WIN32
|
|
struct termios rawattr = ctl->termattr;
|
|
|
|
|
|
if (!ctl->istty) {
|
|
if (report_errors) {
|
|
vshError(ctl, "%s",
|
|
_("unable to make terminal raw: console isn't a tty"));
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
cfmakeraw(&rawattr);
|
|
|
|
if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &rawattr) < 0) {
|
|
if (report_errors)
|
|
vshError(ctl, _("unable to set tty attributes: %1$s"),
|
|
g_strerror(errno));
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
void
|
|
vshError(vshControl *ctl, const char *format, ...)
|
|
{
|
|
va_list ap;
|
|
g_autofree char *str = NULL;
|
|
|
|
if (ctl != NULL) {
|
|
va_start(ap, format);
|
|
vshOutputLogFile(ctl, VSH_ERR_ERROR, format, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
/* Most output is to stdout, but if someone ran virsh 2>&1, then
|
|
* printing to stderr will not interleave correctly with stdout
|
|
* unless we flush between every transition between streams. */
|
|
fflush(stdout);
|
|
fputs(_("error: "), stderr);
|
|
|
|
va_start(ap, format);
|
|
str = g_strdup_vprintf(format, ap);
|
|
va_end(ap);
|
|
|
|
fprintf(stderr, "%s\n", NULLSTR(str));
|
|
fflush(stderr);
|
|
}
|
|
|
|
|
|
void
|
|
vshEventLoop(void *opaque)
|
|
{
|
|
vshControl *ctl = opaque;
|
|
|
|
while (1) {
|
|
bool quit = false;
|
|
VIR_WITH_MUTEX_LOCK_GUARD(&ctl->lock) {
|
|
quit = ctl->quit;
|
|
}
|
|
|
|
if (quit)
|
|
break;
|
|
|
|
if (virEventRunDefaultImpl() < 0)
|
|
vshReportError(ctl);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Helpers for waiting for a libvirt event.
|
|
*/
|
|
|
|
/* We want to use SIGINT to cancel a wait; but as signal handlers
|
|
* don't have an opaque argument, we have to use static storage. */
|
|
#ifndef WIN32
|
|
static int vshEventFd = -1;
|
|
static struct sigaction vshEventOldAction;
|
|
|
|
|
|
/* Signal handler installed in vshEventStart, removed in vshEventCleanup. */
|
|
static void
|
|
vshEventInt(int sig G_GNUC_UNUSED,
|
|
siginfo_t *siginfo G_GNUC_UNUSED,
|
|
void *context G_GNUC_UNUSED)
|
|
{
|
|
char reason = VSH_EVENT_INTERRUPT;
|
|
if (vshEventFd >= 0)
|
|
ignore_value(safewrite(vshEventFd, &reason, 1));
|
|
}
|
|
#endif /* !WIN32 */
|
|
|
|
|
|
/* Event loop handler used to limit length of waiting for any other event. */
|
|
void
|
|
vshEventTimeout(int timer G_GNUC_UNUSED,
|
|
void *opaque)
|
|
{
|
|
vshControl *ctl = opaque;
|
|
char reason = VSH_EVENT_TIMEOUT;
|
|
|
|
if (ctl->eventPipe[1] >= 0)
|
|
ignore_value(safewrite(ctl->eventPipe[1], &reason, 1));
|
|
}
|
|
|
|
|
|
/**
|
|
* vshEventStart:
|
|
* @ctl vsh command struct
|
|
* @timeout_ms max wait time in milliseconds, or 0 for indefinite
|
|
*
|
|
* Set up a wait for a libvirt event. The wait can be canceled by
|
|
* SIGINT or by calling vshEventDone() in your event handler. If
|
|
* @timeout_ms is positive, the wait will also end if the timeout
|
|
* expires. Call vshEventWait() to block the main thread (the event
|
|
* handler runs in the event loop thread). When done (including if
|
|
* there was an error registering for an event), use vshEventCleanup()
|
|
* to quit waiting. Returns 0 on success, -1 on failure. */
|
|
int
|
|
vshEventStart(vshControl *ctl, int timeout_ms)
|
|
{
|
|
#ifndef WIN32
|
|
struct sigaction action;
|
|
assert(vshEventFd == -1);
|
|
#endif /* !WIN32 */
|
|
|
|
assert(ctl->eventPipe[0] == -1 && ctl->eventPipe[1] == -1 &&
|
|
ctl->eventTimerId >= 0);
|
|
if (virPipe(ctl->eventPipe) < 0) {
|
|
vshSaveLibvirtError();
|
|
vshReportError(ctl);
|
|
return -1;
|
|
}
|
|
|
|
#ifndef WIN32
|
|
vshEventFd = ctl->eventPipe[1];
|
|
|
|
action.sa_sigaction = vshEventInt;
|
|
action.sa_flags = SA_SIGINFO;
|
|
sigemptyset(&action.sa_mask);
|
|
sigaction(SIGINT, &action, &vshEventOldAction);
|
|
#endif /* !WIN32 */
|
|
|
|
if (timeout_ms)
|
|
virEventUpdateTimeout(ctl->eventTimerId, timeout_ms);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshEventDone:
|
|
* @ctl vsh command struct
|
|
*
|
|
* Call this from an event callback to let the main thread quit
|
|
* blocking on further events.
|
|
*/
|
|
void
|
|
vshEventDone(vshControl *ctl)
|
|
{
|
|
char reason = VSH_EVENT_DONE;
|
|
|
|
if (ctl->eventPipe[1] >= 0)
|
|
ignore_value(safewrite(ctl->eventPipe[1], &reason, 1));
|
|
}
|
|
|
|
|
|
/**
|
|
* vshEventWait:
|
|
* @ctl vsh command struct
|
|
*
|
|
* Call this in the main thread after calling vshEventStart() then
|
|
* registering for one or more events. This call will block until
|
|
* SIGINT, the timeout registered at the start, or until one of your
|
|
* event handlers calls vshEventDone(). Returns an enum VSH_EVENT_*
|
|
* stating how the wait concluded, or -1 on error.
|
|
*/
|
|
int
|
|
vshEventWait(vshControl *ctl)
|
|
{
|
|
char buf;
|
|
int rv;
|
|
|
|
assert(ctl->eventPipe[0] >= 0);
|
|
while ((rv = read(ctl->eventPipe[0], &buf, 1)) < 0 && errno == EINTR);
|
|
if (rv != 1) {
|
|
if (!rv)
|
|
errno = EPIPE;
|
|
vshError(ctl, _("failed to determine loop exit status: %1$s"),
|
|
g_strerror(errno));
|
|
return -1;
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshEventCleanup:
|
|
* @ctl vsh control struct
|
|
*
|
|
* Call at the end of any function that has used vshEventStart(), to
|
|
* tear down any remaining SIGINT or timeout handlers.
|
|
*/
|
|
void
|
|
vshEventCleanup(vshControl *ctl)
|
|
{
|
|
#ifndef WIN32
|
|
if (vshEventFd >= 0) {
|
|
sigaction(SIGINT, &vshEventOldAction, NULL);
|
|
vshEventFd = -1;
|
|
}
|
|
#endif /* !WIN32 */
|
|
VIR_FORCE_CLOSE(ctl->eventPipe[0]);
|
|
VIR_FORCE_CLOSE(ctl->eventPipe[1]);
|
|
virEventUpdateTimeout(ctl->eventTimerId, -1);
|
|
}
|
|
|
|
#ifdef O_SYNC
|
|
# define LOGFILE_FLAGS (O_WRONLY | O_APPEND | O_CREAT | O_SYNC)
|
|
#else
|
|
# define LOGFILE_FLAGS (O_WRONLY | O_APPEND | O_CREAT)
|
|
#endif
|
|
|
|
/**
|
|
* vshOpenLogFile:
|
|
*
|
|
* Open log file.
|
|
*/
|
|
void
|
|
vshOpenLogFile(vshControl *ctl)
|
|
{
|
|
if (ctl->logfile == NULL)
|
|
return;
|
|
|
|
if ((ctl->log_fd = open(ctl->logfile, LOGFILE_FLAGS, FILE_MODE)) < 0) {
|
|
vshError(ctl, "%s",
|
|
_("failed to open the log file. check the log file path"));
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* vshOutputLogFile:
|
|
*
|
|
* Outputting an error to log file.
|
|
*/
|
|
void
|
|
vshOutputLogFile(vshControl *ctl, int log_level, const char *msg_format,
|
|
va_list ap)
|
|
{
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
g_autofree char *str = NULL;
|
|
size_t len;
|
|
const char *lvl = "";
|
|
g_autoptr(GDateTime) now = g_date_time_new_now_local();
|
|
g_autofree gchar *nowstr = NULL;
|
|
|
|
if (ctl->log_fd == -1)
|
|
return;
|
|
|
|
/**
|
|
* create log format
|
|
*
|
|
* [YYYY.MM.DD HH:MM:SS SIGNATURE PID] LOG_LEVEL message
|
|
*/
|
|
nowstr = g_date_time_format(now, "%Y.%m.%d %H:%M:%S");
|
|
virBufferAsprintf(&buf, "[%s %s %d] ",
|
|
nowstr,
|
|
ctl->progname,
|
|
(int) getpid());
|
|
switch (log_level) {
|
|
case VSH_ERR_DEBUG:
|
|
lvl = LVL_DEBUG;
|
|
break;
|
|
case VSH_ERR_INFO:
|
|
lvl = LVL_INFO;
|
|
break;
|
|
case VSH_ERR_NOTICE:
|
|
lvl = LVL_INFO;
|
|
break;
|
|
case VSH_ERR_WARNING:
|
|
lvl = LVL_WARNING;
|
|
break;
|
|
case VSH_ERR_ERROR:
|
|
lvl = LVL_ERROR;
|
|
break;
|
|
default:
|
|
lvl = LVL_DEBUG;
|
|
break;
|
|
}
|
|
virBufferAsprintf(&buf, "%s ", lvl);
|
|
virBufferVasprintf(&buf, msg_format, ap);
|
|
virBufferTrim(&buf, "\n");
|
|
virBufferAddChar(&buf, '\n');
|
|
|
|
str = virBufferContentAndReset(&buf);
|
|
len = strlen(str);
|
|
|
|
/* write log */
|
|
if (safewrite(ctl->log_fd, str, len) < 0)
|
|
goto error;
|
|
|
|
return;
|
|
|
|
error:
|
|
vshCloseLogFile(ctl);
|
|
vshError(ctl, "%s", _("failed to write the log file"));
|
|
}
|
|
|
|
/**
|
|
* vshCloseLogFile:
|
|
*
|
|
* Close log file.
|
|
*/
|
|
void
|
|
vshCloseLogFile(vshControl *ctl)
|
|
{
|
|
/* log file close */
|
|
if (VIR_CLOSE(ctl->log_fd) < 0) {
|
|
vshError(ctl, _("%1$s: failed to write log file: %2$s"),
|
|
ctl->logfile ? ctl->logfile : "?",
|
|
g_strerror(errno));
|
|
}
|
|
|
|
g_clear_pointer(&ctl->logfile, g_free);
|
|
}
|
|
|
|
#ifndef WIN32
|
|
static void G_GNUC_NULL_TERMINATED
|
|
vshPrintRaw(vshControl *ctl, ...)
|
|
{
|
|
va_list ap;
|
|
char *key;
|
|
|
|
va_start(ap, ctl);
|
|
while ((key = va_arg(ap, char *)) != NULL)
|
|
vshPrint(ctl, "%s\r\n", key);
|
|
va_end(ap);
|
|
}
|
|
|
|
/**
|
|
* vshAskReedit:
|
|
* @msg: Question to ask user
|
|
*
|
|
* Ask user if he wants to return to previously
|
|
* edited file.
|
|
*
|
|
* Returns 'y' if he wants to
|
|
* 'n' if he doesn't want to
|
|
* 'i' if he wants to try defining it again while ignoring validation
|
|
* 'f' if he forcibly wants to
|
|
* -1 on error
|
|
* 0 otherwise
|
|
*/
|
|
int
|
|
vshAskReedit(vshControl *ctl, const char *msg, bool relax_avail)
|
|
{
|
|
int c = -1;
|
|
|
|
if (!isatty(STDIN_FILENO))
|
|
return -1;
|
|
|
|
vshReportError(ctl);
|
|
|
|
if (vshTTYMakeRaw(ctl, false) < 0)
|
|
return -1;
|
|
|
|
while (true) {
|
|
vshPrint(ctl, "\r%s %s %s: ", msg, _("Try again?"),
|
|
relax_avail ? "[y,n,i,f,?]" : "[y,n,f,?]");
|
|
c = g_ascii_tolower(getchar());
|
|
|
|
if (c == '?') {
|
|
vshPrintRaw(ctl,
|
|
"",
|
|
_("y - yes, start editor again"),
|
|
_("n - no, throw away my changes"),
|
|
NULL);
|
|
|
|
if (relax_avail) {
|
|
vshPrintRaw(ctl,
|
|
_("i - turn off validation and try to redefine again"),
|
|
NULL);
|
|
}
|
|
|
|
vshPrintRaw(ctl,
|
|
_("f - force, try to redefine again"),
|
|
_("? - print this help"),
|
|
NULL);
|
|
continue;
|
|
} else if (c == 'y' || c == 'n' || c == 'f' ||
|
|
(relax_avail && c == 'i')) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
vshTTYRestore(ctl);
|
|
|
|
vshPrint(ctl, "\r\n");
|
|
return c;
|
|
}
|
|
#else /* WIN32 */
|
|
int
|
|
vshAskReedit(vshControl *ctl,
|
|
const char *msg G_GNUC_UNUSED,
|
|
bool relax_avail G_GNUC_UNUSED)
|
|
{
|
|
vshDebug(ctl, VSH_ERR_WARNING, "%s", _("This function is not supported on WIN32 platform"));
|
|
return 0;
|
|
}
|
|
#endif /* WIN32 */
|
|
|
|
|
|
void
|
|
vshEditUnlinkTempfile(char *file)
|
|
{
|
|
if (!file)
|
|
return;
|
|
|
|
ignore_value(unlink(file));
|
|
g_free(file);
|
|
}
|
|
|
|
|
|
/* Common code for the edit / net-edit / pool-edit functions which follow. */
|
|
char *
|
|
vshEditWriteToTempFile(vshControl *ctl, const char *doc)
|
|
{
|
|
g_autofree char *filename = NULL;
|
|
g_autoptr(vshTempFile) ret = NULL;
|
|
const char *tmpdir;
|
|
VIR_AUTOCLOSE fd = -1;
|
|
|
|
tmpdir = getenv("TMPDIR");
|
|
if (!tmpdir)
|
|
tmpdir = "/tmp";
|
|
filename = g_strdup_printf("%s/virshXXXXXX.xml", tmpdir);
|
|
fd = g_mkstemp_full(filename, O_RDWR | O_CLOEXEC, S_IRUSR | S_IWUSR);
|
|
if (fd == -1) {
|
|
vshError(ctl, _("g_mkstemp_full: failed to create temporary file: %1$s"),
|
|
g_strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
ret = g_steal_pointer(&filename);
|
|
|
|
if (safewrite(fd, doc, strlen(doc)) == -1) {
|
|
vshError(ctl, _("write: %1$s: failed to write to temporary file: %2$s"),
|
|
ret, g_strerror(errno));
|
|
return NULL;
|
|
}
|
|
if (VIR_CLOSE(fd) < 0) {
|
|
vshError(ctl, _("close: %1$s: failed to write or close temporary file: %2$s"),
|
|
ret, g_strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
/* Temporary filename: caller frees. */
|
|
return g_steal_pointer(&ret);
|
|
}
|
|
|
|
/* Characters permitted in $EDITOR environment variable and temp filename. */
|
|
#define ACCEPTED_CHARS \
|
|
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-/_.:@"
|
|
|
|
/* Hard-code default editor used as a fallback if not configured by
|
|
* VISUAL or EDITOR environment variables. */
|
|
#define DEFAULT_EDITOR "vi"
|
|
|
|
int
|
|
vshEditFile(vshControl *ctl, const char *filename)
|
|
{
|
|
const char *editor;
|
|
g_autoptr(virCommand) cmd = NULL;
|
|
int outfd = STDOUT_FILENO;
|
|
int errfd = STDERR_FILENO;
|
|
|
|
editor = getenv("VISUAL");
|
|
if (!editor)
|
|
editor = getenv("EDITOR");
|
|
if (!editor)
|
|
editor = DEFAULT_EDITOR;
|
|
|
|
/* Check that filename doesn't contain shell meta-characters, and
|
|
* if it does, refuse to run. Follow the Unix conventions for
|
|
* EDITOR: the user can intentionally specify command options, so
|
|
* we don't protect any shell metacharacters there. Lots more
|
|
* than virsh will misbehave if EDITOR has bogus contents (which
|
|
* is why sudo scrubs it by default). Conversely, if the editor
|
|
* is safe, we can run it directly rather than wasting a shell.
|
|
*/
|
|
if (strspn(editor, ACCEPTED_CHARS) != strlen(editor)) {
|
|
if (strspn(filename, ACCEPTED_CHARS) != strlen(filename)) {
|
|
vshError(ctl,
|
|
_("%1$s: temporary filename contains shell meta or other unacceptable characters (is $TMPDIR wrong?)"),
|
|
filename);
|
|
return -1;
|
|
}
|
|
cmd = virCommandNewArgList("sh", "-c", NULL);
|
|
virCommandAddArgFormat(cmd, "%s %s", editor, filename);
|
|
} else {
|
|
cmd = virCommandNewArgList(editor, filename, NULL);
|
|
}
|
|
|
|
virCommandSetInputFD(cmd, STDIN_FILENO);
|
|
virCommandSetOutputFD(cmd, &outfd);
|
|
virCommandSetErrorFD(cmd, &errfd);
|
|
if (virCommandRunAsync(cmd, NULL) < 0 ||
|
|
virCommandWait(cmd, NULL) < 0) {
|
|
vshReportError(ctl);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
char *
|
|
vshEditReadBackFile(vshControl *ctl, const char *filename)
|
|
{
|
|
char *ret;
|
|
|
|
if (virFileReadAll(filename, VSH_MAX_XML_FILE, &ret) == -1) {
|
|
vshError(ctl,
|
|
_("%1$s: failed to read temporary file: %2$s"),
|
|
filename, g_strerror(errno));
|
|
return NULL;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
vshEditString(vshControl *ctl,
|
|
char **output,
|
|
const char *string)
|
|
{
|
|
g_autoptr(vshTempFile) tmp = NULL;
|
|
char *tmpstr;
|
|
|
|
/* Create and open the temporary file. */
|
|
if (!(tmp = vshEditWriteToTempFile(ctl, string)))
|
|
return -1;
|
|
|
|
/* Start the editor. */
|
|
if (vshEditFile(ctl, tmp) == -1)
|
|
return -1;
|
|
|
|
/* Read back the edited file. */
|
|
if (!(*output = vshEditReadBackFile(ctl, tmp)))
|
|
return -1;
|
|
|
|
/* strip a possible newline at the end of file; some
|
|
* editors enforce a newline, this makes editing
|
|
* more convenient */
|
|
if ((tmpstr = strrchr(*output, '\n')) &&
|
|
*(tmpstr+1) == '\0')
|
|
*tmpstr = '\0';
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Tree listing helpers. */
|
|
|
|
static int
|
|
vshTreePrintInternal(vshControl *ctl,
|
|
vshTreeLookup lookup,
|
|
void *opaque,
|
|
int num_devices,
|
|
int devid,
|
|
int lastdev,
|
|
bool root,
|
|
virBuffer *indent)
|
|
{
|
|
size_t i;
|
|
int nextlastdev = -1;
|
|
const char *dev = (lookup)(devid, false, opaque);
|
|
|
|
/* Print this device, with indent if not at root */
|
|
vshPrint(ctl, "%s%s%s\n", virBufferCurrentContent(indent),
|
|
root ? "" : "+- ", dev);
|
|
|
|
/* Update indent to show '|' or ' ' for child devices */
|
|
if (!root) {
|
|
virBufferAddChar(indent, devid == lastdev ? ' ' : '|');
|
|
virBufferAddChar(indent, ' ');
|
|
}
|
|
|
|
/* Determine the index of the last child device */
|
|
for (i = 0; i < num_devices; i++) {
|
|
const char *parent = (lookup)(i, true, opaque);
|
|
|
|
if (parent && STREQ(parent, dev))
|
|
nextlastdev = i;
|
|
}
|
|
|
|
/* If there is a child device, then print another blank line */
|
|
if (nextlastdev != -1)
|
|
vshPrint(ctl, "%s |\n", virBufferCurrentContent(indent));
|
|
|
|
/* Finally print all children */
|
|
virBufferAddLit(indent, " ");
|
|
for (i = 0; i < num_devices; i++) {
|
|
const char *parent = (lookup)(i, true, opaque);
|
|
|
|
if (parent && STREQ(parent, dev) &&
|
|
vshTreePrintInternal(ctl, lookup, opaque,
|
|
num_devices, i, nextlastdev,
|
|
false, indent) < 0)
|
|
return -1;
|
|
}
|
|
virBufferTrim(indent, " ");
|
|
|
|
/* If there was no child device, and we're the last in
|
|
* a list of devices, then print another blank line */
|
|
if (nextlastdev == -1 && devid == lastdev)
|
|
vshPrint(ctl, "%s\n", virBufferCurrentContent(indent));
|
|
|
|
if (!root)
|
|
virBufferTrimLen(indent, 2);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vshTreePrint(vshControl *ctl, vshTreeLookup lookup, void *opaque,
|
|
int num_devices, int devid)
|
|
{
|
|
int ret;
|
|
g_auto(virBuffer) indent = VIR_BUFFER_INITIALIZER;
|
|
|
|
ret = vshTreePrintInternal(ctl, lookup, opaque, num_devices,
|
|
devid, devid, true, &indent);
|
|
if (ret < 0)
|
|
vshError(ctl, "%s", _("Failed to complete tree listing"));
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* vshReadlineCommandGenerator:
|
|
*
|
|
* Generator function for command completion. Used also for completing the
|
|
* '--command' option of the 'help' command.
|
|
*
|
|
* Returns a string list of all commands, or NULL on failure.
|
|
*/
|
|
static char **
|
|
vshReadlineCommandGenerator(void)
|
|
{
|
|
size_t grp_list_index = 0;
|
|
const vshCmdGrp *grp;
|
|
size_t ret_size = 0;
|
|
g_auto(GStrv) ret = NULL;
|
|
|
|
grp = cmdGroups;
|
|
|
|
for (grp_list_index = 0; grp[grp_list_index].name; grp_list_index++) {
|
|
const vshCmdDef *cmds = grp[grp_list_index].commands;
|
|
size_t cmd_list_index;
|
|
|
|
for (cmd_list_index = 0; cmds[cmd_list_index].name; cmd_list_index++) {
|
|
const char *name = cmds[cmd_list_index].name;
|
|
|
|
if (cmds[cmd_list_index].alias ||
|
|
cmds[cmd_list_index].flags & VSH_CMD_FLAG_HIDDEN)
|
|
continue;
|
|
|
|
VIR_REALLOC_N(ret, ret_size + 2);
|
|
|
|
ret[ret_size] = g_strdup(name);
|
|
ret_size++;
|
|
/* Terminate the string list properly. */
|
|
ret[ret_size] = NULL;
|
|
}
|
|
}
|
|
|
|
return g_steal_pointer(&ret);
|
|
}
|
|
|
|
|
|
#if WITH_READLINE
|
|
|
|
/* -----------------
|
|
* Readline stuff
|
|
* -----------------
|
|
*/
|
|
|
|
|
|
static char **
|
|
vshReadlineOptionsGenerator(vshCmd *cmd)
|
|
{
|
|
size_t ret_size = 0;
|
|
g_auto(GStrv) ret = NULL;
|
|
vshCmdOpt *n;
|
|
|
|
for (n = cmd->opts; n && n->def; n++) {
|
|
/* Skip aliases, we do not report them in help output either. */
|
|
if (n->def->type == VSH_OT_ALIAS)
|
|
continue;
|
|
|
|
/* skip already populated single-instance arguments */
|
|
if (n->present && n->def->type != VSH_OT_ARGV)
|
|
continue;
|
|
|
|
VIR_REALLOC_N(ret, ret_size + 2);
|
|
|
|
ret[ret_size] = g_strdup_printf("--%s", n->def->name);
|
|
ret_size++;
|
|
/* Terminate the string list properly. */
|
|
ret[ret_size] = NULL;
|
|
}
|
|
|
|
return g_steal_pointer(&ret);
|
|
}
|
|
|
|
|
|
static int
|
|
vshCompleterFilter(char ***list,
|
|
const char *text)
|
|
{
|
|
char **newList = NULL;
|
|
size_t newList_len = 0;
|
|
size_t list_len;
|
|
size_t i;
|
|
|
|
if (!list || !*list)
|
|
return 0;
|
|
|
|
list_len = g_strv_length(*list);
|
|
newList = g_new0(char *, list_len + 1);
|
|
|
|
for (i = 0; i < list_len; i++) {
|
|
if (text && !STRPREFIX((*list)[i], text)) {
|
|
g_clear_pointer(&(*list)[i], g_free);
|
|
continue;
|
|
}
|
|
|
|
newList[newList_len] = g_steal_pointer(&(*list)[i]);
|
|
newList_len++;
|
|
}
|
|
|
|
newList = g_renew(char *, newList, newList_len + 1);
|
|
g_free(*list);
|
|
*list = newList;
|
|
return 0;
|
|
}
|
|
|
|
|
|
static char *
|
|
vshReadlineParse(const char *text, int state)
|
|
{
|
|
static char **list;
|
|
static size_t list_index;
|
|
char *ret = NULL;
|
|
|
|
/* Readline calls this function until NULL is returned. On
|
|
* the very first call @state is zero which means we should
|
|
* initialize those static variables above. On subsequent
|
|
* calls @state is non zero. */
|
|
if (!state) {
|
|
g_autoptr(vshCmd) partial = NULL;
|
|
const vshCmdDef *cmd = NULL;
|
|
g_autofree char *line = g_strdup(rl_line_buffer);
|
|
|
|
g_clear_pointer(&list, g_strfreev);
|
|
list_index = 0;
|
|
|
|
*(line + rl_point) = '\0';
|
|
|
|
vshCommandStringParse(NULL, line, &partial);
|
|
|
|
if (partial) {
|
|
cmd = partial->def;
|
|
partial->skipChecks = true;
|
|
}
|
|
|
|
if (cmd && STREQ(cmd->name, text)) {
|
|
/* Corner case - some commands share prefix (e.g.
|
|
* dump and dumpxml). If user typed 'dump<TAB><TAB>',
|
|
* then @text = "dump" and we want to offer command
|
|
* completion. If they typed 'dump <TAB><TAB>' then
|
|
* @text = "" (the space after the command) and we
|
|
* want to offer options completion for dump command.
|
|
*/
|
|
cmd = NULL;
|
|
}
|
|
|
|
if (!cmd) {
|
|
list = vshReadlineCommandGenerator();
|
|
} else {
|
|
bool complete_argument = false;
|
|
|
|
/* attempt completion only when:
|
|
- there is an argument
|
|
- it has the 'data' field filled
|
|
- it has a completer (rules out booleans)
|
|
*/
|
|
if (partial->lastopt && partial->lastopt->data && partial->lastopt->def->completer) {
|
|
/* Furthermore we want to do the completion only at the point of
|
|
* user's cursor. This is the case if:
|
|
* - value in 'data' is equal to 'text' (last component of the completed command)
|
|
* - value in 'data' is a space when 'text' is empty (quirk)
|
|
*/
|
|
if (STREQ_NULLABLE(partial->lastopt->data, text))
|
|
complete_argument = true;
|
|
|
|
if (STREQ_NULLABLE(partial->lastopt->data, " ") && *text == '\0')
|
|
complete_argument = true;
|
|
}
|
|
|
|
if (complete_argument) {
|
|
list = partial->lastopt->def->completer(autoCompleteOpaque,
|
|
partial,
|
|
partial->lastopt->def->completer_flags);
|
|
} else {
|
|
list = vshReadlineOptionsGenerator(partial);
|
|
}
|
|
}
|
|
|
|
/* Escape completions, if needed (i.e. argument
|
|
* we are completing wasn't started with a quote
|
|
* character). This also enables filtering done
|
|
* below to work properly. */
|
|
if (list &&
|
|
!rl_completion_quote_character) {
|
|
size_t i;
|
|
|
|
for (i = 0; list[i]; i++) {
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
virBufferEscape(&buf, '\\', " ", "%s", list[i]);
|
|
VIR_FREE(list[i]);
|
|
list[i] = virBufferContentAndReset(&buf);
|
|
}
|
|
}
|
|
|
|
/* For string list returned by completers we have to do
|
|
* filtering based on @text because completers returns all
|
|
* possible strings. */
|
|
if (vshCompleterFilter(&list, text) < 0)
|
|
goto cleanup;
|
|
}
|
|
|
|
if (list) {
|
|
ret = g_strdup(list[list_index]);
|
|
list_index++;
|
|
}
|
|
|
|
cleanup:
|
|
if (!ret) {
|
|
g_clear_pointer(&list, g_strfreev);
|
|
list_index = 0;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static char **
|
|
vshReadlineCompletion(const char *text,
|
|
int start G_GNUC_UNUSED,
|
|
int end G_GNUC_UNUSED)
|
|
{
|
|
return rl_completion_matches(text, vshReadlineParse);
|
|
}
|
|
|
|
|
|
static int
|
|
vshReadlineCharIsQuoted(char *line, int idx)
|
|
{
|
|
return idx > 0 &&
|
|
line[idx - 1] == '\\' &&
|
|
!vshReadlineCharIsQuoted(line, idx - 1);
|
|
}
|
|
|
|
|
|
# define HISTSIZE_MAX 500000
|
|
|
|
static int
|
|
vshReadlineInit(vshControl *ctl)
|
|
{
|
|
g_autofree char *userdir = NULL;
|
|
int max_history = 500;
|
|
g_autofree char *histsize_env = NULL;
|
|
const char *histsize_str = NULL;
|
|
const char *break_characters = " \t\n`@$><=;|&{(";
|
|
const char *quote_characters = "\"'";
|
|
|
|
/* initialize readline stuff only once */
|
|
if (autoCompleteOpaque)
|
|
return 0;
|
|
|
|
/* Opaque data for autocomplete callbacks. */
|
|
autoCompleteOpaque = ctl;
|
|
|
|
rl_readline_name = ctl->name;
|
|
|
|
/* Tell the completer that we want a crack first. */
|
|
rl_attempted_completion_function = vshReadlineCompletion;
|
|
|
|
rl_basic_word_break_characters = break_characters;
|
|
|
|
rl_completer_quote_characters = quote_characters;
|
|
rl_char_is_quoted_p = vshReadlineCharIsQuoted;
|
|
|
|
/* Stuff below is needed only for interactive mode. */
|
|
if (!ctl->imode) {
|
|
return 0;
|
|
}
|
|
|
|
histsize_env = g_strdup_printf("%s_HISTSIZE", ctl->env_prefix);
|
|
|
|
/* Limit the total size of the history buffer */
|
|
if ((histsize_str = getenv(histsize_env))) {
|
|
if (virStrToLong_i(histsize_str, NULL, 10, &max_history) < 0) {
|
|
vshError(ctl, _("Bad $%1$s value."), histsize_env);
|
|
return -1;
|
|
} else if (max_history > HISTSIZE_MAX || max_history < 0) {
|
|
vshError(ctl, _("$%1$s value should be between 0 and %2$d"),
|
|
histsize_env, HISTSIZE_MAX);
|
|
return -1;
|
|
}
|
|
}
|
|
stifle_history(max_history);
|
|
|
|
/* Prepare to read/write history from/to the
|
|
* $XDG_CACHE_HOME/virtshell/history file
|
|
*/
|
|
userdir = virGetUserCacheDirectory();
|
|
|
|
ctl->historydir = g_strdup_printf("%s/%s", userdir, ctl->name);
|
|
|
|
ctl->historyfile = g_strdup_printf("%s/history", ctl->historydir);
|
|
|
|
read_history(ctl->historyfile);
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
vshReadlineDeinit(vshControl *ctl)
|
|
{
|
|
if (ctl->historyfile != NULL) {
|
|
if (g_mkdir_with_parents(ctl->historydir, 0755) < 0 &&
|
|
errno != EEXIST) {
|
|
vshError(ctl, _("Failed to create '%1$s': %2$s"),
|
|
ctl->historydir, g_strerror(errno));
|
|
} else {
|
|
write_history(ctl->historyfile);
|
|
}
|
|
}
|
|
|
|
g_clear_pointer(&ctl->historydir, g_free);
|
|
g_clear_pointer(&ctl->historyfile, g_free);
|
|
|
|
/* Allow vshReadlineInit() to be called again. */
|
|
autoCompleteOpaque = NULL;
|
|
}
|
|
|
|
char *
|
|
vshReadline(vshControl *ctl G_GNUC_UNUSED, const char *prompt)
|
|
{
|
|
return readline(prompt);
|
|
}
|
|
|
|
void
|
|
vshReadlineHistoryAdd(const char *cmd)
|
|
{
|
|
return add_history(cmd);
|
|
}
|
|
|
|
#else /* !WITH_READLINE */
|
|
|
|
static int
|
|
vshReadlineInit(vshControl *ctl G_GNUC_UNUSED)
|
|
{
|
|
/* empty */
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
vshReadlineDeinit(vshControl *ctl G_GNUC_UNUSED)
|
|
{
|
|
/* empty */
|
|
}
|
|
|
|
char *
|
|
vshReadline(vshControl *ctl G_GNUC_UNUSED,
|
|
const char *prompt)
|
|
{
|
|
char line[1024];
|
|
char *r;
|
|
int len;
|
|
|
|
fputs(prompt, stdout);
|
|
fflush(stdout);
|
|
r = fgets(line, sizeof(line), stdin);
|
|
if (r == NULL) return NULL; /* EOF */
|
|
|
|
/* Chomp trailing \n */
|
|
len = strlen(r);
|
|
if (len > 0 && r[len-1] == '\n')
|
|
r[len-1] = '\0';
|
|
|
|
return g_strdup(r);
|
|
}
|
|
|
|
void
|
|
vshReadlineHistoryAdd(const char *cmd G_GNUC_UNUSED)
|
|
{
|
|
/* empty */
|
|
}
|
|
|
|
#endif /* !WITH_READLINE */
|
|
|
|
/*
|
|
* Initialize debug settings.
|
|
*/
|
|
static int
|
|
vshInitDebug(vshControl *ctl)
|
|
{
|
|
const char *debugEnv;
|
|
|
|
if (ctl->debug == VSH_DEBUG_DEFAULT) {
|
|
g_autofree char *env = g_strdup_printf("%s_DEBUG", ctl->env_prefix);
|
|
|
|
/* log level not set from commandline, check env variable */
|
|
debugEnv = getenv(env);
|
|
if (debugEnv) {
|
|
int debug;
|
|
if (virStrToLong_i(debugEnv, NULL, 10, &debug) < 0 ||
|
|
debug < VSH_ERR_DEBUG || debug > VSH_ERR_ERROR) {
|
|
vshError(ctl, _("%1$s_DEBUG not set with a valid numeric value"),
|
|
ctl->env_prefix);
|
|
} else {
|
|
ctl->debug = debug;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ctl->logfile == NULL) {
|
|
g_autofree char *env = g_strdup_printf("%s_LOG_FILE", ctl->env_prefix);
|
|
|
|
/* log file not set from cmdline */
|
|
debugEnv = getenv(env);
|
|
if (debugEnv && *debugEnv) {
|
|
ctl->logfile = g_strdup(debugEnv);
|
|
vshOpenLogFile(ctl);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Initialize global data
|
|
*/
|
|
bool
|
|
vshInit(vshControl *ctl, const vshCmdGrp *groups)
|
|
{
|
|
if (!ctl->hooks) {
|
|
vshError(ctl, "%s", _("client hooks cannot be NULL"));
|
|
return false;
|
|
}
|
|
|
|
if (!groups) {
|
|
vshError(ctl, "%s", _("command groups must be non-NULL"));
|
|
return false;
|
|
}
|
|
|
|
cmdGroups = groups;
|
|
|
|
if (vshInitDebug(ctl) < 0 ||
|
|
vshReadlineInit(ctl) < 0)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
vshInitReload(vshControl *ctl)
|
|
{
|
|
if (!cmdGroups) {
|
|
vshError(ctl, "%s", _("command groups is NULL run vshInit before reloading"));
|
|
return false;
|
|
}
|
|
|
|
if (vshInitDebug(ctl) < 0)
|
|
return false;
|
|
|
|
if (ctl->imode)
|
|
vshReadlineDeinit(ctl);
|
|
if (vshReadlineInit(ctl) < 0)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
vshDeinit(vshControl *ctl)
|
|
{
|
|
/* NB: Don't make calling of vshReadlineDeinit conditional on active
|
|
* interactive mode. */
|
|
vshReadlineDeinit(ctl);
|
|
vshCloseLogFile(ctl);
|
|
}
|
|
|
|
/* -----------------------------------------------
|
|
* Generic commands available to use by any client
|
|
* -----------------------------------------------
|
|
*/
|
|
|
|
static char **
|
|
vshCompleteHelpCommand(vshControl *ctl G_GNUC_UNUSED,
|
|
const vshCmd *cmd G_GNUC_UNUSED,
|
|
unsigned int completerflags G_GNUC_UNUSED)
|
|
{
|
|
return vshReadlineCommandGenerator();
|
|
}
|
|
|
|
|
|
const vshCmdOptDef opts_help[] = {
|
|
{.name = "command",
|
|
.type = VSH_OT_STRING,
|
|
.positional = true,
|
|
.completer = vshCompleteHelpCommand,
|
|
.help = N_("Prints global help, command specific help, or help for a group of related commands")
|
|
},
|
|
{.name = NULL}
|
|
};
|
|
|
|
const vshCmdInfo info_help = {
|
|
.help = N_("print help"),
|
|
.desc = N_("Prints global help, command specific help, or help for a\n"
|
|
" group of related commands"),
|
|
};
|
|
|
|
bool
|
|
cmdHelp(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
const vshCmdDef *def = NULL;
|
|
const vshCmdGrp *grp = NULL;
|
|
const char *name = NULL;
|
|
|
|
if (vshCommandOptStringQuiet(ctl, cmd, "command", &name) <= 0) {
|
|
vshPrint(ctl, "%s", _("Grouped commands:\n\n"));
|
|
|
|
for (grp = cmdGroups; grp->name; grp++) {
|
|
vshPrint(ctl, _(" %1$s (help keyword '%2$s'):\n"), grp->name,
|
|
grp->keyword);
|
|
|
|
for (def = grp->commands; def->name; def++) {
|
|
if (def->alias ||
|
|
def->flags & VSH_CMD_FLAG_HIDDEN)
|
|
continue;
|
|
vshPrint(ctl, " %-30s %s\n", def->name, _(def->info->help));
|
|
}
|
|
|
|
vshPrint(ctl, "\n");
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if ((def = vshCmddefSearch(name))) {
|
|
if (def->alias)
|
|
def = vshCmddefSearch(def->alias);
|
|
}
|
|
|
|
if (def) {
|
|
return vshCmddefHelp(def);
|
|
} else if ((grp = vshCmdGrpSearch(name))) {
|
|
return vshCmdGrpHelp(ctl, grp);
|
|
} else {
|
|
vshError(ctl, _("command or command group '%1$s' doesn't exist"), name);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
const vshCmdOptDef opts_cd[] = {
|
|
{.name = "dir",
|
|
.type = VSH_OT_STRING,
|
|
.positional = true,
|
|
.help = N_("directory to switch to (default: home or else root)")
|
|
},
|
|
{.name = NULL}
|
|
};
|
|
|
|
const vshCmdInfo info_cd = {
|
|
.help = N_("change the current directory"),
|
|
.desc = N_("Change the current directory."),
|
|
};
|
|
|
|
bool
|
|
cmdCd(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
const char *dir = NULL;
|
|
g_autofree char *dir_malloced = NULL;
|
|
|
|
if (vshCommandOptStringQuiet(ctl, cmd, "dir", &dir) <= 0)
|
|
dir = dir_malloced = virGetUserDirectory();
|
|
if (!dir)
|
|
dir = "/";
|
|
|
|
if (chdir(dir) == -1) {
|
|
vshError(ctl, _("cd: %1$s: %2$s"),
|
|
g_strerror(errno), dir);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
const vshCmdOptDef opts_echo[] = {
|
|
{.name = "shell",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("escape for shell use")
|
|
},
|
|
{.name = "xml",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("escape for XML use")
|
|
},
|
|
{.name = "split",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("split each argument on ','; ',,' is an escape sequence")
|
|
},
|
|
{.name = "err",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("output to stderr"),
|
|
},
|
|
{.name = "str",
|
|
.type = VSH_OT_ALIAS,
|
|
.help = "string"
|
|
},
|
|
{.name = "hi",
|
|
.type = VSH_OT_ALIAS,
|
|
.help = "string=hello"
|
|
},
|
|
{.name = "prefix",
|
|
.type = VSH_OT_STRING,
|
|
.help = N_("prefix the message")
|
|
},
|
|
{.name = "string",
|
|
.type = VSH_OT_ARGV,
|
|
.positional = true,
|
|
.help = N_("arguments to echo")
|
|
},
|
|
{.name = NULL}
|
|
};
|
|
|
|
const vshCmdInfo info_echo = {
|
|
.help = N_("echo arguments. Used for internal testing."),
|
|
.desc = N_("Echo back arguments, possibly with quoting. Used for internal testing."),
|
|
};
|
|
|
|
/* Exists mainly for debugging virsh, but also handy for adding back
|
|
* quotes for later evaluation.
|
|
*/
|
|
bool
|
|
cmdEcho(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
bool shell = vshCommandOptBool(cmd, "shell");
|
|
bool xml = vshCommandOptBool(cmd, "xml");
|
|
bool err = vshCommandOptBool(cmd, "err");
|
|
bool split = vshCommandOptBool(cmd, "split");
|
|
const char *prefix;
|
|
g_autofree char *arg = NULL;
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
const char **o;
|
|
|
|
VSH_EXCLUSIVE_OPTIONS_VAR(shell, xml);
|
|
VSH_EXCLUSIVE_OPTIONS_VAR(shell, split);
|
|
VSH_EXCLUSIVE_OPTIONS_VAR(xml, split);
|
|
|
|
ignore_value(vshCommandOptString(ctl, cmd, "prefix", &prefix));
|
|
|
|
if (prefix)
|
|
virBufferAsprintf(&buf, "%s ", prefix);
|
|
|
|
for (o = vshCommandOptArgv(cmd, "string"); o && *o; o++) {
|
|
const char *curr = *o;
|
|
|
|
if (xml) {
|
|
virBufferEscapeString(&buf, "%s", curr);
|
|
} else if (shell) {
|
|
virBufferEscapeShell(&buf, curr);
|
|
} else if (split) {
|
|
g_auto(GStrv) spl = NULL;
|
|
GStrv n;
|
|
|
|
vshStringToArray(curr, &spl);
|
|
|
|
for (n = spl; *n; n++)
|
|
virBufferAsprintf(&buf, "%s\n", *n);
|
|
} else {
|
|
virBufferAdd(&buf, curr, -1);
|
|
}
|
|
|
|
virBufferAddChar(&buf, ' ');
|
|
}
|
|
|
|
virBufferTrim(&buf, " ");
|
|
|
|
arg = virBufferContentAndReset(&buf);
|
|
if (arg) {
|
|
if (err)
|
|
vshError(ctl, "%s", arg);
|
|
else
|
|
vshPrint(ctl, "%s", arg);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
const vshCmdInfo info_pwd = {
|
|
.help = N_("print the current directory"),
|
|
.desc = N_("Print the current directory."),
|
|
};
|
|
|
|
bool
|
|
cmdPwd(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED)
|
|
{
|
|
g_autofree char *cwd = g_get_current_dir();
|
|
|
|
vshPrint(ctl, _("%1$s\n"), cwd);
|
|
|
|
return true;
|
|
}
|
|
|
|
const vshCmdInfo info_quit = {
|
|
.help = N_("quit this interactive terminal"),
|
|
.desc = "",
|
|
};
|
|
|
|
bool
|
|
cmdQuit(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED)
|
|
{
|
|
ctl->imode = false;
|
|
return true;
|
|
}
|
|
|
|
/* -----------------
|
|
* Command self-test
|
|
* ----------------- */
|
|
|
|
const vshCmdOptDef opts_selftest[] = {
|
|
{.name = "completers-missing",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("output the list of options which are missing completers")
|
|
},
|
|
{.name = "dump-help",
|
|
.type = VSH_OT_BOOL,
|
|
.help = N_("output help for each command")
|
|
},
|
|
{.name = NULL}
|
|
};
|
|
const vshCmdInfo info_selftest = {
|
|
.help = N_("internal command for testing virt shells"),
|
|
.desc = N_("internal use only"),
|
|
};
|
|
|
|
bool
|
|
cmdSelfTest(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
const vshCmdGrp *grp;
|
|
const vshCmdDef *def;
|
|
bool completers = vshCommandOptBool(cmd, "completers-missing");
|
|
bool dumphelp = vshCommandOptBool(cmd, "dump-help");
|
|
|
|
for (grp = cmdGroups; grp->name; grp++) {
|
|
for (def = grp->commands; def->name; def++) {
|
|
|
|
if (dumphelp && !def->alias)
|
|
vshCmddefHelp(def);
|
|
|
|
if (vshCmddefCheckInternals(ctl, def, completers) < 0)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* ----------------------
|
|
* Autocompletion command
|
|
* ---------------------- */
|
|
|
|
const vshCmdOptDef opts_complete[] = {
|
|
{.name = "string",
|
|
.type = VSH_OT_ARGV,
|
|
.positional = true,
|
|
.allowEmpty = true,
|
|
.help = N_("partial string to autocomplete")
|
|
},
|
|
{.name = NULL}
|
|
};
|
|
|
|
const vshCmdInfo info_complete = {
|
|
.help = N_("internal command for autocompletion"),
|
|
.desc = N_("internal use only"),
|
|
};
|
|
|
|
|
|
#ifdef WITH_READLINE
|
|
|
|
static virOnceControl vshCmdCompleteCloseStdinStderrOnce = VIR_ONCE_CONTROL_INITIALIZER;
|
|
|
|
static void
|
|
vshCmdCompleteCloseStdinStderr(void)
|
|
{
|
|
/* In non-interactive mode which is how the 'complete' command is intended
|
|
* to be used we need to ensure that any authentication callback will not
|
|
* attempt to read any input which would break the completion. Similarly,
|
|
* printing anything onto stderr should be avoided. */
|
|
int stdin_fileno = STDIN_FILENO;
|
|
int stderr_fileno = STDERR_FILENO;
|
|
|
|
VIR_FORCE_CLOSE(stdin_fileno);
|
|
VIR_FORCE_CLOSE(stderr_fileno);
|
|
}
|
|
|
|
|
|
bool
|
|
cmdComplete(vshControl *ctl, const vshCmd *cmd)
|
|
{
|
|
const vshClientHooks *hooks = ctl->hooks;
|
|
const char *lastArg = NULL;
|
|
const char **args = NULL;
|
|
char *old_rl_line_buffer = NULL;
|
|
g_auto(GStrv) matches = NULL;
|
|
char **iter;
|
|
|
|
/* The completer needs also the last component */
|
|
for (args = vshCommandOptArgv(cmd, "string"); args && *args; args++)
|
|
lastArg = *args;
|
|
|
|
/* This command is flagged VSH_CMD_FLAG_NOCONNECT because we
|
|
* need to prevent auth hooks reading any input. Therefore, we
|
|
* have to close stdin and then connect ourselves. */
|
|
if (!ctl->imode) {
|
|
if (virOnce(&vshCmdCompleteCloseStdinStderrOnce, vshCmdCompleteCloseStdinStderr) < 0)
|
|
return false;
|
|
}
|
|
|
|
if (!(hooks && hooks->connHandler && hooks->connHandler(ctl)))
|
|
return false;
|
|
|
|
vshReadlineInit(ctl);
|
|
|
|
old_rl_line_buffer = g_steal_pointer(&rl_line_buffer);
|
|
if (!(rl_line_buffer = g_strdup(vshCommandOptArgvString(cmd, "string"))))
|
|
rl_line_buffer = g_strdup("");
|
|
|
|
/* rl_point is current cursor position in rl_line_buffer.
|
|
* In our case it's at the end of the whole line. */
|
|
rl_point = strlen(rl_line_buffer);
|
|
|
|
matches = vshReadlineCompletion(lastArg, 0, 0);
|
|
g_clear_pointer(&rl_line_buffer, g_free);
|
|
rl_line_buffer = g_steal_pointer(&old_rl_line_buffer);
|
|
|
|
if (!matches)
|
|
return false;
|
|
|
|
for (iter = matches; *iter; iter++) {
|
|
if (iter == matches && matches[1])
|
|
continue;
|
|
printf("%s\n", *iter);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
#else /* !WITH_READLINE */
|
|
|
|
|
|
bool
|
|
cmdComplete(vshControl *ctl G_GNUC_UNUSED,
|
|
const vshCmd *cmd G_GNUC_UNUSED)
|
|
{
|
|
return false;
|
|
}
|
|
#endif /* !WITH_READLINE */
|