1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-02 12:01:32 +01:00

gpg: Make the use of "--verify FILE" for detached sigs harder.

* g10/openfile.c (open_sigfile): Factor some code out to ...
(get_matching_datafile): new function.
* g10/plaintext.c (hash_datafiles): Do not try to find matching file
in batch mode.
* g10/mainproc.c (check_sig_and_print): Print a warning if a possibly
matching data file is not used by a standard signatures.
--

Allowing to use the abbreviated form for detached signatures is a long
standing bug which has only been noticed by the public with the
release of 2.1.0.  :-(

What we do is to remove the ability to check detached signature in
--batch using the one file abbreviated mode.  This should exhibit
problems in scripts which use this insecure practice.  We also print a
warning if a matching data file exists but was not considered because
the detached signature was actually a standard signature:

  gpgv: Good signature from "Werner Koch (dist sig)"
  gpgv: WARNING: not a detached signature; \
  file 'gnupg-2.1.0.tar.bz2' was NOT verified!

We can only print a warning because it is possible that a standard
signature is indeed to be verified but by coincidence a file with a
matching name is stored alongside the standard signature.

Reported-by: Simon Nicolussi (to gnupg-users on Nov 7)
Signed-off-by: Werner Koch <wk@gnupg.org>

(backported from commit 69384568f66a48eff3968bb1714aa13925580e9f)

Updated doc/gpg.texi.
This commit is contained in:
Werner Koch 2014-11-14 09:36:19 +01:00
parent 42d2474a02
commit fbb50867f8
5 changed files with 129 additions and 47 deletions

View File

@ -198,16 +198,22 @@ files which don't begin with an encrypted message.
@item --verify @item --verify
@opindex verify @opindex verify
Assume that the first argument is a signed file or a detached signature Assume that the first argument is a signed file and verify it without
and verify it without generating any output. With no arguments, the generating any output. With no arguments, the signature packet is
signature packet is read from STDIN. If only a sigfile is given, it may read from STDIN. If only a one argument is given, it is expected to
be a complete signature or a detached signature, in which case the be a complete signature.
signed stuff is expected in a file without the ".sig" or ".asc"
extension. With more than 1 argument, the first should be a detached With more than 1 argument, the first should be a detached signature
signature and the remaining files are the signed stuff. To read the and the remaining files ake up the the signed data. To read the signed
signed stuff from STDIN, use @samp{-} as the second filename. For data from STDIN, use @samp{-} as the second filename. For security
security reasons a detached signature cannot read the signed material reasons a detached signature cannot read the signed material from
from STDIN without denoting it in the above way. STDIN without denoting it in the above way.
Note: If the option @option{--batch} is not used, @command{gpg}
may assume that a single argument is a file with a detached signature
and it will try to find a matching data file by stripping certain
suffixes. Using this historical feature to verify a detached
signature is strongly discouraged; always specify the data file too.
Note: When verifying a cleartext signature, @command{gpg} verifies Note: When verifying a cleartext signature, @command{gpg} verifies
only what makes up the cleartext signed data and not any extra data only what makes up the cleartext signed data and not any extra data
@ -217,6 +223,7 @@ out the actual signed data; but there are other pitfalls with this
format as well. It is suggested to avoid cleartext signatures in format as well. It is suggested to avoid cleartext signatures in
favor of detached signatures. favor of detached signatures.
@item --multifile @item --multifile
@opindex multifile @opindex multifile
This modifies certain other commands to accept multiple files for This modifies certain other commands to accept multiple files for

View File

@ -195,6 +195,7 @@ int overwrite_filep( const char *fname );
char *make_outfile_name( const char *iname ); char *make_outfile_name( const char *iname );
char *ask_outfile_name( const char *name, size_t namelen ); char *ask_outfile_name( const char *name, size_t namelen );
int open_outfile( const char *iname, int mode, IOBUF *a ); int open_outfile( const char *iname, int mode, IOBUF *a );
char *get_matching_datafile (const char *sigfilename);
IOBUF open_sigfile( const char *iname, progress_filter_context_t *pfx ); IOBUF open_sigfile( const char *iname, progress_filter_context_t *pfx );
void try_make_homedir( const char *fname ); void try_make_homedir( const char *fname );

View File

@ -1940,6 +1940,44 @@ check_sig_and_print( CTX c, KBNODE node )
sig->sig_class==0x01?_("textmode"):_("unknown"), sig->sig_class==0x01?_("textmode"):_("unknown"),
digest_algo_to_string(sig->digest_algo)); digest_algo_to_string(sig->digest_algo));
if (!rc && !c->signed_data)
{
/* Signature is basically good but we test whether the
deprecated command
gpg --verify FILE.sig
was used instead of
gpg --verify FILE.sig FILE
to verify a detached signature. If we figure out that a
data file with a matching name exists, we print a warning.
The problem is that the first form would also verify a
standard signature. This behavior could be used to
create a made up .sig file for a tarball by creating a
standard signature from a valid detached signature packet
(for example from a signed git tag). Then replace the
sig file on the FTP server along with a changed tarball.
Using the first form the verify command would correctly
verify the signature but don't even consider the tarball. */
kbnode_t n;
char *dfile;
dfile = get_matching_datafile (c->sigfilename);
if (dfile)
{
for (n = c->list; n; n = n->next)
if (n->pkt->pkttype != PKT_SIGNATURE)
break;
if (n)
{
/* Not only signature packets in the tree thus this
is not a detached signature. */
log_info (_("WARNING: not a detached signature; "
"file '%s' was NOT verified!\n"), dfile);
}
xfree (dfile);
}
}
if( rc ) if( rc )
g10_errors_seen = 1; g10_errors_seen = 1;
if( opt.batch && rc ) if( opt.batch && rc )

View File

@ -199,7 +199,7 @@ open_outfile( const char *iname, int mode, IOBUF *a )
else { else {
char *buf = NULL; char *buf = NULL;
const char *name; const char *name;
if ( opt.dry_run ) if ( opt.dry_run )
{ {
#ifdef HAVE_W32_SYSTEM #ifdef HAVE_W32_SYSTEM
@ -224,12 +224,12 @@ open_outfile( const char *iname, int mode, IOBUF *a )
char *dot; char *dot;
const char *newsfx = mode==1 ? ".asc" : const char *newsfx = mode==1 ? ".asc" :
mode==2 ? ".sig" : ".gpg"; mode==2 ? ".sig" : ".gpg";
buf = xmalloc(strlen(iname)+4+1); buf = xmalloc(strlen(iname)+4+1);
strcpy(buf,iname); strcpy(buf,iname);
dot = strrchr(buf, '.' ); dot = strrchr(buf, '.' );
if ( dot && dot > buf && dot[1] && strlen(dot) <= 4 if ( dot && dot > buf && dot[1] && strlen(dot) <= 4
&& CMP_FILENAME(newsfx, dot) && CMP_FILENAME(newsfx, dot)
&& !(strchr (dot, '/') || strchr (dot, '\\'))) && !(strchr (dot, '/') || strchr (dot, '\\')))
{ {
/* There is a dot, the dot is not the first character, /* There is a dot, the dot is not the first character,
@ -272,7 +272,7 @@ open_outfile( const char *iname, int mode, IOBUF *a )
xfree (buf); xfree (buf);
name = buf = tmp; name = buf = tmp;
} }
if( !rc ) if( !rc )
{ {
if (is_secured_filename (name) ) if (is_secured_filename (name) )
@ -300,41 +300,70 @@ open_outfile( const char *iname, int mode, IOBUF *a )
} }
/* Find a matching data file for the signature file SIGFILENAME and
return it as a malloced string. If no matching data file is found,
return NULL. */
char *
get_matching_datafile (const char *sigfilename)
{
char *fname = NULL;
size_t len;
if (iobuf_is_pipe_filename (sigfilename))
return NULL;
len = strlen (sigfilename);
if (len > 4
&& (!strcmp (sigfilename + len - 4, EXTSEP_S "sig")
|| (len > 5 && !strcmp(sigfilename + len - 5, EXTSEP_S "sign"))
|| !strcmp(sigfilename + len - 4, EXTSEP_S "asc")))
{
fname = xstrdup (sigfilename);
fname[len-(fname[len-1]=='n'?5:4)] = 0 ;
if (access (fname, R_OK ))
{
/* Not found or other error. */
xfree (fname);
fname = NULL;
}
}
return fname;
}
/**************** /****************
* Try to open a file without the extension ".sig" or ".asc" * Try to open a file without the extension ".sig" or ".asc"
* Return NULL if such a file is not available. * Return NULL if such a file is not available.
*/ */
IOBUF IOBUF
open_sigfile( const char *iname, progress_filter_context_t *pfx ) open_sigfile (const char *sigfilename, progress_filter_context_t *pfx)
{ {
IOBUF a = NULL; iobuf_t a = NULL;
size_t len; char *buf;
if( !iobuf_is_pipe_filename (iname) ) { buf = get_matching_datafile (sigfilename);
len = strlen(iname); if (buf)
if( len > 4 && ( !strcmp(iname + len - 4, EXTSEP_S "sig") {
|| ( len > 5 && !strcmp(iname + len - 5, EXTSEP_S "sign") ) a = iobuf_open (buf);
|| !strcmp(iname + len - 4, EXTSEP_S "asc")) ) { if (a && is_secured_file (iobuf_get_fd (a)))
char *buf; {
buf = xstrdup(iname); iobuf_close (a);
buf[len-(buf[len-1]=='n'?5:4)] = 0 ; a = NULL;
a = iobuf_open( buf ); errno = EPERM;
if (a && is_secured_file (iobuf_get_fd (a))) }
{ if (a)
iobuf_close (a); log_info (_("assuming signed data in `%s'\n"), buf);
a = NULL; if (a && pfx)
errno = EPERM; handle_progress (pfx, a, buf);
} xfree (buf);
if( a && opt.verbose )
log_info(_("assuming signed data in `%s'\n"), buf );
if (a && pfx)
handle_progress (pfx, a, buf);
xfree(buf);
}
} }
return a;
return a;
} }
/**************** /****************
* Copy the option file skeleton to the given directory. * Copy the option file skeleton to the given directory.
*/ */
@ -398,7 +427,7 @@ copy_options_file( const char *destdir )
; ;
else if (c == '#') else if (c == '#')
esc = 2; esc = 2;
else else
any_option = 1; any_option = 1;
} }
} }

View File

@ -538,13 +538,20 @@ hash_datafiles( MD_HANDLE md, MD_HANDLE md2, STRLIST files,
STRLIST sl; STRLIST sl;
if( !files ) { if( !files ) {
/* check whether we can open the signed material */ /* Check whether we can open the signed material. We avoid
fp = open_sigfile( sigfilename, &pfx ); trying to open a file if run in batch mode. This assumed
if( fp ) { data file for a sig file feature is just a convenience thing
do_hash( md, md2, fp, textmode ); for the command line and the user needs to read possible
iobuf_close(fp); warning messages. */
return 0; if (!opt.batch) {
} fp = open_sigfile( sigfilename, &pfx );
if( fp ) {
do_hash( md, md2, fp, textmode );
iobuf_close(fp);
return 0;
}
}
log_error (_("no signed data\n")); log_error (_("no signed data\n"));
return G10ERR_OPEN_FILE; return G10ERR_OPEN_FILE;
} }