X-Git-Url: http://git.annexia.org/?p=libguestfs.git;a=blobdiff_plain;f=daemon%2Ftar.c;h=bb0e4830820f346991b10231b7e1ecbcd90f49f4;hp=ecf919db19fba0a3ab5912f04b874f9f03ea7d2a;hb=07f4b20ae959069fca41756b0dc103ec5fa99754;hpb=0232e722826cfda0f6042da983f9eb871f24e946 diff --git a/daemon/tar.c b/daemon/tar.c index ecf919d..bb0e483 100644 --- a/daemon/tar.c +++ b/daemon/tar.c @@ -1,5 +1,5 @@ /* libguestfs - the guestfsd daemon - * Copyright (C) 2009 Red Hat Inc. + * Copyright (C) 2009 Red Hat Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -38,29 +38,27 @@ fwrite_cb (void *fp_ptr, const void *buf, int len) int do_tar_in (const char *dir) { - int err, r, len; + int err, r; FILE *fp; char *cmd; if (!root_mounted || dir[0] != '/') { cancel_receive (); - reply_with_error ("tar-in: root must be mounted and path must be absolute"); + reply_with_error ("root must be mounted and path must be absolute"); return -1; } /* "tar -C /sysroot%s -xf -" but we have to quote the dir. */ - len = 2 * strlen (dir) + 32; - cmd = malloc (len); - if (!cmd) { + if (asprintf_nowarn (&cmd, "tar -C %R -xf -", dir) == -1) { err = errno; cancel_receive (); errno = err; - reply_with_perror ("malloc"); + reply_with_perror ("asprintf"); return -1; } - strcpy (cmd, "tar -C /sysroot"); - shell_quote (cmd+15, len-15, dir); - strcat (cmd, " -xf -"); + + if (verbose) + fprintf (stderr, "%s\n", cmd); fp = popen (cmd, "w"); if (fp == NULL) { @@ -68,8 +66,10 @@ do_tar_in (const char *dir) cancel_receive (); errno = err; reply_with_perror ("%s", cmd); + free (cmd); return -1; } + free (cmd); r = receive_file (fwrite_cb, &fp); if (r == -1) { /* write error */ @@ -86,9 +86,10 @@ do_tar_in (const char *dir) return -1; } - if (pclose (fp) == -1) { + if (pclose (fp) != 0) { err = errno; - cancel_receive (); + if (r == -1) /* if r == 0, file transfer ended already */ + cancel_receive (); errno = err; reply_with_perror ("pclose: %s", dir); return -1; @@ -101,30 +102,27 @@ do_tar_in (const char *dir) int do_tar_out (const char *dir) { - int r, len; + int r; FILE *fp; char *cmd; char buf[GUESTFS_MAX_CHUNK_SIZE]; - NEED_ROOT (-1); - ABS_PATH (dir, -1); - /* "tar -C /sysroot%s -cf - ." but we have to quote the dir. */ - len = 2 * strlen (dir) + 32; - cmd = malloc (len); - if (!cmd) { - reply_with_perror ("malloc"); + if (asprintf_nowarn (&cmd, "tar -C %R -cf - .", dir) == -1) { + reply_with_perror ("asprintf"); return -1; } - strcpy (cmd, "tar -C /sysroot"); - shell_quote (cmd+15, len-15, dir); - strcat (cmd, " -cf - ."); + + if (verbose) + fprintf (stderr, "%s\n", cmd); fp = popen (cmd, "r"); if (fp == NULL) { reply_with_perror ("%s", cmd); + free (cmd); return -1; } + free (cmd); /* Now we must send the reply message, before the file contents. After * this there is no opportunity in the protocol to send any error @@ -139,20 +137,22 @@ do_tar_out (const char *dir) } } - if (r == -1) { + if (ferror (fp)) { perror (dir); send_file_end (1); /* Cancel. */ pclose (fp); return -1; } - if (pclose (fp) == -1) { + if (pclose (fp) != 0) { perror (dir); send_file_end (1); /* Cancel. */ return -1; } - send_file_end (0); /* Normal end of file. */ + if (send_file_end (0)) /* Normal end of file. */ + return -1; + return 0; } @@ -160,29 +160,27 @@ do_tar_out (const char *dir) int do_tgz_in (const char *dir) { - int err, r, len; + int err, r; FILE *fp; char *cmd; if (!root_mounted || dir[0] != '/') { cancel_receive (); - reply_with_error ("tar-in: root must be mounted and path must be absolute"); + reply_with_error ("root must be mounted and path must be absolute"); return -1; } /* "tar -C /sysroot%s -zxf -" but we have to quote the dir. */ - len = 2 * strlen (dir) + 32; - cmd = malloc (len); - if (!cmd) { + if (asprintf_nowarn (&cmd, "tar -C %R -zxf -", dir) == -1) { err = errno; cancel_receive (); errno = err; - reply_with_perror ("malloc"); + reply_with_perror ("asprintf"); return -1; } - strcpy (cmd, "tar -C /sysroot"); - shell_quote (cmd+15, len-15, dir); - strcat (cmd, " -zxf -"); + + if (verbose) + fprintf (stderr, "%s\n", cmd); fp = popen (cmd, "w"); if (fp == NULL) { @@ -190,8 +188,10 @@ do_tgz_in (const char *dir) cancel_receive (); errno = err; reply_with_perror ("%s", cmd); + free (cmd); return -1; } + free (cmd); r = receive_file (fwrite_cb, &fp); if (r == -1) { /* write error */ @@ -208,9 +208,10 @@ do_tgz_in (const char *dir) return -1; } - if (pclose (fp) == -1) { + if (pclose (fp) != 0) { err = errno; - cancel_receive (); + if (r == -1) /* if r == 0, file transfer ended already */ + cancel_receive (); errno = err; reply_with_perror ("pclose: %s", dir); return -1; @@ -223,30 +224,27 @@ do_tgz_in (const char *dir) int do_tgz_out (const char *dir) { - int r, len; + int r; FILE *fp; char *cmd; char buf[GUESTFS_MAX_CHUNK_SIZE]; - NEED_ROOT (-1); - ABS_PATH (dir, -1); - /* "tar -C /sysroot%s -zcf - ." but we have to quote the dir. */ - len = 2 * strlen (dir) + 32; - cmd = malloc (len); - if (!cmd) { - reply_with_perror ("malloc"); + if (asprintf_nowarn (&cmd, "tar -C %R -zcf - .", dir) == -1) { + reply_with_perror ("asprintf"); return -1; } - strcpy (cmd, "tar -C /sysroot"); - shell_quote (cmd+15, len-15, dir); - strcat (cmd, " -zcf - ."); + + if (verbose) + fprintf (stderr, "%s\n", cmd); fp = popen (cmd, "r"); if (fp == NULL) { reply_with_perror ("%s", cmd); + free (cmd); return -1; } + free (cmd); /* Now we must send the reply message, before the file contents. After * this there is no opportunity in the protocol to send any error @@ -261,19 +259,21 @@ do_tgz_out (const char *dir) } } - if (r == -1) { + if (ferror (fp)) { perror (dir); send_file_end (1); /* Cancel. */ pclose (fp); return -1; } - if (pclose (fp) == -1) { + if (pclose (fp) != 0) { perror (dir); send_file_end (1); /* Cancel. */ return -1; } - send_file_end (0); /* Normal end of file. */ + if (send_file_end (0)) /* Normal end of file. */ + return -1; + return 0; }