2004-01-07 00:43:52 +00:00
/*
smbget : a wget - like utility with support for recursive downloading and
smb : // urls
Copyright ( C ) 2003 - 2004 Jelmer Vernooij < jelmer @ samba . org >
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
2004-01-07 00:43:52 +00:00
( at your option ) any later version .
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +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 .
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +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/>. */
2004-01-07 00:43:52 +00:00
# include "includes.h"
2011-04-26 14:58:01 +02:00
# include "system/filesys.h"
2010-08-05 10:49:53 +02:00
# include "popt_common.h"
2004-01-07 00:43:52 +00:00
# include "libsmbclient.h"
# if _FILE_OFFSET_BITS==64
# define OFF_T_FORMAT "%lld"
2005-12-03 20:18:43 +00:00
# define OFF_T_FORMAT_CAST long long
2004-01-07 00:43:52 +00:00
# else
# define OFF_T_FORMAT "%ld"
2005-12-03 20:18:43 +00:00
# define OFF_T_FORMAT_CAST long
2004-01-07 00:43:52 +00:00
# endif
2008-03-09 13:50:56 +01:00
static int columns = 0 ;
2004-01-07 00:43:52 +00:00
2008-03-11 19:49:08 +01:00
static int debuglevel , update ;
2005-11-21 15:52:10 +00:00
static char * outputfile ;
2008-03-09 13:50:56 +01:00
static time_t total_start_time = 0 ;
static off_t total_bytes = 0 ;
2004-01-07 00:43:52 +00:00
# define SMB_MAXPATHLEN MAXPATHLEN
/* Number of bytes to read when checking whether local and remote file are really the same file */
# define RESUME_CHECK_SIZE 512
# define RESUME_DOWNLOAD_OFFSET 1024
# define RESUME_CHECK_OFFSET RESUME_DOWNLOAD_OFFSET+RESUME_CHECK_SIZE
/* Number of bytes to read at once */
# define SMB_DEFAULT_BLOCKSIZE 64000
2008-03-09 13:50:56 +01:00
static const char * username = NULL , * password = NULL , * workgroup = NULL ;
static int nonprompt = 0 , quiet = 0 , dots = 0 , keep_permissions = 0 , verbose = 0 , send_stdout = 0 ;
static int blocksize = SMB_DEFAULT_BLOCKSIZE ;
2004-01-07 00:43:52 +00:00
2010-07-04 11:32:50 +02:00
static int smb_download_file ( const char * base , const char * name , int recursive ,
int resume , int toplevel , char * outfile ) ;
2004-01-07 00:43:52 +00:00
2006-12-19 20:16:52 +00:00
static int get_num_cols ( void )
2004-01-07 00:43:52 +00:00
{
# ifdef TIOCGWINSZ
struct winsize ws ;
if ( ioctl ( STDOUT_FILENO , TIOCGWINSZ , & ws ) < 0 ) {
return 0 ;
}
return ws . ws_col ;
# else
# warning No support for TIOCGWINSZ
char * cols = getenv ( " COLUMNS " ) ;
if ( ! cols ) return 0 ;
return atoi ( cols ) ;
# endif
}
2006-12-19 20:16:52 +00:00
static void change_columns ( int sig )
2004-01-07 00:43:52 +00:00
{
columns = get_num_cols ( ) ;
}
2006-12-19 20:16:52 +00:00
static void human_readable ( off_t s , char * buffer , int l )
2004-01-07 00:43:52 +00:00
{
2011-04-07 22:03:49 +02:00
if ( s > 1024 * 1024 * 1024 ) {
snprintf ( buffer , l , " %.2fGB " , 1.0 * s / ( 1024 * 1024 * 1024 ) ) ;
} else if ( s > 1024 * 1024 ) {
snprintf ( buffer , l , " %.2fMB " , 1.0 * s / ( 1024 * 1024 ) ) ;
} else if ( s > 1024 ) {
snprintf ( buffer , l , " %.2fkB " , 1.0 * s / 1024 ) ;
} else {
snprintf ( buffer , l , OFF_T_FORMAT " b " , ( OFF_T_FORMAT_CAST ) s ) ;
}
2004-01-07 00:43:52 +00:00
}
2006-12-19 20:16:52 +00:00
static void get_auth_data ( const char * srv , const char * shr , char * wg , int wglen , char * un , int unlen , char * pw , int pwlen )
2004-01-07 00:43:52 +00:00
{
static char hasasked = 0 ;
char * wgtmp , * usertmp ;
char tmp [ 128 ] ;
if ( hasasked ) return ;
hasasked = 1 ;
if ( ! nonprompt & & ! username ) {
printf ( " Username for %s at %s [guest] " , shr , srv ) ;
2008-12-31 18:06:57 -08:00
if ( fgets ( tmp , sizeof ( tmp ) , stdin ) = = NULL ) {
return ;
}
2009-11-30 21:48:33 +01:00
if ( ( strlen ( tmp ) > 0 ) & & ( tmp [ strlen ( tmp ) - 1 ] = = ' \n ' ) ) {
tmp [ strlen ( tmp ) - 1 ] = ' \0 ' ;
}
2004-01-07 00:43:52 +00:00
strncpy ( un , tmp , unlen - 1 ) ;
} else if ( username ) strncpy ( un , username , unlen - 1 ) ;
if ( ! nonprompt & & ! password ) {
2012-11-23 14:38:14 +01:00
char * prompt ;
2008-12-31 18:06:57 -08:00
if ( asprintf ( & prompt , " Password for %s at %s: " , shr , srv ) = = - 1 ) {
return ;
}
2012-11-23 14:38:14 +01:00
( void ) samba_getpass ( prompt , pw , pwlen , false , false ) ;
2004-01-07 00:43:52 +00:00
free ( prompt ) ;
} else if ( password ) strncpy ( pw , password , pwlen - 1 ) ;
if ( workgroup ) strncpy ( wg , workgroup , wglen - 1 ) ;
2005-07-25 15:04:25 +00:00
wgtmp = SMB_STRNDUP ( wg , wglen ) ;
usertmp = SMB_STRNDUP ( un , unlen ) ;
2004-01-07 00:43:52 +00:00
if ( ! quiet ) printf ( " Using workgroup %s, %s%s \n " , wgtmp , * usertmp ? " user " : " guest user " , usertmp ) ;
free ( wgtmp ) ; free ( usertmp ) ;
}
2010-01-28 10:38:24 -08:00
/* Return 1 on error, 0 on success. */
2006-12-19 20:16:52 +00:00
static int smb_download_dir ( const char * base , const char * name , int resume )
2004-01-07 00:43:52 +00:00
{
char path [ SMB_MAXPATHLEN ] ;
int dirhandle ;
struct smbc_dirent * dirent ;
const char * relname = name ;
char * tmpname ;
struct stat remotestat ;
2010-01-28 10:38:24 -08:00
int ret = 0 ;
2004-01-07 00:43:52 +00:00
snprintf ( path , SMB_MAXPATHLEN - 1 , " %s%s%s " , base , ( base [ 0 ] & & name [ 0 ] & & name [ 0 ] ! = ' / ' & & base [ strlen ( base ) - 1 ] ! = ' / ' ) ? " / " : " " , name ) ;
/* List files in directory and call smb_download_file on them */
dirhandle = smbc_opendir ( path ) ;
if ( dirhandle < 1 ) {
2010-07-04 11:32:50 +02:00
if ( errno = = ENOTDIR ) {
return smb_download_file ( base , name , 1 , resume ,
0 , NULL ) ;
}
2004-01-07 00:43:52 +00:00
fprintf ( stderr , " Can't open directory %s: %s \n " , path , strerror ( errno ) ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
while ( * relname = = ' / ' ) relname + + ;
mkdir ( relname , 0755 ) ;
2010-07-04 11:02:21 +02:00
2004-12-16 21:06:33 +00:00
tmpname = SMB_STRDUP ( name ) ;
2004-01-07 00:43:52 +00:00
while ( ( dirent = smbc_readdir ( dirhandle ) ) ) {
char * newname ;
if ( ! strcmp ( dirent - > name , " . " ) | | ! strcmp ( dirent - > name , " .. " ) ) continue ;
2008-12-31 18:06:57 -08:00
if ( asprintf ( & newname , " %s/%s " , tmpname , dirent - > name ) = = - 1 ) {
2013-12-04 14:01:55 +01:00
free ( tmpname ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2008-12-31 18:06:57 -08:00
}
2004-01-07 00:43:52 +00:00
switch ( dirent - > smbc_type ) {
case SMBC_DIR :
2010-01-28 10:38:24 -08:00
ret = smb_download_dir ( base , newname , resume ) ;
2004-01-07 00:43:52 +00:00
break ;
case SMBC_WORKGROUP :
2010-01-28 10:38:24 -08:00
ret = smb_download_dir ( " smb:// " , dirent - > name , resume ) ;
2004-01-07 00:43:52 +00:00
break ;
case SMBC_SERVER :
2010-01-28 10:38:24 -08:00
ret = smb_download_dir ( " smb:// " , dirent - > name , resume ) ;
2004-01-07 00:43:52 +00:00
break ;
case SMBC_FILE :
2010-07-04 11:32:50 +02:00
ret = smb_download_file ( base , newname , 1 , resume , 0 ,
NULL ) ;
2004-01-07 00:43:52 +00:00
break ;
case SMBC_FILE_SHARE :
2010-01-28 10:38:24 -08:00
ret = smb_download_dir ( base , newname , resume ) ;
2004-01-07 00:43:52 +00:00
break ;
case SMBC_PRINTER_SHARE :
if ( ! quiet ) printf ( " Ignoring printer share %s \n " , dirent - > name ) ;
break ;
case SMBC_COMMS_SHARE :
if ( ! quiet ) printf ( " Ignoring comms share %s \n " , dirent - > name ) ;
break ;
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +00:00
case SMBC_IPC_SHARE :
if ( ! quiet ) printf ( " Ignoring ipc$ share %s \n " , dirent - > name ) ;
break ;
default :
fprintf ( stderr , " Ignoring file '%s' of type '%d' \n " , newname , dirent - > smbc_type ) ;
break ;
}
free ( newname ) ;
}
free ( tmpname ) ;
if ( keep_permissions ) {
if ( smbc_fstat ( dirhandle , & remotestat ) < 0 ) {
fprintf ( stderr , " Unable to get stats on %s on remote server \n " , path ) ;
smbc_closedir ( dirhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +00:00
if ( chmod ( relname , remotestat . st_mode ) < 0 ) {
2009-02-23 16:22:43 -08:00
fprintf ( stderr , " Unable to change mode of local dir %s to %o \n " , relname ,
( unsigned int ) remotestat . st_mode ) ;
2004-01-07 00:43:52 +00:00
smbc_closedir ( dirhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
}
smbc_closedir ( dirhandle ) ;
2010-01-28 10:38:24 -08:00
return ret ;
2004-01-07 00:43:52 +00:00
}
2006-12-19 20:16:52 +00:00
static char * print_time ( long t )
2004-01-07 00:43:52 +00:00
{
static char buffer [ 100 ] ;
int secs , mins , hours ;
if ( t < - 1 ) {
strncpy ( buffer , " Unknown " , sizeof ( buffer ) ) ;
return buffer ;
}
secs = ( int ) t % 60 ;
mins = ( int ) t / 60 % 60 ;
hours = ( int ) t / ( 60 * 60 ) ;
snprintf ( buffer , sizeof ( buffer ) - 1 , " %02d:%02d:%02d " , hours , mins , secs ) ;
return buffer ;
}
2006-12-19 20:16:52 +00:00
static void print_progress ( const char * name , time_t start , time_t now , off_t start_pos , off_t pos , off_t total )
2004-01-07 00:43:52 +00:00
{
double avg = 0.0 ;
long eta = - 1 ;
double prcnt = 0.0 ;
char hpos [ 20 ] , htotal [ 20 ] , havg [ 20 ] ;
char * status , * filename ;
int len ;
if ( now - start ) avg = 1.0 * ( pos - start_pos ) / ( now - start ) ;
2004-02-13 22:09:53 +00:00
eta = ( total - pos ) / avg ;
2004-01-07 00:43:52 +00:00
if ( total ) prcnt = 100.0 * pos / total ;
human_readable ( pos , hpos , sizeof ( hpos ) ) ;
human_readable ( total , htotal , sizeof ( htotal ) ) ;
human_readable ( avg , havg , sizeof ( havg ) ) ;
len = asprintf ( & status , " %s of %s (%.2f%%) at %s/s ETA: %s " , hpos , htotal , prcnt , havg , print_time ( eta ) ) ;
2008-12-31 18:06:57 -08:00
if ( len = = - 1 ) {
return ;
}
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +00:00
if ( columns ) {
int required = strlen ( name ) , available = columns - len - strlen ( " [] " ) ;
2008-12-31 18:06:57 -08:00
if ( required > available ) {
if ( asprintf ( & filename , " ...%s " , name + required - available + 3 ) = = - 1 ) {
return ;
}
} else {
filename = SMB_STRNDUP ( name , available ) ;
}
2004-12-16 21:06:33 +00:00
} else filename = SMB_STRDUP ( name ) ;
2004-01-07 00:43:52 +00:00
fprintf ( stderr , " \r [%s] %s " , filename , status ) ;
free ( filename ) ; free ( status ) ;
}
2010-01-28 10:38:24 -08:00
/* Return 1 on error, 0 on success. */
2010-07-04 11:32:50 +02:00
static int smb_download_file ( const char * base , const char * name , int recursive ,
int resume , int toplevel , char * outfile )
{
2004-01-07 00:43:52 +00:00
int remotehandle , localhandle ;
2010-09-08 22:29:00 +02:00
time_t start_time = time_mono ( NULL ) ;
2004-01-07 00:43:52 +00:00
const char * newpath ;
char path [ SMB_MAXPATHLEN ] ;
char checkbuf [ 2 ] [ RESUME_CHECK_SIZE ] ;
char * readbuf = NULL ;
off_t offset_download = 0 , offset_check = 0 , curpos = 0 , start_offset = 0 ;
struct stat localstat , remotestat ;
snprintf ( path , SMB_MAXPATHLEN - 1 , " %s%s%s " , base , ( * base & & * name & & name [ 0 ] ! = ' / ' & & base [ strlen ( base ) - 1 ] ! = ' / ' ) ? " / " : " " , name ) ;
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +00:00
remotehandle = smbc_open ( path , O_RDONLY , 0755 ) ;
if ( remotehandle < 0 ) {
switch ( errno ) {
case EISDIR :
if ( ! recursive ) {
fprintf ( stderr , " %s is a directory. Specify -R to download recursively \n " , path ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
2010-01-28 10:38:24 -08:00
return smb_download_dir ( base , name , resume ) ;
2004-01-07 00:43:52 +00:00
case ENOENT :
fprintf ( stderr , " %s can't be found on the remote server \n " , path ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
case ENOMEM :
fprintf ( stderr , " Not enough memory \n " ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
case ENODEV :
fprintf ( stderr , " The share name used in %s does not exist \n " , path ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
case EACCES :
fprintf ( stderr , " You don't have enough permissions to access %s \n " , path ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
default :
perror ( " smbc_open " ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
}
if ( smbc_fstat ( remotehandle , & remotestat ) < 0 ) {
fprintf ( stderr , " Can't stat %s: %s \n " , path , strerror ( errno ) ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
if ( outfile ) newpath = outfile ;
else if ( ! name [ 0 ] ) {
newpath = strrchr ( base , ' / ' ) ;
if ( newpath ) newpath + + ; else newpath = base ;
} else newpath = name ;
2010-07-04 11:32:50 +02:00
if ( ! toplevel & & ( newpath [ 0 ] = = ' / ' ) ) {
newpath + + ;
}
2010-07-04 11:02:21 +02:00
2008-03-05 15:20:29 +01:00
/* Open local file according to the mode */
if ( update ) {
/* if it is up-to-date, skip */
if ( stat ( newpath , & localstat ) = = 0 & &
localstat . st_mtime > = remotestat . st_mtime ) {
if ( verbose )
printf ( " %s is up-to-date, skipping \n " , newpath ) ;
smbc_close ( remotehandle ) ;
return 0 ;
}
2008-03-05 17:30:18 +01:00
/* else open it for writing and truncate if it exists */
localhandle = open ( newpath , O_CREAT | O_NONBLOCK | O_RDWR | O_TRUNC , 0775 ) ;
2008-03-05 15:20:29 +01:00
if ( localhandle < 0 ) {
fprintf ( stderr , " Can't open %s : %s \n " , newpath ,
strerror ( errno ) ) ;
smbc_close ( remotehandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2008-03-05 15:20:29 +01:00
}
/* no offset */
} else if ( ! send_stdout ) {
2004-03-05 17:17:31 +00:00
localhandle = open ( newpath , O_CREAT | O_NONBLOCK | O_RDWR | ( ! resume ? O_EXCL : 0 ) , 0755 ) ;
if ( localhandle < 0 ) {
fprintf ( stderr , " Can't open %s: %s \n " , newpath , strerror ( errno ) ) ;
smbc_close ( remotehandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
2010-07-04 11:02:21 +02:00
2008-03-14 14:26:28 -08:00
if ( fstat ( localhandle , & localstat ) ! = 0 ) {
fprintf ( stderr , " Can't fstat %s: %s \n " , newpath , strerror ( errno ) ) ;
smbc_close ( remotehandle ) ;
close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2008-03-14 14:26:28 -08:00
}
2004-01-07 00:43:52 +00:00
2004-03-05 17:17:31 +00:00
start_offset = localstat . st_size ;
2004-01-07 00:43:52 +00:00
2004-03-05 17:17:31 +00:00
if ( localstat . st_size & & localstat . st_size = = remotestat . st_size ) {
if ( verbose ) fprintf ( stderr , " %s is already downloaded completely. \n " , path ) ;
else if ( ! quiet ) fprintf ( stderr , " %s \n " , path ) ;
smbc_close ( remotehandle ) ;
close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 0 ;
2004-01-07 00:43:52 +00:00
}
2004-03-05 17:17:31 +00:00
if ( localstat . st_size > RESUME_CHECK_OFFSET & & remotestat . st_size > RESUME_CHECK_OFFSET ) {
offset_download = localstat . st_size - RESUME_DOWNLOAD_OFFSET ;
offset_check = localstat . st_size - RESUME_CHECK_OFFSET ;
if ( verbose ) printf ( " Trying to start resume of %s at " OFF_T_FORMAT " \n "
2005-12-03 20:18:43 +00:00
" At the moment " OFF_T_FORMAT " of " OFF_T_FORMAT " bytes have been retrieved \n " ,
newpath , ( OFF_T_FORMAT_CAST ) offset_check ,
( OFF_T_FORMAT_CAST ) localstat . st_size ,
( OFF_T_FORMAT_CAST ) remotestat . st_size ) ;
2004-01-07 00:43:52 +00:00
}
2004-03-05 17:17:31 +00:00
if ( offset_check ) {
off_t off1 , off2 ;
/* First, check all bytes from offset_check to offset_download */
off1 = lseek ( localhandle , offset_check , SEEK_SET ) ;
if ( off1 < 0 ) {
2005-12-03 20:18:43 +00:00
fprintf ( stderr , " Can't seek to " OFF_T_FORMAT " in local file %s \n " ,
( OFF_T_FORMAT_CAST ) offset_check , newpath ) ;
2004-03-05 17:17:31 +00:00
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
2004-01-07 00:43:52 +00:00
2004-03-05 17:17:31 +00:00
off2 = smbc_lseek ( remotehandle , offset_check , SEEK_SET ) ;
if ( off2 < 0 ) {
2005-12-03 20:18:43 +00:00
fprintf ( stderr , " Can't seek to " OFF_T_FORMAT " in remote file %s \n " ,
( OFF_T_FORMAT_CAST ) offset_check , newpath ) ;
2004-03-05 17:17:31 +00:00
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
2004-01-07 00:43:52 +00:00
2004-03-05 17:17:31 +00:00
if ( off1 ! = off2 ) {
2005-12-03 20:18:43 +00:00
fprintf ( stderr , " Offset in local and remote files is different (local: " OFF_T_FORMAT " , remote: " OFF_T_FORMAT " ) \n " ,
( OFF_T_FORMAT_CAST ) off1 ,
( OFF_T_FORMAT_CAST ) off2 ) ;
2011-01-19 15:18:10 +01:00
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
2004-01-07 00:43:52 +00:00
2004-03-05 17:17:31 +00:00
if ( smbc_read ( remotehandle , checkbuf [ 0 ] , RESUME_CHECK_SIZE ) ! = RESUME_CHECK_SIZE ) {
fprintf ( stderr , " Can't read %d bytes from remote file %s \n " , RESUME_CHECK_SIZE , path ) ;
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
if ( read ( localhandle , checkbuf [ 1 ] , RESUME_CHECK_SIZE ) ! = RESUME_CHECK_SIZE ) {
fprintf ( stderr , " Can't read %d bytes from local file %s \n " , RESUME_CHECK_SIZE , name ) ;
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
if ( memcmp ( checkbuf [ 0 ] , checkbuf [ 1 ] , RESUME_CHECK_SIZE ) = = 0 ) {
2005-12-03 20:18:43 +00:00
if ( verbose ) printf ( " Current local and remote file appear to be the same. Starting download from offset " OFF_T_FORMAT " \n " , ( OFF_T_FORMAT_CAST ) offset_download ) ;
2004-03-05 17:17:31 +00:00
} else {
fprintf ( stderr , " Local and remote file appear to be different, not doing resume for %s \n " , path ) ;
smbc_close ( remotehandle ) ; close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-03-05 17:17:31 +00:00
}
2004-01-07 00:43:52 +00:00
}
2004-03-05 17:17:31 +00:00
} else {
localhandle = STDOUT_FILENO ;
start_offset = 0 ;
offset_download = 0 ;
offset_check = 0 ;
2004-01-07 00:43:52 +00:00
}
2006-08-26 22:59:58 +00:00
readbuf = ( char * ) SMB_MALLOC ( blocksize ) ;
2010-09-10 11:56:26 -07:00
if ( ! readbuf ) {
2013-02-20 10:23:45 +01:00
if ( localhandle ! = STDOUT_FILENO ) {
close ( localhandle ) ;
}
2010-09-10 11:56:26 -07:00
return 1 ;
}
2004-01-07 00:43:52 +00:00
/* Now, download all bytes from offset_download to the end */
for ( curpos = offset_download ; curpos < remotestat . st_size ; curpos + = blocksize ) {
ssize_t bytesread = smbc_read ( remotehandle , readbuf , blocksize ) ;
if ( bytesread < 0 ) {
2005-12-03 20:18:43 +00:00
fprintf ( stderr , " Can't read %u bytes at offset " OFF_T_FORMAT " , file %s \n " , ( unsigned int ) blocksize , ( OFF_T_FORMAT_CAST ) curpos , path ) ;
2004-03-05 17:17:31 +00:00
smbc_close ( remotehandle ) ;
if ( localhandle ! = STDOUT_FILENO ) close ( localhandle ) ;
2004-01-07 00:43:52 +00:00
free ( readbuf ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
total_bytes + = bytesread ;
if ( write ( localhandle , readbuf , bytesread ) < 0 ) {
2005-12-03 20:18:43 +00:00
fprintf ( stderr , " Can't write %u bytes to local file %s at offset " OFF_T_FORMAT " \n " , ( unsigned int ) bytesread , path , ( OFF_T_FORMAT_CAST ) curpos ) ;
2004-01-07 00:43:52 +00:00
free ( readbuf ) ;
2004-03-05 17:17:31 +00:00
smbc_close ( remotehandle ) ;
if ( localhandle ! = STDOUT_FILENO ) close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
if ( dots ) fputc ( ' . ' , stderr ) ;
else if ( ! quiet ) {
2010-09-08 22:29:00 +02:00
print_progress ( newpath , start_time , time_mono ( NULL ) ,
2010-08-31 13:10:45 +02:00
start_offset , curpos , remotestat . st_size ) ;
2004-01-07 00:43:52 +00:00
}
}
free ( readbuf ) ;
if ( dots ) {
fputc ( ' \n ' , stderr ) ;
printf ( " %s downloaded \n " , path ) ;
} else if ( ! quiet ) {
int i ;
fprintf ( stderr , " \r %s " , path ) ;
if ( columns ) {
for ( i = strlen ( path ) ; i < columns ; i + + ) {
fputc ( ' ' , stderr ) ;
}
}
fputc ( ' \n ' , stderr ) ;
}
2004-03-05 17:17:31 +00:00
if ( keep_permissions & & ! send_stdout ) {
2004-01-07 00:43:52 +00:00
if ( fchmod ( localhandle , remotestat . st_mode ) < 0 ) {
2009-02-23 16:22:43 -08:00
fprintf ( stderr , " Unable to change mode of local file %s to %o \n " , path ,
( unsigned int ) remotestat . st_mode ) ;
2004-01-07 00:43:52 +00:00
smbc_close ( remotehandle ) ;
close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 1 ;
2004-01-07 00:43:52 +00:00
}
}
2004-03-05 17:17:31 +00:00
2004-01-07 00:43:52 +00:00
smbc_close ( remotehandle ) ;
2004-03-05 17:17:31 +00:00
if ( localhandle ! = STDOUT_FILENO ) close ( localhandle ) ;
2010-01-28 10:38:24 -08:00
return 0 ;
2004-01-07 00:43:52 +00:00
}
2006-12-19 20:16:52 +00:00
static void clean_exit ( void )
2004-01-07 00:43:52 +00:00
{
char bs [ 100 ] ;
human_readable ( total_bytes , bs , sizeof ( bs ) ) ;
2009-02-19 13:36:20 -08:00
if ( ! quiet ) fprintf ( stderr , " Downloaded %s in %lu seconds \n " , bs ,
2010-09-08 22:29:00 +02:00
( unsigned long ) ( time_mono ( NULL ) - total_start_time ) ) ;
2004-01-07 00:43:52 +00:00
exit ( 0 ) ;
}
2006-12-19 20:16:52 +00:00
static void signal_quit ( int v )
2004-01-07 00:43:52 +00:00
{
clean_exit ( ) ;
}
2006-12-19 20:16:52 +00:00
static int readrcfile ( const char * name , const struct poptOption long_options [ ] )
2004-01-07 00:43:52 +00:00
{
FILE * fd = fopen ( name , " r " ) ;
int lineno = 0 , i ;
char var [ 101 ] , val [ 101 ] ;
char found ;
int * intdata ; char * * stringdata ;
if ( ! fd ) {
fprintf ( stderr , " Can't open RC file %s \n " , name ) ;
return 1 ;
}
while ( ! feof ( fd ) ) {
lineno + + ;
if ( fscanf ( fd , " %100s %100s \n " , var , val ) < 2 ) {
fprintf ( stderr , " Can't parse line %d of %s, ignoring. \n " , lineno , name ) ;
continue ;
}
found = 0 ;
for ( i = 0 ; long_options [ i ] . shortName ; i + + ) {
if ( ! long_options [ i ] . longName ) continue ;
if ( strcmp ( long_options [ i ] . longName , var ) ) continue ;
if ( ! long_options [ i ] . arg ) continue ;
switch ( long_options [ i ] . argInfo ) {
case POPT_ARG_NONE :
intdata = ( int * ) long_options [ i ] . arg ;
if ( ! strcmp ( val , " on " ) ) * intdata = 1 ;
else if ( ! strcmp ( val , " off " ) ) * intdata = 0 ;
else fprintf ( stderr , " Illegal value %s for %s at line %d in %s \n " , val , var , lineno , name ) ;
break ;
case POPT_ARG_INT :
intdata = ( int * ) long_options [ i ] . arg ;
* intdata = atoi ( val ) ;
break ;
case POPT_ARG_STRING :
stringdata = ( char * * ) long_options [ i ] . arg ;
2004-12-16 21:06:33 +00:00
* stringdata = SMB_STRDUP ( val ) ;
2004-01-07 00:43:52 +00:00
break ;
default :
fprintf ( stderr , " Invalid variable %s at line %d in %s \n " , var , lineno , name ) ;
break ;
}
found = 1 ;
}
if ( ! found ) {
fprintf ( stderr , " Invalid variable %s at line %d in %s \n " , var , lineno , name ) ;
}
}
fclose ( fd ) ;
return 0 ;
}
int main ( int argc , const char * * argv )
{
int c = 0 ;
const char * file = NULL ;
char * rcfile = NULL ;
2008-01-05 00:51:50 -08:00
bool smb_encrypt = false ;
2008-03-11 19:49:08 +01:00
int resume = 0 , recursive = 0 ;
2007-09-04 05:39:06 +00:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2010-01-28 10:38:24 -08:00
int ret = 0 ;
2004-01-07 00:43:52 +00:00
struct poptOption long_options [ ] = {
{ " guest " , ' a ' , POPT_ARG_NONE , NULL , ' a ' , " Work as user guest " } ,
2008-01-05 00:51:50 -08:00
{ " encrypt " , ' e ' , POPT_ARG_NONE , NULL , ' e ' , " Encrypt SMB transport (UNIX extended servers only) " } ,
2008-03-11 19:49:08 +01:00
{ " resume " , ' r ' , POPT_ARG_NONE , & resume , 0 , " Automatically resume aborted files " } ,
2008-03-05 15:20:29 +01:00
{ " update " , ' U ' , POPT_ARG_NONE , & update , 0 , " Download only when remote file is newer than local file or local file is missing " } ,
2008-03-11 19:49:08 +01:00
{ " recursive " , ' R ' , POPT_ARG_NONE , & recursive , 0 , " Recursively download files " } ,
2004-01-07 00:43:52 +00:00
{ " username " , ' u ' , POPT_ARG_STRING , & username , ' u ' , " Username to use " } ,
{ " password " , ' p ' , POPT_ARG_STRING , & password , ' p ' , " Password to use " } ,
{ " workgroup " , ' w ' , POPT_ARG_STRING , & workgroup , ' w ' , " Workgroup to use (optional) " } ,
{ " nonprompt " , ' n ' , POPT_ARG_NONE , & nonprompt , ' n ' , " Don't ask anything (non-interactive) " } ,
{ " debuglevel " , ' d ' , POPT_ARG_INT , & debuglevel , ' d ' , " Debuglevel to use " } ,
{ " outputfile " , ' o ' , POPT_ARG_STRING , & outputfile , ' o ' , " Write downloaded data to specified file " } ,
2004-03-05 17:17:31 +00:00
{ " stdout " , ' O ' , POPT_ARG_NONE , & send_stdout , ' O ' , " Write data to stdout " } ,
2004-01-07 00:43:52 +00:00
{ " dots " , ' D ' , POPT_ARG_NONE , & dots , ' D ' , " Show dots as progress indication " } ,
{ " quiet " , ' q ' , POPT_ARG_NONE , & quiet , ' q ' , " Be quiet " } ,
{ " verbose " , ' v ' , POPT_ARG_NONE , & verbose , ' v ' , " Be verbose " } ,
{ " keep-permissions " , ' P ' , POPT_ARG_NONE , & keep_permissions , ' P ' , " Keep permissions " } ,
{ " blocksize " , ' b ' , POPT_ARG_INT , & blocksize , ' b ' , " Change number of bytes in a block " } ,
2009-05-15 21:02:08 +02:00
{ " rcfile " , ' f ' , POPT_ARG_STRING , NULL , ' f ' , " Use specified rc file " } ,
2004-01-07 00:43:52 +00:00
POPT_AUTOHELP
POPT_TABLEEND
} ;
poptContext pc ;
2005-12-28 22:48:54 +00:00
load_case_tables ( ) ;
2004-01-07 00:43:52 +00:00
/* only read rcfile if it exists */
2008-12-31 18:06:57 -08:00
if ( asprintf ( & rcfile , " %s/.smbgetrc " , getenv ( " HOME " ) ) = = - 1 ) {
return 1 ;
}
2005-11-21 15:52:10 +00:00
if ( access ( rcfile , F_OK ) = = 0 )
readrcfile ( rcfile , long_options ) ;
2004-01-07 00:43:52 +00:00
free ( rcfile ) ;
# ifdef SIGWINCH
signal ( SIGWINCH , change_columns ) ;
# endif
signal ( SIGINT , signal_quit ) ;
signal ( SIGTERM , signal_quit ) ;
pc = poptGetContext ( argv [ 0 ] , argc , argv , long_options , 0 ) ;
while ( ( c = poptGetNextOpt ( pc ) ) > = 0 ) {
switch ( c ) {
case ' f ' :
readrcfile ( poptGetOptArg ( pc ) , long_options ) ;
break ;
case ' a ' :
username = " " ; password = " " ;
break ;
2008-01-05 00:51:50 -08:00
case ' e ' :
smb_encrypt = true ;
break ;
2004-01-07 00:43:52 +00:00
}
}
2008-03-11 19:49:08 +01:00
if ( ( send_stdout | | resume | | outputfile ) & & update ) {
2008-03-05 15:20:29 +01:00
fprintf ( stderr , " The -o, -R or -O and -U options can not be used together. \n " ) ;
return 1 ;
}
2008-03-11 19:49:08 +01:00
if ( ( send_stdout | | outputfile ) & & recursive ) {
2004-03-05 17:17:31 +00:00
fprintf ( stderr , " The -o or -O and -R options can not be used together. \n " ) ;
return 1 ;
}
if ( outputfile & & send_stdout ) {
fprintf ( stderr , " The -o and -O options cannot be used together. \n " ) ;
2004-01-07 00:43:52 +00:00
return 1 ;
}
if ( smbc_init ( get_auth_data , debuglevel ) < 0 ) {
fprintf ( stderr , " Unable to initialize libsmbclient \n " ) ;
return 1 ;
}
2008-01-05 00:51:50 -08:00
if ( smb_encrypt ) {
SMBCCTX * smb_ctx = smbc_set_context ( NULL ) ;
smbc_option_set ( smb_ctx ,
2011-05-05 16:19:49 -07:00
discard_const_p ( char , " smb_encrypt_level " ) ,
2008-01-05 00:51:50 -08:00
" require " ) ;
}
2010-07-04 11:02:21 +02:00
2004-01-07 00:43:52 +00:00
columns = get_num_cols ( ) ;
2010-09-08 22:29:00 +02:00
total_start_time = time_mono ( NULL ) ;
2004-01-07 00:43:52 +00:00
2005-11-21 15:52:10 +00:00
while ( ( file = poptGetArg ( pc ) ) ) {
2008-03-11 19:49:08 +01:00
if ( ! recursive )
2010-07-04 11:32:50 +02:00
ret = smb_download_file ( file , " " , recursive , resume ,
1 , outputfile ) ;
2005-11-21 15:52:10 +00:00
else
2010-01-28 10:38:24 -08:00
ret = smb_download_dir ( file , " " , resume ) ;
2004-01-07 00:43:52 +00:00
}
2007-09-04 05:39:06 +00:00
TALLOC_FREE ( frame ) ;
2010-01-28 10:38:24 -08:00
if ( ret = = 0 ) {
clean_exit ( ) ;
}
return ret ;
2004-01-07 00:43:52 +00:00
}