313
313
class KZip::KZipPrivate
321
m_extraField( KZip::NoExtraField ),
321
m_extraField( KZip::NoExtraField ),
325
unsigned long m_crc; // checksum
326
KZipFileEntry* m_currentFile; // file currently being written
327
QIODevice* m_currentDev; // filterdev used to write to the above file
328
QList<KZipFileEntry*> m_fileList; // flat list of all files, for the index (saves a recursive method ;)
330
KZip::ExtraField m_extraField;
331
// m_offset holds the offset of the place in the zip,
332
// where new data can be appended. after openarchive it points to 0, when in
333
// writeonly mode, or it points to the beginning of the central directory.
334
// each call to writefile updates this value.
335
unsigned int m_offset;
325
unsigned long m_crc; // checksum
326
KZipFileEntry* m_currentFile; // file currently being written
327
QIODevice* m_currentDev; // filterdev used to write to the above file
328
QList<KZipFileEntry*> m_fileList; // flat list of all files, for the index (saves a recursive method ;)
330
KZip::ExtraField m_extraField;
331
// m_offset holds the offset of the place in the zip,
332
// where new data can be appended. after openarchive it points to 0, when in
333
// writeonly mode, or it points to the beginning of the central directory.
334
// each call to writefile updates this value.
335
unsigned int m_offset;
338
338
KZip::KZip( const QString& fileName )
339
: KArchive( fileName ),d(new KZipPrivate)
339
: KArchive( fileName ),d(new KZipPrivate)
343
343
KZip::KZip( QIODevice * dev )
344
: KArchive( dev ),d(new KZipPrivate)
344
: KArchive( dev ),d(new KZipPrivate)
356
356
bool KZip::openArchive( QIODevice::OpenMode mode )
359
d->m_fileList.clear();
361
if ( mode == QIODevice::WriteOnly )
366
// Check that it's a valid ZIP file
367
// KArchive::open() opened the underlying device already.
369
uint offset = 0; // holds offset, where we read
372
// contains information gathered from the local file headers
373
QHash<QByteArray, ParseFileInfo> pfi_map;
375
QIODevice* dev = device();
377
// We set a bool for knowing if we are allowed to skip the start of the file
378
bool startOfFile = true;
380
for (;;) // repeat until 'end of entries' signature is reached
382
//qDebug() << "loop starts";
383
//qDebug() << "dev->pos() now : " << dev->pos();
384
n = dev->read( buffer, 4 );
388
qWarning() << "Invalid ZIP file. Unexpected end of file. (#1)";
393
if ( !memcmp( buffer, "PK\5\6", 4 ) ) // 'end of entries'
395
//qDebug() << "PK56 found end of archive";
359
d->m_fileList.clear();
361
if ( mode == QIODevice::WriteOnly )
366
// Check that it's a valid ZIP file
367
// KArchive::open() opened the underlying device already.
369
uint offset = 0; // holds offset, where we read
372
// contains information gathered from the local file headers
373
QHash<QByteArray, ParseFileInfo> pfi_map;
375
QIODevice* dev = device();
377
// We set a bool for knowing if we are allowed to skip the start of the file
378
bool startOfFile = true;
380
for (;;) // repeat until 'end of entries' signature is reached
382
//qDebug() << "loop starts";
383
//qDebug() << "dev->pos() now : " << dev->pos();
384
n = dev->read( buffer, 4 );
388
qWarning() << "Invalid ZIP file. Unexpected end of file. (#1)";
393
if ( !memcmp( buffer, "PK\5\6", 4 ) ) // 'end of entries'
395
//qDebug() << "PK56 found end of archive";
400
400
if ( !memcmp( buffer, "PK\3\4", 4 ) ) // local file header
402
//qDebug() << "PK34 found local file header";
405
dev->seek( dev->pos() + 2 ); // skip 'version needed to extract'
407
// read static header stuff
408
n = dev->read( buffer, 24 );
410
qWarning() << "Invalid ZIP file. Unexpected end of file. (#4)";
414
int gpf = (uchar)buffer[0]; // "general purpose flag" not "general protection fault" ;-)
415
int compression_mode = (uchar)buffer[2] | (uchar)buffer[3] << 8;
416
time_t mtime = transformFromMsDos( buffer+4 );
418
qint64 compr_size = (uchar)buffer[12] | (uchar)buffer[13] << 8
419
| (uchar)buffer[14] << 16 | (uchar)buffer[15] << 24;
420
qint64 uncomp_size = (uchar)buffer[16] | (uchar)buffer[17] << 8
421
| (uchar)buffer[18] << 16 | (uchar)buffer[19] << 24;
422
int namelen = (uchar)buffer[20] | (uchar)buffer[21] << 8;
423
int extralen = (uchar)buffer[22] | (uchar)buffer[23] << 8;
426
qDebug() << "general purpose bit flag: " << gpf;
427
qDebug() << "compressed size: " << compr_size;
428
qDebug() << "uncompressed size: " << uncomp_size;
429
qDebug() << "namelen: " << namelen;
430
qDebug() << "extralen: " << extralen;
431
qDebug() << "archive size: " << dev->size();
435
Q_ASSERT( namelen > 0 );
436
QByteArray fileName = dev->read(namelen);
437
if ( fileName.size() < namelen ) {
438
qWarning() << "Invalid ZIP file. Name not completely read (#2)";
402
//qDebug() << "PK34 found local file header";
405
dev->seek( dev->pos() + 2 ); // skip 'version needed to extract'
407
// read static header stuff
408
n = dev->read( buffer, 24 );
410
qWarning() << "Invalid ZIP file. Unexpected end of file. (#4)";
414
int gpf = (uchar)buffer[0]; // "general purpose flag" not "general protection fault" ;-)
415
int compression_mode = (uchar)buffer[2] | (uchar)buffer[3] << 8;
416
time_t mtime = transformFromMsDos( buffer+4 );
418
qint64 compr_size = (uchar)buffer[12] | (uchar)buffer[13] << 8
419
| (uchar)buffer[14] << 16 | (uchar)buffer[15] << 24;
420
qint64 uncomp_size = (uchar)buffer[16] | (uchar)buffer[17] << 8
421
| (uchar)buffer[18] << 16 | (uchar)buffer[19] << 24;
422
int namelen = (uchar)buffer[20] | (uchar)buffer[21] << 8;
423
int extralen = (uchar)buffer[22] | (uchar)buffer[23] << 8;
426
qDebug() << "general purpose bit flag: " << gpf;
427
qDebug() << "compressed size: " << compr_size;
428
qDebug() << "uncompressed size: " << uncomp_size;
429
qDebug() << "namelen: " << namelen;
430
qDebug() << "extralen: " << extralen;
431
qDebug() << "archive size: " << dev->size();
435
Q_ASSERT( namelen > 0 );
436
QByteArray fileName = dev->read(namelen);
437
if ( fileName.size() < namelen ) {
438
qWarning() << "Invalid ZIP file. Name not completely read (#2)";
445
// read and parse the beginning of the extra field,
446
// skip rest of extra field in case it is too long
447
unsigned int extraFieldEnd = dev->pos() + extralen;
448
pfi.extralen = extralen;
449
int handledextralen = qMin(extralen, (int)sizeof buffer);
451
//if ( handledextralen )
452
// qDebug() << "handledextralen: " << handledextralen;
454
n = dev->read(buffer, handledextralen);
455
// no error msg necessary as we deliberately truncate the extra field
456
if (!parseExtraField(buffer, handledextralen, true, pfi))
458
qWarning() << "Invalid ZIP File. Broken ExtraField.";
462
// jump to end of extra field
463
dev->seek( extraFieldEnd );
465
// we have to take care of the 'general purpose bit flag'.
466
// if bit 3 is set, the header doesn't contain the length of
467
// the file and we look for the signature 'PK\7\8'.
470
// here we have to read through the compressed data to find
445
// read and parse the beginning of the extra field,
446
// skip rest of extra field in case it is too long
447
unsigned int extraFieldEnd = dev->pos() + extralen;
448
pfi.extralen = extralen;
449
int handledextralen = qMin(extralen, (int)sizeof buffer);
451
//if ( handledextralen )
452
// qDebug() << "handledextralen: " << handledextralen;
454
n = dev->read(buffer, handledextralen);
455
// no error msg necessary as we deliberately truncate the extra field
456
if (!parseExtraField(buffer, handledextralen, true, pfi))
458
qWarning() << "Invalid ZIP File. Broken ExtraField.";
462
// jump to end of extra field
463
dev->seek( extraFieldEnd );
465
// we have to take care of the 'general purpose bit flag'.
466
// if bit 3 is set, the header doesn't contain the length of
467
// the file and we look for the signature 'PK\7\8'.
470
// here we have to read through the compressed data to find
472
qDebug() << "trying to seek for next PK78";
473
bool foundSignature = false;
475
while (!foundSignature)
477
n = dev->read( buffer, 1 );
480
qWarning() << "Invalid ZIP file. Unexpected end of file. (#2)";
484
if ( buffer[0] != 'P' )
487
n = dev->read( buffer, 3 );
490
qWarning() << "Invalid ZIP file. Unexpected end of file. (#3)";
494
// we have to detect three magic tokens here:
495
// PK34 for the next local header in case there is no data descriptor
496
// PK12 for the central header in case there is no data descriptor
497
// PK78 for the data descriptor in case it is following the compressed data
499
if ( buffer[0] == 'K' && buffer[1] == 7 && buffer[2] == 8 )
501
foundSignature = true;
502
dev->seek( dev->pos() + 12 ); // skip the 'data_descriptor'
504
else if ( ( buffer[0] == 'K' && buffer[1] == 1 && buffer[2] == 2 )
505
|| ( buffer[0] == 'K' && buffer[1] == 3 && buffer[2] == 4 ) )
507
foundSignature = true;
508
dev->seek( dev->pos() - 4 ); // go back 4 bytes, so that the magic bytes can be found...
510
else if ( buffer[0] == 'P' || buffer[1] == 'P' || buffer[2] == 'P' )
512
// We have another P character so we must go back a little to check if it is a magic
513
dev->seek( dev->pos() - 3 );
520
// here we skip the compressed data and jump to the next header
521
//qDebug() << "general purpose bit flag indicates, that local file header contains valid size";
472
qDebug() << "trying to seek for next PK78";
473
bool foundSignature = false;
475
while (!foundSignature)
477
n = dev->read( buffer, 1 );
480
qWarning() << "Invalid ZIP file. Unexpected end of file. (#2)";
484
if ( buffer[0] != 'P' )
487
n = dev->read( buffer, 3 );
490
qWarning() << "Invalid ZIP file. Unexpected end of file. (#3)";
494
// we have to detect three magic tokens here:
495
// PK34 for the next local header in case there is no data descriptor
496
// PK12 for the central header in case there is no data descriptor
497
// PK78 for the data descriptor in case it is following the compressed data
499
if ( buffer[0] == 'K' && buffer[1] == 7 && buffer[2] == 8 )
501
foundSignature = true;
502
dev->seek( dev->pos() + 12 ); // skip the 'data_descriptor'
504
else if ( ( buffer[0] == 'K' && buffer[1] == 1 && buffer[2] == 2 )
505
|| ( buffer[0] == 'K' && buffer[1] == 3 && buffer[2] == 4 ) )
507
foundSignature = true;
508
dev->seek( dev->pos() - 4 ); // go back 4 bytes, so that the magic bytes can be found...
510
else if ( buffer[0] == 'P' || buffer[1] == 'P' || buffer[2] == 'P' )
512
// We have another P character so we must go back a little to check if it is a magic
513
dev->seek( dev->pos() - 3 );
520
// here we skip the compressed data and jump to the next header
521
//qDebug() << "general purpose bit flag indicates, that local file header contains valid size";
522
522
// check if this could be a symbolic link
523
523
if (compression_mode == NoCompression
524
&& uncomp_size <= max_path_len
524
&& uncomp_size <= max_path_len
525
525
&& uncomp_size > 0) {
526
// read content and store it
527
// If it's not a symlink, then we'll just discard the data for now.
528
pfi.guessed_symlink = dev->read(uncomp_size);
529
if (pfi.guessed_symlink.size() < uncomp_size) {
526
// read content and store it
527
// If it's not a symlink, then we'll just discard the data for now.
528
pfi.guessed_symlink = dev->read(uncomp_size);
529
if (pfi.guessed_symlink.size() < uncomp_size) {
530
530
qWarning() << "Invalid ZIP file. Unexpected end of file. (#5)";
535
if ( compr_size > dev->size() )
537
// here we cannot trust the compressed size, so scan through the compressed
535
if ( compr_size > dev->size() )
537
// here we cannot trust the compressed size, so scan through the compressed
538
538
// data to find the next header
539
539
bool foundSignature = false;
577
577
// in the next cycle...
583
583
// qDebug() << "before interesting dev->pos(): " << dev->pos();
584
584
bool success = dev->seek( dev->pos() + compr_size ); // can this fail ???
585
585
success = success; // annoying unused variable warning
586
Q_ASSERT( success ); // let's see...
586
Q_ASSERT( success ); // let's see...
587
587
/* qDebug() << "after interesting dev->pos(): " << dev->pos();
589
589
qDebug() << "dev->at was successful... ";
591
591
qDebug() << "dev->at failed... ";*/
596
596
// not needed any more
597
597
/* // here we calculate the length of the file in the zip
598
// with headers and jump to the next header.
599
uint skip = compr_size + namelen + extralen;
600
offset += 30 + skip;*/
602
pfi_map.insert(fileName, pfi);
604
else if ( !memcmp( buffer, "PK\1\2", 4 ) ) // central block
606
//qDebug() << "PK12 found central block";
609
// so we reached the central header at the end of the zip file
610
// here we get all interesting data out of the central header
612
offset = dev->pos() - 4;
614
//set offset for appending new files
615
if ( d->m_offset == 0L ) d->m_offset = offset;
617
n = dev->read( buffer + 4, 42 );
619
qWarning() << "Invalid ZIP file, central entry too short"; // not long enough for valid entry
623
//int gpf = (uchar)buffer[9] << 8 | (uchar)buffer[10];
624
//kDebug() << "general purpose flag=" << gpf;
625
// length of the fileName (well, pathname indeed)
626
int namelen = (uchar)buffer[29] << 8 | (uchar)buffer[28];
627
Q_ASSERT( namelen > 0 );
628
QByteArray bufferName = dev->read( namelen );
629
if ( bufferName.size() < namelen )
630
qWarning() << "Invalid ZIP file. Name not completely read";
632
ParseFileInfo pfi = pfi_map.value( bufferName, ParseFileInfo() );
634
QString name( QFile::decodeName(bufferName) );
636
//qDebug() << "name: " << name;
637
// only in central header ! see below.
638
// length of extra attributes
639
int extralen = (uchar)buffer[31] << 8 | (uchar)buffer[30];
640
// length of comment for this file
641
int commlen = (uchar)buffer[33] << 8 | (uchar)buffer[32];
642
// compression method of this file
643
int cmethod = (uchar)buffer[11] << 8 | (uchar)buffer[10];
645
//qDebug() << "cmethod: " << cmethod;
646
//qDebug() << "extralen: " << extralen;
649
uint crc32 = (uchar)buffer[19] << 24 | (uchar)buffer[18] << 16 |
650
(uchar)buffer[17] << 8 | (uchar)buffer[16];
652
// uncompressed file size
653
uint ucsize = (uchar)buffer[27] << 24 | (uchar)buffer[26] << 16 |
654
(uchar)buffer[25] << 8 | (uchar)buffer[24];
655
// compressed file size
656
uint csize = (uchar)buffer[23] << 24 | (uchar)buffer[22] << 16 |
657
(uchar)buffer[21] << 8 | (uchar)buffer[20];
659
// offset of local header
660
uint localheaderoffset = (uchar)buffer[45] << 24 | (uchar)buffer[44] << 16 |
598
// with headers and jump to the next header.
599
uint skip = compr_size + namelen + extralen;
600
offset += 30 + skip;*/
602
pfi_map.insert(fileName, pfi);
604
else if ( !memcmp( buffer, "PK\1\2", 4 ) ) // central block
606
//qDebug() << "PK12 found central block";
609
// so we reached the central header at the end of the zip file
610
// here we get all interesting data out of the central header
612
offset = dev->pos() - 4;
614
//set offset for appending new files
615
if ( d->m_offset == 0L ) d->m_offset = offset;
617
n = dev->read( buffer + 4, 42 );
619
qWarning() << "Invalid ZIP file, central entry too short"; // not long enough for valid entry
623
//int gpf = (uchar)buffer[9] << 8 | (uchar)buffer[10];
624
//kDebug() << "general purpose flag=" << gpf;
625
// length of the fileName (well, pathname indeed)
626
int namelen = (uchar)buffer[29] << 8 | (uchar)buffer[28];
627
Q_ASSERT( namelen > 0 );
628
QByteArray bufferName = dev->read( namelen );
629
if ( bufferName.size() < namelen )
630
qWarning() << "Invalid ZIP file. Name not completely read";
632
ParseFileInfo pfi = pfi_map.value( bufferName, ParseFileInfo() );
634
QString name( QFile::decodeName(bufferName) );
636
//qDebug() << "name: " << name;
637
// only in central header ! see below.
638
// length of extra attributes
639
int extralen = (uchar)buffer[31] << 8 | (uchar)buffer[30];
640
// length of comment for this file
641
int commlen = (uchar)buffer[33] << 8 | (uchar)buffer[32];
642
// compression method of this file
643
int cmethod = (uchar)buffer[11] << 8 | (uchar)buffer[10];
645
//qDebug() << "cmethod: " << cmethod;
646
//qDebug() << "extralen: " << extralen;
649
uint crc32 = (uchar)buffer[19] << 24 | (uchar)buffer[18] << 16 |
650
(uchar)buffer[17] << 8 | (uchar)buffer[16];
652
// uncompressed file size
653
uint ucsize = (uchar)buffer[27] << 24 | (uchar)buffer[26] << 16 |
654
(uchar)buffer[25] << 8 | (uchar)buffer[24];
655
// compressed file size
656
uint csize = (uchar)buffer[23] << 24 | (uchar)buffer[22] << 16 |
657
(uchar)buffer[21] << 8 | (uchar)buffer[20];
659
// offset of local header
660
uint localheaderoffset = (uchar)buffer[45] << 24 | (uchar)buffer[44] << 16 |
661
661
(uchar)buffer[43] << 8 | (uchar)buffer[42];
663
// some clever people use different extra field lengths
664
// in the central header and in the local header... funny.
665
// so we need to get the localextralen to calculate the offset
666
// from localheaderstart to dataoffset
667
int localextralen = pfi.extralen; // FIXME: this will not work if
668
// no local header exists
670
//qDebug() << "localextralen: " << localextralen;
672
// offset, where the real data for uncompression starts
673
uint dataoffset = localheaderoffset + 30 + localextralen + namelen; //comment only in central header
675
//qDebug() << "esize: " << esize;
676
//qDebug() << "eoffset: " << eoffset;
677
//qDebug() << "csize: " << csize;
679
int os_madeby = (uchar)buffer[5];
681
int access = 0100644;
683
if (os_madeby == 3) { // good ole unix
684
access = (uchar)buffer[40] | (uchar)buffer[41] << 8;
689
if ( name.endsWith( '/' ) ) // Entries with a trailing slash are directories
692
name = name.left( name.length() - 1 );
693
if (os_madeby != 3) access = S_IFDIR | 0755;
663
// some clever people use different extra field lengths
664
// in the central header and in the local header... funny.
665
// so we need to get the localextralen to calculate the offset
666
// from localheaderstart to dataoffset
667
int localextralen = pfi.extralen; // FIXME: this will not work if
668
// no local header exists
670
//qDebug() << "localextralen: " << localextralen;
672
// offset, where the real data for uncompression starts
673
uint dataoffset = localheaderoffset + 30 + localextralen + namelen; //comment only in central header
675
//qDebug() << "esize: " << esize;
676
//qDebug() << "eoffset: " << eoffset;
677
//qDebug() << "csize: " << csize;
679
int os_madeby = (uchar)buffer[5];
681
int access = 0100644;
683
if (os_madeby == 3) { // good ole unix
684
access = (uchar)buffer[40] | (uchar)buffer[41] << 8;
689
if ( name.endsWith( '/' ) ) // Entries with a trailing slash are directories
692
name = name.left( name.length() - 1 );
693
if (os_madeby != 3) access = S_IFDIR | 0755;
694
694
else Q_ASSERT(access & S_IFDIR);
697
int pos = name.lastIndexOf( '/' );
701
entryName = name.mid( pos + 1 );
702
Q_ASSERT( !entryName.isEmpty() );
704
KArchiveEntry* entry;
707
QString path = QDir::cleanPath( name );
708
const KArchiveEntry* ent = rootDir()->entry( path );
709
if ( ent && ent->isDirectory() )
711
//qDebug() << "Directory already exists, NOT going to add it again";
716
entry = new KArchiveDirectory( this, entryName, access, (int)pfi.mtime, rootDir()->user(), rootDir()->group(), QString() );
717
//qDebug() << "KArchiveDirectory created, entryName= " << entryName << ", name=" << name;
697
int pos = name.lastIndexOf( '/' );
701
entryName = name.mid( pos + 1 );
702
Q_ASSERT( !entryName.isEmpty() );
704
KArchiveEntry* entry;
707
QString path = QDir::cleanPath( name );
708
const KArchiveEntry* ent = rootDir()->entry( path );
709
if ( ent && ent->isDirectory() )
711
//qDebug() << "Directory already exists, NOT going to add it again";
716
entry = new KArchiveDirectory( this, entryName, access, (int)pfi.mtime, rootDir()->user(), rootDir()->group(), QString() );
717
//qDebug() << "KArchiveDirectory created, entryName= " << entryName << ", name=" << name;
723
723
if (S_ISLNK(access)) {
724
symlink = QFile::decodeName(pfi.guessed_symlink);
724
symlink = QFile::decodeName(pfi.guessed_symlink);
726
entry = new KZipFileEntry( this, entryName, access, pfi.mtime,
726
entry = new KZipFileEntry( this, entryName, access, pfi.mtime,
727
727
rootDir()->user(), rootDir()->group(),
728
728
symlink, name, dataoffset,
729
729
ucsize, cmethod, csize );
730
static_cast<KZipFileEntry *>(entry)->setHeaderStart( localheaderoffset );
731
static_cast<KZipFileEntry*>(entry)->setCRC32(crc32);
732
//qDebug() << "KZipFileEntry created, entryName= " << entryName << ", name=" << name;
733
d->m_fileList.append( static_cast<KZipFileEntry *>( entry ) );
740
rootDir()->addEntry(entry);
744
// In some tar files we can find dir/./file => call cleanPath
745
QString path = QDir::cleanPath( name.left( pos ) );
746
// Ensure container directory exists, create otherwise
747
KArchiveDirectory * tdir = findOrCreate( path );
748
tdir->addEntry(entry);
752
//calculate offset to next entry
753
offset += 46 + commlen + extralen + namelen;
754
bool b = dev->seek(offset);
759
else if ( startOfFile )
761
// The file does not start with any ZIP header (e.g. self-extractable ZIP files)
762
// Therefore we need to find the first PK\003\004 (local header)
763
qDebug() << "Try to skip start of file";
765
bool foundSignature = false;
767
while (!foundSignature)
769
n = dev->read( buffer, 1 );
772
qWarning() << "Invalid ZIP file. Unexpected end of file. " ;
776
if ( buffer[0] != 'P' )
779
n = dev->read( buffer, 3 );
782
qWarning() << "Invalid ZIP file. Unexpected end of file. " ;
786
// We have to detect the magic token for a local header: PK\003\004
788
* Note: we do not need to check the other magics, if the ZIP file has no
789
* local header, then it has not any files!
791
if ( buffer[0] == 'K' && buffer[1] == 3 && buffer[2] == 4 )
793
foundSignature = true;
794
dev->seek( dev->pos() - 4 ); // go back 4 bytes, so that the magic bytes can be found...
796
else if ( buffer[0] == 'P' || buffer[1] == 'P' || buffer[2] == 'P' )
798
// We have another P character so we must go back a little to check if it is a magic
799
dev->seek( dev->pos() - 3 );
805
qWarning() << "Invalid ZIP file. Unrecognized header at offset " << offset;
810
//qDebug() << "*** done *** ";
730
static_cast<KZipFileEntry *>(entry)->setHeaderStart( localheaderoffset );
731
static_cast<KZipFileEntry*>(entry)->setCRC32(crc32);
732
//qDebug() << "KZipFileEntry created, entryName= " << entryName << ", name=" << name;
733
d->m_fileList.append( static_cast<KZipFileEntry *>( entry ) );
740
rootDir()->addEntry(entry);
744
// In some tar files we can find dir/./file => call cleanPath
745
QString path = QDir::cleanPath( name.left( pos ) );
746
// Ensure container directory exists, create otherwise
747
KArchiveDirectory * tdir = findOrCreate( path );
748
tdir->addEntry(entry);
752
//calculate offset to next entry
753
offset += 46 + commlen + extralen + namelen;
754
bool b = dev->seek(offset);
759
else if ( startOfFile )
761
// The file does not start with any ZIP header (e.g. self-extractable ZIP files)
762
// Therefore we need to find the first PK\003\004 (local header)
763
qDebug() << "Try to skip start of file";
765
bool foundSignature = false;
767
while (!foundSignature)
769
n = dev->read( buffer, 1 );
772
qWarning() << "Invalid ZIP file. Unexpected end of file. " ;
776
if ( buffer[0] != 'P' )
779
n = dev->read( buffer, 3 );
782
qWarning() << "Invalid ZIP file. Unexpected end of file. " ;
786
// We have to detect the magic token for a local header: PK\003\004
788
* Note: we do not need to check the other magics, if the ZIP file has no
789
* local header, then it has not any files!
791
if ( buffer[0] == 'K' && buffer[1] == 3 && buffer[2] == 4 )
793
foundSignature = true;
794
dev->seek( dev->pos() - 4 ); // go back 4 bytes, so that the magic bytes can be found...
796
else if ( buffer[0] == 'P' || buffer[1] == 'P' || buffer[2] == 'P' )
798
// We have another P character so we must go back a little to check if it is a magic
799
dev->seek( dev->pos() - 3 );
805
qWarning() << "Invalid ZIP file. Unrecognized header at offset " << offset;
810
//qDebug() << "*** done *** ";
814
814
bool KZip::closeArchive()
816
if ( ! ( mode() & QIODevice::WriteOnly ) )
818
//qDebug() << "readonly";
822
//ReadWrite or WriteOnly
823
//write all central dir file entries
825
// to be written at the end of the file...
826
char buffer[ 22 ]; // first used for 12, then for 22 at the end
827
uLong crc = crc32(0L, Z_NULL, 0);
829
qint64 centraldiroffset = device()->pos();
830
//qDebug() << "closearchive: centraldiroffset: " << centraldiroffset;
831
qint64 atbackup = centraldiroffset;
832
QMutableListIterator<KZipFileEntry*> it( d->m_fileList );
835
{ //set crc and compressed size in each local file header
837
if ( !device()->seek( it.value()->headerStart() + 14 ) )
816
if ( ! ( mode() & QIODevice::WriteOnly ) )
818
//qDebug() << "readonly";
822
//ReadWrite or WriteOnly
823
//write all central dir file entries
825
// to be written at the end of the file...
826
char buffer[ 22 ]; // first used for 12, then for 22 at the end
827
uLong crc = crc32(0L, Z_NULL, 0);
829
qint64 centraldiroffset = device()->pos();
830
//qDebug() << "closearchive: centraldiroffset: " << centraldiroffset;
831
qint64 atbackup = centraldiroffset;
832
QMutableListIterator<KZipFileEntry*> it( d->m_fileList );
835
{ //set crc and compressed size in each local file header
837
if ( !device()->seek( it.value()->headerStart() + 14 ) )
839
839
//qDebug() << "closearchive setcrcandcsize: fileName:"
840
840
// << it.current()->path()
841
841
// << "encoding:" << it.current()->encoding();
843
uLong mycrc = it.value()->crc32();
844
buffer[0] = char(mycrc); // crc checksum, at headerStart+14
845
buffer[1] = char(mycrc >> 8);
846
buffer[2] = char(mycrc >> 16);
847
buffer[3] = char(mycrc >> 24);
849
int mysize1 = it.value()->compressedSize();
850
buffer[4] = char(mysize1); // compressed file size, at headerStart+18
851
buffer[5] = char(mysize1 >> 8);
852
buffer[6] = char(mysize1 >> 16);
853
buffer[7] = char(mysize1 >> 24);
855
int myusize = it.value()->size();
856
buffer[8] = char(myusize); // uncompressed file size, at headerStart+22
857
buffer[9] = char(myusize >> 8);
858
buffer[10] = char(myusize >> 16);
859
buffer[11] = char(myusize >> 24);
861
if ( device()->write( buffer, 12 ) != 12 )
864
device()->seek( atbackup );
843
uLong mycrc = it.value()->crc32();
844
buffer[0] = char(mycrc); // crc checksum, at headerStart+14
845
buffer[1] = char(mycrc >> 8);
846
buffer[2] = char(mycrc >> 16);
847
buffer[3] = char(mycrc >> 24);
849
int mysize1 = it.value()->compressedSize();
850
buffer[4] = char(mysize1); // compressed file size, at headerStart+18
851
buffer[5] = char(mysize1 >> 8);
852
buffer[6] = char(mysize1 >> 16);
853
buffer[7] = char(mysize1 >> 24);
855
int myusize = it.value()->size();
856
buffer[8] = char(myusize); // uncompressed file size, at headerStart+22
857
buffer[9] = char(myusize >> 8);
858
buffer[10] = char(myusize >> 16);
859
buffer[11] = char(myusize >> 24);
861
if ( device()->write( buffer, 12 ) != 12 )
864
device()->seek( atbackup );
870
//qDebug() << "fileName:" << it.current()->path()
871
// << "encoding:" << it.current()->encoding();
870
//qDebug() << "fileName:" << it.current()->path()
871
// << "encoding:" << it.current()->encoding();
873
QByteArray path = QFile::encodeName(it.value()->path());
873
QByteArray path = QFile::encodeName(it.value()->path());
875
875
const int extra_field_len = 9;
876
int bufferSize = extra_field_len + path.length() + 46;
877
char* buffer = new char[ bufferSize ];
879
memset(buffer, 0, 46); // zero is a nice default for most header fields
883
'P', 'K', 1, 2, // central file header signature
884
0x14, 3, // version made by (3 == UNIX)
885
0x14, 0 // version needed to extract
876
int bufferSize = extra_field_len + path.length() + 46;
877
char* buffer = new char[ bufferSize ];
879
memset(buffer, 0, 46); // zero is a nice default for most header fields
883
'P', 'K', 1, 2, // central file header signature
884
0x14, 3, // version made by (3 == UNIX)
885
0x14, 0 // version needed to extract
888
888
// I do not know why memcpy is not working here
889
//memcpy(buffer, head, sizeof(head));
890
memmove(buffer, head, sizeof(head));
892
buffer[ 10 ] = char(it.value()->encoding()); // compression method
893
buffer[ 11 ] = char(it.value()->encoding() >> 8);
895
transformToMsDos( it.value()->datetime(), &buffer[ 12 ] );
897
uLong mycrc = it.value()->crc32();
898
buffer[ 16 ] = char(mycrc); // crc checksum
899
buffer[ 17 ] = char(mycrc >> 8);
900
buffer[ 18 ] = char(mycrc >> 16);
901
buffer[ 19 ] = char(mycrc >> 24);
903
int mysize1 = it.value()->compressedSize();
904
buffer[ 20 ] = char(mysize1); // compressed file size
905
buffer[ 21 ] = char(mysize1 >> 8);
906
buffer[ 22 ] = char(mysize1 >> 16);
907
buffer[ 23 ] = char(mysize1 >> 24);
909
int mysize = it.value()->size();
910
buffer[ 24 ] = char(mysize); // uncompressed file size
911
buffer[ 25 ] = char(mysize >> 8);
912
buffer[ 26 ] = char(mysize >> 16);
913
buffer[ 27 ] = char(mysize >> 24);
915
buffer[ 28 ] = char(it.value()->path().length()); // fileName length
916
buffer[ 29 ] = char(it.value()->path().length() >> 8);
889
//memcpy(buffer, head, sizeof(head));
890
memmove(buffer, head, sizeof(head));
892
buffer[ 10 ] = char(it.value()->encoding()); // compression method
893
buffer[ 11 ] = char(it.value()->encoding() >> 8);
895
transformToMsDos( it.value()->datetime(), &buffer[ 12 ] );
897
uLong mycrc = it.value()->crc32();
898
buffer[ 16 ] = char(mycrc); // crc checksum
899
buffer[ 17 ] = char(mycrc >> 8);
900
buffer[ 18 ] = char(mycrc >> 16);
901
buffer[ 19 ] = char(mycrc >> 24);
903
int mysize1 = it.value()->compressedSize();
904
buffer[ 20 ] = char(mysize1); // compressed file size
905
buffer[ 21 ] = char(mysize1 >> 8);
906
buffer[ 22 ] = char(mysize1 >> 16);
907
buffer[ 23 ] = char(mysize1 >> 24);
909
int mysize = it.value()->size();
910
buffer[ 24 ] = char(mysize); // uncompressed file size
911
buffer[ 25 ] = char(mysize >> 8);
912
buffer[ 26 ] = char(mysize >> 16);
913
buffer[ 27 ] = char(mysize >> 24);
915
buffer[ 28 ] = char(it.value()->path().length()); // fileName length
916
buffer[ 29 ] = char(it.value()->path().length() >> 8);
918
918
buffer[ 30 ] = char(extra_field_len);
919
919
buffer[ 31 ] = char(extra_field_len >> 8);
946
946
extfield[7] = char(time >> 16);
947
947
extfield[8] = char(time >> 24);
949
crc = crc32(crc, (Bytef *)buffer, bufferSize );
950
bool ok = ( device()->write( buffer, bufferSize ) == bufferSize );
955
qint64 centraldirendoffset = device()->pos();
956
//qDebug() << "closearchive: centraldirendoffset: " << centraldirendoffset;
957
//qDebug() << "closearchive: device()->pos(): " << device()->pos();
959
//write end of central dir record.
960
buffer[ 0 ] = 'P'; //end of central dir signature
965
buffer[ 4 ] = 0; // number of this disk
968
buffer[ 6 ] = 0; // number of disk with start of central dir
971
int count = d->m_fileList.count();
972
//qDebug() << "number of files (count): " << count;
975
buffer[ 8 ] = char(count); // total number of entries in central dir of
976
buffer[ 9 ] = char(count >> 8); // this disk
978
buffer[ 10 ] = buffer[ 8 ]; // total number of entries in the central dir
979
buffer[ 11 ] = buffer[ 9 ];
981
int cdsize = centraldirendoffset - centraldiroffset;
982
buffer[ 12 ] = char(cdsize); // size of the central dir
983
buffer[ 13 ] = char(cdsize >> 8);
984
buffer[ 14 ] = char(cdsize >> 16);
985
buffer[ 15 ] = char(cdsize >> 24);
987
//qDebug() << "end : centraldiroffset: " << centraldiroffset;
988
//qDebug() << "end : centraldirsize: " << cdsize;
990
buffer[ 16 ] = char(centraldiroffset); // central dir offset
991
buffer[ 17 ] = char(centraldiroffset >> 8);
992
buffer[ 18 ] = char(centraldiroffset >> 16);
993
buffer[ 19 ] = char(centraldiroffset >> 24);
995
buffer[ 20 ] = 0; //zipfile comment length
998
if ( device()->write( buffer, 22 ) != 22 )
949
crc = crc32(crc, (Bytef *)buffer, bufferSize );
950
bool ok = ( device()->write( buffer, bufferSize ) == bufferSize );
955
qint64 centraldirendoffset = device()->pos();
956
//qDebug() << "closearchive: centraldirendoffset: " << centraldirendoffset;
957
//qDebug() << "closearchive: device()->pos(): " << device()->pos();
959
//write end of central dir record.
960
buffer[ 0 ] = 'P'; //end of central dir signature
965
buffer[ 4 ] = 0; // number of this disk
968
buffer[ 6 ] = 0; // number of disk with start of central dir
971
int count = d->m_fileList.count();
972
//qDebug() << "number of files (count): " << count;
975
buffer[ 8 ] = char(count); // total number of entries in central dir of
976
buffer[ 9 ] = char(count >> 8); // this disk
978
buffer[ 10 ] = buffer[ 8 ]; // total number of entries in the central dir
979
buffer[ 11 ] = buffer[ 9 ];
981
int cdsize = centraldirendoffset - centraldiroffset;
982
buffer[ 12 ] = char(cdsize); // size of the central dir
983
buffer[ 13 ] = char(cdsize >> 8);
984
buffer[ 14 ] = char(cdsize >> 16);
985
buffer[ 15 ] = char(cdsize >> 24);
987
//qDebug() << "end : centraldiroffset: " << centraldiroffset;
988
//qDebug() << "end : centraldirsize: " << cdsize;
990
buffer[ 16 ] = char(centraldiroffset); // central dir offset
991
buffer[ 17 ] = char(centraldiroffset >> 8);
992
buffer[ 18 ] = char(centraldiroffset >> 16);
993
buffer[ 19 ] = char(centraldiroffset >> 24);
995
buffer[ 20 ] = 0; //zipfile comment length
998
if ( device()->write( buffer, 22 ) != 22 )
1004
1004
bool KZip::doWriteDir( const QString&, const QString&, const QString&,
1005
mode_t, time_t, time_t, time_t ) {
1005
mode_t, time_t, time_t, time_t ) {
1009
1009
bool KZip::doPrepareWriting(const QString &name, const QString &user,
1010
const QString &group, qint64 /*size*/, mode_t perm,
1011
time_t atime, time_t mtime, time_t ctime) {
1015
qWarning( "KZip::writeFile: You must open the zip file before writing to it\n");
1019
if ( ! ( mode() & QIODevice::WriteOnly ) ) // accept WriteOnly and ReadWrite
1021
qWarning( "KZip::writeFile: You must open the zip file for writing\n");
1025
Q_ASSERT( device() );
1027
// set right offset in zip.
1028
if ( !device()->seek( d->m_offset ) ) {
1029
qWarning() << "doPrepareWriting: cannot seek in ZIP file. Disk full?";
1033
// delete entries in the filelist with the same fileName as the one we want
1034
// to save, so that we don't have duplicate file entries when viewing the zip
1036
// CAUTION: the old file itself is still in the zip and won't be removed !!!
1037
QMutableListIterator<KZipFileEntry*> it( d->m_fileList );
1010
const QString &group, qint64 /*size*/, mode_t perm,
1011
time_t atime, time_t mtime, time_t ctime) {
1015
qWarning( "KZip::writeFile: You must open the zip file before writing to it\n");
1019
if ( ! ( mode() & QIODevice::WriteOnly ) ) // accept WriteOnly and ReadWrite
1021
qWarning( "KZip::writeFile: You must open the zip file for writing\n");
1025
Q_ASSERT( device() );
1027
// set right offset in zip.
1028
if ( !device()->seek( d->m_offset ) ) {
1029
qWarning() << "doPrepareWriting: cannot seek in ZIP file. Disk full?";
1033
// delete entries in the filelist with the same fileName as the one we want
1034
// to save, so that we don't have duplicate file entries when viewing the zip
1036
// CAUTION: the old file itself is still in the zip and won't be removed !!!
1037
QMutableListIterator<KZipFileEntry*> it( d->m_fileList );
1038
1038
//qDebug() << "fileName to write: " << name;
1042
//qDebug() << "prepfileName: " << it.current()->path();
1042
//qDebug() << "prepfileName: " << it.current()->path();
1043
1043
if (name == it.value()->path() )
1045
//qDebug() << "removing following entry: " << it.current()->path();
1045
//qDebug() << "removing following entry: " << it.current()->path();
1046
1046
delete it.value();
1051
// Find or create parent dir
1052
KArchiveDirectory* parentDir = rootDir();
1053
QString fileName( name );
1054
int i = name.lastIndexOf( '/' );
1057
QString dir = name.left( i );
1058
fileName = name.mid( i + 1 );
1059
//qDebug() << "ensuring" << dir << "exists. fileName=" << fileName;
1060
parentDir = findOrCreate( dir );
1063
// construct a KZipFileEntry and add it to list
1064
KZipFileEntry * e = new KZipFileEntry( this, fileName, perm, mtime, user, group, QString(),
1065
name, device()->pos() + 30 + name.length(), // start
1066
0 /*size unknown yet*/, d->m_compression, 0 /*csize unknown yet*/ );
1067
e->setHeaderStart( device()->pos() );
1068
//qDebug() << "wrote file start: " << e->position() << " name: " << name;
1069
parentDir->addEntry( e );
1071
d->m_currentFile = e;
1072
d->m_fileList.append( e );
1074
int extra_field_len = 0;
1075
if ( d->m_extraField == ModificationTime )
1076
extra_field_len = 17; // value also used in finishWriting()
1078
// write out zip header
1079
QByteArray encodedName = QFile::encodeName(name);
1080
int bufferSize = extra_field_len + encodedName.length() + 30;
1081
//qDebug() << "bufferSize=" << bufferSize;
1082
char* buffer = new char[ bufferSize ];
1084
buffer[ 0 ] = 'P'; //local file header signature
1089
buffer[ 4 ] = 0x14; // version needed to extract
1092
buffer[ 6 ] = 0; // general purpose bit flag
1095
buffer[ 8 ] = char(e->encoding()); // compression method
1096
buffer[ 9 ] = char(e->encoding() >> 8);
1098
transformToMsDos( e->datetime(), &buffer[ 10 ] );
1100
buffer[ 14 ] = 'C'; //dummy crc
1105
buffer[ 18 ] = 'C'; //compressed file size
1110
buffer[ 22 ] = 'U'; //uncompressed file size
1115
buffer[ 26 ] = (uchar)(encodedName.length()); //fileName length
1116
buffer[ 27 ] = (uchar)(encodedName.length() >> 8);
1118
buffer[ 28 ] = (uchar)(extra_field_len); // extra field length
1119
buffer[ 29 ] = (uchar)(extra_field_len >> 8);
1122
strncpy( buffer + 30, encodedName, encodedName.length() );
1125
if ( d->m_extraField == ModificationTime )
1127
char *extfield = buffer + 30 + encodedName.length();
1128
// "Extended timestamp" header (0x5455)
1131
extfield[2] = 13; // data size
1133
extfield[4] = 1 | 2 | 4; // contains mtime, atime, ctime
1135
extfield[5] = char(mtime);
1136
extfield[6] = char(mtime >> 8);
1137
extfield[7] = char(mtime >> 16);
1138
extfield[8] = char(mtime >> 24);
1140
extfield[9] = char(atime);
1141
extfield[10] = char(atime >> 8);
1142
extfield[11] = char(atime >> 16);
1143
extfield[12] = char(atime >> 24);
1145
extfield[13] = char(ctime);
1146
extfield[14] = char(ctime >> 8);
1147
extfield[15] = char(ctime >> 16);
1148
extfield[16] = char(ctime >> 24);
1152
bool b = (device()->write( buffer, bufferSize ) == bufferSize );
1161
// Prepare device for writing the data
1162
// Either device() if no compression, or a KFilterDev to compress
1163
if ( d->m_compression == 0 ) {
1164
d->m_currentDev = device();
1168
d->m_currentDev = KFilterDev::device( device(), "application/x-gzip", false );
1169
Q_ASSERT( d->m_currentDev );
1170
if ( !d->m_currentDev ) {
1171
return false; // ouch
1173
static_cast<KFilterDev *>(d->m_currentDev)->setSkipHeaders(); // Just zlib, not gzip
1175
b = d->m_currentDev->open( QIODevice::WriteOnly );
1051
// Find or create parent dir
1052
KArchiveDirectory* parentDir = rootDir();
1053
QString fileName( name );
1054
int i = name.lastIndexOf( '/' );
1057
QString dir = name.left( i );
1058
fileName = name.mid( i + 1 );
1059
//qDebug() << "ensuring" << dir << "exists. fileName=" << fileName;
1060
parentDir = findOrCreate( dir );
1063
// construct a KZipFileEntry and add it to list
1064
KZipFileEntry * e = new KZipFileEntry( this, fileName, perm, mtime, user, group, QString(),
1065
name, device()->pos() + 30 + name.length(), // start
1066
0 /*size unknown yet*/, d->m_compression, 0 /*csize unknown yet*/ );
1067
e->setHeaderStart( device()->pos() );
1068
//qDebug() << "wrote file start: " << e->position() << " name: " << name;
1069
parentDir->addEntry( e );
1071
d->m_currentFile = e;
1072
d->m_fileList.append( e );
1074
int extra_field_len = 0;
1075
if ( d->m_extraField == ModificationTime )
1076
extra_field_len = 17; // value also used in finishWriting()
1078
// write out zip header
1079
QByteArray encodedName = QFile::encodeName(name);
1080
int bufferSize = extra_field_len + encodedName.length() + 30;
1081
//qDebug() << "bufferSize=" << bufferSize;
1082
char* buffer = new char[ bufferSize ];
1084
buffer[ 0 ] = 'P'; //local file header signature
1089
buffer[ 4 ] = 0x14; // version needed to extract
1092
buffer[ 6 ] = 0; // general purpose bit flag
1095
buffer[ 8 ] = char(e->encoding()); // compression method
1096
buffer[ 9 ] = char(e->encoding() >> 8);
1098
transformToMsDos( e->datetime(), &buffer[ 10 ] );
1100
buffer[ 14 ] = 'C'; //dummy crc
1105
buffer[ 18 ] = 'C'; //compressed file size
1110
buffer[ 22 ] = 'U'; //uncompressed file size
1115
buffer[ 26 ] = (uchar)(encodedName.length()); //fileName length
1116
buffer[ 27 ] = (uchar)(encodedName.length() >> 8);
1118
buffer[ 28 ] = (uchar)(extra_field_len); // extra field length
1119
buffer[ 29 ] = (uchar)(extra_field_len >> 8);
1122
strncpy( buffer + 30, encodedName, encodedName.length() );
1125
if ( d->m_extraField == ModificationTime )
1127
char *extfield = buffer + 30 + encodedName.length();
1128
// "Extended timestamp" header (0x5455)
1131
extfield[2] = 13; // data size
1133
extfield[4] = 1 | 2 | 4; // contains mtime, atime, ctime
1135
extfield[5] = char(mtime);
1136
extfield[6] = char(mtime >> 8);
1137
extfield[7] = char(mtime >> 16);
1138
extfield[8] = char(mtime >> 24);
1140
extfield[9] = char(atime);
1141
extfield[10] = char(atime >> 8);
1142
extfield[11] = char(atime >> 16);
1143
extfield[12] = char(atime >> 24);
1145
extfield[13] = char(ctime);
1146
extfield[14] = char(ctime >> 8);
1147
extfield[15] = char(ctime >> 16);
1148
extfield[16] = char(ctime >> 24);
1152
bool b = (device()->write( buffer, bufferSize ) == bufferSize );
1161
// Prepare device for writing the data
1162
// Either device() if no compression, or a KFilterDev to compress
1163
if ( d->m_compression == 0 ) {
1164
d->m_currentDev = device();
1168
d->m_currentDev = KFilterDev::device( device(), "application/x-gzip", false );
1169
Q_ASSERT( d->m_currentDev );
1170
if ( !d->m_currentDev ) {
1171
return false; // ouch
1173
static_cast<KFilterDev *>(d->m_currentDev)->setSkipHeaders(); // Just zlib, not gzip
1175
b = d->m_currentDev->open( QIODevice::WriteOnly );
1180
1180
bool KZip::doFinishWriting( qint64 size )
1182
if ( d->m_currentFile->encoding() == 8 ) {
1184
(void)d->m_currentDev->write( 0, 0 );
1185
delete d->m_currentDev;
1187
// If 0, d->m_currentDev was device() - don't delete ;)
1188
d->m_currentDev = 0L;
1190
Q_ASSERT( d->m_currentFile );
1191
//qDebug() << "fileName: " << d->m_currentFile->path();
1192
//qDebug() << "getpos (at): " << device()->pos();
1193
d->m_currentFile->setSize(size);
1194
int extra_field_len = 0;
1195
if ( d->m_extraField == ModificationTime )
1196
extra_field_len = 17; // value also used in finishWriting()
1198
int csize = device()->pos() -
1199
d->m_currentFile->headerStart() - 30 -
1182
if ( d->m_currentFile->encoding() == 8 ) {
1184
(void)d->m_currentDev->write( 0, 0 );
1185
delete d->m_currentDev;
1187
// If 0, d->m_currentDev was device() - don't delete ;)
1188
d->m_currentDev = 0L;
1190
Q_ASSERT( d->m_currentFile );
1191
//qDebug() << "fileName: " << d->m_currentFile->path();
1192
//qDebug() << "getpos (at): " << device()->pos();
1193
d->m_currentFile->setSize(size);
1194
int extra_field_len = 0;
1195
if ( d->m_extraField == ModificationTime )
1196
extra_field_len = 17; // value also used in finishWriting()
1198
int csize = device()->pos() -
1199
d->m_currentFile->headerStart() - 30 -
1200
1200
d->m_currentFile->path().length() - extra_field_len;
1201
d->m_currentFile->setCompressedSize(csize);
1202
//qDebug() << "usize: " << d->m_currentFile->size();
1203
//qDebug() << "csize: " << d->m_currentFile->compressedSize();
1204
//qDebug() << "headerstart: " << d->m_currentFile->headerStart();
1206
//qDebug() << "crc: " << d->m_crc;
1207
d->m_currentFile->setCRC32( d->m_crc );
1209
d->m_currentFile = 0L;
1211
// update saved offset for appending new files
1212
d->m_offset = device()->pos();
1201
d->m_currentFile->setCompressedSize(csize);
1202
//qDebug() << "usize: " << d->m_currentFile->size();
1203
//qDebug() << "csize: " << d->m_currentFile->compressedSize();
1204
//qDebug() << "headerstart: " << d->m_currentFile->headerStart();
1206
//qDebug() << "crc: " << d->m_crc;
1207
d->m_currentFile->setCRC32( d->m_crc );
1209
d->m_currentFile = 0L;
1211
// update saved offset for appending new files
1212
d->m_offset = device()->pos();
1216
1216
bool KZip::doWriteSymLink(const QString &name, const QString &target,
1217
const QString &user, const QString &group,
1218
mode_t perm, time_t atime, time_t mtime, time_t ctime) {
1217
const QString &user, const QString &group,
1218
mode_t perm, time_t atime, time_t mtime, time_t ctime) {
1219
1219
// reassure that symlink flag is set, otherwise strange things happen on
1221
1221
perm |= S_IFLNK;
1293
1293
class KZipFileEntry::KZipFileEntryPrivate
1296
KZipFileEntryPrivate()
1303
qint64 compressedSize;
1296
KZipFileEntryPrivate()
1303
qint64 compressedSize;
1309
1309
KZipFileEntry::KZipFileEntry(KZip* zip, const QString& name, int access, int date,
1310
const QString& user, const QString& group, const QString& symlink,
1311
const QString& path, qint64 start, qint64 uncompressedSize,
1312
int encoding, qint64 compressedSize)
1310
const QString& user, const QString& group, const QString& symlink,
1311
const QString& path, qint64 start, qint64 uncompressedSize,
1312
int encoding, qint64 compressedSize)
1313
1313
: KArchiveFile(zip, name, access, date, user, group, symlink, start, uncompressedSize ),
1314
1314
d(new KZipFileEntryPrivate)
1317
d->encoding = encoding;
1318
d->compressedSize = compressedSize;
1317
d->encoding = encoding;
1318
d->compressedSize = compressedSize;
1321
1321
KZipFileEntry::~KZipFileEntry()
1326
1326
int KZipFileEntry::encoding() const
1331
1331
qint64 KZipFileEntry::compressedSize() const
1333
return d->compressedSize;
1333
return d->compressedSize;
1336
1336
void KZipFileEntry::setCompressedSize(qint64 compressedSize)
1338
d->compressedSize = compressedSize;
1338
d->compressedSize = compressedSize;
1341
1341
void KZipFileEntry::setHeaderStart(qint64 headerstart)
1343
d->headerStart = headerstart;
1343
d->headerStart = headerstart;
1346
1346
qint64 KZipFileEntry::headerStart() const
1348
return d->headerStart;
1348
return d->headerStart;
1351
1351
unsigned long KZipFileEntry::crc32() const
1356
1356
void KZipFileEntry::setCRC32(unsigned long crc32)
1361
1361
const QString &KZipFileEntry::path() const
1366
1366
QByteArray KZipFileEntry::data() const
1368
QIODevice* dev = createDevice();
1371
arr = dev->readAll();
1368
QIODevice* dev = createDevice();
1371
arr = dev->readAll();
1377
1377
QIODevice* KZipFileEntry::createDevice() const
1379
//qDebug() << "creating iodevice limited to pos=" << position() << ", csize=" << compressedSize();
1380
// Limit the reading to the appropriate part of the underlying device (e.g. file)
1381
KLimitedIODevice* limitedDev = new KLimitedIODevice( archive()->device(), position(), compressedSize() );
1382
if ( encoding() == 0 || compressedSize() == 0 ) // no compression (or even no data)
1379
//qDebug() << "creating iodevice limited to pos=" << position() << ", csize=" << compressedSize();
1380
// Limit the reading to the appropriate part of the underlying device (e.g. file)
1381
KLimitedIODevice* limitedDev = new KLimitedIODevice( archive()->device(), position(), compressedSize() );
1382
if ( encoding() == 0 || compressedSize() == 0 ) // no compression (or even no data)
1385
if ( encoding() == 8 )
1387
// On top of that, create a device that uncompresses the zlib data
1388
QIODevice* filterDev = KFilterDev::device( limitedDev, "application/x-gzip" );
1391
static_cast<KFilterDev *>(filterDev)->setSkipHeaders(); // Just zlib, not gzip
1392
bool b = filterDev->open( QIODevice::ReadOnly );
1385
if ( encoding() == 8 )
1387
// On top of that, create a device that uncompresses the zlib data
1388
QIODevice* filterDev = KFilterDev::device( limitedDev, "application/x-gzip" );
1391
static_cast<KFilterDev *>(filterDev)->setSkipHeaders(); // Just zlib, not gzip
1392
bool b = filterDev->open( QIODevice::ReadOnly );
1393
1393
b = b; // annoying unused variable warning
1398
qWarning() << "This zip file contains files compressed with method"
1399
<< encoding() << ", this method is currently not supported by KZip,"
1400
<< "please use a command-line tool to handle this file.";
1398
qWarning() << "This zip file contains files compressed with method"
1399
<< encoding() << ", this method is currently not supported by KZip,"
1400
<< "please use a command-line tool to handle this file.";