1
0
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:
Werner Koch 2024-11-07 15:06:17 +01:00
parent 6c58694a88
commit fc47bdad59
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B
2 changed files with 74 additions and 32 deletions

View File

@ -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

View File

@ -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));