mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-18 14:17:03 +01:00
gpgtar: Make sure to create upper directories for regular files.
* tools/gpgtar-extract.c (extract_directory): Factor parent directory creation out to .. (try_mkdir_p): new. (extract_regular): Create directory on ENOENT. * g10/pubkey-enc.c (get_it): Use log_info instead of log_error if the public key was not found for preference checking. -- If tarball was created with tar cf tarball file1.txt foo/file2.txt the tarball has no entry for foo/ and thus the extraction fails. This patch fixes this. GnuPG-bug-id: 7380 The second patch avoid a wrong exit status status line due to the use of log_error. But the actual cause needs stuill needs tobe investigated.
This commit is contained in:
parent
6c58694a88
commit
fc47bdad59
@ -373,8 +373,8 @@ get_it (ctrl_t ctrl,
|
|||||||
|
|
||||||
if (!pkb)
|
if (!pkb)
|
||||||
{
|
{
|
||||||
err = -1;
|
err = gpg_error (GPG_ERR_UNEXPECTED);
|
||||||
log_error ("oops: public key not found for preference check\n");
|
log_info ("oops: public key not found for preference check\n");
|
||||||
}
|
}
|
||||||
else if (pkb->pkt->pkt.public_key->selfsigversion > 3
|
else if (pkb->pkt->pkt.public_key->selfsigversion > 3
|
||||||
&& dek->algo != CIPHER_ALGO_3DES
|
&& dek->algo != CIPHER_ALGO_3DES
|
||||||
|
@ -68,6 +68,60 @@ check_suspicious_name (const char *name, tarinfo_t info)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* This is our version of mkdir -p. DIRECTORY is the full filename of
|
||||||
|
* the directory and PREFIXLEN is the length of an intial directory
|
||||||
|
* part which already exists. If STRIP is set filename is removed.
|
||||||
|
* If VERBOSE is set a diagnostic is printed to show the created
|
||||||
|
* directory. */
|
||||||
|
static gpg_error_t
|
||||||
|
try_mkdir_p (const char *directory, size_t prefixlen, int strip, int verbose)
|
||||||
|
{
|
||||||
|
gpg_error_t err = 0;
|
||||||
|
char *fname;
|
||||||
|
char *p;
|
||||||
|
|
||||||
|
fname = xtrystrdup (directory);
|
||||||
|
if (!fname)
|
||||||
|
return gpg_error_from_syserror ();
|
||||||
|
|
||||||
|
if (strip) /* Strip last file name. */
|
||||||
|
{
|
||||||
|
p = strrchr (fname, '/');
|
||||||
|
if (p)
|
||||||
|
*p = 0;
|
||||||
|
}
|
||||||
|
else /* Remove a possible trailing slash. */
|
||||||
|
{
|
||||||
|
if (fname[strlen (fname)-1] == '/')
|
||||||
|
fname[strlen (fname)-1] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prefixlen >= strlen (fname))
|
||||||
|
goto leave; /* Nothing to create */
|
||||||
|
|
||||||
|
for (p = fname+prefixlen; (p = strchr (p, '/')); p++)
|
||||||
|
{
|
||||||
|
*p = 0;
|
||||||
|
err = gnupg_mkdir (fname, "-rwx------");
|
||||||
|
if (gpg_err_code (err) == GPG_ERR_EEXIST)
|
||||||
|
err = 0;
|
||||||
|
*p = '/';
|
||||||
|
if (err)
|
||||||
|
goto leave;
|
||||||
|
}
|
||||||
|
err = gnupg_mkdir (fname, "-rwx------");
|
||||||
|
if (gpg_err_code (err) == GPG_ERR_EEXIST)
|
||||||
|
err = 0;
|
||||||
|
if (!err && verbose)
|
||||||
|
log_info ("created '%s/'\n", fname);
|
||||||
|
|
||||||
|
leave:
|
||||||
|
xfree (fname);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static gpg_error_t
|
static gpg_error_t
|
||||||
extract_regular (estream_t stream, const char *dirname,
|
extract_regular (estream_t stream, const char *dirname,
|
||||||
tarinfo_t info, tar_header_t hdr, strlist_t exthdr)
|
tarinfo_t info, tar_header_t hdr, strlist_t exthdr)
|
||||||
@ -98,7 +152,6 @@ extract_regular (estream_t stream, const char *dirname,
|
|||||||
}
|
}
|
||||||
fname = fname_buffer;
|
fname = fname_buffer;
|
||||||
|
|
||||||
|
|
||||||
if (opt.dry_run)
|
if (opt.dry_run)
|
||||||
outfp = es_fopen ("/dev/null", "wb");
|
outfp = es_fopen ("/dev/null", "wb");
|
||||||
else
|
else
|
||||||
@ -106,9 +159,19 @@ extract_regular (estream_t stream, const char *dirname,
|
|||||||
if (!outfp)
|
if (!outfp)
|
||||||
{
|
{
|
||||||
err = gpg_error_from_syserror ();
|
err = gpg_error_from_syserror ();
|
||||||
|
/* On ENOENT, try afain after trying to create the directories. */
|
||||||
|
if (!opt.dry_run && gpg_err_code (GPG_ERR_ENOENT)
|
||||||
|
&& !try_mkdir_p (fname, strlen (dirname) + 1, 1, opt.verbose))
|
||||||
|
{
|
||||||
|
outfp = es_fopen (fname, "wb,sysopen");
|
||||||
|
err = outfp? 0 : gpg_error_from_syserror ();
|
||||||
|
}
|
||||||
|
if (err)
|
||||||
|
{
|
||||||
log_error ("error creating '%s': %s\n", fname, gpg_strerror (err));
|
log_error ("error creating '%s': %s\n", fname, gpg_strerror (err));
|
||||||
goto leave;
|
goto leave;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (n=0; n < hdr->nrecords;)
|
for (n=0; n < hdr->nrecords;)
|
||||||
{
|
{
|
||||||
@ -180,41 +243,20 @@ extract_directory (const char *dirname, tarinfo_t info,
|
|||||||
if (fname[strlen (fname)-1] == '/')
|
if (fname[strlen (fname)-1] == '/')
|
||||||
fname[strlen (fname)-1] = 0;
|
fname[strlen (fname)-1] = 0;
|
||||||
|
|
||||||
if (! opt.dry_run && gnupg_mkdir (fname, "-rwx------"))
|
if (!opt.dry_run && gnupg_mkdir (fname, "-rwx------"))
|
||||||
{
|
{
|
||||||
err = gpg_error_from_syserror ();
|
err = gpg_error_from_syserror ();
|
||||||
if (gpg_err_code (err) == GPG_ERR_EEXIST)
|
|
||||||
{
|
|
||||||
/* Ignore existing directories while extracting. */
|
/* Ignore existing directories while extracting. */
|
||||||
|
if (gpg_err_code (err) == GPG_ERR_EEXIST)
|
||||||
err = 0;
|
err = 0;
|
||||||
}
|
else if (gpg_err_code (err) == GPG_ERR_ENOENT)
|
||||||
|
|
||||||
if (gpg_err_code (err) == GPG_ERR_ENOENT)
|
|
||||||
{
|
{
|
||||||
/* Try to create the directory with parents but keep the
|
/* Try to create the directory with parents but keep the
|
||||||
original error code in case of a failure. */
|
original error code in case of a failure. */
|
||||||
int rc = 0;
|
if (!try_mkdir_p (fname, strlen (dirname) + 1, 0, 0))
|
||||||
char *p;
|
|
||||||
size_t prefixlen;
|
|
||||||
|
|
||||||
/* (PREFIXLEN is the length of the new directory we use to
|
|
||||||
* extract the tarball.) */
|
|
||||||
prefixlen = strlen (dirname) + 1;
|
|
||||||
|
|
||||||
for (p = fname+prefixlen; (p = strchr (p, '/')); p++)
|
|
||||||
{
|
|
||||||
*p = 0;
|
|
||||||
rc = gnupg_mkdir (fname, "-rwx------");
|
|
||||||
if (rc && (gpg_err_code (gpg_error_from_syserror ())
|
|
||||||
== GPG_ERR_EEXIST))
|
|
||||||
rc = 0;
|
|
||||||
*p = '/';
|
|
||||||
if (rc)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (!rc && !gnupg_mkdir (fname, "-rwx------"))
|
|
||||||
err = 0;
|
err = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
log_error ("error creating directory '%s': %s\n",
|
log_error ("error creating directory '%s': %s\n",
|
||||||
fname, gpg_strerror (err));
|
fname, gpg_strerror (err));
|
||||||
|
Loading…
x
Reference in New Issue
Block a user