summaryrefslogtreecommitdiffstats
path: root/ark/sevenzip.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'ark/sevenzip.cpp')
-rw-r--r--ark/sevenzip.cpp367
1 files changed, 367 insertions, 0 deletions
diff --git a/ark/sevenzip.cpp b/ark/sevenzip.cpp
new file mode 100644
index 0000000..30747a3
--- /dev/null
+++ b/ark/sevenzip.cpp
@@ -0,0 +1,367 @@
+/*
+
+ ark -- archiver for the KDE project
+
+ Copyright (C)
+
+ 2004: Henrique Pinto <henrique.pinto@kdemail.net>
+ 2003: Helio Chissini de Castro <helio@conectiva.com>
+ 2000: Corel Corporation (author: Emily Ezust, emilye@corel.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 <qdir.h>
+
+#include <kglobal.h>
+#include <klocale.h>
+#include <kdebug.h>
+#include <kurl.h>
+#include <kmessagebox.h>
+#include <kprocess.h>
+#include <kstandarddirs.h>
+
+#include "sevenzip.h"
+#include "arkwidget.h"
+#include "settings.h"
+#include "arkutils.h"
+#include "filelistview.h"
+
+SevenZipArch::SevenZipArch( ArkWidget *gui, const QString &filename )
+ : Arch( gui, filename ), m_nameColumnPos( -1 )
+{
+ // Check if 7z is available
+ bool have_7z = !KGlobal::dirs()->findExe( "7z" ).isNull();
+ // Check if 7za is available
+ bool have_7za = !KGlobal::dirs()->findExe( "7za" ).isNull();
+
+ if ( have_7z )
+ m_archiver_program = m_unarchiver_program = "7z"; // Use 7z
+ else if ( have_7za )
+ m_archiver_program = m_unarchiver_program = "7za"; // Try 7za
+ else
+ m_archiver_program = m_unarchiver_program = "7zr";
+
+ verifyCompressUtilityIsAvailable( m_archiver_program );
+ verifyUncompressUtilityIsAvailable( m_unarchiver_program );
+
+ m_headerString = "------------------";
+
+ m_repairYear = 5; m_fixMonth = 6; m_fixDay = 7; m_fixTime = 8;
+ m_dateCol = 3;
+ m_numCols = 5;
+
+ m_archCols.append( new ArchColumns( 5, QRegExp( "[0-2][0-9][0-9][0-9]" ), 4 ) ); // Year
+ m_archCols.append( new ArchColumns( 6, QRegExp( "[01][0-9]" ), 2 ) ); // Month
+ m_archCols.append( new ArchColumns( 7, QRegExp( "[0-3][0-9]" ), 2 ) ); // Day
+ m_archCols.append( new ArchColumns( 8, QRegExp( "[0-9:]+" ), 8 ) ); // Time
+ m_archCols.append( new ArchColumns( 4, QRegExp( "[^\\s]+" ) ) ); // Attributes
+ m_archCols.append( new ArchColumns( 1, QRegExp( "[0-9]+" ) ) ); // Size
+ m_archCols.append( new ArchColumns( 2, QRegExp( "[0-9]+" ), 64, true ) ); // Compressed Size
+}
+
+SevenZipArch::~SevenZipArch()
+{
+}
+
+void SevenZipArch::setHeaders()
+{
+ ColumnList columns;
+ columns.append( FILENAME_COLUMN );
+ columns.append( SIZE_COLUMN );
+ columns.append( PACKED_COLUMN );
+ columns.append( TIMESTAMP_COLUMN );
+ columns.append( PERMISSION_COLUMN );
+
+ emit headers( columns );
+}
+
+void SevenZipArch::open()
+{
+ setHeaders();
+
+ m_buffer = "";
+ m_header_removed = false;
+ m_finished = false;
+
+ KProcess *kp = m_currentProcess = new KProcess;
+ *kp << m_archiver_program << "l" << m_filename;
+
+ connect( kp, SIGNAL( receivedStdout(KProcess*, char*, int) ),
+ SLOT( slotReceivedTOC(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( receivedStderr(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( processExited(KProcess*) ),
+ SLOT( slotOpenExited(KProcess*) ) );
+
+ if ( !kp->start( KProcess::NotifyOnExit, KProcess::AllOutput ) )
+ {
+ KMessageBox::error( 0, i18n( "Could not start a subprocess." ) );
+ emit sigOpen( this, false, QString::null, 0 );
+ }
+}
+
+void SevenZipArch::create()
+{
+ emit sigCreate( this, true, m_filename,
+ Arch::Extract | Arch::Delete | Arch::Add | Arch::View );
+}
+
+void SevenZipArch::addFile( const QStringList & urls )
+{
+ KProcess *kp = m_currentProcess = new KProcess;
+
+ kp->clearArguments();
+ *kp << m_archiver_program << "a" ;
+
+ KURL url( urls.first() );
+ QDir::setCurrent( url.directory() );
+
+ *kp << m_filename;
+
+ QStringList::ConstIterator iter;
+ for ( iter = urls.begin(); iter != urls.end(); ++iter )
+ {
+ KURL url( *iter );
+ *kp << url.fileName();
+ }
+
+ connect( kp, SIGNAL( receivedStdout(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( receivedStderr(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( processExited(KProcess*) ),
+ SLOT( slotAddExited(KProcess*) ) );
+
+ if ( !kp->start( KProcess::NotifyOnExit, KProcess::AllOutput ) )
+ {
+ KMessageBox::error( 0, i18n( "Could not start a subprocess." ) );
+ emit sigAdd( false );
+ }
+}
+
+void SevenZipArch::addDir( const QString & dirName )
+{
+ if ( !dirName.isEmpty() )
+ {
+ QStringList list;
+ list.append( dirName );
+ addFile( list );
+ }
+}
+
+void SevenZipArch::remove( QStringList *list )
+{
+ if ( !list )
+ return;
+
+ KProcess *kp = m_currentProcess = new KProcess;
+ kp->clearArguments();
+
+ *kp << m_archiver_program << "d" << m_filename;
+
+ QStringList::Iterator it;
+ for ( it = list->begin(); it != list->end(); ++it )
+ {
+ *kp << *it;
+ }
+
+ connect( kp, SIGNAL( receivedStdout(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( receivedStderr(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( processExited(KProcess*) ),
+ SLOT( slotDeleteExited(KProcess*) ) );
+
+ if ( !kp->start( KProcess::NotifyOnExit, KProcess::AllOutput ) )
+ {
+ KMessageBox::error( 0, i18n( "Could not start a subprocess." ) );
+ emit sigDelete( false );
+ }
+}
+
+void SevenZipArch::unarchFileInternal( )
+{
+ if ( m_destDir.isEmpty() || m_destDir.isNull() )
+ {
+ kdError( 1601 ) << "There was no extract directory given." << endl;
+ return;
+ }
+
+ KProcess *kp = m_currentProcess = new KProcess;
+ kp->clearArguments();
+
+ // extract (and maybe overwrite)
+ *kp << m_unarchiver_program << "x";
+
+ if ( ArkSettings::extractOverwrite() )
+ {
+ //*kp << "-ao";
+ }
+
+ *kp << m_filename;
+
+ // if the file list is empty, no filenames go on the command line,
+ // and we then extract everything in the archive.
+ if ( m_fileList )
+ {
+ QStringList::Iterator it;
+ for ( it = m_fileList->begin(); it != m_fileList->end(); ++it )
+ {
+ *kp << (*it);
+ }
+ }
+
+ *kp << "-o" + m_destDir ;
+
+ connect( kp, SIGNAL( receivedStdout(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( receivedStderr(KProcess*, char*, int) ),
+ SLOT( slotReceivedOutput(KProcess*, char*, int) ) );
+ connect( kp, SIGNAL( processExited(KProcess*) ),
+ SLOT( slotExtractExited(KProcess*) ) );
+
+ if ( !kp->start( KProcess::NotifyOnExit, KProcess::AllOutput ) )
+ {
+ KMessageBox::error( 0, i18n( "Could not start a subprocess." ) );
+ emit sigExtract( false );
+ }
+}
+
+bool SevenZipArch::processLine( const QCString& _line )
+{
+ QCString line( _line );
+ QString columns[ 11 ];
+ unsigned int pos = 0;
+ int strpos, len;
+
+ columns[ 0 ] = line.right( line.length() - m_nameColumnPos +1);
+ line.truncate( m_nameColumnPos );
+
+ // Go through our columns, try to pick out data, return silently on failure
+ for ( QPtrListIterator <ArchColumns>col( m_archCols ); col.current(); ++col )
+ {
+ ArchColumns *curCol = *col;
+
+ strpos = curCol->pattern.search( line, pos );
+ len = curCol->pattern.matchedLength();
+
+ if ( ( strpos == -1 ) || ( len > curCol->maxLength ) )
+ {
+ if ( curCol->optional )
+ continue; // More?
+ else
+ {
+ kdDebug(1601) << "processLine failed to match critical column" << endl;
+ return false;
+ }
+ }
+
+ pos = strpos + len;
+
+ columns[ curCol->colRef ] = line.mid( strpos, len );
+ }
+
+
+ if ( m_dateCol >= 0 )
+ {
+ QString year = ( m_repairYear >= 0 ) ?
+ ArkUtils::fixYear( columns[ m_repairYear ].ascii())
+ : columns[ m_fixYear ];
+ QString month = ( m_repairMonth >= 0 ) ?
+ QString( "%1" )
+ .arg( ArkUtils::getMonth( columns[ m_repairMonth ].ascii() ) )
+ : columns[ m_fixMonth ];
+ QString timestamp = QString::fromLatin1( "%1-%2-%3 %4" )
+ .arg( year )
+ .arg( month )
+ .arg( columns[ m_fixDay ] )
+ .arg( columns[ m_fixTime ] );
+
+ columns[ m_dateCol ] = timestamp;
+ }
+
+ QStringList list;
+
+ for ( int i = 0; i < m_numCols; ++i )
+ {
+ list.append( columns[ i ] );
+ }
+
+ m_gui->fileList()->addItem( list ); // send the entry to the GUI
+
+ return true;
+}
+
+void SevenZipArch::slotReceivedTOC( KProcess*, char* data, int length )
+{
+ char endchar = data[ length ];
+ data[ length ] = '\0';
+
+ appendShellOutputData( data );
+
+ int startChar = 0;
+
+ while ( !m_finished )
+ {
+ int lfChar;
+ for ( lfChar = startChar; data[ lfChar ] != '\n' && lfChar < length;
+ lfChar++ );
+
+ if ( data[ lfChar ] != '\n')
+ break; // We are done all the complete lines
+
+ data[ lfChar ] = '\0';
+ m_buffer.append( data + startChar );
+ data[ lfChar ] = '\n';
+ startChar = lfChar + 1;
+
+ // Check if the header was found
+ if ( m_buffer.find( m_headerString ) != -1 )
+ {
+ if ( !m_header_removed )
+ {
+ m_nameColumnPos = m_buffer.findRev( ' ' ) + 1;
+ m_header_removed = true;
+ }
+ else
+ {
+ m_finished = true;
+ }
+ }
+ else
+ {
+ // If the header was not found, process the line
+ if ( m_header_removed && !m_finished )
+ {
+ if ( !processLine( m_buffer ) )
+ {
+ m_header_removed = false;
+ m_error = true;
+ }
+ }
+ }
+
+ m_buffer.resize( 0 );
+ }
+
+ if ( !m_finished )
+ m_buffer.append( data + startChar); // Append what's left of the buffer
+
+ data[ length ] = endchar;
+}
+
+#include "sevenzip.moc"