mirror of
git://git.gnupg.org/gnupg.git
synced 2024-12-22 10:19:57 +01:00
Add unfinished gpgtar.
Collected changes and ports of bug fixes from stable.
This commit is contained in:
parent
29cc88db7d
commit
bbe388b5db
@ -1,3 +1,11 @@
|
||||
2010-06-07 Werner Koch <wk@g10code.com>
|
||||
|
||||
* configure.ac: Add option --enable-gpgtar.
|
||||
|
||||
2010-05-31 Werner Koch <wk@g10code.com>
|
||||
|
||||
* configure.ac (AC_CHECK_FUNCS): Check for lstat.
|
||||
|
||||
2010-04-30 Werner Koch <wk@g10code.com>
|
||||
|
||||
* configure.ac: Add option --enable-standard-socket.
|
||||
|
@ -1,3 +1,8 @@
|
||||
2010-05-12 Werner Koch <wk@g10code.com>
|
||||
|
||||
* preset-passphrase.c (forget_passphrase): Actually implement
|
||||
this. Fixes bug#1198.
|
||||
|
||||
2010-05-11 Werner Koch <wk@g10code.com>
|
||||
|
||||
* agent.h (opt): Add field USE_STANDARD_SOCKET.
|
||||
|
@ -189,11 +189,15 @@ forget_passphrase (const char *keygrip)
|
||||
|
||||
rc = asprintf (&line, "CLEAR_PASSPHRASE %s\n", keygrip);
|
||||
if (rc < 0)
|
||||
rc = gpg_error_from_syserror ();
|
||||
else
|
||||
rc = map_spwq_error (simple_query (line));
|
||||
if (rc)
|
||||
{
|
||||
log_error ("clearing passphrase failed: %s\n",
|
||||
gpg_strerror (gpg_error_from_syserror ()));
|
||||
log_error ("clearing passphrase failed: %s\n", gpg_strerror (rc));
|
||||
return;
|
||||
}
|
||||
|
||||
xfree (line);
|
||||
}
|
||||
|
||||
|
@ -1,3 +1,22 @@
|
||||
2010-06-07 Werner Koch <wk@g10code.com>
|
||||
|
||||
* estream.c (es_fname_get, es_fname_set): New.
|
||||
(fname_set_internal): New.
|
||||
(struct estream_internal): Add fields printable_fname and
|
||||
printable_fname_inuse.
|
||||
(_es_get_std_stream): Set stream name.
|
||||
(es_fopen, es_freopen, es_deinitialize): Set fname.
|
||||
|
||||
* exechelp-posix.c (gnupg_spawn_process): Allow passing INFILE or
|
||||
OUTFILE as NULL.
|
||||
* exechelp-w32.c (gnupg_spawn_process): Ditto.
|
||||
* exechelp-w32ce.c (gnupg_spawn_process): Return an error for
|
||||
INFILE or OUTFILE passed as NULL.
|
||||
|
||||
2010-06-01 Werner Koch <wk@g10code.com>
|
||||
|
||||
* logging.c (log_get_stream): Make sture a log stream is available.
|
||||
|
||||
2010-05-30 Werner Koch <wk@g10code.com>
|
||||
|
||||
* init.c (writestring_via_estream): New.
|
||||
|
@ -211,6 +211,7 @@ struct estream_internal
|
||||
void *cookie; /* Cookie. */
|
||||
void *opaque; /* Opaque data. */
|
||||
unsigned int modeflags; /* Flags for the backend. */
|
||||
char *printable_fname; /* Malloced filename for es_fname_get. */
|
||||
off_t offset;
|
||||
es_cookie_read_function_t func_read;
|
||||
es_cookie_write_function_t func_write;
|
||||
@ -227,6 +228,7 @@ struct estream_internal
|
||||
unsigned int is_stdstream:1; /* This is a standard stream. */
|
||||
unsigned int stdstream_fd:2; /* 0, 1 or 2 for a standard stream. */
|
||||
unsigned int print_err: 1; /* Error in print_fun_writer. */
|
||||
unsigned int printable_fname_inuse: 1; /* es_fname_get has been used. */
|
||||
int print_errno; /* Errno from print_fun_writer. */
|
||||
size_t print_ntotal; /* Bytes written from in print_fun_writer. */
|
||||
FILE *print_fp; /* Stdio stream used by print_fun_writer. */
|
||||
@ -266,8 +268,12 @@ static unsigned char custom_std_fds_valid[3];
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
/* Local prototypes. */
|
||||
static void fname_set_internal (estream_t stream, const char *fname, int quote);
|
||||
|
||||
|
||||
|
||||
|
||||
/* Macros. */
|
||||
|
||||
/* Calculate array dimension. */
|
||||
@ -1275,6 +1281,8 @@ es_initialize (estream_t stream,
|
||||
stream->intern->is_stdstream = 0;
|
||||
stream->intern->stdstream_fd = 0;
|
||||
stream->intern->deallocate_buffer = 0;
|
||||
stream->intern->printable_fname = NULL;
|
||||
stream->intern->printable_fname_inuse = 0;
|
||||
|
||||
stream->data_len = 0;
|
||||
stream->data_offset = 0;
|
||||
@ -1314,6 +1322,9 @@ es_deinitialize (estream_t stream)
|
||||
if (func_close)
|
||||
SET_UNLESS_NONZERO (err, tmp_err, (*func_close) (stream->intern->cookie));
|
||||
|
||||
mem_free (stream->intern->printable_fname);
|
||||
stream->intern->printable_fname = NULL;
|
||||
stream->intern->printable_fname_inuse = 0;
|
||||
|
||||
return err;
|
||||
}
|
||||
@ -2190,6 +2201,9 @@ es_fopen (const char *ES__RESTRICT path, const char *ES__RESTRICT mode)
|
||||
if (err)
|
||||
goto out;
|
||||
|
||||
if (stream && path)
|
||||
fname_set_internal (stream, path, 1);
|
||||
|
||||
out:
|
||||
|
||||
if (err && create_called)
|
||||
@ -2467,6 +2481,9 @@ _es_get_std_stream (int fd)
|
||||
stream->intern->stdstream_fd = fd;
|
||||
if (fd == 2)
|
||||
es_set_buffering (stream, NULL, _IOLBF, 0);
|
||||
fname_set_internal (stream,
|
||||
fd == 0? "[stdin]" :
|
||||
fd == 1? "[stdout]" : "[stderr]", 0);
|
||||
}
|
||||
ESTREAM_LIST_UNLOCK;
|
||||
return stream;
|
||||
@ -2515,7 +2532,11 @@ es_freopen (const char *ES__RESTRICT path, const char *ES__RESTRICT mode,
|
||||
stream = NULL;
|
||||
}
|
||||
else
|
||||
ESTREAM_UNLOCK (stream);
|
||||
{
|
||||
if (stream && path)
|
||||
fname_set_internal (stream, path, 1);
|
||||
ESTREAM_UNLOCK (stream);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -3407,6 +3428,68 @@ es_opaque_get (estream_t stream)
|
||||
return opaque;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
fname_set_internal (estream_t stream, const char *fname, int quote)
|
||||
{
|
||||
if (stream->intern->printable_fname
|
||||
&& !stream->intern->printable_fname_inuse)
|
||||
{
|
||||
mem_free (stream->intern->printable_fname);
|
||||
stream->intern->printable_fname = NULL;
|
||||
}
|
||||
if (stream->intern->printable_fname)
|
||||
return; /* Can't change because it is in use. */
|
||||
|
||||
if (*fname != '[')
|
||||
quote = 0;
|
||||
else
|
||||
quote = !!quote;
|
||||
|
||||
stream->intern->printable_fname = mem_alloc (strlen (fname) + quote + 1);
|
||||
if (fname)
|
||||
{
|
||||
if (quote)
|
||||
stream->intern->printable_fname[0] = '\\';
|
||||
strcpy (stream->intern->printable_fname+quote, fname);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/* Set the filename attribute of STREAM. There is no error return.
|
||||
as long as STREAM is valid. This function is called internally by
|
||||
functions which open a filename. */
|
||||
void
|
||||
es_fname_set (estream_t stream, const char *fname)
|
||||
{
|
||||
if (fname)
|
||||
{
|
||||
ESTREAM_LOCK (stream);
|
||||
fname_set_internal (stream, fname, 1);
|
||||
ESTREAM_UNLOCK (stream);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/* Return the filename attribute of STREAM. In case no filename has
|
||||
been set, "[?]" will be returned. The returned file name is valid
|
||||
as long as STREAM is valid. */
|
||||
const char *
|
||||
es_fname_get (estream_t stream)
|
||||
{
|
||||
const char *fname;
|
||||
|
||||
ESTREAM_LOCK (stream);
|
||||
fname = stream->intern->printable_fname;
|
||||
if (fname)
|
||||
stream->intern->printable_fname_inuse = 1;
|
||||
ESTREAM_UNLOCK (stream);
|
||||
if (!fname)
|
||||
fname = "[?]";
|
||||
return fname;
|
||||
}
|
||||
|
||||
|
||||
/* Print a BUFFER to STREAM while replacing all control characters and
|
||||
the characters in DELIMITERS by standard C escape sequences.
|
||||
Returns 0 on success or -1 on error. If BYTES_WRITTEN is not NULL
|
||||
|
@ -128,6 +128,8 @@
|
||||
#define es_tmpfile _ESTREAM_PREFIX(es_tmpfile)
|
||||
#define es_opaque_set _ESTREAM_PREFIX(es_opaque_set)
|
||||
#define es_opaque_get _ESTREAM_PREFIX(es_opaque_get)
|
||||
#define es_fname_set _ESTREAM_PREFIX(es_fname_set)
|
||||
#define es_fname_get _ESTREAM_PREFIX(es_fname_get)
|
||||
#define es_write_sanitized_utf8_buffer \
|
||||
_ESTREAM_PREFIX(es_write_sanitized_utf8_buffer)
|
||||
#endif /*_ESTREAM_EXT_SYM_PREFIX*/
|
||||
@ -358,6 +360,9 @@ estream_t es_tmpfile (void);
|
||||
void es_opaque_set (estream_t ES__RESTRICT stream, void *ES__RESTRICT opaque);
|
||||
void *es_opaque_get (estream_t stream);
|
||||
|
||||
void es_fname_set (estream_t stream, const char *fname);
|
||||
const char *es_fname_get (estream_t stream);
|
||||
|
||||
|
||||
#ifdef GNUPG_MAJOR_VERSION
|
||||
int es_write_sanitized_utf8_buffer (estream_t stream,
|
||||
|
@ -313,11 +313,22 @@ gnupg_spawn_process (const char *pgmname, const char *argv[],
|
||||
|
||||
*statusfile = NULL;
|
||||
*pid = (pid_t)(-1);
|
||||
es_fflush (infile);
|
||||
es_rewind (infile);
|
||||
fd = es_fileno (infile);
|
||||
fdout = es_fileno (outfile);
|
||||
if (fd == -1 || fdout == -1)
|
||||
|
||||
if (infile)
|
||||
{
|
||||
es_fflush (infile);
|
||||
es_rewind (infile);
|
||||
fd = es_fileno (infile);
|
||||
}
|
||||
else
|
||||
fd = -1;
|
||||
|
||||
if (outfile)
|
||||
fdout = es_fileno (outfile);
|
||||
else
|
||||
fdout = -1;
|
||||
|
||||
if ((infile && fd == -1) || (outfile && fdout == -1))
|
||||
log_fatal ("no file descriptor for file passed to gnupg_spawn_process\n");
|
||||
|
||||
if (pipe (rp) == -1)
|
||||
|
@ -382,17 +382,30 @@ gnupg_spawn_process (const char *pgmname, const char *argv[],
|
||||
int cr_flags;
|
||||
char *cmdline;
|
||||
int fd, fdout, rp[2];
|
||||
HANDLE nullhd[];
|
||||
int i;
|
||||
|
||||
(void)preexec;
|
||||
|
||||
/* Setup return values. */
|
||||
*statusfile = NULL;
|
||||
*pid = (pid_t)(-1);
|
||||
es_fflush (infile);
|
||||
es_rewind (infile);
|
||||
fd = _get_osfhandle (es_fileno (infile));
|
||||
fdout = _get_osfhandle (es_fileno (outfile));
|
||||
if (fd == -1 || fdout == -1)
|
||||
|
||||
if (infile)
|
||||
{
|
||||
es_fflush (infile);
|
||||
es_rewind (infile);
|
||||
fd = _get_osfhandle (es_fileno (infile));
|
||||
}
|
||||
else
|
||||
fd = -1;
|
||||
|
||||
if (outfile)
|
||||
fdout = _get_osfhandle (es_fileno (outfile));
|
||||
else
|
||||
fdout = -1;
|
||||
|
||||
if ( (infile && fd == -1) || (outfile && fdout == -1))
|
||||
log_fatal ("no file descriptor for file passed to gnupg_spawn_process\n");
|
||||
|
||||
/* Prepare security attributes. */
|
||||
@ -414,14 +427,17 @@ gnupg_spawn_process (const char *pgmname, const char *argv[],
|
||||
return err;
|
||||
}
|
||||
|
||||
nullhd[0] = fd == -1? w32_open_null (0) : INVALID_HANDLE_VALUE;
|
||||
nullhd[1] = outfd == -1? w32_open_null (1) : INVALID_HANDLE_VALUE;
|
||||
|
||||
/* Start the process. Note that we can't run the PREEXEC function
|
||||
because this would change our own environment. */
|
||||
memset (&si, 0, sizeof si);
|
||||
si.cb = sizeof (si);
|
||||
si.dwFlags = STARTF_USESTDHANDLES | STARTF_USESHOWWINDOW;
|
||||
si.wShowWindow = DEBUG_W32_SPAWN? SW_SHOW : SW_MINIMIZE;
|
||||
si.hStdInput = fd_to_handle (fd);
|
||||
si.hStdOutput = fd_to_handle (fdout);
|
||||
si.hStdInput = fd == -1? nullhd[0] : fd_to_handle (fd);
|
||||
si.hStdOutput = outfd == -1? nullhd[1] : fd_to_handle (fdout);
|
||||
si.hStdError = fd_to_handle (rp[1]);
|
||||
|
||||
cr_flags = (CREATE_DEFAULT_ERROR_MODE
|
||||
@ -450,6 +466,11 @@ gnupg_spawn_process (const char *pgmname, const char *argv[],
|
||||
xfree (cmdline);
|
||||
cmdline = NULL;
|
||||
|
||||
/* Close the inherited handles to /dev/null. */
|
||||
for (i=0; i < DIM (nullhd); i++)
|
||||
if (nullhd[i] != INVALID_HANDLE_VALUE)
|
||||
CloseHandle (nullhd[i]);
|
||||
|
||||
/* Close the other end of the pipe. */
|
||||
CloseHandle (fd_to_handle (rp[1]));
|
||||
|
||||
|
@ -501,6 +501,11 @@ gnupg_spawn_process (const char *pgmname, const char *argv[],
|
||||
*statusfile = NULL;
|
||||
*pid = (pid_t)(-1);
|
||||
|
||||
/* A NULL INFILE or OUTFILE is only used by gpgtar thus we don't
|
||||
need to implement this for CE. */
|
||||
if (!infile || !outfile)
|
||||
return gpg_error (GPG_ERR_NOT_IMPLEMENTED);
|
||||
|
||||
es_fflush (infile);
|
||||
es_rewind (infile);
|
||||
|
||||
|
@ -53,13 +53,14 @@ gpg_error_t gnupg_create_outbound_pipe (int filedes[2]);
|
||||
|
||||
|
||||
/* Fork and exec the PGMNAME, connect the file descriptor of INFILE to
|
||||
stdin, write the output to OUTFILE, return a new stream in
|
||||
STATUSFILE for stderr and the pid of the process in PID. The
|
||||
arguments for the process are expected in the NULL terminated array
|
||||
ARGV. The program name itself should not be included there. If
|
||||
PREEXEC is not NULL, that function will be called right before the
|
||||
exec. Calling gnupg_wait_process is required. Returns 0 on
|
||||
success or an error code.
|
||||
stdin, write the output to OUTFILE. INFILE or PUTFILE may be NULL
|
||||
to connect thenm to /dev/null. Returns a new stream in STATUSFILE
|
||||
for stderr and the pid of the process in PID. The arguments for the
|
||||
process are expected in the NULL terminated array ARGV. The
|
||||
program name itself should not be included there. If PREEXEC is
|
||||
not NULL, that function will be called right before the exec.
|
||||
Calling gnupg_wait_process is required. Returns 0 on success or an
|
||||
error code.
|
||||
|
||||
FLAGS is a bit vector:
|
||||
|
||||
|
@ -402,7 +402,11 @@ log_get_fd ()
|
||||
estream_t
|
||||
log_get_stream ()
|
||||
{
|
||||
assert (logstream);
|
||||
if (!logstream)
|
||||
{
|
||||
log_set_file (NULL); /* Make sure a log stream has been set. */
|
||||
assert (logstream);
|
||||
}
|
||||
return logstream;
|
||||
}
|
||||
|
||||
|
@ -89,6 +89,7 @@ GNUPG_BUILD_PROGRAM(g13, yes)
|
||||
GNUPG_BUILD_PROGRAM(tools, yes)
|
||||
GNUPG_BUILD_PROGRAM(doc, yes)
|
||||
GNUPG_BUILD_PROGRAM(symcryptrun, no)
|
||||
GNUPG_BUILD_PROGRAM(gpgtar, no)
|
||||
|
||||
AC_SUBST(PACKAGE)
|
||||
AC_SUBST(PACKAGE_GT)
|
||||
@ -1156,8 +1157,8 @@ AC_CHECK_FUNCS([strcasecmp strncasecmp ctermid times gmtime_r])
|
||||
AC_CHECK_FUNCS([unsetenv fcntl ftruncate])
|
||||
AC_CHECK_FUNCS([gettimeofday getrusage getrlimit setrlimit clock_gettime])
|
||||
AC_CHECK_FUNCS([atexit raise getpagesize strftime nl_langinfo setlocale])
|
||||
AC_CHECK_FUNCS([waitpid wait4 sigaction sigprocmask pipe stat getaddrinfo])
|
||||
AC_CHECK_FUNCS([ttyname rand ftello fsync])
|
||||
AC_CHECK_FUNCS([waitpid wait4 sigaction sigprocmask pipe getaddrinfo])
|
||||
AC_CHECK_FUNCS([ttyname rand ftello fsync stat lstat])
|
||||
|
||||
AC_CHECK_TYPES([struct sigaction, sigset_t],,,[#include <signal.h>])
|
||||
|
||||
@ -1473,6 +1474,7 @@ AM_CONDITIONAL(BUILD_G13, test "$build_g13" = "yes")
|
||||
AM_CONDITIONAL(BUILD_TOOLS, test "$build_tools" = "yes")
|
||||
AM_CONDITIONAL(BUILD_DOC, test "$build_doc" = "yes")
|
||||
AM_CONDITIONAL(BUILD_SYMCRYPTRUN, test "$build_symcryptrun" = "yes")
|
||||
AM_CONDITIONAL(BUILD_GPGTAR,test "$build_gpgtar" = "yes")
|
||||
|
||||
AM_CONDITIONAL(RUN_GPG_TESTS,
|
||||
test x$cross_compiling = xno -a "$build_gpg" = yes )
|
||||
@ -1583,6 +1585,7 @@ echo "
|
||||
Agent: $build_agent $build_agent_threaded
|
||||
Smartcard: $build_scdaemon $build_scdaemon_extra
|
||||
G13: $build_g13
|
||||
Gpgtar: $build_gpgtar
|
||||
|
||||
Protect tool: $show_gnupg_protect_tool_pgm
|
||||
Default agent: $show_gnupg_agent_pgm
|
||||
|
@ -845,7 +845,8 @@ key rings.
|
||||
|
||||
@c man:.RS
|
||||
The listing shows you the key with its secondary keys and all user
|
||||
ids. Selected keys or user ids are indicated by an asterisk. The trust
|
||||
ids. The primary user id is indicated by a dot, and selected keys or
|
||||
user ids are indicated by an asterisk. The trust
|
||||
value is displayed with the primary key: the first is the assigned owner
|
||||
trust and the second is the calculated trust value. Letters are used for
|
||||
the values:
|
||||
|
@ -1,3 +1,10 @@
|
||||
2010-05-12 Werner Koch <wk@g10code.com>
|
||||
|
||||
* armor.c (radix64_read): Extended 2006-04-28 fix to fix bug#1179.
|
||||
|
||||
* plaintext.c (handle_plaintext): Check return code of fflush.
|
||||
Fixes bug#1207.
|
||||
|
||||
2010-05-07 Werner Koch <wk@g10code.com>
|
||||
|
||||
* import.c (fix_bad_direct_key_sigs): New.
|
||||
|
@ -798,8 +798,9 @@ radix64_read( armor_filter_context_t *afx, IOBUF a, size_t *retn,
|
||||
goto again;
|
||||
}
|
||||
}
|
||||
else if(n==0)
|
||||
onlypad=1;
|
||||
|
||||
if (!n)
|
||||
onlypad = 1;
|
||||
|
||||
if( idx == 1 )
|
||||
buf[n++] = val;
|
||||
|
@ -1360,6 +1360,8 @@ generate_card_keys (void)
|
||||
{
|
||||
char *answer;
|
||||
|
||||
/* FIXME: Should be something like cpr_get_bool so that a status
|
||||
GET_BOOL will be emitted. */
|
||||
answer = cpr_get ("cardedit.genkeys.backup_enc",
|
||||
_("Make off-card backup of encryption key? (Y/n) "));
|
||||
|
||||
|
@ -448,7 +448,15 @@ handle_plaintext (PKT_plaintext * pt, md_filter_context_t * mfx,
|
||||
/* Make sure that stdout gets flushed after the plaintext has been
|
||||
handled. This is for extra security as we do a flush anyway
|
||||
before checking the signature. */
|
||||
fflush (stdout);
|
||||
if (fflush (stdout))
|
||||
{
|
||||
/* We need to check the return code to detect errors like disk
|
||||
full for short plaintexts. See bug#1207. Checking return
|
||||
values is a good idea in any case. */
|
||||
if (!rc)
|
||||
rc = gpg_error_from_syserror ();
|
||||
log_error ("error flushing `%s': %s\n", "[stdout]", strerror (errno));
|
||||
}
|
||||
|
||||
if (fp && fp != stdout && fp != opt.outfp)
|
||||
fclose (fp);
|
||||
|
@ -48,7 +48,7 @@ struct runner_s
|
||||
|
||||
|
||||
/* We use a reference counter to know when it is safe to remove the
|
||||
object. Lackiong an explicit ref fucntion this counter will take
|
||||
object. Lacking an explicit ref function this counter will take
|
||||
only these two values:
|
||||
|
||||
1 = Thread not running or only the thread is still running.
|
||||
|
@ -1,3 +1,7 @@
|
||||
2010-05-12 Werner Koch <wk@g10code.com>
|
||||
|
||||
* armor.test (Version): Add test for bug#1179.
|
||||
|
||||
2010-05-11 Werner Koch <wk@g10code.com>
|
||||
|
||||
* genkey1024.test: Use GPG macro.
|
||||
|
@ -185,4 +185,579 @@ $GPG --import x || error "the $i bug is back in town"
|
||||
|
||||
|
||||
|
||||
nopad_armored_msg='-----BEGIN PGP MESSAGE-----
|
||||
Version: GnuPG v1.4.11-svn5139 (GNU/Linux)
|
||||
|
||||
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
|
||||
S96L4PDmXseFkVZOH/7dNWyKuvSA/MmthkN14lWJiYJebaXb7oZG3XDC70o7bG6Y
|
||||
mtGOrOwVthFereg1Ii98ZA4nKgqeu2paMju/t03hQXHY5iqyV9ax0A3B0rdivz4U
|
||||
VbqCcO7pGNb/Ki3gGc3hfVw9YXnR7E+tra19eE7UM7o+YlunKF1Q5dJaLKY1z3L9
|
||||
3UAJmXG/sZcLqrDn9zHfb/YxqbyIkM7VIU2kKztWCCNiNKgLQsIoVzK4sbS2LVDI
|
||||
grmvZg+Q7EkwxZ1FXeBGJmsiyjKYPU3PI8ZBU1MXjwTjtnQRuBSxh+Ok9glPeE+1
|
||||
rUDdlVoogUGgPAvLV3BM43E/Q5VE+X04KvPNKINvvS1nZpGKyyy1MayOfstc7Hsw
|
||||
W/RgllTEd7VW5jgZKg7YyA8r8cUjMqE7zewrq6MtkoFk6ZVslWxfoADBW3ivwdwe
|
||||
I4XwNboOgBx77qxf95aHRRdqWZa0JF8zvK5BH6EpBI29hlJU4leax696ACA7QIIM
|
||||
Rpc+ulkvHsNxsUlrUut9AXoob0MqDJRmZvU4gSVivHjvKLEIlymJw4pLbMWhcrNa
|
||||
v94TExP21Fy/zYcWiZT2nCX6qZVXTcUcQGIrBTWDbR8bxKll4FNo/2zmelyFmeXt
|
||||
iH+Zp2EwFAUELsrxx9plNd5WceyU9VnZoeucNFd2QFl7lV9KL8AlQaaMdsYOt27m
|
||||
8j7iNBID4HtCYM0xLwE/5uqVQPxX6qzPPZc66MSRIi8ZHZPFvVilFVzFyZSrq5Lo
|
||||
ojQQmAycQGOzx7dwx7vi9SeTrBRY2PK3WVurcLiRxM+2u9vjlxigwimpnK3VU1dh
|
||||
GDoPghB3O34bfiV5GndcnPDLJxsSGWz2z2WiyCFgCkVp3shHUN5c2PLWDIOf/Ejz
|
||||
LzNJXFKjO6/6ZGpMmGf6bZCa9MBnvXMNic9/k0lmtbgj8SS+2//gko1EDa/Gpaqo
|
||||
J2xgEuHLvp9KQABWt9VRI+tNUJQCS/Jq10ZiT1TzhczyJEqbtVDOo0l5A5sMV0PO
|
||||
HszFNxw2BmuM8RYGgAe5pkrdzVdtB8TLmhys7P+xRXNXnsUb+878kt2EyiXjMl1g
|
||||
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
|
||||
=3akT
|
||||
-----END PGP MESSAGE-----
|
||||
|
||||
'
|
||||
|
||||
alpha_seckey='-----BEGIN PGP PRIVATE KEY BLOCK-----
|
||||
Version: GnuPG v1.4.8 (GNU/Linux)
|
||||
|
||||
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
|
||||
wg7Md81a5RI3F2FG8747t9gX
|
||||
=VM1e
|
||||
-----END PGP PRIVATE KEY BLOCK-----
|
||||
'
|
||||
|
||||
# Bug 1179 solved 2010-05-12:
|
||||
# It occured for messages of a multiple of the iobuf block size where
|
||||
# the last line had no pad character. Due to premature poppng of thea
|
||||
# rmor filter gpg swalled the CRC line and passed the '-----END...'
|
||||
# line on to the decryption layer.
|
||||
|
||||
i=alpha_seckey
|
||||
info "importing: $i"
|
||||
eval "(IFS=; echo \"\$$i\")" >x
|
||||
$GPG --import x || true
|
||||
|
||||
i=nopad_armored_msg
|
||||
info "checking: $i"
|
||||
eval "(IFS=; echo \"\$$i\")" >x
|
||||
if $GPG -o - x > /dev/null ; then
|
||||
:
|
||||
else
|
||||
error "bug#1179 is back in town"
|
||||
fi
|
||||
|
||||
|
@ -1,3 +1,11 @@
|
||||
2010-06-07 Werner Koch <wk@g10code.com>
|
||||
|
||||
* gpgtar.c, gpgtar.h, gpgtar-list.c, gpgtar-create.c
|
||||
* gpgtar-extract.c: New.
|
||||
* Makefile.am (commonpth_libs): New.
|
||||
(gpgtar_SOURCES, gpgtar_CFLAGS, gpgtar_LDADD): New.
|
||||
(bin_PROGRAMS) [!W32CE]: Add gpgtar.
|
||||
|
||||
2010-04-20 Marcus Brinkmann <marcus@g10code.de>
|
||||
|
||||
* gpgconf-comp.c (option_check_validity): Use dummy variables to
|
||||
@ -1104,7 +1112,7 @@
|
||||
|
||||
|
||||
Copyright 2003, 2004, 2005, 2006, 2007, 2008,
|
||||
2009 Free Software Foundation, Inc.
|
||||
2009, 2010 Free Software Foundation, Inc.
|
||||
|
||||
This file is free software; as a special exception the author gives
|
||||
unlimited permission to copy and/or distribute it, with or without
|
||||
|
@ -42,13 +42,19 @@ else
|
||||
symcryptrun =
|
||||
endif
|
||||
|
||||
if BUILD_GPGTAR
|
||||
gpgtar = gpgtar
|
||||
else
|
||||
gpgtar =
|
||||
endif
|
||||
|
||||
# Fixme: We should remove the gpgkey2ssh tool.
|
||||
bin_PROGRAMS = gpgconf gpg-connect-agent ${symcryptrun}
|
||||
if !HAVE_W32_SYSTEM
|
||||
bin_PROGRAMS += watchgnupg gpgparsemail
|
||||
endif
|
||||
if !HAVE_W32CE_SYSTEM
|
||||
bin_PROGRAMS += gpgkey2ssh
|
||||
bin_PROGRAMS += gpgkey2ssh ${gpgtar}
|
||||
endif
|
||||
|
||||
if !DISABLE_REGEX
|
||||
@ -60,6 +66,7 @@ noinst_PROGRAMS = clean-sat mk-tdata make-dns-cert gpgsplit
|
||||
endif
|
||||
|
||||
common_libs = $(libcommon) ../gl/libgnu.a
|
||||
commonpth_libs = $(libcommonpth) ../gl/libgnu.a
|
||||
if HAVE_W32CE_SYSTEM
|
||||
pwquery_libs =
|
||||
else
|
||||
@ -114,14 +121,15 @@ gpg_check_pattern_LDADD = $(common_libs) $(LIBGCRYPT_LIBS) $(GPG_ERROR_LIBS) \
|
||||
$(LIBINTL) $(LIBICONV) $(W32SOCKLIBS)
|
||||
endif
|
||||
|
||||
#gpgtar_SOURCES = \
|
||||
# gpgtar.c gpgtar.h \
|
||||
# gpgtar-create.c \
|
||||
# gpgtar-extract.c \
|
||||
# gpgtar-list.c \
|
||||
# no-libgcrypt.c
|
||||
#gpgtar_LDADD = $(common_libs)
|
||||
#
|
||||
gpgtar_SOURCES = \
|
||||
gpgtar.c gpgtar.h \
|
||||
gpgtar-create.c \
|
||||
gpgtar-extract.c \
|
||||
gpgtar-list.c \
|
||||
no-libgcrypt.c
|
||||
gpgtar_CFLAGS = $(GPG_ERROR_CFLAGS) $(PTH_CFLAGS)
|
||||
gpgtar_LDADD = $(commonpth_libs) $(PTH_LIBS) $(GPG_ERROR_LIBS)
|
||||
|
||||
|
||||
# Make sure that all libs are build before we use them. This is
|
||||
# important for things like make -j2.
|
||||
|
643
tools/gpgtar-create.c
Normal file
643
tools/gpgtar-create.c
Normal file
@ -0,0 +1,643 @@
|
||||
/* gpgtar-create.c - Create a TAR archive
|
||||
* Copyright (C) 2010 Free Software Foundation, Inc.
|
||||
*
|
||||
* This file is part of GnuPG.
|
||||
*
|
||||
* GnuPG is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* GnuPG is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <config.h>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <dirent.h>
|
||||
/* #ifdef HAVE_W32_SYSTEM */
|
||||
/* # define WIN32_LEAN_AND_MEAN */
|
||||
/* # include <windows.h> */
|
||||
/* #else /\*!HAVE_W32_SYSTEM*\/ */
|
||||
# include <unistd.h>
|
||||
# include <pwd.h>
|
||||
# include <grp.h>
|
||||
/* #endif /\*!HAVE_W32_SYSTEM*\/ */
|
||||
#include <assert.h>
|
||||
|
||||
#include "i18n.h"
|
||||
#include "../common/sysutils.h"
|
||||
#include "gpgtar.h"
|
||||
|
||||
#ifndef HAVE_LSTAT
|
||||
#define lstat(a,b) stat ((a), (b))
|
||||
#endif
|
||||
|
||||
|
||||
/* Object to control the file scanning. */
|
||||
struct scanctrl_s;
|
||||
typedef struct scanctrl_s *scanctrl_t;
|
||||
struct scanctrl_s
|
||||
{
|
||||
tar_header_t flist;
|
||||
tar_header_t *flist_tail;
|
||||
int nestlevel;
|
||||
};
|
||||
|
||||
|
||||
|
||||
|
||||
/* Given a fresh header object HDR with only the name field set, try
|
||||
to gather all available info. */
|
||||
static gpg_error_t
|
||||
fillup_entry (tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
struct stat sbuf;
|
||||
|
||||
if (lstat (hdr->name, &sbuf))
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error stat-ing `%s': %s\n", hdr->name, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
|
||||
if (S_ISREG (sbuf.st_mode))
|
||||
hdr->typeflag = TF_REGULAR;
|
||||
else if (S_ISDIR (sbuf.st_mode))
|
||||
hdr->typeflag = TF_DIRECTORY;
|
||||
else if (S_ISCHR (sbuf.st_mode))
|
||||
hdr->typeflag = TF_CHARDEV;
|
||||
else if (S_ISBLK (sbuf.st_mode))
|
||||
hdr->typeflag = TF_BLOCKDEV;
|
||||
else if (S_ISFIFO (sbuf.st_mode))
|
||||
hdr->typeflag = TF_FIFO;
|
||||
else if (S_ISLNK (sbuf.st_mode))
|
||||
hdr->typeflag = TF_SYMLINK;
|
||||
else
|
||||
hdr->typeflag = TF_NOTSUP;
|
||||
|
||||
/* FIXME: Save DEV and INO? */
|
||||
|
||||
/* Set the USTAR defined mode bits using the system macros. */
|
||||
if (sbuf.st_mode & S_IRUSR)
|
||||
hdr->mode |= 0400;
|
||||
if (sbuf.st_mode & S_IWUSR)
|
||||
hdr->mode |= 0200;
|
||||
if (sbuf.st_mode & S_IXUSR)
|
||||
hdr->mode |= 0100;
|
||||
if (sbuf.st_mode & S_IRGRP)
|
||||
hdr->mode |= 0040;
|
||||
if (sbuf.st_mode & S_IWGRP)
|
||||
hdr->mode |= 0020;
|
||||
if (sbuf.st_mode & S_IXGRP)
|
||||
hdr->mode |= 0010;
|
||||
if (sbuf.st_mode & S_IROTH)
|
||||
hdr->mode |= 0004;
|
||||
if (sbuf.st_mode & S_IWOTH)
|
||||
hdr->mode |= 0002;
|
||||
if (sbuf.st_mode & S_IXOTH)
|
||||
hdr->mode |= 0001;
|
||||
#ifdef S_IXUID
|
||||
if (sbuf.st_mode & S_IXUID)
|
||||
hdr->mode |= 04000;
|
||||
#endif
|
||||
#ifdef S_IXGID
|
||||
if (sbuf.st_mode & S_IXGID)
|
||||
hdr->mode |= 02000;
|
||||
#endif
|
||||
#ifdef S_ISVTX
|
||||
if (sbuf.st_mode & S_ISVTX)
|
||||
hdr->mode |= 01000;
|
||||
#endif
|
||||
|
||||
hdr->nlink = sbuf.st_nlink;
|
||||
|
||||
hdr->uid = sbuf.st_uid;
|
||||
hdr->gid = sbuf.st_gid;
|
||||
|
||||
/* Only set the size for a regular file. */
|
||||
if (hdr->typeflag == TF_REGULAR)
|
||||
hdr->size = sbuf.st_size;
|
||||
|
||||
hdr->mtime = sbuf.st_mtime;
|
||||
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
add_entry (const char *dname, size_t dnamelen, struct dirent *de,
|
||||
scanctrl_t scanctrl)
|
||||
{
|
||||
gpg_error_t err;
|
||||
tar_header_t hdr;
|
||||
char *p;
|
||||
|
||||
assert (dnamelen);
|
||||
|
||||
hdr = xtrycalloc (1, sizeof *hdr + dnamelen + 1
|
||||
+ (de? strlen (de->d_name) : 0));
|
||||
if (!hdr)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error (_("error reading directory `%s': %s\n"),
|
||||
dname, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
|
||||
p = stpcpy (hdr->name, dname);
|
||||
if (de)
|
||||
{
|
||||
if (dname[dnamelen-1] != '/')
|
||||
*p++ = '/';
|
||||
strcpy (p, de->d_name);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (hdr->name[dnamelen-1] == '/')
|
||||
hdr->name[dnamelen-1] = 0;
|
||||
}
|
||||
#ifdef HAVE_DOSISH_SYSTEM
|
||||
for (p=hdr->name; *p; p++)
|
||||
if (*p == '\\')
|
||||
*p = '/';
|
||||
#endif
|
||||
err = fillup_entry (hdr);
|
||||
if (err)
|
||||
xfree (hdr);
|
||||
else
|
||||
{
|
||||
if (opt.verbose)
|
||||
gpgtar_print_header (hdr, log_get_stream ());
|
||||
*scanctrl->flist_tail = hdr;
|
||||
scanctrl->flist_tail = &hdr->next;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
scan_directory (const char *dname, scanctrl_t scanctrl)
|
||||
{
|
||||
gpg_error_t err = 0;
|
||||
size_t dnamelen;
|
||||
DIR *dir;
|
||||
struct dirent *de;
|
||||
|
||||
dnamelen = strlen (dname);
|
||||
if (!dnamelen)
|
||||
return 0; /* An empty directory name has no entries. */
|
||||
|
||||
dir = opendir (dname);
|
||||
if (!dir)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error (_("error reading directory `%s': %s\n"),
|
||||
dname, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
|
||||
while ((de = readdir (dir)))
|
||||
{
|
||||
if (!strcmp (de->d_name, "." ) || !strcmp (de->d_name, ".."))
|
||||
continue; /* Skip self and parent dir entry. */
|
||||
|
||||
err = add_entry (dname, dnamelen, de, scanctrl);
|
||||
if (err)
|
||||
goto leave;
|
||||
}
|
||||
|
||||
leave:
|
||||
closedir (dir);
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
scan_recursive (const char *dname, scanctrl_t scanctrl)
|
||||
{
|
||||
gpg_error_t err = 0;
|
||||
tar_header_t hdr, *start_tail, *stop_tail;
|
||||
|
||||
if (scanctrl->nestlevel > 200)
|
||||
{
|
||||
log_error ("directories too deeply nested\n");
|
||||
return gpg_error (GPG_ERR_RESOURCE_LIMIT);
|
||||
}
|
||||
scanctrl->nestlevel++;
|
||||
|
||||
assert (scanctrl->flist_tail);
|
||||
start_tail = scanctrl->flist_tail;
|
||||
scan_directory (dname, scanctrl);
|
||||
stop_tail = scanctrl->flist_tail;
|
||||
hdr = *start_tail;
|
||||
for (; hdr && hdr != *stop_tail; hdr = hdr->next)
|
||||
if (hdr->typeflag == TF_DIRECTORY)
|
||||
{
|
||||
if (opt.verbose > 1)
|
||||
log_info ("scanning directory `%s'\n", hdr->name);
|
||||
scan_recursive (hdr->name, scanctrl);
|
||||
}
|
||||
|
||||
scanctrl->nestlevel--;
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
/* Returns true if PATTERN is acceptable. */
|
||||
static int
|
||||
pattern_valid_p (const char *pattern)
|
||||
{
|
||||
if (!*pattern)
|
||||
return 0;
|
||||
if (*pattern == '.' && pattern[1] == '.')
|
||||
return 0;
|
||||
if (*pattern == '/' || *pattern == DIRSEP_C)
|
||||
return 0; /* Absolute filenames are not supported. */
|
||||
#ifdef HAVE_DRIVE_LETTERS
|
||||
if (((*pattern >= 'a' && *pattern <= 'z')
|
||||
|| (*pattern >= 'A' && *pattern <= 'Z'))
|
||||
&& pattern[1] == ':')
|
||||
return 0; /* Drive letter are not allowed either. */
|
||||
#endif /*HAVE_DRIVE_LETTERS*/
|
||||
|
||||
return 1; /* Okay. */
|
||||
}
|
||||
|
||||
|
||||
|
||||
static void
|
||||
store_xoctal (char *buffer, size_t length, unsigned long long value)
|
||||
{
|
||||
char *p, *pend;
|
||||
size_t n;
|
||||
unsigned long long v;
|
||||
|
||||
assert (length > 1);
|
||||
|
||||
v = value;
|
||||
n = length;
|
||||
p = pend = buffer + length;
|
||||
*--p = 0; /* Nul byte. */
|
||||
n--;
|
||||
do
|
||||
{
|
||||
*--p = '0' + (v % 8);
|
||||
v /= 8;
|
||||
n--;
|
||||
}
|
||||
while (v && n);
|
||||
if (!v)
|
||||
{
|
||||
/* Pad. */
|
||||
for ( ; n; n--)
|
||||
*--p = '0';
|
||||
}
|
||||
else /* Does not fit into the field. Store as binary number. */
|
||||
{
|
||||
v = value;
|
||||
n = length;
|
||||
p = pend = buffer + length;
|
||||
do
|
||||
{
|
||||
*--p = v;
|
||||
v /= 256;
|
||||
n--;
|
||||
}
|
||||
while (v && n);
|
||||
if (!v)
|
||||
{
|
||||
/* Pad. */
|
||||
for ( ; n; n--)
|
||||
*--p = 0;
|
||||
if (*p & 0x80)
|
||||
BUG ();
|
||||
*p |= 0x80; /* Set binary flag. */
|
||||
}
|
||||
else
|
||||
BUG ();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
store_uname (char *buffer, size_t length, unsigned long uid)
|
||||
{
|
||||
static int initialized;
|
||||
static unsigned long lastuid;
|
||||
static char lastuname[32];
|
||||
|
||||
if (!initialized || uid != lastuid)
|
||||
{
|
||||
struct passwd *pw = getpwuid (uid);
|
||||
|
||||
lastuid = uid;
|
||||
initialized = 1;
|
||||
if (pw)
|
||||
mem2str (lastuname, pw->pw_name, sizeof lastuname);
|
||||
else
|
||||
{
|
||||
log_info ("failed to get name for uid %lu\n", uid);
|
||||
*lastuname = 0;
|
||||
}
|
||||
}
|
||||
mem2str (buffer, lastuname, length);
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
store_gname (char *buffer, size_t length, unsigned long gid)
|
||||
{
|
||||
static int initialized;
|
||||
static unsigned long lastgid;
|
||||
static char lastgname[32];
|
||||
|
||||
if (!initialized || gid != lastgid)
|
||||
{
|
||||
struct group *gr = getgrgid (gid);
|
||||
|
||||
lastgid = gid;
|
||||
initialized = 1;
|
||||
if (gr)
|
||||
mem2str (lastgname, gr->gr_name, sizeof lastgname);
|
||||
else
|
||||
{
|
||||
log_info ("failed to get name for gid %lu\n", gid);
|
||||
*lastgname = 0;
|
||||
}
|
||||
}
|
||||
mem2str (buffer, lastgname, length);
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
build_header (void *record, tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
struct ustar_raw_header *raw = record;
|
||||
size_t namelen, n;
|
||||
unsigned long chksum;
|
||||
unsigned char *p;
|
||||
|
||||
memset (record, 0, RECORDSIZE);
|
||||
|
||||
/* Store name and prefix. */
|
||||
namelen = strlen (hdr->name);
|
||||
if (namelen < sizeof raw->name)
|
||||
memcpy (raw->name, hdr->name, namelen);
|
||||
else
|
||||
{
|
||||
n = (namelen < sizeof raw->prefix)? namelen : sizeof raw->prefix;
|
||||
for (n--; n ; n--)
|
||||
if (hdr->name[n] == '/')
|
||||
break;
|
||||
if (namelen - n < sizeof raw->name)
|
||||
{
|
||||
/* Note that the N is < sizeof prefix and that the
|
||||
delimiting slash is not stored. */
|
||||
memcpy (raw->prefix, hdr->name, n);
|
||||
memcpy (raw->name, hdr->name+n+1, namelen - n);
|
||||
}
|
||||
else
|
||||
{
|
||||
err = gpg_error (GPG_ERR_TOO_LARGE);
|
||||
log_error ("error storing file `%s': %s\n",
|
||||
hdr->name, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
}
|
||||
|
||||
store_xoctal (raw->mode, sizeof raw->mode, hdr->mode);
|
||||
store_xoctal (raw->uid, sizeof raw->uid, hdr->uid);
|
||||
store_xoctal (raw->gid, sizeof raw->gid, hdr->gid);
|
||||
store_xoctal (raw->size, sizeof raw->size, hdr->size);
|
||||
store_xoctal (raw->mtime, sizeof raw->mtime, hdr->mtime);
|
||||
|
||||
switch (hdr->typeflag)
|
||||
{
|
||||
case TF_REGULAR: raw->typeflag[0] = '0'; break;
|
||||
case TF_HARDLINK: raw->typeflag[0] = '1'; break;
|
||||
case TF_SYMLINK: raw->typeflag[0] = '2'; break;
|
||||
case TF_CHARDEV: raw->typeflag[0] = '3'; break;
|
||||
case TF_BLOCKDEV: raw->typeflag[0] = '4'; break;
|
||||
case TF_DIRECTORY: raw->typeflag[0] = '5'; break;
|
||||
case TF_FIFO: raw->typeflag[0] = '6'; break;
|
||||
default: return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
||||
}
|
||||
|
||||
memcpy (raw->magic, "ustar", 6);
|
||||
raw->version[0] = '0';
|
||||
raw->version[1] = '0';
|
||||
|
||||
store_uname (raw->uname, sizeof raw->uname, hdr->uid);
|
||||
store_gname (raw->gname, sizeof raw->gname, hdr->gid);
|
||||
|
||||
if (hdr->typeflag == TF_SYMLINK)
|
||||
{
|
||||
int nread;
|
||||
|
||||
nread = readlink (hdr->name, raw->linkname, sizeof raw->linkname -1);
|
||||
if (nread < 0)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error reading symlink `%s': %s\n",
|
||||
hdr->name, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
raw->linkname[nread] = 0;
|
||||
}
|
||||
|
||||
|
||||
/* Compute the checksum. */
|
||||
memset (raw->checksum, ' ', sizeof raw->checksum);
|
||||
chksum = 0;
|
||||
p = record;
|
||||
for (n=0; n < RECORDSIZE; n++)
|
||||
chksum += *p++;
|
||||
store_xoctal (raw->checksum, sizeof raw->checksum - 1, chksum);
|
||||
raw->checksum[7] = ' ';
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
write_file (estream_t stream, tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
char record[RECORDSIZE];
|
||||
estream_t infp;
|
||||
size_t nread, nbytes;
|
||||
int any;
|
||||
|
||||
err = build_header (record, hdr);
|
||||
if (err)
|
||||
{
|
||||
if (gpg_err_code (err) == GPG_ERR_NOT_SUPPORTED)
|
||||
{
|
||||
log_info ("skipping unsupported file `%s'\n", hdr->name);
|
||||
err = 0;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
if (hdr->typeflag == TF_REGULAR)
|
||||
{
|
||||
infp = es_fopen (hdr->name, "rb");
|
||||
if (!infp)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("can't open `%s': %s - skipped\n",
|
||||
hdr->name, gpg_strerror (err));
|
||||
return err;
|
||||
}
|
||||
}
|
||||
else
|
||||
infp = NULL;
|
||||
|
||||
err = write_record (stream, record);
|
||||
if (err)
|
||||
goto leave;
|
||||
|
||||
if (hdr->typeflag == TF_REGULAR)
|
||||
{
|
||||
hdr->nrecords = (hdr->size + RECORDSIZE-1)/RECORDSIZE;
|
||||
any = 0;
|
||||
while (hdr->nrecords--)
|
||||
{
|
||||
nbytes = hdr->nrecords? RECORDSIZE : (hdr->size % RECORDSIZE);
|
||||
nread = es_fread (record, 1, nbytes, infp);
|
||||
if (nread != nbytes)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error reading file `%s': %s%s\n",
|
||||
hdr->name, gpg_strerror (err),
|
||||
any? " (file shrunk?)":"");
|
||||
goto leave;
|
||||
}
|
||||
any = 1;
|
||||
err = write_record (stream, record);
|
||||
if (err)
|
||||
goto leave;
|
||||
}
|
||||
nread = es_fread (record, 1, 1, infp);
|
||||
if (nread)
|
||||
log_info ("note: file `%s' has grown\n", hdr->name);
|
||||
}
|
||||
|
||||
leave:
|
||||
if (err)
|
||||
es_fclose (infp);
|
||||
else if ((err = es_fclose (infp)))
|
||||
log_error ("error closing file `%s': %s\n", hdr->name, gpg_strerror (err));
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
write_eof_mark (estream_t stream)
|
||||
{
|
||||
gpg_error_t err;
|
||||
char record[RECORDSIZE];
|
||||
|
||||
memset (record, 0, sizeof record);
|
||||
err = write_record (stream, record);
|
||||
if (!err)
|
||||
err = write_record (stream, record);
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
|
||||
void
|
||||
gpgtar_create (char **inpattern)
|
||||
{
|
||||
gpg_error_t err = 0;
|
||||
const char *pattern;
|
||||
struct scanctrl_s scanctrl_buffer;
|
||||
scanctrl_t scanctrl = &scanctrl_buffer;
|
||||
tar_header_t hdr, *start_tail;
|
||||
estream_t outstream;
|
||||
|
||||
memset (scanctrl, 0, sizeof *scanctrl);
|
||||
scanctrl->flist_tail = &scanctrl->flist;
|
||||
|
||||
for (; (pattern = *inpattern); inpattern++)
|
||||
{
|
||||
if (!*pattern)
|
||||
continue;
|
||||
if (opt.verbose > 1)
|
||||
log_info ("scanning `%s'\n", pattern);
|
||||
|
||||
start_tail = scanctrl->flist_tail;
|
||||
if (!pattern_valid_p (pattern))
|
||||
log_error ("skipping invalid name `%s'\n", pattern);
|
||||
else if (!add_entry (pattern, strlen (pattern), NULL, scanctrl)
|
||||
&& *start_tail && ((*start_tail)->typeflag & TF_DIRECTORY))
|
||||
scan_recursive (pattern, scanctrl);
|
||||
}
|
||||
|
||||
if (opt.outfile)
|
||||
{
|
||||
outstream = es_fopen (opt.outfile, "wb");
|
||||
if (!outstream)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error (_("can't create `%s': %s\n"),
|
||||
opt.outfile, gpg_strerror (err));
|
||||
goto leave;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
outstream = es_stdout;
|
||||
}
|
||||
|
||||
for (hdr = scanctrl->flist; hdr; hdr = hdr->next)
|
||||
{
|
||||
err = write_file (outstream, hdr);
|
||||
if (err)
|
||||
goto leave;
|
||||
}
|
||||
err = write_eof_mark (outstream);
|
||||
|
||||
leave:
|
||||
if (!err)
|
||||
{
|
||||
if (outstream != es_stdout)
|
||||
err = es_fclose (outstream);
|
||||
else
|
||||
err = es_fflush (outstream);
|
||||
outstream = NULL;
|
||||
}
|
||||
if (err)
|
||||
{
|
||||
log_error ("creating tarball `%s' failed: %s\n",
|
||||
es_fname_get (outstream), gpg_strerror (err));
|
||||
if (outstream && outstream != es_stdout)
|
||||
es_fclose (outstream);
|
||||
if (opt.outfile)
|
||||
gnupg_remove (opt.outfile);
|
||||
}
|
||||
scanctrl->flist_tail = NULL;
|
||||
while ( (hdr = scanctrl->flist) )
|
||||
{
|
||||
scanctrl->flist = hdr->next;
|
||||
xfree (hdr);
|
||||
}
|
||||
}
|
266
tools/gpgtar-extract.c
Normal file
266
tools/gpgtar-extract.c
Normal file
@ -0,0 +1,266 @@
|
||||
/* gpgtar-extract.c - Extract from a TAR archive
|
||||
* Copyright (C) 2010 Free Software Foundation, Inc.
|
||||
*
|
||||
* This file is part of GnuPG.
|
||||
*
|
||||
* GnuPG is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* GnuPG is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <config.h>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <unistd.h>
|
||||
#include <assert.h>
|
||||
|
||||
#include "i18n.h"
|
||||
#include "../common/sysutils.h"
|
||||
#include "gpgtar.h"
|
||||
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
extract_regular (estream_t stream, const char *dirname,
|
||||
tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
char record[RECORDSIZE];
|
||||
size_t n, nbytes, nwritten;
|
||||
char *fname;
|
||||
estream_t outfp = NULL;
|
||||
|
||||
fname = strconcat (dirname, "/", hdr->name, NULL);
|
||||
if (!fname)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error creating filename: %s\n", gpg_strerror (err));
|
||||
goto leave;
|
||||
}
|
||||
else
|
||||
err = 0;
|
||||
|
||||
outfp = es_fopen (fname, "wb");
|
||||
if (!outfp)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error creating `%s': %s\n", fname, gpg_strerror (err));
|
||||
goto leave;
|
||||
}
|
||||
|
||||
for (n=0; n < hdr->nrecords;)
|
||||
{
|
||||
err = read_record (stream, record);
|
||||
if (err)
|
||||
goto leave;
|
||||
n++;
|
||||
nbytes = (n < hdr->nrecords)? RECORDSIZE : (hdr->size % RECORDSIZE);
|
||||
nwritten = es_fwrite (record, 1, nbytes, outfp);
|
||||
if (nwritten != nbytes)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error writing `%s': %s\n", fname, gpg_strerror (err));
|
||||
goto leave;
|
||||
}
|
||||
}
|
||||
/* Fixme: Set permissions etc. */
|
||||
|
||||
leave:
|
||||
es_fclose (outfp);
|
||||
if (err && fname && outfp)
|
||||
{
|
||||
if (gnupg_remove (fname))
|
||||
log_error ("error removing incomplete file `%s': %s\n",
|
||||
fname, gpg_strerror (gpg_error_from_syserror ()));
|
||||
}
|
||||
xfree (fname);
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
extract_directory (const char *dirname, tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
char *fname;
|
||||
|
||||
fname = strconcat (dirname, "/", hdr->name, NULL);
|
||||
if (!fname)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error creating filename: %s\n", gpg_strerror (err));
|
||||
goto leave;
|
||||
}
|
||||
else
|
||||
err = 0;
|
||||
|
||||
if (gnupg_mkdir (fname, "-rwx------"))
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error creating directory `%s': %s\n",
|
||||
fname, gpg_strerror (err));
|
||||
}
|
||||
|
||||
leave:
|
||||
xfree (fname);
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
static gpg_error_t
|
||||
extract (estream_t stream, const char *dirname, tar_header_t hdr)
|
||||
{
|
||||
gpg_error_t err;
|
||||
size_t n;
|
||||
|
||||
n = strlen (hdr->name);
|
||||
#ifdef HAVE_DOSISH_SYSTEM
|
||||
if (strchr (hdr->name, '\\'))
|
||||
{
|
||||
log_error ("filename `%s' contains a backslash - "
|
||||
"can't extract on this system\n", hdr->name);
|
||||
return gpg_error (GPG_ERR_INV_NAME);
|
||||
}
|
||||
#endif /*HAVE_DOSISH_SYSTEM*/
|
||||
|
||||
if (!n
|
||||
|| strstr (hdr->name, "//")
|
||||
|| strstr (hdr->name, "/../")
|
||||
|| !strncmp (hdr->name, "../", 3)
|
||||
|| (n >= 3 && !strcmp (hdr->name+n-3, "/.." )))
|
||||
{
|
||||
log_error ("filename `%s' as suspicious parts - not extracting\n",
|
||||
hdr->name);
|
||||
return gpg_error (GPG_ERR_INV_NAME);
|
||||
}
|
||||
|
||||
if (hdr->typeflag == TF_REGULAR || hdr->typeflag == TF_UNKNOWN)
|
||||
err = extract_regular (stream, dirname, hdr);
|
||||
else if (hdr->typeflag == TF_DIRECTORY)
|
||||
err = extract_directory (dirname, hdr);
|
||||
else
|
||||
{
|
||||
char record[RECORDSIZE];
|
||||
|
||||
log_info ("unsupported file type for `%s' - skipped\n", hdr->name);
|
||||
for (err = 0, n=0; !err && n < hdr->nrecords; n++)
|
||||
err = read_record (stream, record);
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
/* Create a new directory to be used for extracting the tarball.
|
||||
Returns the name of the directory which must be freed by the
|
||||
caller. In case of an error a diagnostic is printed and NULL
|
||||
returned. */
|
||||
static char *
|
||||
create_directory (const char *dirprefix)
|
||||
{
|
||||
gpg_error_t err = 0;
|
||||
char *dirname = NULL;
|
||||
int idx;
|
||||
|
||||
for (idx=1; idx < 5000; idx++)
|
||||
{
|
||||
xfree (dirname);
|
||||
dirname = xtryasprintf ("%s_%d_", dirprefix, idx);
|
||||
if (!dirname)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
goto leave;
|
||||
}
|
||||
if (!gnupg_mkdir (dirname, "-rwx------"))
|
||||
goto leave;
|
||||
if (errno != EEXIST && errno != ENOTDIR)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
goto leave;
|
||||
}
|
||||
}
|
||||
err = gpg_error_from_syserror ();
|
||||
|
||||
leave:
|
||||
if (err)
|
||||
{
|
||||
log_error ("error creating an extract directory: %s\n",
|
||||
gpg_strerror (err));
|
||||
xfree (dirname);
|
||||
dirname = NULL;
|
||||
}
|
||||
return dirname;
|
||||
}
|
||||
|
||||
|
||||
|
||||
void
|
||||
gpgtar_extract (const char *filename)
|
||||
{
|
||||
gpg_error_t err;
|
||||
estream_t stream;
|
||||
tar_header_t header = NULL;
|
||||
const char *dirprefix = NULL;
|
||||
char *dirname = NULL;
|
||||
|
||||
if (filename)
|
||||
{
|
||||
dirprefix = strrchr (filename, '/');
|
||||
if (dirprefix)
|
||||
dirprefix++;
|
||||
stream = es_fopen (filename, "rb");
|
||||
if (!stream)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error opening `%s': %s\n", filename, gpg_strerror (err));
|
||||
return;
|
||||
}
|
||||
}
|
||||
else
|
||||
stream = es_stdin; /* FIXME: How can we enforce binary mode? */
|
||||
|
||||
if (!dirprefix || !*dirprefix)
|
||||
dirprefix = "GPGARCH";
|
||||
|
||||
dirname = create_directory (dirprefix);
|
||||
if (!dirname)
|
||||
{
|
||||
err = gpg_error (GPG_ERR_GENERAL);
|
||||
goto leave;
|
||||
}
|
||||
|
||||
if (opt.verbose)
|
||||
log_info ("extracting to `%s/'\n", dirname);
|
||||
|
||||
for (;;)
|
||||
{
|
||||
header = gpgtar_read_header (stream);
|
||||
if (!header)
|
||||
goto leave;
|
||||
|
||||
if (extract (stream, dirname, header))
|
||||
goto leave;
|
||||
xfree (header);
|
||||
header = NULL;
|
||||
}
|
||||
|
||||
|
||||
leave:
|
||||
xfree (header);
|
||||
xfree (dirname);
|
||||
if (stream != es_stdin)
|
||||
es_fclose (stream);
|
||||
return;
|
||||
}
|
324
tools/gpgtar-list.c
Normal file
324
tools/gpgtar-list.c
Normal file
@ -0,0 +1,324 @@
|
||||
/* gpgtar-list.c - List a TAR archive
|
||||
* Copyright (C) 2010 Free Software Foundation, Inc.
|
||||
*
|
||||
* This file is part of GnuPG.
|
||||
*
|
||||
* GnuPG is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* GnuPG is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <config.h>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <assert.h>
|
||||
|
||||
#include "i18n.h"
|
||||
#include "gpgtar.h"
|
||||
|
||||
|
||||
|
||||
static unsigned long long
|
||||
parse_xoctal (const void *data, size_t length, const char *filename)
|
||||
{
|
||||
const unsigned char *p = data;
|
||||
unsigned long long value;
|
||||
|
||||
if (!length)
|
||||
value = 0;
|
||||
else if ( (*p & 0x80))
|
||||
{
|
||||
/* Binary format. */
|
||||
value = (*p++ & 0x7f);
|
||||
while (--length)
|
||||
{
|
||||
value <<= 8;
|
||||
value |= *p++;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Octal format */
|
||||
value = 0;
|
||||
/* Skip leading spaces and zeroes. */
|
||||
for (; length && (*p == ' ' || *p == '0'); length--, p++)
|
||||
;
|
||||
for (; length && *p; length--, p++)
|
||||
{
|
||||
if (*p >= '0' && *p <= '7')
|
||||
{
|
||||
value <<= 3;
|
||||
value += (*p - '0');
|
||||
}
|
||||
else
|
||||
{
|
||||
log_error ("%s: invalid octal number encountered - assuming 0\n",
|
||||
filename);
|
||||
value = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return value;
|
||||
}
|
||||
|
||||
|
||||
static tar_header_t
|
||||
parse_header (const void *record, const char *filename)
|
||||
{
|
||||
const struct ustar_raw_header *raw = record;
|
||||
size_t n, namelen, prefixlen;
|
||||
tar_header_t header;
|
||||
int use_prefix;
|
||||
|
||||
use_prefix = (!memcmp (raw->magic, "ustar", 5)
|
||||
&& (raw->magic[5] == ' ' || !raw->magic[5]));
|
||||
|
||||
|
||||
for (namelen=0; namelen < sizeof raw->name && raw->name[namelen]; namelen++)
|
||||
;
|
||||
if (namelen == sizeof raw->name)
|
||||
log_info ("%s: warning: name not terminated by a nul byte\n",
|
||||
filename);
|
||||
for (n=namelen+1; n < sizeof raw->name; n++)
|
||||
if (raw->name[n])
|
||||
{
|
||||
log_info ("%s: warning: garbage after name\n", filename);
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
if (use_prefix && raw->prefix[0])
|
||||
{
|
||||
for (prefixlen=0; (prefixlen < sizeof raw->prefix
|
||||
&& raw->prefix[prefixlen]); prefixlen++)
|
||||
;
|
||||
if (prefixlen == sizeof raw->prefix)
|
||||
log_info ("%s: warning: prefix not terminated by a nul byte\n",
|
||||
filename);
|
||||
for (n=prefixlen+1; n < sizeof raw->prefix; n++)
|
||||
if (raw->prefix[n])
|
||||
{
|
||||
log_info ("%s: warning: garbage after prefix\n", filename);
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
prefixlen = 0;
|
||||
|
||||
header = xtrycalloc (1, sizeof *header + prefixlen + 1 + namelen);
|
||||
if (!header)
|
||||
{
|
||||
log_error ("%s: error allocating header: %s\n",
|
||||
filename, gpg_strerror (gpg_error_from_syserror ()));
|
||||
return NULL;
|
||||
}
|
||||
if (prefixlen)
|
||||
{
|
||||
n = prefixlen;
|
||||
memcpy (header->name, raw->prefix, n);
|
||||
if (raw->prefix[n-1] != '/')
|
||||
header->name[n++] = '/';
|
||||
}
|
||||
else
|
||||
n = 0;
|
||||
memcpy (header->name+n, raw->name, namelen);
|
||||
header->name[n+namelen] = 0;
|
||||
|
||||
header->mode = parse_xoctal (raw->mode, sizeof raw->mode, filename);
|
||||
header->uid = parse_xoctal (raw->uid, sizeof raw->uid, filename);
|
||||
header->gid = parse_xoctal (raw->gid, sizeof raw->gid, filename);
|
||||
header->size = parse_xoctal (raw->size, sizeof raw->size, filename);
|
||||
header->mtime = parse_xoctal (raw->mtime, sizeof raw->mtime, filename);
|
||||
/* checksum = */
|
||||
switch (raw->typeflag[0])
|
||||
{
|
||||
case '0': header->typeflag = TF_REGULAR; break;
|
||||
case '1': header->typeflag = TF_HARDLINK; break;
|
||||
case '2': header->typeflag = TF_SYMLINK; break;
|
||||
case '3': header->typeflag = TF_CHARDEV; break;
|
||||
case '4': header->typeflag = TF_BLOCKDEV; break;
|
||||
case '5': header->typeflag = TF_DIRECTORY; break;
|
||||
case '6': header->typeflag = TF_FIFO; break;
|
||||
case '7': header->typeflag = TF_RESERVED; break;
|
||||
default: header->typeflag = TF_UNKNOWN; break;
|
||||
}
|
||||
|
||||
|
||||
/* Compute the number of data records following this header. */
|
||||
if (header->typeflag == TF_REGULAR || header->typeflag == TF_UNKNOWN)
|
||||
header->nrecords = (header->size + RECORDSIZE-1)/RECORDSIZE;
|
||||
else
|
||||
header->nrecords = 0;
|
||||
|
||||
|
||||
return header;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/* Read the next block, assming it is a tar header. Returns a header
|
||||
object on success or NULL one error. In case of an error an error
|
||||
message has been printed. */
|
||||
static tar_header_t
|
||||
read_header (estream_t stream)
|
||||
{
|
||||
gpg_error_t err;
|
||||
char record[RECORDSIZE];
|
||||
int i;
|
||||
|
||||
err = read_record (stream, record);
|
||||
if (err)
|
||||
return NULL;
|
||||
|
||||
for (i=0; i < RECORDSIZE && !record[i]; i++)
|
||||
;
|
||||
if (i == RECORDSIZE)
|
||||
{
|
||||
/* All zero header - check whether it is the first part of an
|
||||
end of archive mark. */
|
||||
err = read_record (stream, record);
|
||||
if (err)
|
||||
return NULL;
|
||||
|
||||
for (i=0; i < RECORDSIZE && !record[i]; i++)
|
||||
;
|
||||
if (i != RECORDSIZE)
|
||||
log_info ("%s: warning: skipping empty header\n",
|
||||
es_fname_get (stream));
|
||||
else
|
||||
{
|
||||
/* End of archive - FIXME: we might want to check for garbage. */
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
return parse_header (record, es_fname_get (stream));
|
||||
}
|
||||
|
||||
|
||||
/* Skip the data records according to HEADER. Prints an error message
|
||||
on error and return -1. */
|
||||
static int
|
||||
skip_data (estream_t stream, tar_header_t header)
|
||||
{
|
||||
char record[RECORDSIZE];
|
||||
unsigned long long n;
|
||||
|
||||
for (n=0; n < header->nrecords; n++)
|
||||
{
|
||||
if (read_record (stream, record))
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
static void
|
||||
print_header (tar_header_t header, estream_t out)
|
||||
{
|
||||
unsigned long mask;
|
||||
char modestr[10+1];
|
||||
int i;
|
||||
|
||||
*modestr = '?';
|
||||
switch (header->typeflag)
|
||||
{
|
||||
case TF_REGULAR: *modestr = '-'; break;
|
||||
case TF_HARDLINK: *modestr = 'h'; break;
|
||||
case TF_SYMLINK: *modestr = 'l'; break;
|
||||
case TF_CHARDEV: *modestr = 'c'; break;
|
||||
case TF_BLOCKDEV: *modestr = 'b'; break;
|
||||
case TF_DIRECTORY:*modestr = 'd'; break;
|
||||
case TF_FIFO: *modestr = 'f'; break;
|
||||
case TF_RESERVED: *modestr = '='; break;
|
||||
case TF_UNKNOWN: break;
|
||||
case TF_NOTSUP: break;
|
||||
}
|
||||
for (mask = 0400, i = 0; i < 9; i++, mask >>= 1)
|
||||
modestr[1+i] = (header->mode & mask)? "rwxrwxrwx"[i]:'-';
|
||||
if ((header->typeflag & 04000))
|
||||
modestr[3] = modestr[3] == 'x'? 's':'S';
|
||||
if ((header->typeflag & 02000))
|
||||
modestr[6] = modestr[6] == 'x'? 's':'S';
|
||||
if ((header->typeflag & 01000))
|
||||
modestr[9] = modestr[9] == 'x'? 't':'T';
|
||||
modestr[10] = 0;
|
||||
|
||||
es_fprintf (out, "%s %lu %lu/%lu %12llu %s %s\n",
|
||||
modestr, header->nlink, header->uid, header->gid, header->size,
|
||||
isotimestamp (header->mtime), header->name);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/* List the tarball FILENAME or, if FILENAME is NULL, the tarball read
|
||||
from stdin. */
|
||||
void
|
||||
gpgtar_list (const char *filename)
|
||||
{
|
||||
gpg_error_t err;
|
||||
estream_t stream;
|
||||
tar_header_t header;
|
||||
|
||||
if (filename)
|
||||
{
|
||||
stream = es_fopen (filename, "rb");
|
||||
if (!stream)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error opening `%s': %s\n", filename, gpg_strerror (err));
|
||||
return;
|
||||
}
|
||||
}
|
||||
else
|
||||
stream = es_stdin; /* FIXME: How can we enforce binary mode? */
|
||||
|
||||
for (;;)
|
||||
{
|
||||
header = read_header (stream);
|
||||
if (!header)
|
||||
goto leave;
|
||||
|
||||
print_header (header, es_stdout);
|
||||
|
||||
if (skip_data (stream, header))
|
||||
goto leave;
|
||||
xfree (header);
|
||||
header = NULL;
|
||||
}
|
||||
|
||||
|
||||
leave:
|
||||
xfree (header);
|
||||
if (filename)
|
||||
es_fclose (stream);
|
||||
return;
|
||||
}
|
||||
|
||||
tar_header_t
|
||||
gpgtar_read_header (estream_t stream)
|
||||
{
|
||||
/*FIXME: Change to return an error code. */
|
||||
return read_header (stream);
|
||||
}
|
||||
|
||||
void
|
||||
gpgtar_print_header (tar_header_t header, estream_t out)
|
||||
{
|
||||
if (header && out)
|
||||
print_header (header, out);
|
||||
}
|
361
tools/gpgtar.c
Normal file
361
tools/gpgtar.c
Normal file
@ -0,0 +1,361 @@
|
||||
/* gpgtar.c - A simple TAR implementation mainly useful for Windows.
|
||||
* Copyright (C) 2010 Free Software Foundation, Inc.
|
||||
*
|
||||
* This file is part of GnuPG.
|
||||
*
|
||||
* GnuPG is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* GnuPG is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
/* GnuPG comes with a shell script gpg-zip which creates archive files
|
||||
in the same format as PGP Zip, which is actually a USTAR format.
|
||||
That is fine and works nicely on all Unices but for Windows we
|
||||
don't have a compatible shell and the supply of tar programs is
|
||||
limited. Given that we need just a few tar option and it is an
|
||||
open question how many Unix concepts are to be mapped to Windows,
|
||||
we might as well write our own little tar customized for use with
|
||||
gpg. So here we go. */
|
||||
|
||||
#include <config.h>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <assert.h>
|
||||
|
||||
#include "util.h"
|
||||
#include "i18n.h"
|
||||
#include "sysutils.h"
|
||||
#include "../common/openpgpdefs.h"
|
||||
|
||||
#include "gpgtar.h"
|
||||
|
||||
|
||||
/* Constants to identify the commands and options. */
|
||||
enum cmd_and_opt_values
|
||||
{
|
||||
aNull = 0,
|
||||
aEncrypt = 'e',
|
||||
aDecrypt = 'd',
|
||||
aSign = 's',
|
||||
|
||||
oSymmetric = 'c',
|
||||
oRecipient = 'r',
|
||||
oUser = 'u',
|
||||
oOutput = 'o',
|
||||
oQuiet = 'q',
|
||||
oVerbose = 'v',
|
||||
oNoVerbose = 500,
|
||||
|
||||
aSignEncrypt,
|
||||
oSkipCrypto,
|
||||
aList
|
||||
};
|
||||
|
||||
|
||||
/* The list of commands and options. */
|
||||
static ARGPARSE_OPTS opts[] = {
|
||||
ARGPARSE_group (300, N_("@Commands:\n ")),
|
||||
|
||||
ARGPARSE_c (aEncrypt, "encrypt", N_("create an archive")),
|
||||
ARGPARSE_c (aDecrypt, "decrypt", N_("extract an archive")),
|
||||
ARGPARSE_c (aSign, "sign", N_("create a signed archive")),
|
||||
ARGPARSE_c (aList, "list-archive", N_("list an archive")),
|
||||
|
||||
ARGPARSE_group (301, N_("@\nOptions:\n ")),
|
||||
|
||||
ARGPARSE_s_n (oSymmetric, "symmetric", N_("use symmetric encryption")),
|
||||
ARGPARSE_s_s (oRecipient, "recipient", N_("|USER-ID|encrypt for USER-ID")),
|
||||
ARGPARSE_s_s (oUser, "local-user",
|
||||
N_("|USER-ID|use USER-ID to sign or decrypt")),
|
||||
ARGPARSE_s_s (oOutput, "output", N_("|FILE|write output to FILE")),
|
||||
ARGPARSE_s_n (oVerbose, "verbose", N_("verbose")),
|
||||
ARGPARSE_s_n (oQuiet, "quiet", N_("be somewhat more quiet")),
|
||||
ARGPARSE_s_n (oSkipCrypto, "skip-crypto", N_("skip the crypto processing")),
|
||||
|
||||
ARGPARSE_end ()
|
||||
};
|
||||
|
||||
|
||||
|
||||
static void tar_and_encrypt (char **inpattern);
|
||||
static void decrypt_and_untar (const char *fname);
|
||||
static void decrypt_and_list (const char *fname);
|
||||
|
||||
|
||||
|
||||
|
||||
/* Print usage information and and provide strings for help. */
|
||||
static const char *
|
||||
my_strusage( int level )
|
||||
{
|
||||
const char *p;
|
||||
|
||||
switch (level)
|
||||
{
|
||||
case 11: p = "gpgtar (GnuPG)";
|
||||
break;
|
||||
case 13: p = VERSION; break;
|
||||
case 17: p = PRINTABLE_OS_NAME; break;
|
||||
case 19: p = _("Please report bugs to <@EMAIL@>.\n"); break;
|
||||
|
||||
case 1:
|
||||
case 40:
|
||||
p = _("Usage: gpgtar [options] [files] [directories] (-h for help)");
|
||||
break;
|
||||
case 41:
|
||||
p = _("Syntax: gpgtar [options] [files] [directories]\n"
|
||||
"Encrypt or sign files into an archive\n");
|
||||
break;
|
||||
|
||||
default: p = NULL; break;
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
set_cmd (enum cmd_and_opt_values *ret_cmd, enum cmd_and_opt_values new_cmd)
|
||||
{
|
||||
enum cmd_and_opt_values cmd = *ret_cmd;
|
||||
|
||||
if (!cmd || cmd == new_cmd)
|
||||
cmd = new_cmd;
|
||||
else if (cmd == aSign && new_cmd == aEncrypt)
|
||||
cmd = aSignEncrypt;
|
||||
else if (cmd == aEncrypt && new_cmd == aSign)
|
||||
cmd = aSignEncrypt;
|
||||
else
|
||||
{
|
||||
log_error (_("conflicting commands\n"));
|
||||
exit (2);
|
||||
}
|
||||
|
||||
*ret_cmd = cmd;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/* gpgtar main. */
|
||||
int
|
||||
main (int argc, char **argv)
|
||||
{
|
||||
ARGPARSE_ARGS pargs;
|
||||
const char *fname;
|
||||
int no_more_options = 0;
|
||||
enum cmd_and_opt_values cmd = 0;
|
||||
int skip_crypto = 0;
|
||||
|
||||
assert (sizeof (struct ustar_raw_header) == 512);
|
||||
|
||||
gnupg_reopen_std ("gpgtar");
|
||||
set_strusage (my_strusage);
|
||||
log_set_prefix ("gpgtar", 1);
|
||||
|
||||
/* Make sure that our subsystems are ready. */
|
||||
i18n_init();
|
||||
init_common_subsystems (&argc, &argv);
|
||||
|
||||
/* Parse the command line. */
|
||||
pargs.argc = &argc;
|
||||
pargs.argv = &argv;
|
||||
pargs.flags = ARGPARSE_FLAG_KEEP;
|
||||
while (!no_more_options && optfile_parse (NULL, NULL, NULL, &pargs, opts))
|
||||
{
|
||||
switch (pargs.r_opt)
|
||||
{
|
||||
case oOutput: opt.outfile = pargs.r.ret_str; break;
|
||||
case oQuiet: opt.quiet = 1; break;
|
||||
case oVerbose: opt.verbose++; break;
|
||||
case oNoVerbose: opt.verbose = 0; break;
|
||||
|
||||
case aList:
|
||||
case aDecrypt:
|
||||
case aEncrypt:
|
||||
case aSign:
|
||||
set_cmd (&cmd, pargs.r_opt);
|
||||
break;
|
||||
|
||||
case oSymmetric:
|
||||
set_cmd (&cmd, aEncrypt);
|
||||
opt.symmetric = 1;
|
||||
break;
|
||||
|
||||
case oSkipCrypto:
|
||||
skip_crypto = 1;
|
||||
break;
|
||||
|
||||
default: pargs.err = 2; break;
|
||||
}
|
||||
}
|
||||
|
||||
if (log_get_errorcount (0))
|
||||
exit (2);
|
||||
|
||||
switch (cmd)
|
||||
{
|
||||
case aList:
|
||||
if (argc > 1)
|
||||
usage (1);
|
||||
fname = argc ? *argv : NULL;
|
||||
if (skip_crypto)
|
||||
gpgtar_list (fname);
|
||||
else
|
||||
decrypt_and_list (fname);
|
||||
break;
|
||||
|
||||
case aEncrypt:
|
||||
if (!argc)
|
||||
usage (1);
|
||||
if (skip_crypto)
|
||||
gpgtar_create (argv);
|
||||
else
|
||||
tar_and_encrypt (argv);
|
||||
break;
|
||||
|
||||
case aDecrypt:
|
||||
if (argc != 1)
|
||||
usage (1);
|
||||
if (opt.outfile)
|
||||
log_info ("note: ignoring option --output\n");
|
||||
fname = argc ? *argv : NULL;
|
||||
if (skip_crypto)
|
||||
gpgtar_extract (fname);
|
||||
else
|
||||
decrypt_and_untar (fname);
|
||||
break;
|
||||
|
||||
default:
|
||||
log_error (_("invalid command (there is no implicit command)\n"));
|
||||
break;
|
||||
}
|
||||
|
||||
return log_get_errorcount (0)? 1:0;
|
||||
}
|
||||
|
||||
|
||||
/* Read the next record from STREAM. RECORD is a buffer provided by
|
||||
the caller and must be at leadt of size RECORDSIZE. The function
|
||||
return 0 on success and and error code on failure; a diagnostic
|
||||
printed as well. Note that there is no need for an EOF indicator
|
||||
because a tarball has an explicit EOF record. */
|
||||
gpg_error_t
|
||||
read_record (estream_t stream, void *record)
|
||||
{
|
||||
gpg_error_t err;
|
||||
size_t nread;
|
||||
|
||||
nread = es_fread (record, 1, RECORDSIZE, stream);
|
||||
if (nread != RECORDSIZE)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
if (es_ferror (stream))
|
||||
log_error ("error reading `%s': %s\n",
|
||||
es_fname_get (stream), gpg_strerror (err));
|
||||
else
|
||||
log_error ("error reading `%s': premature EOF "
|
||||
"(size of last record: %zu)\n",
|
||||
es_fname_get (stream), nread);
|
||||
}
|
||||
else
|
||||
err = 0;
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
/* Write the RECORD of size RECORDSIZE to STREAM. FILENAME is the
|
||||
name of the file used for diagnostics. */
|
||||
gpg_error_t
|
||||
write_record (estream_t stream, const void *record)
|
||||
{
|
||||
gpg_error_t err;
|
||||
size_t nwritten;
|
||||
|
||||
nwritten = es_fwrite (record, 1, RECORDSIZE, stream);
|
||||
if (nwritten != RECORDSIZE)
|
||||
{
|
||||
err = gpg_error_from_syserror ();
|
||||
log_error ("error writing `%s': %s\n",
|
||||
es_fname_get (stream), gpg_strerror (err));
|
||||
}
|
||||
else
|
||||
err = 0;
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
|
||||
/* Return true if FP is an unarmored OpenPGP message. Note that this
|
||||
fucntion reads a few bytes from FP but pushes them back. */
|
||||
static int
|
||||
openpgp_message_p (estream_t fp)
|
||||
{
|
||||
int ctb;
|
||||
|
||||
ctb = es_getc (fp);
|
||||
if (ctb != EOF)
|
||||
{
|
||||
if (es_ungetc (ctb, fp))
|
||||
log_fatal ("error ungetting first byte: %s\n",
|
||||
gpg_strerror (gpg_error_from_syserror ()));
|
||||
|
||||
if ((ctb & 0x80))
|
||||
{
|
||||
switch ((ctb & 0x40) ? (ctb & 0x3f) : ((ctb>>2)&0xf))
|
||||
{
|
||||
case PKT_MARKER:
|
||||
case PKT_SYMKEY_ENC:
|
||||
case PKT_ONEPASS_SIG:
|
||||
case PKT_PUBKEY_ENC:
|
||||
case PKT_SIGNATURE:
|
||||
case PKT_COMMENT:
|
||||
case PKT_OLD_COMMENT:
|
||||
case PKT_PLAINTEXT:
|
||||
case PKT_COMPRESSED:
|
||||
case PKT_ENCRYPTED:
|
||||
return 1; /* Yes, this seems to be an OpenPGP message. */
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
static void
|
||||
tar_and_encrypt (char **inpattern)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
static void
|
||||
decrypt_and_untar (const char *fname)
|
||||
{
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
static void
|
||||
decrypt_and_list (const char *fname)
|
||||
{
|
||||
|
||||
}
|
124
tools/gpgtar.h
Normal file
124
tools/gpgtar.h
Normal file
@ -0,0 +1,124 @@
|
||||
/* gpgtar.h - Global definitions for gpgtar
|
||||
* Copyright (C) 2010 Free Software Foundation, Inc.
|
||||
*
|
||||
* This file is part of GnuPG.
|
||||
*
|
||||
* GnuPG is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* GnuPG is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef GPGTAR_H
|
||||
#define GPGTAR_H
|
||||
|
||||
#include "../common/util.h"
|
||||
|
||||
/* We keep all global options in the structure OPT. */
|
||||
struct
|
||||
{
|
||||
int verbose;
|
||||
int quiet;
|
||||
char *outfile;
|
||||
int symmetric;
|
||||
} opt;
|
||||
|
||||
|
||||
/* The size of a tar record. All IO is done in chunks of this size.
|
||||
Note that we don't care about blocking because this version of tar
|
||||
is not expected to be used directly on a tape drive in fact it is
|
||||
used in a pipeline with GPG and thus any blocking would be
|
||||
useless. */
|
||||
#define RECORDSIZE 512
|
||||
|
||||
|
||||
/* Description of the USTAR header format. */
|
||||
struct ustar_raw_header
|
||||
{
|
||||
char name[100];
|
||||
char mode[8];
|
||||
char uid[8];
|
||||
char gid[8];
|
||||
char size[12];
|
||||
char mtime[12];
|
||||
char checksum[8];
|
||||
char typeflag[1];
|
||||
char linkname[100];
|
||||
char magic[6];
|
||||
char version[2];
|
||||
char uname[32];
|
||||
char gname[32];
|
||||
char devmajor[8];
|
||||
char devminor[8];
|
||||
char prefix[155];
|
||||
char pad[12];
|
||||
};
|
||||
|
||||
|
||||
/* Filetypes as defined by USTAR. */
|
||||
typedef enum
|
||||
{
|
||||
TF_REGULAR,
|
||||
TF_HARDLINK,
|
||||
TF_SYMLINK,
|
||||
TF_CHARDEV,
|
||||
TF_BLOCKDEV,
|
||||
TF_DIRECTORY,
|
||||
TF_FIFO,
|
||||
TF_RESERVED,
|
||||
TF_UNKNOWN, /* Needs to be treated as regular file. */
|
||||
TF_NOTSUP /* Not supported (used with --create). */
|
||||
} typeflag_t;
|
||||
|
||||
|
||||
/* The internal represenation of a TAR header. */
|
||||
struct tar_header_s;
|
||||
typedef struct tar_header_s *tar_header_t;
|
||||
struct tar_header_s
|
||||
{
|
||||
tar_header_t next; /* Used to build a linked list iof entries. */
|
||||
|
||||
unsigned long mode; /* The file mode. */
|
||||
unsigned long nlink; /* Number of hard links. */
|
||||
unsigned long uid; /* The user id of the file. */
|
||||
unsigned long gid; /* The group id of the file. */
|
||||
unsigned long long size; /* The size of the file. */
|
||||
unsigned long long mtime; /* Modification time since Epoch. Note
|
||||
that we don't use time_t here but a
|
||||
type which is more likely to be larger
|
||||
that 32 bit and thus allows to track
|
||||
times beyond 2106. */
|
||||
typeflag_t typeflag; /* The type of the file. */
|
||||
|
||||
|
||||
unsigned long long nrecords; /* Number of data records. */
|
||||
|
||||
char name[1]; /* Filename (dynamically extended). */
|
||||
};
|
||||
|
||||
|
||||
/*-- gpgtar.c --*/
|
||||
gpg_error_t read_record (estream_t stream, void *record);
|
||||
gpg_error_t write_record (estream_t stream, const void *record);
|
||||
|
||||
/*-- gpgtar-create.c --*/
|
||||
void gpgtar_create (char **inpattern);
|
||||
|
||||
/*-- gpgtar-extract.c --*/
|
||||
void gpgtar_extract (const char *filename);
|
||||
|
||||
/*-- gpgtar-list.c --*/
|
||||
void gpgtar_list (const char *filename);
|
||||
tar_header_t gpgtar_read_header (estream_t stream);
|
||||
void gpgtar_print_header (tar_header_t header, estream_t out);
|
||||
|
||||
|
||||
#endif /*GPGTAR_H*/
|
Loading…
x
Reference in New Issue
Block a user