2007-12-07 11:28:16 -08:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2013-07-03 16:47:05 +02:00
Tar backup command extension
Copyright ( C ) Aurélien Aptel 2013
2007-12-07 11:28:16 -08:00
1996-05-04 07:50:46 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1996-05-04 07:50:46 +00:00
( at your option ) any later version .
2007-12-07 11:28:16 -08:00
1996-05-04 07:50:46 +00:00
This program 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 .
2007-12-07 11:28:16 -08:00
1996-05-04 07:50:46 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1996-05-04 07:50:46 +00:00
*/
2013-07-23 16:55:50 +02:00
/**
* # General overview of the tar extension
*
* All tar_xxx ( ) functions work on a ` struct tar ` which store most of
* the context of the backup process .
*
* The current tar context can be accessed via the global variable
* ` tar_ctx ` . It ' s not static but you should avoid accessing it
* directly .
*
* A tar context is first configured through tar_parse_args ( ) which
* can be called from either the CLI ( in client . c ) or the interactive
* session ( via the cmd_tar ( ) callback ) .
*
* Once the configuration is done ( successfully ) , the context is ready
* for processing and tar_to_process ( ) returns true .
*
* The next step is to call tar_process ( ) which dispatch the
* processing to either tar_create ( ) or tar_extract ( ) , depending on
* the context .
*
* # # Archive creation
*
* tar_create ( ) creates an archive using the libarchive API then
*
* - iterates on the requested paths if the context is in inclusion
* mode with tar_create_from_list ( )
*
* - or iterates on the whole share ( starting from the current dir ) if
* in exclusion mode or if no specific path were requested
*
* The do_list ( ) function from client . c is used to list recursively
* the share . In particular it takes a DOS path mask ( eg . \ mydir \ * )
* and a callback function which will be called with each file name
* and attributes . The tar callback function is get_file_callback ( ) .
*
* The callback function checks whether the file should be skipped
* according the the configuration via tar_create_skip_path ( ) . If it ' s
* not skipped it ' s downloaded and written to the archive in
* tar_get_file ( ) .
*
* # # Archive extraction
*
* tar_extract ( ) opens the archive and iterates on each file in
* it . For each file tar_extract_skip_path ( ) checks whether it should
* be skipped according to the config . If it ' s not skipped it ' s
* uploaded on the server in tar_send_file ( ) .
*/
1996-05-04 07:50:46 +00:00
# include "includes.h"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2004-10-07 04:01:18 +00:00
# include "client/client_proto.h"
2013-07-08 18:09:47 +02:00
# include "client/clitar_proto.h"
2011-05-06 11:47:43 +02:00
# include "libsmb/libsmb.h"
2013-07-05 09:51:43 +02:00
# include <archive.h>
2013-07-09 18:01:47 +02:00
# include <archive_entry.h>
2013-07-05 09:51:43 +02:00
# define LEN(x) (sizeof(x) / sizeof((x)[0]))
2013-07-17 16:25:34 +02:00
# define DBG(a, b) (DEBUG(a, ("tar:%-4d ", __LINE__)), DEBUG(a, b))
2013-07-09 18:01:47 +02:00
2013-07-18 17:06:33 +02:00
/**
* Number of byte in a block unit .
*/
# define TAR_BLOCK_UNIT 512
/**
* Default tar block size in TAR_BLOCK_UNIT .
*/
# define TAR_DEFAULT_BLOCK_SIZE 20
2013-07-09 18:01:47 +02:00
/**
* Maximum value for the blocksize field
*/
# define TAR_MAX_BLOCK_SIZE 0xffff
/**
2013-07-18 17:06:33 +02:00
* Size of the buffer used when downloading a file
2013-07-09 18:01:47 +02:00
*/
2013-07-11 00:57:40 +02:00
# define TAR_CLI_READ_SIZE 0xff00
2013-07-16 16:32:29 +02:00
# define TAR_DO_LIST_ATTR (FILE_ATTRIBUTE_DIRECTORY \
| FILE_ATTRIBUTE_SYSTEM \
| FILE_ATTRIBUTE_HIDDEN )
2013-07-11 00:57:40 +02:00
2013-07-11 15:56:03 +02:00
2013-07-05 18:14:50 +02:00
enum tar_operation {
TAR_NO_OPERATION ,
TAR_CREATE , /* c flag */
TAR_EXTRACT , /* x flag */
} ;
enum tar_selection {
TAR_NO_SELECTION ,
2013-07-16 16:22:13 +02:00
TAR_INCLUDE , /* I and F flag, default */
2013-07-05 18:14:50 +02:00
TAR_EXCLUDE , /* X flag */
2013-07-03 18:18:25 +02:00
} ;
2013-07-05 09:51:43 +02:00
enum {
ATTR_UNSET ,
ATTR_SET ,
} ;
struct tar {
2013-07-09 18:01:47 +02:00
/* in state that needs/can be processed? */
2013-07-09 15:10:44 +02:00
bool to_process ;
2013-07-03 18:18:25 +02:00
/* flags */
2013-07-05 18:14:50 +02:00
struct tar_mode {
enum tar_operation operation ; /* create, extract */
2013-07-23 16:55:50 +02:00
enum tar_selection selection ; /* include, exclude */
int blocksize ; /* size in TAR_BLOCK_UNIT of a tar file block */
2013-07-03 18:18:25 +02:00
bool hidden ; /* backup hidden file? */
bool system ; /* backup system file? */
bool incremental ; /* backup _only_ archived file? */
bool reset ; /* unset archive bit? */
bool dry ; /* don't write tar file? */
2013-07-05 18:14:50 +02:00
bool regex ; /* XXX: never actually using regex... */
2013-07-23 16:55:50 +02:00
bool verbose ; /* XXX: ignored */
2013-07-03 18:18:25 +02:00
} mode ;
2013-07-11 18:17:25 +02:00
/* nb of bytes received */
uint64_t total_size ;
2013-07-03 18:18:25 +02:00
/* path to tar archive name */
char * tar_path ;
2013-07-08 18:09:47 +02:00
/* list of path to include or exclude */
char * * path_list ;
2013-07-09 23:17:46 +02:00
int path_list_size ;
1998-09-18 12:47:46 +00:00
2013-07-05 09:51:43 +02:00
/* archive handle */
struct archive * archive ;
} ;
2013-07-03 18:18:25 +02:00
2013-07-23 16:55:50 +02:00
/**
* Global context imported in client . c when needed .
*
* Default options .
*/
2013-07-08 18:09:47 +02:00
struct tar tar_ctx = {
2013-07-05 18:14:50 +02:00
. mode . selection = TAR_INCLUDE ,
2013-07-09 18:01:47 +02:00
. mode . blocksize = TAR_DEFAULT_BLOCK_SIZE ,
2013-07-09 13:09:56 +02:00
. mode . hidden = true ,
. mode . system = true ,
. mode . incremental = false ,
2013-07-23 16:55:50 +02:00
. mode . reset = false ,
2013-07-09 13:09:56 +02:00
. mode . dry = false ,
2013-07-23 16:55:50 +02:00
. mode . regex = false ,
. mode . verbose = false ,
2013-07-05 09:51:43 +02:00
} ;
2013-07-03 18:18:25 +02:00
2013-07-23 17:39:09 +02:00
/* tar, local function */
static int tar_create ( struct tar * t ) ;
static int tar_create_from_list ( struct tar * t ) ;
static int tar_extract ( struct tar * t ) ;
static int tar_read_inclusion_file ( struct tar * t , const char * filename ) ;
static int tar_send_file ( struct tar * t , struct archive_entry * entry ) ;
static int tar_set_blocksize ( struct tar * t , int size ) ;
static int tar_set_newer_than ( struct tar * t , const char * filename ) ;
static void tar_add_selection_path ( struct tar * t , const char * path ) ;
static void tar_dump ( struct tar * t ) ;
static bool tar_extract_skip_path ( struct tar * t , struct archive_entry * entry ) ;
static bool tar_create_skip_path ( struct tar * t ,
const char * fullpath ,
const struct file_info * finfo ) ;
2013-07-23 16:55:50 +02:00
2013-07-23 17:39:09 +02:00
static bool tar_path_in_list ( struct tar * t ,
const char * path ,
bool reverse ) ;
2013-07-09 14:27:55 +02:00
2013-07-23 17:39:09 +02:00
static int tar_get_file ( struct tar * t ,
const char * full_dos_path ,
struct file_info * finfo ) ;
2013-07-09 14:27:55 +02:00
2013-07-23 17:39:09 +02:00
static NTSTATUS get_file_callback ( struct cli_state * cli ,
struct file_info * finfo ,
const char * dir ) ;
2013-07-09 14:27:55 +02:00
2013-07-23 17:39:09 +02:00
/* utilities */
static char * fix_unix_path ( char * path , bool removeprefix ) ;
static char * path_base_name ( const char * path ) ;
static const char * skip_useless_char_in_path ( const char * p ) ;
static int make_remote_path ( const char * full_path ) ;
static int max_token ( const char * str ) ;
static bool is_subpath ( const char * sub , const char * full ) ;
static int set_remote_attr ( const char * filename , uint16 new_attr , int mode ) ;
2013-07-09 14:27:55 +02:00
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* cmd_block - interactive command to change tar blocksize
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* Read a size from the client command line and update the current
* blocksize .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
int cmd_block ( void )
2013-07-17 16:25:34 +02:00
{
2013-07-23 17:39:09 +02:00
/* XXX: from client.c */
const extern char * cmd_ptr ;
char * buf ;
2013-07-17 16:25:34 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
if ( ! next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
DBG ( 0 , ( " blocksize <n> \n " ) ) ;
return 1 ;
2013-07-17 16:25:34 +02:00
}
2013-07-23 17:39:09 +02:00
if ( tar_set_blocksize ( & tar_ctx , atoi ( buf ) ) ) {
DBG ( 0 , ( " invalid blocksize \n " ) ) ;
2013-07-17 16:25:34 +02:00
}
2013-07-23 17:39:09 +02:00
DBG ( 2 , ( " blocksize is now %d \n " , tar_ctx . mode . blocksize ) ) ;
2013-07-17 16:25:34 +02:00
2013-07-23 17:39:09 +02:00
return 0 ;
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* cmd_tarmode - interactive command to change tar behaviour
*
* Read one or more modes from the client command line and update the
* current tar mode .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
int cmd_tarmode ( void )
2013-07-09 11:41:06 +02:00
{
2013-07-23 17:39:09 +02:00
const extern char * cmd_ptr ;
char * buf ;
2013-07-09 11:41:06 +02:00
int i ;
2013-07-23 17:39:09 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-09 11:41:06 +02:00
2013-07-23 17:39:09 +02:00
struct {
const char * cmd ;
bool * p ;
bool value ;
} table [ ] = {
{ " full " , & tar_ctx . mode . incremental , false } ,
{ " inc " , & tar_ctx . mode . incremental , true } ,
{ " reset " , & tar_ctx . mode . reset , true } ,
{ " noreset " , & tar_ctx . mode . reset , false } ,
{ " system " , & tar_ctx . mode . system , true } ,
{ " nosystem " , & tar_ctx . mode . system , false } ,
{ " hidden " , & tar_ctx . mode . hidden , true } ,
{ " nohidden " , & tar_ctx . mode . hidden , false } ,
{ " verbose " , & tar_ctx . mode . verbose , true } ,
{ " noquiet " , & tar_ctx . mode . verbose , true } ,
{ " quiet " , & tar_ctx . mode . verbose , false } ,
{ " noverbose " , & tar_ctx . mode . verbose , false } ,
2013-07-09 11:41:06 +02:00
} ;
2013-07-23 17:39:09 +02:00
while ( next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
for ( i = 0 ; i < LEN ( table ) ; i + + ) {
if ( strequal ( table [ i ] . cmd , buf ) ) {
* table [ i ] . p = table [ i ] . value ;
break ;
}
}
2013-07-09 11:41:06 +02:00
2013-07-23 17:39:09 +02:00
if ( i = = LEN ( table ) )
DBG ( 0 , ( " tarmode: unrecognised option %s \n " , buf ) ) ;
TALLOC_FREE ( buf ) ;
2013-07-09 11:41:06 +02:00
}
2013-07-23 17:39:09 +02:00
DBG ( 0 , ( " tarmode is now %s, %s, %s, %s, %s \n " ,
tar_ctx . mode . incremental ? " incremental " : " full " ,
tar_ctx . mode . system ? " system " : " nosystem " ,
tar_ctx . mode . hidden ? " hidden " : " nohidden " ,
tar_ctx . mode . reset ? " reset " : " noreset " ,
tar_ctx . mode . verbose ? " verbose " : " quiet " ) ) ;
return 0 ;
2013-07-09 11:41:06 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* cmd_tar - interactive command to start a tar backup / restoration
*
* Check presence of argument , parse them and handle the request .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
int cmd_tar ( void )
2013-07-09 23:17:46 +02:00
{
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
const extern char * cmd_ptr ;
const char * flag ;
const char * * val ;
char * buf ;
int maxtok = max_token ( cmd_ptr ) ;
int i = 0 ;
int err = 0 ;
2013-07-09 23:17:46 +02:00
2013-07-23 17:39:09 +02:00
if ( ! next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
DBG ( 0 , ( " tar <c|x>[IXFbganN] [options] <tar file> [path list] \n " ) ) ;
2013-07-18 16:22:26 +02:00
return 1 ;
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
flag = buf ;
val = talloc_array ( ctx , const char * , maxtok ) ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
while ( next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
val [ i + + ] = buf ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
if ( tar_parse_args ( & tar_ctx , flag , val , i ) ) {
DBG ( 0 , ( " parse_args failed \n " ) ) ;
err = 1 ;
goto out ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
if ( tar_process ( & tar_ctx ) ) {
DBG ( 0 , ( " tar_process failed \n " ) ) ;
err = 1 ;
goto out ;
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
out :
return err ;
2013-07-05 18:14:50 +02:00
}
1996-05-04 07:50:46 +00:00
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* cmd_setmode - interactive command to set DOS attributes
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* Read a filename and mode from the client command line and update
* the file DOS attributes .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
int cmd_setmode ( void )
2013-07-08 18:09:47 +02:00
{
2013-07-23 17:39:09 +02:00
const extern char * cmd_ptr ;
char * buf ;
char * fname = NULL ;
uint16 attr [ 2 ] = { 0 } ;
int mode = ATTR_SET ;
2013-07-08 18:09:47 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
if ( ! next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
DBG ( 0 , ( " setmode <filename> <[+|-]rsha> \n " ) ) ;
2013-07-18 16:22:26 +02:00
return 1 ;
2013-07-08 18:09:47 +02:00
}
2013-07-23 17:39:09 +02:00
fname = talloc_asprintf ( ctx ,
" %s%s " ,
client_get_cur_dir ( ) ,
buf ) ;
if ( ! fname ) {
return 1 ;
2013-07-08 18:09:47 +02:00
}
2013-07-23 17:39:09 +02:00
while ( next_token_talloc ( ctx , & cmd_ptr , & buf , NULL ) ) {
const char * s = buf ;
2013-07-08 18:09:47 +02:00
2013-07-23 17:39:09 +02:00
while ( * s ) {
switch ( * s + + ) {
case ' + ' :
mode = ATTR_SET ;
break ;
case ' - ' :
mode = ATTR_UNSET ;
break ;
case ' r ' :
attr [ mode ] | = FILE_ATTRIBUTE_READONLY ;
break ;
case ' h ' :
attr [ mode ] | = FILE_ATTRIBUTE_HIDDEN ;
break ;
case ' s ' :
attr [ mode ] | = FILE_ATTRIBUTE_SYSTEM ;
break ;
case ' a ' :
attr [ mode ] | = FILE_ATTRIBUTE_ARCHIVE ;
break ;
default :
DBG ( 0 , ( " setmode <filename> <perm=[+|-]rsha> \n " ) ) ;
return 1 ;
}
2013-07-11 15:56:03 +02:00
}
}
2013-07-23 17:39:09 +02:00
if ( attr [ ATTR_SET ] = = 0 & & attr [ ATTR_UNSET ] = = 0 ) {
DBG ( 0 , ( " setmode <filename> <[+|-]rsha> \n " ) ) ;
return 1 ;
2013-07-11 15:56:03 +02:00
}
2013-07-23 17:39:09 +02:00
DBG ( 2 , ( " perm set %d %d \n " , attr [ ATTR_SET ] , attr [ ATTR_UNSET ] ) ) ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* ignore return value: server might not store DOS attributes */
set_remote_attr ( fname , attr [ ATTR_SET ] , ATTR_SET ) ;
set_remote_attr ( fname , attr [ ATTR_UNSET ] , ATTR_UNSET ) ;
return 0 ;
2013-07-11 15:56:03 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_parse_args - parse and set tar command line arguments
* @ flag : string pointing to tar options
* @ val : number of tar arguments
* @ valsize : table of arguments after the flags ( number of element in val )
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* tar arguments work in a weird way . For each flag f that takes a
* value v , the user is supposed to type :
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* on the CLI :
* - Tf1f2f3 v1 v2 v3 TARFILE PATHS . . .
*
* in the interactive session :
* tar f1f2f3 v1 v2 v3 TARFILE PATHS . . .
*
* @ flag has only flags ( eg . " f1f2f3 " ) and @ val has the arguments
* ( values ) following them ( eg . [ " v1 " , " v2 " , " v3 " , " TARFILE " , " PATH1 " ,
* " PATH2 " ] ) .
*
* There are only 2 flags that take an arg : b and N . The other flags
* just change the semantic of PATH or TARFILE .
*
* PATH can be a list of included / excluded paths , the path to a file
* containing a list of included / excluded paths to use ( F flag ) . If no
* PATH is provided , the whole share is used ( / ) .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
int tar_parse_args ( struct tar * t , const char * flag ,
const char * * val , int valsize )
2013-07-11 15:56:03 +02:00
{
2013-07-23 17:39:09 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
bool list = false ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* index of next value to use */
int ival = 0 ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/*
* Reset back some options - could be from interactive version
* all other modes are left as they are
*/
t - > mode . operation = TAR_NO_OPERATION ;
t - > mode . selection = TAR_NO_SELECTION ;
t - > mode . dry = false ;
t - > to_process = false ;
t - > total_size = 0 ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
while ( * flag ) {
switch ( * flag + + ) {
/* operation */
case ' c ' :
if ( t - > mode . operation ! = TAR_NO_OPERATION ) {
printf ( " Tar must be followed by only one of c or x. \n " ) ;
return 1 ;
}
t - > mode . operation = TAR_CREATE ;
break ;
case ' x ' :
if ( t - > mode . operation ! = TAR_NO_OPERATION ) {
printf ( " Tar must be followed by only one of c or x. \n " ) ;
return 1 ;
}
t - > mode . operation = TAR_EXTRACT ;
break ;
2013-07-15 18:58:36 +02:00
2013-07-23 17:39:09 +02:00
/* selection */
case ' I ' :
if ( t - > mode . selection ! = TAR_NO_SELECTION ) {
DBG ( 0 , ( " Only one of I,X,F must be specified \n " ) ) ;
return 1 ;
}
t - > mode . selection = TAR_INCLUDE ;
break ;
case ' X ' :
if ( t - > mode . selection ! = TAR_NO_SELECTION ) {
DBG ( 0 , ( " Only one of I,X,F must be specified \n " ) ) ;
return 1 ;
}
t - > mode . selection = TAR_EXCLUDE ;
break ;
case ' F ' :
if ( t - > mode . selection ! = TAR_NO_SELECTION ) {
DBG ( 0 , ( " Only one of I,X,F must be specified \n " ) ) ;
return 1 ;
}
t - > mode . selection = TAR_INCLUDE ;
list = true ;
break ;
2013-07-19 18:35:01 +02:00
2013-07-23 17:39:09 +02:00
/* blocksize */
case ' b ' :
if ( ival > = valsize ) {
DBG ( 0 , ( " Option b must be followed by a blocksize \n " ) ) ;
return 1 ;
}
2013-07-15 18:58:36 +02:00
2013-07-23 17:39:09 +02:00
if ( tar_set_blocksize ( t , atoi ( val [ ival ] ) ) ) {
DBG ( 0 , ( " Option b must be followed by a valid blocksize \n " ) ) ;
return 1 ;
}
2013-07-15 18:58:36 +02:00
2013-07-23 17:39:09 +02:00
ival + + ;
break ;
2013-07-15 18:58:36 +02:00
2013-07-23 17:39:09 +02:00
/* incremental mode */
case ' g ' :
t - > mode . incremental = true ;
break ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* newer than */
case ' N ' :
if ( ival > = valsize ) {
DBG ( 0 , ( " Option N must be followed by valid file name \n " ) ) ;
return 1 ;
}
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
if ( tar_set_newer_than ( t , val [ ival ] ) ) {
DBG ( 0 , ( " Error setting newer-than time \n " ) ) ;
return 1 ;
}
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
ival + + ;
break ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* reset mode */
case ' a ' :
t - > mode . reset = true ;
break ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* verbose */
case ' q ' :
t - > mode . verbose = true ;
break ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* regex match */
case ' r ' :
t - > mode . regex = true ;
break ;
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
/* dry run mode */
case ' n ' :
if ( t - > mode . operation ! = TAR_CREATE ) {
DBG ( 0 , ( " n is only meaningful when creating a tar-file \n " ) ) ;
return 1 ;
}
2013-07-11 15:56:03 +02:00
2013-07-23 17:39:09 +02:00
t - > mode . dry = true ;
DBG ( 0 , ( " dry_run set \n " ) ) ;
break ;
2013-07-19 18:35:01 +02:00
2013-07-23 17:39:09 +02:00
default :
DBG ( 0 , ( " Unknown tar option \n " ) ) ;
return 1 ;
}
2013-07-19 18:35:01 +02:00
}
2013-07-23 17:39:09 +02:00
/* no selection given? default selection is include */
if ( t - > mode . selection = = TAR_NO_SELECTION ) {
t - > mode . selection = TAR_INCLUDE ;
2013-07-11 15:56:03 +02:00
}
2013-07-23 17:39:09 +02:00
if ( valsize - ival < 1 ) {
DBG ( 0 , ( " No tar file given. \n " ) ) ;
2013-07-05 09:51:43 +02:00
return 1 ;
}
2013-07-03 18:18:25 +02:00
2013-07-23 17:39:09 +02:00
/* handle TARFILE */
t - > tar_path = talloc_strdup ( ctx , val [ ival ] ) ;
ival + + ;
1996-05-04 07:50:46 +00:00
2013-07-23 17:39:09 +02:00
/*
* Make sure that dbf points to stderr if we are using stdout for
* tar output
*/
if ( t - > mode . operation = = TAR_CREATE & & strequal ( t - > tar_path , " - " ) ) {
setup_logging ( " smbclient " , DEBUG_STDERR ) ;
}
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
/* handle PATHs... */
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
/* flag F -> read file list */
if ( list ) {
if ( valsize - ival ! = 1 ) {
DBG ( 0 , ( " Option F must be followed by exactly one filename. \n " ) ) ;
return 1 ;
2013-07-05 09:51:43 +02:00
}
2013-07-23 17:39:09 +02:00
if ( tar_read_inclusion_file ( t , val [ ival ] ) ) {
return 1 ;
}
ival + + ;
}
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
/* otherwise store all the PATHs on the command line */
else {
int i ;
for ( i = ival ; i < valsize ; i + + ) {
tar_add_selection_path ( t , val [ i ] ) ;
}
2013-07-05 09:51:43 +02:00
}
2013-07-23 17:39:09 +02:00
t - > to_process = true ;
tar_dump ( t ) ;
2013-07-03 18:18:25 +02:00
return 0 ;
1996-05-04 07:50:46 +00:00
}
2013-07-05 11:33:55 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_process - start processing archive
2013-07-05 11:33:55 +02:00
*/
2013-07-23 17:39:09 +02:00
int tar_process ( struct tar * t )
2013-07-05 09:51:43 +02:00
{
2013-07-23 17:39:09 +02:00
int rc = 0 ;
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
switch ( t - > mode . operation ) {
case TAR_EXTRACT :
rc = tar_extract ( t ) ;
break ;
case TAR_CREATE :
rc = tar_create ( t ) ;
break ;
default :
DBG ( 0 , ( " Invalid tar state \n " ) ) ;
rc = 1 ;
2013-07-05 09:51:43 +02:00
}
2013-07-18 16:22:26 +02:00
2013-07-23 17:39:09 +02:00
DBG ( 5 , ( " tar_process done, err = %d \n " , rc ) ) ;
return rc ;
2013-07-05 09:51:43 +02:00
}
2013-07-05 11:33:55 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_create - create archive and fetch files
2013-07-05 11:33:55 +02:00
*/
2013-07-23 17:39:09 +02:00
static int tar_create ( struct tar * t )
1996-05-04 07:50:46 +00:00
{
2013-07-05 09:51:43 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
int r ;
int err = 0 ;
NTSTATUS status ;
const char * mask ;
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
t - > archive = archive_write_new ( ) ;
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
if ( ! t - > mode . dry ) {
const int bsize = t - > mode . blocksize * TAR_BLOCK_UNIT ;
r = archive_write_set_bytes_per_block ( t - > archive , bsize ) ;
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Can't use a block size of %d bytes " , bsize ) ) ;
err = 1 ;
goto out ;
}
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
/*
* Use PAX restricted format which is not the most
* conservative choice but has useful extensions and is widely
* supported
*/
r = archive_write_set_format_pax_restricted ( t - > archive ) ;
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Can't use pax restricted format: %s \n " ,
archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto out ;
}
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
if ( strequal ( t - > tar_path , " - " ) ) {
r = archive_write_open_fd ( t - > archive , STDOUT_FILENO ) ;
} else {
r = archive_write_open_filename ( t - > archive , t - > tar_path ) ;
}
2013-07-05 09:51:43 +02:00
2013-07-23 17:39:09 +02:00
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Can't open %s: %s \n " , t - > tar_path ,
archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto out_close ;
2013-07-05 09:51:43 +02:00
}
}
2013-07-23 17:39:09 +02:00
/*
* In inclusion mode , iterate on the inclusion list
*/
if ( t - > mode . selection = = TAR_INCLUDE & & t - > path_list_size > 0 ) {
if ( tar_create_from_list ( t ) ) {
err = 1 ;
goto out_close ;
}
} else {
mask = talloc_asprintf ( ctx , " %s \\ * " , client_get_cur_dir ( ) ) ;
DBG ( 5 , ( " tar_process do_list with mask: %s \n " , mask ) ) ;
status = do_list ( mask , TAR_DO_LIST_ATTR , get_file_callback , false , true ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " do_list fail %s \n " , nt_errstr ( status ) ) ) ;
err = 1 ;
goto out_close ;
}
2013-07-05 09:51:43 +02:00
}
2013-07-23 17:39:09 +02:00
out_close :
DBG ( 0 , ( " Total bytes received: % " PRIu64 " \n " , t - > total_size ) ) ;
2013-07-18 16:22:26 +02:00
2013-07-23 17:39:09 +02:00
if ( ! t - > mode . dry ) {
r = archive_write_close ( t - > archive ) ;
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Fatal: %s \n " , archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto out ;
}
}
out :
archive_write_free ( t - > archive ) ;
return err ;
1996-05-04 07:50:46 +00:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_create_from_list - fetch from path list in include mode
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static int tar_create_from_list ( struct tar * t )
2013-07-10 13:16:08 +02:00
{
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-10 14:12:25 +02:00
int err = 0 ;
2013-07-23 17:39:09 +02:00
NTSTATUS status ;
const char * path , * mask , * base , * start_dir ;
int i ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
start_dir = talloc_strdup ( ctx , client_get_cur_dir ( ) ) ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
for ( i = 0 ; i < t - > path_list_size ; i + + ) {
path = t - > path_list [ i ] ;
base = path_base_name ( path ) ;
mask = talloc_asprintf ( ctx , " %s \\ %s " , client_get_cur_dir ( ) , path ) ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
DBG ( 5 , ( " incl. path='%s', base='%s', mask='%s' \n " ,
path , base ? base : " NULL " , mask ) ) ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
if ( base ) {
base = talloc_asprintf ( ctx , " %s%s \\ " ,
client_get_cur_dir ( ) , path_base_name ( path ) ) ;
DBG ( 5 , ( " cd '%s' before do_list \n " , base ) ) ;
client_set_cur_dir ( base ) ;
}
status = do_list ( mask , TAR_DO_LIST_ATTR , get_file_callback , false , true ) ;
if ( base ) {
client_set_cur_dir ( start_dir ) ;
}
2013-07-10 13:16:08 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-07-23 17:39:09 +02:00
DBG ( 0 , ( " do_list failed on %s (%s) \n " , path , nt_errstr ( status ) ) ) ;
err = 1 ;
goto out ;
2013-07-10 13:16:08 +02:00
}
}
2013-07-10 14:12:25 +02:00
out :
return err ;
2013-07-10 13:16:08 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* get_file_callback - do_list callback
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* Callback for client . c do_list ( ) . Called for each file found on the
* share matching do_list mask . Recursively call do_list ( ) with itself
* as callback when the current file is a directory .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static NTSTATUS get_file_callback ( struct cli_state * cli ,
struct file_info * finfo ,
const char * dir )
2013-07-10 13:16:08 +02:00
{
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
NTSTATUS err = NT_STATUS_OK ;
char * remote_name ;
const char * initial_dir = client_get_cur_dir ( ) ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
remote_name = talloc_asprintf ( ctx , " %s%s " , initial_dir , finfo - > name ) ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
if ( strequal ( finfo - > name , " .. " ) | | strequal ( finfo - > name , " . " ) ) {
2013-07-10 14:12:25 +02:00
goto out ;
}
2013-07-23 17:39:09 +02:00
if ( tar_create_skip_path ( & tar_ctx , remote_name , finfo ) ) {
DBG ( 5 , ( " --- %s \n " , remote_name ) ) ;
2013-07-10 14:12:25 +02:00
goto out ;
}
2013-07-23 17:39:09 +02:00
if ( finfo - > mode & FILE_ATTRIBUTE_DIRECTORY ) {
char * old_dir ;
char * new_dir ;
char * mask ;
2013-07-10 13:16:08 +02:00
2013-07-23 17:39:09 +02:00
old_dir = talloc_strdup ( ctx , initial_dir ) ;
new_dir = talloc_asprintf ( ctx , " %s%s \\ " , initial_dir , finfo - > name ) ;
mask = talloc_asprintf ( ctx , " %s* " , new_dir ) ;
2013-07-10 14:12:25 +02:00
2013-07-23 17:39:09 +02:00
if ( tar_get_file ( & tar_ctx , remote_name , finfo ) ) {
err = NT_STATUS_UNSUCCESSFUL ;
goto out ;
2013-07-10 13:16:08 +02:00
}
2013-07-23 17:39:09 +02:00
client_set_cur_dir ( new_dir ) ;
do_list ( mask , TAR_DO_LIST_ATTR , get_file_callback , false , true ) ;
client_set_cur_dir ( old_dir ) ;
2013-07-10 13:16:08 +02:00
}
2013-07-23 17:39:09 +02:00
else {
if ( tar_get_file ( & tar_ctx , remote_name , finfo ) ) {
err = NT_STATUS_UNSUCCESSFUL ;
goto out ;
}
2013-07-11 00:57:40 +02:00
}
out :
return err ;
}
2013-07-23 16:55:50 +02:00
/**
* tar_get_file - fetch a remote file to the local archive
* @ full_dos_path : path to the file to fetch
* @ finfo : attributes of the file to fetch
*/
2013-07-11 00:57:40 +02:00
static int tar_get_file ( struct tar * t , const char * full_dos_path ,
struct file_info * finfo )
{
extern struct cli_state * cli ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
NTSTATUS status ;
struct archive_entry * entry ;
char * full_unix_path ;
char buf [ TAR_CLI_READ_SIZE ] ;
size_t len ;
uint64_t off = 0 ;
uint16_t remote_fd = ( uint16_t ) - 1 ;
int err = 0 , r ;
2013-07-11 18:17:25 +02:00
const bool isdir = finfo - > mode & FILE_ATTRIBUTE_DIRECTORY ;
2013-07-11 00:57:40 +02:00
2013-07-11 15:56:03 +02:00
DBG ( 5 , ( " +++ %s \n " , full_dos_path ) ) ;
2013-07-11 18:17:25 +02:00
t - > total_size + = finfo - > size ;
if ( t - > mode . dry ) {
goto out ;
}
2013-07-11 00:57:40 +02:00
2013-07-16 14:47:29 +02:00
if ( t - > mode . reset ) {
2013-07-18 16:22:26 +02:00
/* ignore return value: server might not store DOS attributes */
2013-07-16 14:47:29 +02:00
set_remote_attr ( full_dos_path , FILE_ATTRIBUTE_ARCHIVE , ATTR_UNSET ) ;
}
2013-07-11 00:57:40 +02:00
full_unix_path = talloc_asprintf ( ctx , " .%s " , full_dos_path ) ;
string_replace ( full_unix_path , ' \\ ' , ' / ' ) ;
entry = archive_entry_new ( ) ;
archive_entry_copy_pathname ( entry , full_unix_path ) ;
2013-07-11 18:17:25 +02:00
archive_entry_set_filetype ( entry , isdir ? AE_IFDIR : AE_IFREG ) ;
archive_entry_set_atime ( entry ,
finfo - > atime_ts . tv_sec ,
finfo - > atime_ts . tv_nsec ) ;
archive_entry_set_mtime ( entry ,
finfo - > mtime_ts . tv_sec ,
finfo - > mtime_ts . tv_nsec ) ;
archive_entry_set_ctime ( entry ,
finfo - > ctime_ts . tv_sec ,
finfo - > ctime_ts . tv_nsec ) ;
archive_entry_set_perm ( entry , isdir ? 0755 : 0644 ) ;
/*
* check if we can safely cast unsigned file size to libarchive
* signed size . Very unlikely problem ( > 9 exabyte file )
*/
if ( finfo - > size > INT64_MAX ) {
DBG ( 0 , ( " Remote file %s too big \n " , full_dos_path ) ) ;
goto out_entry ;
}
archive_entry_set_size ( entry , ( int64_t ) finfo - > size ) ;
2013-07-11 00:57:40 +02:00
r = archive_write_header ( t - > archive , entry ) ;
if ( r ! = ARCHIVE_OK ) {
2013-07-11 15:56:03 +02:00
DBG ( 0 , ( " Fatal: %s \n " , archive_error_string ( t - > archive ) ) ) ;
2013-07-11 00:57:40 +02:00
err = 1 ;
goto out_entry ;
}
if ( isdir ) {
2013-07-11 18:17:25 +02:00
DBG ( 5 , ( " get_file skip dir %s \n " , full_dos_path ) ) ;
2013-07-11 00:57:40 +02:00
goto out_entry ;
}
status = cli_open ( cli , full_dos_path , O_RDONLY , DENY_NONE , & remote_fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-07-11 15:56:03 +02:00
DBG ( 0 , ( " %s opening remote file %s \n " ,
2013-07-11 00:57:40 +02:00
nt_errstr ( status ) , full_dos_path ) ) ;
goto out_entry ;
}
do {
status = cli_read ( cli , remote_fd , buf , off , sizeof ( buf ) , & len ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-07-11 15:56:03 +02:00
DBG ( 0 , ( " Error reading file %s : %s \n " ,
2013-07-11 00:57:40 +02:00
full_dos_path , nt_errstr ( status ) ) ) ;
err = 1 ;
goto out_close ;
}
off + = len ;
r = archive_write_data ( t - > archive , buf , len ) ;
2013-07-11 18:17:25 +02:00
if ( r < 0 ) {
2013-07-11 15:56:03 +02:00
DBG ( 0 , ( " Fatal: %s \n " , archive_error_string ( t - > archive ) ) ) ;
2013-07-11 00:57:40 +02:00
err = 1 ;
2013-07-11 18:17:25 +02:00
goto out_close ;
2013-07-11 00:57:40 +02:00
}
} while ( off < finfo - > size ) ;
out_close :
cli_close ( cli , remote_fd ) ;
out_entry :
archive_entry_free ( entry ) ;
2013-07-11 18:17:25 +02:00
out :
2013-07-11 00:57:40 +02:00
return err ;
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_extract - open archive and send files .
*/
static int tar_extract ( struct tar * t )
{
int err = 0 ;
int r ;
struct archive_entry * entry ;
const size_t bsize = t - > mode . blocksize * TAR_BLOCK_UNIT ;
t - > archive = archive_read_new ( ) ;
archive_read_support_format_all ( t - > archive ) ;
archive_read_support_filter_all ( t - > archive ) ;
if ( strequal ( t - > tar_path , " - " ) ) {
r = archive_read_open_fd ( t - > archive , STDIN_FILENO , bsize ) ;
} else {
r = archive_read_open_filename ( t - > archive , t - > tar_path , bsize ) ;
}
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Can't open %s : %s \n " , t - > tar_path ,
archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto out ;
}
for ( ; ; ) {
r = archive_read_next_header ( t - > archive , & entry ) ;
if ( r = = ARCHIVE_EOF ) {
break ;
}
if ( r = = ARCHIVE_WARN ) {
DBG ( 0 , ( " Warning: %s \n " , archive_error_string ( t - > archive ) ) ) ;
}
if ( r = = ARCHIVE_FATAL ) {
DBG ( 0 , ( " Fatal: %s \n " , archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto out ;
}
if ( tar_extract_skip_path ( t , entry ) ) {
DBG ( 5 , ( " --- %s \n " , archive_entry_pathname ( entry ) ) ) ;
continue ;
}
DBG ( 5 , ( " +++ %s \n " , archive_entry_pathname ( entry ) ) ) ;
if ( tar_send_file ( t , entry ) ) {
err = 1 ;
goto out ;
}
}
out :
r = archive_read_free ( t - > archive ) ;
if ( r ! = ARCHIVE_OK ) {
DBG ( 0 , ( " Can't close %s : %s \n " , t - > tar_path ,
archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
}
return err ;
}
/**
* tar_send_file - send @ entry to the remote server
* @ entry : current archive entry
2013-07-23 16:55:50 +02:00
*
2013-07-23 17:39:09 +02:00
* Handle the creation of the parent directories and transfer the
* entry to a new remote file .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static int tar_send_file ( struct tar * t , struct archive_entry * entry )
2013-07-11 00:57:40 +02:00
{
2013-07-23 17:39:09 +02:00
extern struct cli_state * cli ;
2013-07-11 00:57:40 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
char * dos_path ;
char * full_path ;
NTSTATUS status ;
uint16_t remote_fd = ( uint16_t ) - 1 ;
int err = 0 ;
int flags = O_RDWR | O_CREAT | O_TRUNC ;
mode_t mode = archive_entry_filetype ( entry ) ;
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
dos_path = talloc_strdup ( ctx , archive_entry_pathname ( entry ) ) ;
fix_unix_path ( dos_path , true ) ;
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
full_path = talloc_strdup ( ctx , client_get_cur_dir ( ) ) ;
full_path = talloc_strdup_append ( full_path , dos_path ) ;
if ( mode ! = AE_IFREG & & mode ! = AE_IFDIR ) {
DBG ( 0 , ( " Skipping non-dir & non-regular file %s \n " , full_path ) ) ;
2013-07-11 15:56:03 +02:00
goto out ;
}
2013-07-23 17:39:09 +02:00
if ( make_remote_path ( full_path ) ) {
err = 1 ;
2013-07-11 00:57:40 +02:00
goto out ;
}
2013-07-23 17:39:09 +02:00
if ( mode = = AE_IFDIR ) {
goto out ;
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
status = cli_open ( cli , full_path , flags , DENY_NONE , & remote_fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " Error opening remote file %s: %s \n " ,
full_path , nt_errstr ( status ) ) ) ;
err = 1 ;
goto out ;
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
for ( ; ; ) {
const void * buf ;
size_t len ;
off_t off ;
int r ;
r = archive_read_data_block ( t - > archive , & buf , & len , & off ) ;
if ( r = = ARCHIVE_EOF ) {
break ;
}
if ( r = = ARCHIVE_WARN ) {
DBG ( 0 , ( " Warning: %s \n " , archive_error_string ( t - > archive ) ) ) ;
}
if ( r = = ARCHIVE_FATAL ) {
DBG ( 0 , ( " Fatal: %s \n " , archive_error_string ( t - > archive ) ) ) ;
err = 1 ;
goto close_out ;
2013-07-11 00:57:40 +02:00
}
2013-07-23 17:39:09 +02:00
status = cli_writeall ( cli , remote_fd , 0 , buf , off , len , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " Error writing remote file %s: %s \n " ,
full_path , nt_errstr ( status ) ) ) ;
err = 1 ;
goto close_out ;
}
2013-07-11 00:57:40 +02:00
}
2013-07-23 17:39:09 +02:00
close_out :
status = cli_close ( cli , remote_fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " Error losing remote file %s: %s \n " ,
full_path , nt_errstr ( status ) ) ) ;
err = 1 ;
}
out :
return err ;
}
/**
* tar_add_selection_path - add a path to the path list
* @ path : path to add
*/
static void tar_add_selection_path ( struct tar * t , const char * path )
{
TALLOC_CTX * ctx = talloc_tos ( ) ;
if ( ! t - > path_list ) {
t - > path_list = str_list_make_empty ( ctx ) ;
t - > path_list_size = 0 ;
2013-07-11 00:57:40 +02:00
}
2013-07-23 17:39:09 +02:00
t - > path_list = str_list_add ( ( const char * * ) t - > path_list , path ) ;
t - > path_list_size + + ;
fix_unix_path ( t - > path_list [ t - > path_list_size - 1 ] , true ) ;
2013-07-11 00:57:40 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_set_blocksize - set block size in TAR_BLOCK_UNIT
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static int tar_set_blocksize ( struct tar * t , int size )
2013-07-17 16:25:34 +02:00
{
2013-07-23 17:39:09 +02:00
if ( size < = 0 | | size > TAR_MAX_BLOCK_SIZE ) {
return 1 ;
}
2013-07-17 16:25:34 +02:00
2013-07-23 17:39:09 +02:00
t - > mode . blocksize = size ;
2013-07-17 16:25:34 +02:00
2013-07-23 17:39:09 +02:00
return 0 ;
}
2013-07-17 16:25:34 +02:00
2013-07-23 17:39:09 +02:00
/**
* tar_set_newer_than - set date threshold of saved files
* @ filename : local path to a file
*
* Only files newer than the modification time of @ filename will be
* saved .
*
* Note : this function set the global variable newer_than from
* client . c . Thus the time is not a field of the tar structure . See
* cmd_newer ( ) to change its value from an interactive session .
*/
static int tar_set_newer_than ( struct tar * t , const char * filename )
{
extern time_t newer_than ;
SMB_STRUCT_STAT stbuf ;
2013-07-17 16:25:34 +02:00
2013-07-23 17:39:09 +02:00
if ( sys_stat ( filename , & stbuf , false ) ! = 0 ) {
DBG ( 0 , ( " Error setting newer-than time \n " ) ) ;
return 1 ;
2013-07-17 16:25:34 +02:00
}
2013-07-23 17:39:09 +02:00
newer_than = convert_timespec_to_time_t ( stbuf . st_ex_mtime ) ;
DBG ( 1 , ( " Getting files newer than %s \n " , time_to_asc ( newer_than ) ) ) ;
return 0 ;
2013-07-17 16:25:34 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_read_inclusion_file - set path list from file
* @ filename : path to the list file
*
* Read and add each line of @ filename to the path list .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static int tar_read_inclusion_file ( struct tar * t , const char * filename )
2013-07-11 00:57:40 +02:00
{
2013-07-23 17:39:09 +02:00
char * line ;
2013-07-11 00:57:40 +02:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2013-07-23 17:39:09 +02:00
int fd = open ( filename , O_RDONLY ) ;
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
if ( fd < 0 ) {
DBG ( 0 , ( " Can't open inclusion file '%s': %s \n " , filename , strerror ( errno ) ) ) ;
return 1 ;
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
while ( ( line = afdgets ( fd , ctx , 0 ) ) ) {
tar_add_selection_path ( t , line ) ;
}
2013-07-18 17:06:33 +02:00
2013-07-23 17:39:09 +02:00
close ( fd ) ;
return 0 ;
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
/**
* tar_path_in_list - return true if @ path is in the path list
* @ path : path to find
* @ reverse : when true also try to find path list element in @ path
*
* Look at each path of the path list and return true if @ path is a
* subpath of one of them .
*
* If you want / path to be in the path list ( path / a / , path / b / ) set
* @ reverse to true to try to match the other way around .
*/
static bool tar_path_in_list ( struct tar * t , const char * path , bool reverse )
{
int i ;
const char * p = path ;
const char * pattern ;
bool res ;
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
if ( ! p | | ! p [ 0 ] )
return false ;
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
p = skip_useless_char_in_path ( p ) ;
for ( i = 0 ; i < t - > path_list_size ; i + + ) {
pattern = skip_useless_char_in_path ( t - > path_list [ i ] ) ;
res = is_subpath ( p , pattern ) ;
if ( reverse ) {
res = res | | is_subpath ( pattern , p ) ;
2013-07-17 16:25:34 +02:00
}
2013-07-23 17:39:09 +02:00
if ( res ) {
return true ;
2013-07-17 16:25:34 +02:00
}
2013-07-11 18:17:25 +02:00
}
2013-07-23 17:39:09 +02:00
return false ;
2013-07-10 13:16:08 +02:00
}
2007-12-19 21:59:28 +01:00
2013-07-16 19:15:48 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_extract_skip_path - return true if @ entry should be skipped
* @ entry : current tar entry
2013-07-16 19:15:48 +02:00
*
2013-07-23 17:39:09 +02:00
* Skip predicate for tar extraction ( archive to server ) only .
2013-07-16 19:15:48 +02:00
*/
2013-07-23 17:39:09 +02:00
static bool tar_extract_skip_path ( struct tar * t ,
struct archive_entry * entry )
2013-07-16 19:15:48 +02:00
{
2013-07-23 17:39:09 +02:00
const bool skip = true ;
const char * fullpath = archive_entry_pathname ( entry ) ;
bool in = true ;
2013-07-16 19:15:48 +02:00
2013-07-23 17:39:09 +02:00
if ( t - > path_list_size < = 0 ) {
return ! skip ;
2013-07-16 19:15:48 +02:00
}
2013-07-23 17:39:09 +02:00
if ( t - > mode . regex ) {
in = mask_match_list ( fullpath , t - > path_list , t - > path_list_size , true ) ;
} else {
in = tar_path_in_list ( t , fullpath , false ) ;
2013-07-16 19:15:48 +02:00
}
2013-07-23 17:39:09 +02:00
if ( t - > mode . selection = = TAR_EXCLUDE ) {
in = ! in ;
}
2013-07-16 19:15:48 +02:00
2013-07-23 17:39:09 +02:00
return in ? ! skip : skip ;
2013-07-16 19:15:48 +02:00
}
2013-07-08 18:09:47 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_create_skip_path - return true if @ fullpath shoud be skipped
* @ fullpath : full remote path of the current file
* @ finfo : remote file attributes
2013-07-08 18:09:47 +02:00
*
2013-07-23 17:39:09 +02:00
* Skip predicate for tar creation ( server to archive ) only .
2013-07-08 18:09:47 +02:00
*/
2013-07-23 17:39:09 +02:00
static bool tar_create_skip_path ( struct tar * t ,
const char * fullpath ,
const struct file_info * finfo )
1996-05-04 07:50:46 +00:00
{
2013-07-23 17:39:09 +02:00
/* syntaxic sugar */
const bool skip = true ;
const mode_t mode = finfo - > mode ;
const bool isdir = mode & FILE_ATTRIBUTE_DIRECTORY ;
const bool exclude = t - > mode . selection = = TAR_EXCLUDE ;
bool in = true ;
2013-07-16 19:15:48 +02:00
2013-07-23 17:39:09 +02:00
if ( ! isdir ) {
/* 1. if we dont want X and we have X, skip */
if ( ! t - > mode . system & & ( mode & FILE_ATTRIBUTE_SYSTEM ) ) {
return skip ;
}
if ( ! t - > mode . hidden & & ( mode & FILE_ATTRIBUTE_HIDDEN ) ) {
return skip ;
}
/* 2. if we only want archive and it's not, skip */
if ( t - > mode . incremental & & ! ( mode & FILE_ATTRIBUTE_ARCHIVE ) ) {
return skip ;
}
2013-07-18 16:22:26 +02:00
}
2013-07-16 19:15:48 +02:00
2013-07-23 17:39:09 +02:00
/* 3. is it in the selection list? */
2013-07-16 19:15:48 +02:00
2013-07-23 17:39:09 +02:00
/*
* tar_create_from_list ( ) use the include list as a starting
* point , no need to check
*/
if ( ! exclude ) {
return ! skip ;
2013-07-16 19:15:48 +02:00
}
2013-07-23 17:39:09 +02:00
/* we are now in exclude mode */
/* no matter the selection, no list => include everything */
if ( t - > path_list_size < = 0 ) {
return ! skip ;
2013-07-16 19:15:48 +02:00
}
2013-07-23 17:39:09 +02:00
if ( t - > mode . regex ) {
in = mask_match_list ( fullpath , t - > path_list , t - > path_list_size , true ) ;
} else {
in = tar_path_in_list ( t , fullpath , isdir & & ! exclude ) ;
2013-07-16 19:15:48 +02:00
}
2013-07-23 17:39:09 +02:00
return in ? skip : ! skip ;
1996-05-04 07:50:46 +00:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* tar_to_process - return true if @ t is ready to be processed
*
* @ t is ready if it properly parsed command line arguments .
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
bool tar_to_process ( struct tar * t )
2013-07-09 18:01:47 +02:00
{
2013-07-23 17:39:09 +02:00
return t - > to_process ;
}
2013-07-09 18:01:47 +02:00
2013-07-23 17:39:09 +02:00
/**
* skip_useless_char_in_path - skip leading slashes / dots
*
* Skip leading slashes , backslashes and dot - slashes .
*/
static const char * skip_useless_char_in_path ( const char * p )
{
while ( p ) {
if ( * p = = ' / ' | | * p = = ' \\ ' ) {
p + + ;
2013-07-15 18:58:36 +02:00
}
2013-07-23 17:39:09 +02:00
else if ( p [ 0 ] = = ' . ' & & ( p [ 1 ] = = ' / ' | | p [ 1 ] = = ' \\ ' ) ) {
p + = 2 ;
2013-07-10 13:16:08 +02:00
}
2013-07-23 17:39:09 +02:00
else
return p ;
2013-07-09 18:01:47 +02:00
}
2013-07-23 17:39:09 +02:00
return p ;
2013-07-09 18:01:47 +02:00
}
2013-07-23 16:55:50 +02:00
/**
2013-07-23 17:39:09 +02:00
* is_subpath - return true if the path @ sub is a subpath of @ full .
* @ sub : path to test
* @ full : container path
*
* String comparaison is case - insensitive .
*
* Return true if @ sub = @ full
2013-07-23 16:55:50 +02:00
*/
2013-07-23 17:39:09 +02:00
static bool is_subpath ( const char * sub , const char * full )
1996-05-04 07:50:46 +00:00
{
2013-07-23 17:39:09 +02:00
const char * full_copy = full ;
2013-07-09 15:24:40 +02:00
2013-07-23 17:39:09 +02:00
while ( * full & & * sub & &
( * full = = * sub | | tolower_m ( * full ) = = tolower_m ( * sub ) | |
( * full = = ' \\ ' & & * sub = = ' / ' ) | | ( * full = = ' / ' & & * sub = = ' \\ ' ) ) ) {
full + + ; sub + + ;
2013-07-09 15:24:40 +02:00
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
/* if full has a trailing slash, it compared equal, so full is an "initial"
string of sub .
*/
if ( ! * full & & full ! = full_copy & & ( * ( full - 1 ) = = ' / ' | | * ( full - 1 ) = = ' \\ ' ) )
return true ;
/* ignore trailing slash on full */
if ( ! * sub & & ( * full = = ' / ' | | * full = = ' \\ ' ) & & ! * ( full + 1 ) )
return true ;
/* check for full is an "initial" string of sub */
if ( ( * sub = = ' / ' | | * sub = = ' \\ ' ) & & ! * full )
return true ;
return * full = = * sub ;
1998-06-17 01:52:57 +00:00
}
2013-07-08 18:09:47 +02:00
/**
2013-07-23 17:39:09 +02:00
* set_remote_attr - set DOS attributes of a remote file
* @ filename : path to the file name
* @ new_attr : attribute bit mask to use
* @ mode : one of ATTR_SET or ATTR_UNSET
2013-07-08 18:09:47 +02:00
*
2013-07-23 17:39:09 +02:00
* Update the file attributes with the one provided .
2013-07-08 18:09:47 +02:00
*/
2013-07-23 17:39:09 +02:00
static int set_remote_attr ( const char * filename , uint16 new_attr , int mode )
1996-05-04 07:50:46 +00:00
{
2013-07-23 17:39:09 +02:00
extern struct cli_state * cli ;
uint16 old_attr ;
NTSTATUS status ;
2013-07-08 18:09:47 +02:00
2013-07-23 17:39:09 +02:00
status = cli_getatr ( cli , filename , & old_attr , NULL , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " cli_getatr failed: %s \n " , nt_errstr ( status ) ) ) ;
return 1 ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
if ( mode = = ATTR_SET ) {
new_attr | = old_attr ;
} else {
new_attr = old_attr & ~ new_attr ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
status = cli_setatr ( cli , filename , new_attr , 0 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 1 , ( " cli_setatr failed: %s \n " , nt_errstr ( status ) ) ) ;
return 1 ;
}
return 0 ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
/**
* make_remote_path - recursively make remote dirs
* @ full_path : full hierarchy to create
*
* Create @ full_path and each parent directories as needed .
*/
static int make_remote_path ( const char * full_path )
{
extern struct cli_state * cli ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
char * path ;
char * subpath ;
char * state ;
char * last_backslash ;
char * p ;
int len ;
NTSTATUS status ;
int err = 0 ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
subpath = talloc_strdup ( ctx , full_path ) ;
path = talloc_strdup ( ctx , full_path ) ;
len = talloc_get_size ( path ) - 1 ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
last_backslash = strrchr_m ( path , ' \\ ' ) ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
if ( ! last_backslash ) {
goto out ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
* last_backslash = 0 ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
subpath [ 0 ] = 0 ;
p = strtok_r ( path , " \\ " , & state ) ;
while ( p ) {
strlcat ( subpath , p , len ) ;
status = cli_chkpath ( cli , subpath ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
status = cli_mkdir ( cli , subpath ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG ( 0 , ( " Can't mkdir %s: %s \n " , subpath , nt_errstr ( status ) ) ) ;
err = 1 ;
goto out ;
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
DBG ( 3 , ( " mkdir %s \n " , subpath ) ) ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
strlcat ( subpath , " \\ " , len ) ;
p = strtok_r ( NULL , " / \\ " , & state ) ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
out :
return err ;
}
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
# define XSET(v) [v] = #v
# define XTABLE(v, t) DBG(2, ("DUMP:%-20.20s = %s\n", #v, t[v]))
# define XBOOL(v) DBG(2, ("DUMP:%-20.20s = %d\n", #v, v ? 1 : 0))
# define XSTR(v) DBG(2, ("DUMP:%-20.20s = %s\n", #v, v ? v : "NULL"))
# define XINT(v) DBG(2, ("DUMP:%-20.20s = %d\n", #v, v))
# define XUINT64(v) DBG(2, ("DUMP:%-20.20s = %" PRIu64 "\n", #v, v))
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
/**
* tar_dump - dump tar structure on stdout
*/
static void tar_dump ( struct tar * t )
{
int i ;
const char * op [ ] = {
XSET ( TAR_NO_OPERATION ) ,
XSET ( TAR_CREATE ) ,
XSET ( TAR_EXTRACT ) ,
} ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
const char * sel [ ] = {
XSET ( TAR_NO_SELECTION ) ,
XSET ( TAR_INCLUDE ) ,
XSET ( TAR_EXCLUDE ) ,
} ;
XBOOL ( t - > to_process ) ;
XTABLE ( t - > mode . operation , op ) ;
XTABLE ( t - > mode . selection , sel ) ;
XINT ( t - > mode . blocksize ) ;
XBOOL ( t - > mode . hidden ) ;
XBOOL ( t - > mode . system ) ;
XBOOL ( t - > mode . incremental ) ;
XBOOL ( t - > mode . reset ) ;
XBOOL ( t - > mode . dry ) ;
XBOOL ( t - > mode . verbose ) ;
XUINT64 ( t - > total_size ) ;
XSTR ( t - > tar_path ) ;
XINT ( t - > path_list_size ) ;
for ( i = 0 ; t - > path_list & & t - > path_list [ i ] ; i + + ) {
DBG ( 2 , ( " DUMP: t->path_list[%2d] = %s \n " , i , t - > path_list [ i ] ) ) ;
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
DBG ( 2 , ( " DUMP:t->path_list @ %p (%d elem) \n " , t - > path_list , i ) ) ;
}
# undef XSET
# undef XTABLE
# undef XBOOL
# undef XSTR
# undef XINT
/**
* max_token - return upper limit for the number of token in @ str
*
* The result is not exact , the actual number of token might be less
* than what is returned .
*/
static int max_token ( const char * str )
{
const char * s = str ;
int nb = 0 ;
if ( ! str ) {
return 0 ;
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
while ( * s ) {
if ( isspace ( * s ) ) {
nb + + ;
}
s + + ;
2013-07-08 18:09:47 +02:00
}
2013-07-23 17:39:09 +02:00
nb + + ;
2013-07-08 18:09:47 +02:00
2013-07-23 17:39:09 +02:00
return nb ;
}
2013-07-11 00:57:40 +02:00
2013-07-23 17:39:09 +02:00
/**
* fix_unix_path - convert @ path to a DOS path
* @ path : path to convert
* @ removeprefix : if true , remove leading . / or / .
*/
static char * fix_unix_path ( char * path , bool removeprefix )
{
char * from = path , * to = path ;
2013-07-08 18:09:47 +02:00
2013-07-23 17:39:09 +02:00
if ( ! path | | ! * path )
return path ;
/* remove prefix:
* . / path = > path
* / path = > path
*/
if ( removeprefix ) {
/* /path */
if ( path [ 0 ] = = ' / ' | | path [ 0 ] = = ' \\ ' ) {
from + = 1 ;
2013-07-05 18:14:50 +02:00
}
2013-07-08 18:09:47 +02:00
2013-07-23 17:39:09 +02:00
/* ./path */
if ( path [ 1 ] & & path [ 0 ] = = ' . ' & & ( path [ 1 ] = = ' / ' | | path [ 1 ] = = ' \\ ' ) ) {
from + = 2 ;
2013-07-08 18:09:47 +02:00
}
2013-07-05 18:14:50 +02:00
}
2013-07-23 17:39:09 +02:00
/* replace / with \ */
while ( * from ) {
if ( * from = = ' / ' ) {
* to = ' \\ ' ;
} else {
* to = * from ;
2013-07-08 18:09:47 +02:00
}
2013-07-23 17:39:09 +02:00
from + + ; to + + ;
2013-07-08 18:09:47 +02:00
}
2013-07-23 17:39:09 +02:00
* to = 0 ;
2013-07-05 18:14:50 +02:00
2013-07-23 17:39:09 +02:00
return path ;
}
/**
* path_base_name - return @ path basename
*
* If @ path doesn ' t contain any directory separator return NULL .
*/
static char * path_base_name ( const char * path )
{
TALLOC_CTX * ctx = talloc_tos ( ) ;
char * base = NULL ;
int last = - 1 ;
int i ;
for ( i = 0 ; path [ i ] ; i + + ) {
if ( path [ i ] = = ' \\ ' | | path [ i ] = = ' / ' ) {
last = i ;
}
}
if ( last > = 0 ) {
base = talloc_strdup ( ctx , path ) ;
base [ last ] = 0 ;
}
return base ;
1996-05-04 07:50:46 +00:00
}