lib/download.c: add basic support for vcdiffs

This commit is contained in:
Enno Boland 2014-08-11 18:40:44 +02:00
parent c301cc32e5
commit 040e114ad4
3 changed files with 131 additions and 18 deletions

View File

@ -59,6 +59,7 @@ usage(void)
" pkgmatch\t\t<pkg-version> <pkg-pattern>\n" " pkgmatch\t\t<pkg-version> <pkg-pattern>\n"
" version\t\t<pkgname>\n" " version\t\t<pkgname>\n"
" real-version\t<pkgname>\n" " real-version\t<pkgname>\n"
" xfetch\t\t<oldfile> <URL[>filename]>\n"
"\n" "\n"
" Options shared by all actions:\n" " Options shared by all actions:\n"
" -C\t\tPath to xbps.conf file.\n" " -C\t\tPath to xbps.conf file.\n"
@ -81,6 +82,20 @@ usage(void)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
static char*
fname(char *url) {
char *filename;
if( (filename = strrchr(url, '>')) ) {
*filename = '\0';
} else {
filename = strrchr(url, '/');
}
if(filename == NULL)
return NULL;
return filename + 1;
}
int int
main(int argc, char **argv) main(int argc, char **argv)
{ {
@ -88,7 +103,7 @@ main(int argc, char **argv)
struct xbps_handle xh; struct xbps_handle xh;
struct xferstat xfer; struct xferstat xfer;
const char *version, *rootdir = NULL, *conffile = NULL; const char *version, *rootdir = NULL, *conffile = NULL;
char *pkgname, *hash, *sep; char *pkgname, *hash, *filename;
int flags = 0, c, rv = 0; int flags = 0, c, rv = 0;
while ((c = getopt(argc, argv, "C:dr:V")) != -1) { while ((c = getopt(argc, argv, "C:dr:V")) != -1) {
@ -244,18 +259,30 @@ main(int argc, char **argv)
} }
printf("%s\n", hash); printf("%s\n", hash);
} }
} else if (strcmp(argv[0], "xfetch") == 0) {
/* apply a delta from specified URL */
if (argc != 3)
usage();
filename = fname(argv[2]);
rv = xbps_fetch_delta(&xh, argv[1], argv[2], filename, "v");
if (rv == -1) {
printf("%s: %s\n", argv[2],
xbps_fetch_error_string());
} else if (rv == 0) {
printf("%s: file is identical than remote.\n",
argv[3]);
} else
rv = 0;
} else if (strcmp(argv[0], "fetch") == 0) { } else if (strcmp(argv[0], "fetch") == 0) {
/* Fetch a file from specified URL */ /* Fetch a file from specified URL */
if (argc != 2) if (argc != 2)
usage(); usage();
for (int i = 1; i < argc; i++) { for (int i = 1; i < argc; i++) {
if( (sep = strrchr(argv[i], '>')) ) { filename = fname(argv[i]);
*sep = '\0'; rv = xbps_fetch_file_dest(&xh, argv[i], filename, "v");
rv = xbps_fetch_file_dest(&xh, argv[i], sep+1, "v");
} else {
rv = xbps_fetch_file(&xh, argv[i], "v");
}
if (rv == -1) { if (rv == -1) {
printf("%s: %s\n", argv[i], printf("%s: %s\n", argv[i],

View File

@ -713,6 +713,22 @@ int xbps_fetch_file(struct xbps_handle *xhp, const char *uri,
int xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, int xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri,
const char *filename, const char *flags); const char *filename, const char *flags);
/**
* Download a vcdiff from a remote URL to current working directory. And apply
* it to a given file
*
* @param[in] xhp Pointer to an xbps_handle struct.
* @param[in] uri Remote URI string.
* @param[in] basefile Local basefile to apply the delta to.
* @param[in] filename Local filename to safe the file.
* @param[in] flags Flags passed to libfetch's fetchXget().
*
* @return -1 on error, 0 if not downloaded (because local/remote size/mtime
* do not match) and 1 if downloaded successfully.
**/
int xbps_fetch_delta(struct xbps_handle *xhp, const char *basefile,
const char *uri, const char *filename, const char *flags);
/** /**
* Returns last error string reported by xbps_fetch_file(). * Returns last error string reported by xbps_fetch_file().
* *

View File

@ -41,6 +41,8 @@
#undef _BSD_SOURCE #undef _BSD_SOURCE
#include <errno.h> #include <errno.h>
#include <fcntl.h> #include <fcntl.h>
#include <sys/wait.h>
#include <libgen.h>
#include "xbps_api_impl.h" #include "xbps_api_impl.h"
#include "fetch.h" #include "fetch.h"
@ -129,7 +131,7 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
} else { } else {
if (errno != ENOENT) { if (errno != ENOENT) {
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
} }
/* /*
@ -143,7 +145,7 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
} else { } else {
if (errno != ENOENT) { if (errno != ENOENT) {
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
} }
if (refetch && !restart) { if (refetch && !restart) {
@ -178,10 +180,10 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
if (fio == NULL) { if (fio == NULL) {
if (fetchLastErrCode == FETCH_UNCHANGED) { if (fetchLastErrCode == FETCH_UNCHANGED) {
/* Last-Modified matched */ /* Last-Modified matched */
goto out; goto fetch_file_out;
} }
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
if (url_st.size == -1) { if (url_st.size == -1) {
xbps_dbg_printf(xhp, "Remote file size is unknown, resume " xbps_dbg_printf(xhp, "Remote file size is unknown, resume "
@ -207,7 +209,7 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
if (fd == -1) { if (fd == -1) {
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
/* /*
* Initialize data for the fetch progress function callback * Initialize data for the fetch progress function callback
@ -225,7 +227,7 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
xbps_dbg_printf(xhp, xbps_dbg_printf(xhp,
"Couldn't write to %s!\n", tempfile); "Couldn't write to %s!\n", tempfile);
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
bytes_dload += bytes_read; bytes_dload += bytes_read;
/* /*
@ -241,12 +243,12 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
filename, fetchLastErrString); filename, fetchLastErrString);
errno = EIO; errno = EIO;
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} else if (url_st.size > 0 && ((bytes_dload + url->offset) != url_st.size)) { } else if (url_st.size > 0 && ((bytes_dload + url->offset) != url_st.size)) {
xbps_dbg_printf(xhp, "file %s is truncated\n", filename); xbps_dbg_printf(xhp, "file %s is truncated\n", filename);
errno = EIO; errno = EIO;
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
/* /*
@ -264,7 +266,7 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
ts[0].tv_nsec = ts[1].tv_nsec = 0; ts[0].tv_nsec = ts[1].tv_nsec = 0;
if (futimens(fd, ts) == -1) { if (futimens(fd, ts) == -1) {
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
(void)close(fd); (void)close(fd);
fd = -1; fd = -1;
@ -274,11 +276,11 @@ xbps_fetch_file_dest(struct xbps_handle *xhp, const char *uri, const char *filen
xbps_dbg_printf(xhp, "failed to rename %s to %s: %s", xbps_dbg_printf(xhp, "failed to rename %s to %s: %s",
tempfile, filename, strerror(errno)); tempfile, filename, strerror(errno));
rv = -1; rv = -1;
goto out; goto fetch_file_out;
} }
rv = 1; rv = 1;
out: fetch_file_out:
if (fio != NULL) if (fio != NULL)
fetchIO_close(fio); fetchIO_close(fio);
if (fd != -1) if (fd != -1)
@ -290,6 +292,7 @@ out:
return rv; return rv;
} }
int int
xbps_fetch_file(struct xbps_handle *xhp, const char *uri, const char *flags) xbps_fetch_file(struct xbps_handle *xhp, const char *uri, const char *flags)
{ {
@ -305,3 +308,70 @@ xbps_fetch_file(struct xbps_handle *xhp, const char *uri, const char *flags)
return xbps_fetch_file_dest(xhp, uri, filename, flags); return xbps_fetch_file_dest(xhp, uri, filename, flags);
} }
int
xbps_fetch_delta(struct xbps_handle *xhp, const char *basefile, const char *uri, const char *filename, const char *flags)
{
const char xdelta[] = "/usr/bin/xdelta3";
char *basehash = NULL, *dname = NULL, *durl = NULL;
int status, exitcode;
pid_t pid;
int rv = 0;
struct stat xdelta_stat;
if (basefile == NULL || stat(xdelta, &xdelta_stat)) {
goto fetch_delta_fallback;
}
basehash = xbps_file_hash(basefile);
assert(basehash);
dname = xbps_xasprintf("%s.%s.vcdiff", basename(uri), basehash);
durl = xbps_xasprintf("%s.%s.vcdiff", uri, basehash);
if (xbps_fetch_file_dest(xhp, durl, dname, flags) < 0) {
xbps_dbg_printf(xhp, "error while download vcdiff, fallback to full "
"download\n", xdelta);
goto fetch_delta_fallback;
}
if ((pid = fork()) == 0) {
execl (xdelta, xdelta, "-d", "-f", "-s", basefile, dname, filename, NULL);
exit(127);
} else if (pid < 0) {
xbps_dbg_printf(xhp, "error while forking, fallback to full "
"download\n", xdelta);
goto fetch_delta_fallback;
}
// wait for termination of background process
waitpid(pid, &status, 0);
exitcode = WEXITSTATUS(status);
switch(exitcode) {
case 0: // success
rv = 1;
goto fetch_delta_out;
case 127: // cannot execute binary
xbps_dbg_printf(xhp, "failed to `%s`, fallback to full download\n",
xdelta);
goto fetch_delta_fallback;
default: // other error
xbps_dbg_printf(xhp, "`%s` exited with code %d, fallback to full "
"download\n", xdelta, exitcode);
goto fetch_delta_fallback;
}
fetch_delta_fallback:
rv = xbps_fetch_file_dest(xhp, uri, filename, flags);
fetch_delta_out:
if(dname != NULL)
free(dname);
if(durl != NULL)
free(durl);
if(basehash != NULL)
free(basehash);
return rv;
}