Simply the copy-in & copy-out commands of guestfish using the new
implementations of copy-in & copy-out in the library.
---
fish/copy.c | 279 +-----------------------------------------------------------
1 file changed, 4 insertions(+), 275 deletions(-)
diff --git a/fish/copy.c b/fish/copy.c
index 8394dba..d3abb2e 100644
--- a/fish/copy.c
+++ b/fish/copy.c
@@ -22,25 +22,11 @@
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
-#include <limits.h>
#include <libintl.h>
#include <errno.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <sys/wait.h>
-#include <assert.h>
#include "fish.h"
-struct fd_pid {
- int fd; /* -1 == error */
- pid_t pid;
-};
-
-static struct fd_pid make_tar_from_local (const char *local);
-static struct fd_pid make_tar_output (const char *local, const char *basename);
-static int split_path (char *buf, size_t buf_size, const char *path, const char
**dirname, const char **basename);
-
int
run_copy_in (const char *cmd, size_t argc, char *argv[])
{
@@ -61,40 +47,10 @@ run_copy_in (const char *cmd, size_t argc, char *argv[])
int nr_locals = argc-1;
- int remote_is_dir = guestfs_is_dir (g, remote);
- if (remote_is_dir == -1)
- return -1;
-
- if (!remote_is_dir) {
- fprintf (stderr, _("copy-in: target '%s' is not a directory\n"),
remote);
- return -1;
- }
-
- /* Upload each local one at a time using tar-in. */
+ /* Upload each local one at a time using copy-in. */
int i;
for (i = 0; i < nr_locals; ++i) {
- struct fd_pid fdpid = make_tar_from_local (argv[i]);
- if (fdpid.fd == -1)
- return -1;
-
- char fdbuf[64];
- snprintf (fdbuf, sizeof fdbuf, "/dev/fd/%d", fdpid.fd);
-
- int r = guestfs_tar_in (g, fdbuf, remote);
-
- if (close (fdpid.fd) == -1) {
- perror ("close (tar-from-local subprocess)");
- r = -1;
- }
-
- int status;
- if (waitpid (fdpid.pid, &status, 0) == -1) {
- perror ("wait (tar-from-local subprocess)");
- return -1;
- }
- if (!(WIFEXITED (status) && WEXITSTATUS (status) == 0))
- return -1;
-
+ int r = guestfs_copy_in (g, argv[i], remote);
if (r == -1)
return -1;
}
@@ -102,98 +58,6 @@ run_copy_in (const char *cmd, size_t argc, char *argv[])
return 0;
}
-static void tar_create (const char *dir, const char *path) __attribute__((noreturn));
-
-/* This creates a subprocess which feeds a tar file back to the
- * main guestfish process.
- */
-static struct fd_pid
-make_tar_from_local (const char *local)
-{
- int fd[2];
- struct fd_pid r = { .fd = -1 };
-
- if (pipe (fd) == -1) {
- perror ("pipe");
- return r;
- }
-
- r.pid = fork ();
- if (r.pid == -1) {
- perror ("fork");
- return r;
- }
-
- if (r.pid > 0) { /* Parent */
- close (fd[1]);
- r.fd = fd[0];
- return r;
- }
-
- /* Child. */
- close (fd[0]);
- dup2 (fd[1], 1);
- close (fd[1]);
-
- size_t buf_len = strlen (local) + 1;
- char buf[buf_len];
- const char *dirname, *basename;
- if (split_path (buf, buf_len, local, &dirname, &basename) == -1)
- _exit (EXIT_FAILURE);
-
- tar_create (dirname, basename);
-}
-
-/* Split path into directory name and base name, using the buffer
- * provided as a working area. If there is no directory name
- * (eg. path == "/") then this can return dirname as NULL.
- */
-static int
-split_path (char *buf, size_t buf_size,
- const char *path, const char **dirname, const char **basename)
-{
- size_t len = strlen (path);
- if (len == 0 || len > buf_size - 1) {
- fprintf (stderr, _("error: argument is zero length or longer than maximum
permitted\n"));
- return -1;
- }
-
- strcpy (buf, path);
-
- if (len >= 2 && buf[len-1] == '/') {
- buf[len-1] = '\0';
- len--;
- }
-
- char *p = strrchr (buf, '/');
- if (p && p > buf) { /* "foo/bar" */
- *p = '\0';
- p++;
- if (dirname) *dirname = buf;
- if (basename) *basename = p;
- } else if (p && p == buf) { /* "/foo" */
- if (dirname) *dirname = "/";
- if (basename) *basename = buf+1;
- } else {
- if (dirname) *dirname = NULL;
- if (basename) *basename = buf;
- }
-
- return 0;
-}
-
-static void
-tar_create (const char *dir, const char *path)
-{
- if (dir)
- execlp ("tar", "tar", "-C", dir, "-cf",
"-", path, NULL);
- else
- execlp ("tar", "tar", "-cf", "-", path,
NULL);
-
- perror ("execlp: tar");
- _exit (EXIT_FAILURE);
-}
-
int
run_copy_out (const char *cmd, size_t argc, char *argv[])
{
@@ -207,14 +71,7 @@ run_copy_out (const char *cmd, size_t argc, char *argv[])
const char *local = argv[argc-1];
int nr_remotes = argc-1;
- struct stat statbuf;
- if (stat (local, &statbuf) == -1 ||
- ! (S_ISDIR (statbuf.st_mode))) {
- fprintf (stderr, _("copy-out: target '%s' is not a directory\n"),
local);
- return -1;
- }
-
- /* Download each remote one at a time using tar-out. */
+ /* Download each remote one at a time using copy-out. */
int i, r;
for (i = 0; i < nr_remotes; ++i) {
CLEANUP_FREE char *remote = NULL;
@@ -224,138 +81,10 @@ run_copy_out (const char *cmd, size_t argc, char *argv[])
if (remote == NULL)
return -1;
- /* If the remote is a file, download it. If it's a directory,
- * create the directory in local first before using tar-out.
- */
- r = guestfs_is_file (g, remote);
+ r = guestfs_copy_out (g, remote, local);
if (r == -1)
return -1;
- if (r == 1) { /* is file */
- CLEANUP_FREE char *filename = NULL;
- size_t buf_len = strlen (remote) + 1;
- char buf[buf_len];
- const char *basename;
-
- if (split_path (buf, buf_len, remote, NULL, &basename) == -1)
- return -1;
-
- if (asprintf (&filename, "%s/%s", local, basename) == -1) {
- perror ("asprintf");
- return -1;
- }
- if (guestfs_download (g, remote, filename) == -1)
- return -1;
- }
- else { /* not a regular file */
- r = guestfs_is_dir (g, remote);
- if (r == -1)
- return -1;
-
- if (r == 0) {
- fprintf (stderr, _("copy-out: '%s' is not a file or
directory\n"),
- remote);
- return -1;
- }
-
- size_t buf_len = strlen (remote) + 1;
- char buf[buf_len];
- const char *basename;
- if (split_path (buf, buf_len, remote, NULL, &basename) == -1)
- return -1;
-
- /* RHBZ#845522: If remote == "/" then basename would be an empty
- * string. Replace it with "." so that make_tar_output writes
- * to "local/."
- */
- if (STREQ (basename, ""))
- basename = ".";
-
- struct fd_pid fdpid = make_tar_output (local, basename);
- if (fdpid.fd == -1)
- return -1;
-
- char fdbuf[64];
- snprintf (fdbuf, sizeof fdbuf, "/dev/fd/%d", fdpid.fd);
-
- int r = guestfs_tar_out (g, remote, fdbuf);
-
- if (close (fdpid.fd) == -1) {
- perror ("close (tar-output subprocess)");
- r = -1;
- }
-
- int status;
- if (waitpid (fdpid.pid, &status, 0) == -1) {
- perror ("wait (tar-output subprocess)");
- return -1;
- }
- if (!(WIFEXITED (status) && WEXITSTATUS (status) == 0))
- return -1;
-
- if (r == -1)
- return -1;
- }
}
return 0;
}
-
-/* This creates a subprocess which takes a tar file from the main
- * guestfish process and unpacks it into the 'local/basename'
- * directory.
- */
-static struct fd_pid
-make_tar_output (const char *local, const char *basename)
-{
- int fd[2];
- struct fd_pid r = { .fd = -1 };
-
- /* local can't be an empty string because the caller stats it and
- * checks it is a directory.
- */
- assert (STRNEQ (local, ""));
-
- /* basename must not be an empty string (see RHBZ#845522). */
- assert (STRNEQ (basename, ""));
-
- if (pipe (fd) == -1) {
- perror ("pipe");
- return r;
- }
-
- r.pid = fork ();
- if (r.pid == -1) {
- perror ("fork");
- return r;
- }
-
- if (r.pid > 0) { /* Parent */
- close (fd[0]);
- r.fd = fd[1];
- return r;
- }
-
- /* Child. */
- close (fd[1]);
- dup2 (fd[0], 0);
- close (fd[0]);
-
- if (chdir (local) == -1) {
- perror (local);
- _exit (EXIT_FAILURE);
- }
-
- if (mkdir (basename, 0777) == -1 && errno != EEXIST) {
- perror (basename);
- _exit (EXIT_FAILURE);
- }
-
- if (chdir (basename) == -1) {
- perror (basename);
- _exit (EXIT_FAILURE);
- }
-
- execlp ("tar", "tar", "-xf", "-", NULL);
- perror ("execlp: tar");
- _exit (EXIT_FAILURE);
-}
--
1.9.3