summaryrefslogtreecommitdiffstats
path: root/fish/fish.c
diff options
context:
space:
mode:
authorRichard W.M. Jones <rjones@redhat.com>2011-01-18 11:46:03 +0000
committerRichard W.M. Jones <rjones@redhat.com>2011-01-18 13:16:28 +0000
commitc3887285abbb3606822626ba396a51cb5df80d43 (patch)
tree6710d2b44d0cd9268ad78fc7718b7c4483a63575 /fish/fish.c
parent61a4db138e4f85033c655bf6b24df0949683c24c (diff)
downloadlibguestfs-c3887285abbb3606822626ba396a51cb5df80d43.tar.gz
libguestfs-c3887285abbb3606822626ba396a51cb5df80d43.tar.xz
libguestfs-c3887285abbb3606822626ba396a51cb5df80d43.zip
fish: <! cmd executes a shell command and inlines the resulting commands.
The new guestfish construct "<! cmd" executes the shell command "cmd", and then anything printed to stdout by "cmd" is parsed and executed as a guestfish command. This allows some very hairy shell scripting with guestfish.
Diffstat (limited to 'fish/fish.c')
-rw-r--r--fish/fish.c57
1 files changed, 57 insertions, 0 deletions
diff --git a/fish/fish.c b/fish/fish.c
index 566d769f..4a960dc0 100644
--- a/fish/fish.c
+++ b/fish/fish.c
@@ -61,6 +61,7 @@ static void shell_script (void);
static void script (int prompt);
static void cmdline (char *argv[], int optind, int argc);
static struct parsed_command parse_command_line (char *buf, int *exit_on_error_rtn);
+static int execute_and_inline (const char *cmd, int exit_on_error);
static void initialize_readline (void);
static void cleanup_readline (void);
#ifdef HAVE_LIBREADLINE
@@ -708,6 +709,18 @@ parse_command_line (char *buf, int *exit_on_error_rtn)
return pcmd;
}
+ /* If the next two characters are "<!" then pass the command to
+ * popen(3), read the result and execute it as guestfish commands.
+ */
+ if (buf[0] == '<' && buf[1] == '!') {
+ int r = execute_and_inline (&buf[2], *exit_on_error_rtn);
+ if (r == -1)
+ pcmd.status = -1;
+ else
+ pcmd.status = 0;
+ return pcmd;
+ }
+
/* If the next character is '-' allow the command to fail without
* exiting on error (just for this one command though).
*/
@@ -823,6 +836,50 @@ parse_command_line (char *buf, int *exit_on_error_rtn)
return pcmd;
}
+/* Used to handle "<!" (execute command and inline result). */
+static int
+execute_and_inline (const char *cmd, int global_exit_on_error)
+{
+ FILE *pp;
+ char *line = NULL;
+ size_t len = 0;
+ ssize_t n;
+ int exit_on_error;
+ struct parsed_command pcmd;
+
+ pp = popen (cmd, "r");
+ if (!pp) {
+ perror ("popen");
+ return -1;
+ }
+
+ while ((n = getline (&line, &len, pp)) != -1) {
+ exit_on_error = global_exit_on_error;
+
+ /* Chomp final line ending which parse_command_line would not expect. */
+ if (n > 0 && line[n-1] == '\n')
+ line[n-1] = '\0';
+
+ pcmd = parse_command_line (line, &exit_on_error);
+ if (pcmd.status == -1 && exit_on_error)
+ exit (EXIT_FAILURE);
+ if (pcmd.status == 1) {
+ if (issue_command (pcmd.cmd, pcmd.argv, pcmd.pipe, exit_on_error) == -1) {
+ if (exit_on_error) exit (EXIT_FAILURE);
+ }
+ }
+ }
+
+ free (line);
+
+ if (pclose (pp) == -1) {
+ perror ("pclose");
+ return -1;
+ }
+
+ return 0;
+}
+
static void
cmdline (char *argv[], int optind, int argc)
{