diff options
author | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
---|---|---|
committer | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
commit | 37333bf25ad9a4c538250f5af2f9f1d666362883 (patch) | |
tree | c45e8df5b9efbffe07eb3d9340df7811c7e16943 /kdat/BackupDlg.cpp | |
download | tdeadmin-37333bf25ad9a4c538250f5af2f9f1d666362883.tar.gz tdeadmin-37333bf25ad9a4c538250f5af2f9f1d666362883.zip |
Copy the KDE 3.5 branch to branches/trinity for new KDE 3.5 features.
BUG:215923
git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdeadmin@1054174 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'kdat/BackupDlg.cpp')
-rw-r--r-- | kdat/BackupDlg.cpp | 432 |
1 files changed, 432 insertions, 0 deletions
diff --git a/kdat/BackupDlg.cpp b/kdat/BackupDlg.cpp new file mode 100644 index 0000000..1f237ab --- /dev/null +++ b/kdat/BackupDlg.cpp @@ -0,0 +1,432 @@ +// $Id$ +// +// KDat - a tar-based DAT archiver +// Copyright (C) 1998-2000 Sean Vyain, svyain@mail.tds.net +// Copyright (C) 2001-2002 Lawrence Widman, kdat@cardiothink.com +// +// 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 +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// 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. +// +// You should have received a copy of the GNU General Public License +// along with this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + +#include <assert.h> +#include <stdio.h> +#include <time.h> +#include <unistd.h> +#include <sys/errno.h> +#include <sys/stat.h> + +#include <qdir.h> +#include <qlabel.h> +#include <qlayout.h> +#include <kpushbutton.h> +#include <kstdguiitem.h> + +#include <kapplication.h> +#include <kprocess.h> +#include <kmessagebox.h> +#include <kpushbutton.h> +#include <kstdguiitem.h> + +#include "Archive.h" +#include "BackupDlg.h" +#include "LoggerWidget.h" +#include "Options.h" +#include "Tape.h" +#include "TapeDrive.h" +#include "TarParser.h" +#include "Util.h" +#include <klocale.h> + +#include "BackupDlg.moc" + +BackupDlg::BackupDlg( const QString & archiveName, const QString & workingDir, const QStringList& files, bool oneFilesystem, bool incremental, + const QString & snapshot, bool removeSnapshot, int archiveSize, Tape* tape, + QWidget* parent, const char* name ) + : QDialog( parent, name, TRUE ), + _proc( NULL ), + _tarParser( NULL ), + _archiveName( archiveName ), + _workingDir( workingDir ), + _files(files), + _oneFilesystem( oneFilesystem ), + _incremental( incremental ), + _snapshot( snapshot ), + _removeSnapshot( removeSnapshot ), + _archiveSize( archiveSize ), + _tape( tape ), + _totalKBytes( 0.0 ), + _totalRecords( 0 ), + _startTime( 0 ), + _archive( NULL ), + _aborted( FALSE ), + _numFiles( 0 ), + _fileSize( -1 ), + _fileMTime( -1 ), + _fileStartRecord( -1 ) +{ + // Copy the list of files to archive. + + setCaption( i18n( "KDat: Backup" ) ); + setIconText( i18n( "KDat: Backup" ) ); + + resize( 515, 300 ); + + /* 2002-01-26 LEW: "Time remaining" was cut off in mid-"g" + so we'll provide that plus some space beyond it. */ + // const int labelWidth = 96; + const int labelWidth = 110; + + QFrame* f1 = new QFrame( this ); + f1->setFrameStyle( QFrame::Panel | QFrame::Sunken ); + + QFrame* f2 = new QFrame( this ); + f2->setFrameStyle( QFrame::Panel | QFrame::Sunken ); + + QLabel* lbl1 = new QLabel( i18n( "Elapsed time:" ), f1 ); + lbl1->setFixedSize( labelWidth, lbl1->sizeHint().height() ); + + _elapsedTime = new QLabel( i18n( "00:00:00" ), f1 ); + _elapsedTime->setFixedHeight( _elapsedTime->sizeHint().height() ); + + QLabel* lbl2 = new QLabel( i18n( "Time remaining:" ), f2 ); + lbl2->setFixedSize( labelWidth, lbl2->sizeHint().height() ); + + _timeRemaining = new QLabel( i18n( "00:00:00" ), f2 ); + _timeRemaining->setFixedHeight( _timeRemaining->sizeHint().height() ); + + QLabel* lbl3 = new QLabel( i18n( "Total KB:" ), f1 ); + lbl3->setFixedSize( labelWidth, lbl3->sizeHint().height() ); + + QLabel* totalKbytes = new QLabel( Util::kbytesToString( archiveSize ), f1 ); + totalKbytes->setFixedHeight( totalKbytes->sizeHint().height() ); + + QLabel* lbl4 = new QLabel( i18n( "KB written:" ), f2 ); + lbl4->setFixedSize( labelWidth, lbl4->sizeHint().height() ); + + _kbytesWritten = new QLabel( i18n( "0KB" ), f2 ); + _kbytesWritten->setFixedHeight( _kbytesWritten->sizeHint().height() ); + + QLabel* lbl5 = new QLabel( i18n( "Transfer rate:" ), f1 ); + lbl5->setFixedSize( labelWidth, lbl5->sizeHint().height() ); + + _transferRate = new QLabel( i18n( "0KB/min" ), f1 ); + _transferRate->setFixedHeight( _transferRate->sizeHint().height() ); + + QLabel* lbl6 = new QLabel( i18n( "Files:" ), f2 ); + lbl6->setFixedSize( labelWidth, lbl6->sizeHint().height() ); + + _fileCount = new QLabel( i18n( "0" ), f2 ); + _fileCount->setFixedHeight( _fileCount->sizeHint().height() ); + + _log = new LoggerWidget( i18n( "Backup log:" ), this ); + + _ok = new KPushButton( KStdGuiItem::ok(), this ); + _ok->setFixedSize( 80, _ok->sizeHint().height() ); + connect( _ok, SIGNAL( clicked() ), this, SLOT( slotOK() ) ); + _ok->setEnabled( FALSE ); + + _save = new QPushButton( i18n( "Save Log..." ), this ); + _save->setFixedSize( 80, _save->sizeHint().height() ); + connect( _save, SIGNAL( clicked() ), _log, SLOT( save() ) ); + _save->setEnabled( FALSE ); + + _abort = new KPushButton( KStdGuiItem::cancel(), this ); + _abort->setFixedSize( 80, _abort->sizeHint().height() ); + connect( _abort, SIGNAL( clicked() ), this, SLOT( slotAbort() ) ); + + QVBoxLayout* l1 = new QVBoxLayout( this, 8, 4 ); + + QHBoxLayout* l1_1 = new QHBoxLayout(); + l1->addLayout( l1_1 ); + l1_1->addStrut( 3 * lbl1->height() + 16 ); + l1_1->addWidget( f1 ); + l1_1->addWidget( f2 ); + + QVBoxLayout* l1_1_1 = new QVBoxLayout( f1, 4, 4 ); + + QHBoxLayout* l1_1_1_1 = new QHBoxLayout(); + l1_1_1->addLayout( l1_1_1_1 ); + l1_1_1_1->addWidget( lbl1 ); + l1_1_1_1->addWidget( _elapsedTime, 1 ); + + QHBoxLayout* l1_1_1_2 = new QHBoxLayout(); + l1_1_1->addLayout( l1_1_1_2 ); + l1_1_1_2->addWidget( lbl3 ); + l1_1_1_2->addWidget( totalKbytes, 1 ); + + QHBoxLayout* l1_1_1_3 = new QHBoxLayout(); + l1_1_1->addLayout( l1_1_1_3 ); + l1_1_1_3->addWidget( lbl5 ); + l1_1_1_3->addWidget( _transferRate, 1 ); + + QVBoxLayout* l1_1_2 = new QVBoxLayout( f2, 4, 4 ); + + QHBoxLayout* l1_1_2_1 = new QHBoxLayout(); + l1_1_2->addLayout( l1_1_2_1 ); + l1_1_2_1->addWidget( lbl2 ); + l1_1_2_1->addWidget( _timeRemaining, 1 ); + + QHBoxLayout* l1_1_2_2 = new QHBoxLayout(); + l1_1_2->addLayout( l1_1_2_2 ); + l1_1_2_2->addWidget( lbl4 ); + l1_1_2_2->addWidget( _kbytesWritten, 1 ); + + QHBoxLayout* l1_1_2_3 = new QHBoxLayout(); + l1_1_2->addLayout( l1_1_2_3 ); + l1_1_2_3->addWidget( lbl6 ); + l1_1_2_3->addWidget( _fileCount, 1 ); + + l1->addWidget( _log, 1 ); + + QHBoxLayout* l1_2 = new QHBoxLayout(); + l1->addLayout( l1_2 ); + l1_2->addStretch( 1 ); + l1_2->addWidget( _ok ); + l1_2->addWidget( _save ); + l1_2->addWidget( _abort ); +} + +BackupDlg::~BackupDlg() +{ + delete _tarParser; +} + +void BackupDlg::show() +{ + bool bGoOn = false; + + _archive = new Archive( _tape, time( NULL ), _archiveName.utf8() ); + + chdir( QFile::encodeName(_workingDir) ); + + if ( _removeSnapshot ) { + unlink( QFile::encodeName(_snapshot) ); + } + + _tarParser = new TarParser(); + connect( _tarParser, SIGNAL( sigEntry( const QString &, int, int, int ) ), this, SLOT( slotEntry( const QString &, int, int, int ) ) ); + + _proc = new KProcess(); + *_proc << Options::instance()->getTarCommand(); + if ( _oneFilesystem ) { + *_proc << "-l"; + } + if ( _incremental ) { + *_proc << "-g" << _snapshot; + } + *_proc << "-Spcf" << "-"; + + // Append the list of files to archive. + if ( _files.count() == 1 && _files.first() == "." ) { + dev_t device = 0; + struct stat info; + if ( _oneFilesystem ) { + if ( lstat( ".", &info ) == 0 ) { + device = info.st_dev; + } + } + + // Backup all files in current working directory. + QDir dir; + //roland + //QStringList::Iterator i = dir.entryList( QDir::All, QDir::Name | QDir::DirsFirst ).begin(); + QStringList FilesList = dir.entryList( QDir::All, QDir::Name | QDir::DirsFirst ); + QStringList::Iterator i = FilesList.begin(); + //roland + for ( ; !(*i).isNull() ; ++i ) { + if ( *i != "." && *i != ".." ) { + if ( _oneFilesystem ) { + if ( lstat( QFile::encodeName(*i), &info ) == 0 ) + { + if ( info.st_dev == device ) { + *_proc << *i; + bGoOn = true; + } + } + } else { + *_proc << *i; + bGoOn = true; + } + } + } + } else { + // Backup listed files only. + /* 2002-01-28 LEW */ + // printf("Fixing to list the files/dirs to be dumped:\n"); + /* 2002-01-28 LEW */ + for ( QStringList::Iterator it = _files.begin(); + it != _files.end(); + ++it ) { + /* 2002-01-28 LEW */ + // printf("tar argument: \"%s\"\n", (*it).latin1()); + /* 2002-01-28 LEW */ + *_proc << *it; + bGoOn = true; + } + } + + if (bGoOn == false) { + KMessageBox::information(this, i18n("No files to back up. Aborting.")); + slotAbort(); + return; + } + + connect( _proc, SIGNAL( processExited( KProcess* ) ), this, SLOT( slotProcessExited( KProcess* ) ) ); + connect( _proc, SIGNAL( receivedStdout( KProcess*, char*, int ) ), this, SLOT( slotStdout( KProcess*, char*, int ) ) ); + + startTimer( 1000 ); + + _proc->start( KProcess::NotifyOnExit, KProcess::Stdout ); + + QDialog::show(); +} + +void BackupDlg::slotProcessExited( KProcess* ) +{ + updateStats(); + + _archive->setEndBlock( _totalRecords / ( Options::instance()->getTapeBlockSize() / 512 ) ); + + if ( _fileName.length() > 0 ) { + _archive->addFile( _fileSize, _fileMTime, _fileStartRecord, _totalRecords, _fileName ); + _fileName = QString::null; + } + + TapeDrive::instance()->close(); + TapeDrive::instance()->open(); + + killTimers(); + delete _proc; + + _tape->addChild( _archive ); + + _ok->setEnabled( TRUE ); + _ok->setDefault( TRUE ); + _save->setEnabled( TRUE ); + _abort->setEnabled( FALSE ); +} + +// the KProcess passes the arguments to tar, and tar's output is piped here. +// The output is shown to _tarParser->slotData(), which figures out which files +// are going to tape and their file parameters, and saves these data into +// kdat's archive. The raw data are then written to tape with write(). +// 2002-01-27 LEW +void BackupDlg::slotStdout( KProcess*, char* buf, int len ) +{ + // Don't start throughput timer until the first block of data is written. + if ( _startTime == 0 ) { + _startTime = time( NULL ); + } + + /* 2002-01-26 LEW */ + // printf("got a line from tar, length %d bytes...\n", len); + /* 2002-01-26 LEW */ + + // Pass the data through the tar parser to extract the file info. + _tarParser->slotData( buf, len ); + + _totalKBytes += (float)len / 1024.0; + assert( len % 512 == 0 ); + _totalRecords += len / 512; + if ( TapeDrive::instance()->write( buf, len ) < len ) { + _log->append( i18n( "*** Write failed, giving up." ) ); + + if ( _proc ) { + _proc->kill(); + } + + slotProcessExited( 0 ); + } +} + +void BackupDlg::slotEntry( const QString& name, int size, int mtime, int record ) +{ + if ( _fileName.length() > 0 ) { + _archive->addFile( _fileSize, _fileMTime, _fileStartRecord, record, _fileName ); + } + + /* 2002-01-28 LEW */ + // printf("BackupDlg::slotEntry called with \"%s\", %d bytes\n", name.latin1(), size); + /* 2002-01-28 LEW */ + + _fileName = name; + _fileSize = size; + _fileMTime = mtime; + _fileStartRecord = record; + + QString tmp; + tmp.setNum( ++_numFiles ); + _fileCount->setText( tmp ); + _log->append( name ); +} + +void BackupDlg::slotOK() +{ + if ( _aborted ) { + reject(); + } else { + accept(); + } +} + +void BackupDlg::slotAbort() +{ + killTimers(); + if ( _proc ) { + _proc->kill(); + delete _proc; + } + delete _archive; + _aborted = TRUE; + + _ok->setEnabled( TRUE ); + _ok->setDefault( TRUE ); + _save->setEnabled( TRUE ); + _abort->setEnabled( FALSE ); +} + +void BackupDlg::timerEvent( QTimerEvent* ) +{ + updateStats(); +} + +void BackupDlg::updateStats() +{ + if ( _startTime == 0 ) { + return; + } + + QString str; + + int elapsed = time( NULL ) - _startTime; + str = QString::fromUtf8( QCString().sprintf( i18n( "%02d:%02d:%02d" ).utf8().data(), elapsed / 3600, elapsed / 60 % 60, elapsed % 60 ) ); + _elapsedTime->setText( str ); + + int remain = 0; + if ( (int)_totalKBytes > 0 ) { + remain = (int)(( (float)_archiveSize - _totalKBytes ) * (float)elapsed / _totalKBytes); + } + if ( remain < 0 ) { + remain = 0; + } + str = QString::fromUtf8( QCString().sprintf( i18n( "%02d:%02d:%02d" ).utf8().data(), remain / 3600, remain / 60 % 60, remain % 60 ) ); + _timeRemaining->setText( str ); + + str = Util::kbytesToString( (int)_totalKBytes ); + _kbytesWritten->setText( str ); + + if ( elapsed > 0 ) { + str = i18n( "%1/min" ).arg(Util::kbytesToString( (int)_totalKBytes *60 / elapsed ) ); + _transferRate->setText( str ); + } +} |