10#include "kcompressiondevice.h"
11#include "klimitediodevice_p.h"
12#include "loggingcategory.h"
22#include <qplatformdefs.h>
29#define QT_STAT_LNK 0120000
32static const int max_path_len = 4095;
34static void transformToMsDos(
const QDateTime &_dt,
char *buffer)
38 const quint16 time = (dt.
time().hour() << 11)
39 | (dt.
time().minute() << 5)
40 | (dt.
time().second() >> 1);
43 buffer[0] = char(time);
44 buffer[1] = char(time >> 8);
47 const quint16 date = ((dt.
date().year() - 1980) << 9)
48 | (dt.
date().month() << 5)
52 buffer[2] = char(date);
53 buffer[3] = char(date >> 8);
56static uint transformFromMsDos(
const char *buffer)
58 quint16 time = (uchar)buffer[0] | ((uchar)buffer[1] << 8);
60 int m = (time & 0x7ff) >> 5;
61 int s = (time & 0x1f) * 2;
64 quint16 date = (uchar)buffer[2] | ((uchar)buffer[3] << 8);
65 int y = (date >> 9) + 1980;
66 int o = (date & 0x1ff) >> 5;
67 int d = (date & 0x1f);
74static uint parseUi32(
const char *buffer)
76 return uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
79static quint64 parseUi64(
const char *buffer)
81 const uint a = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
82 const uint b = uint((uchar)buffer[4] | (uchar)buffer[5] << 8 | (uchar)buffer[6] << 16 | (uchar)buffer[7] << 24);
83 return (a | (quint64)b << 32);
98 QByteArray guessed_symlink;
102 bool exttimestamp_seen;
104 bool newinfounix_seen;
108 quint64 uncompressedSize = 0;
109 quint64 compressedSize = 0;
110 quint64 localheaderoffset = 0;
116 , exttimestamp_seen(false)
117 , newinfounix_seen(false)
119 ctime = mtime = atime = time(
nullptr);
131static bool parseExtTimestamp(
const char *buffer,
int size,
bool islocal, ParseFileInfo &pfi)
146 pfi.mtime = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
153 pfi.exttimestamp_seen =
true;
162 pfi.atime = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
172 pfi.ctime = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
176 pfi.exttimestamp_seen =
true;
188static bool parseInfoZipUnixOld(
const char *buffer,
int size,
bool islocal, ParseFileInfo &pfi)
191 if (pfi.exttimestamp_seen || pfi.newinfounix_seen) {
200 pfi.atime = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
202 pfi.mtime = uint((uchar)buffer[0] | (uchar)buffer[1] << 8 | (uchar)buffer[2] << 16 | (uchar)buffer[3] << 24);
204 if (islocal && size >= 12) {
205 pfi.uid = (uchar)buffer[0] | (uchar)buffer[1] << 8;
207 pfi.gid = (uchar)buffer[0] | (uchar)buffer[1] << 8;
222static bool parseInfoZipUnixNew(
const char *buffer,
int size,
bool islocal,
226 pfi.newinfounix =
true;
231 qCDebug(KArchiveLog) <<
"premature end of Info-ZIP unix extra field new";
235 pfi.uid = (uchar)buffer[0] | (uchar)buffer[1] << 8;
237 pfi.gid = (uchar)buffer[0] | (uchar)buffer[1] << 8;
240 pfi.newinfounix =
true;
252static bool parseExtraField(
const char *buffer,
int size, ParseFileInfo &pfi)
255 int magic = (uchar)buffer[0] | (uchar)buffer[1] << 8;
257 int fieldsize = (uchar)buffer[0] | (uchar)buffer[1] << 8;
261 if (fieldsize > size) {
270 pfi.uncompressedSize = parseUi64(buffer);
273 pfi.compressedSize = parseUi64(buffer + 8);
276 pfi.localheaderoffset = parseUi64(buffer + 16);
280 if (!parseExtTimestamp(buffer, fieldsize,
true, pfi)) {
285 if (!parseInfoZipUnixOld(buffer, fieldsize,
true, pfi)) {
291 if (!parseInfoZipUnixNew(buffer, fieldsize, islocal, pfi)) {
321 while (n >= minSize) {
322 if (
auto i = header.
indexOf(
"PK"); i >= 0) {
324 if ((i + minSize) < n) {
329 n = dev->
peek(header.
data(), minSize);
334 dev->
seek(dev->
pos() + (n - 1));
350static bool seekPostDataDescriptor(
QIODevice *dev,
bool isZip64)
353 const qsizetype descriptorSize = isZip64 ? 24 : 16;
356 auto oldPos = dev->
pos();
358 while (seekAnyHeader(dev, header, descriptorSize)) {
361 qint64 compressedSize = isZip64 ? parseUi64(header.
constData() + 8)
362 : parseUi32(header.constData() + 8);
363 if (oldPos + compressedSize == dev->
pos()) {
364 dev->seek(dev->pos() + descriptorSize);
380static bool seekToNextHeaderToken(
QIODevice *dev)
384 while (seekAnyHeader(dev, header, 4)) {
399class Q_DECL_HIDDEN
KZip::KZipPrivate
404 , m_currentFile(nullptr)
405 , m_currentDev(nullptr)
407 , m_extraField(KZip::NoExtraField)
413 KZipFileEntry *m_currentFile;
414 QIODevice *m_currentDev;
415 QList<KZipFileEntry *> m_fileList;
449 d->m_fileList.clear();
466 bool startOfFile =
true;
471 int n = dev->
read(buffer, 4);
474 setErrorString(tr(
"Invalid ZIP file. Unexpected end of file. (Error code: %1)").arg(1));
478 if (!memcmp(buffer,
"PK\6\6", 4)) {
483 if (!memcmp(buffer,
"PK\5\6", 4)) {
489 if (!memcmp(buffer,
"PK\3\4", 4)) {
494 n = dev->
read(buffer, 26);
496 setErrorString(tr(
"Invalid ZIP file. Unexpected end of file. (Error code: %1)").arg(4));
499 int neededVersion = (uchar)buffer[0] | (uchar)buffer[1] << 8;
500 bool isZip64 = neededVersion >= 45;
502 int gpf = (uchar)buffer[2];
503 int compression_mode = (uchar)buffer[4] | (uchar)buffer[5] << 8;
504 uint mtime = transformFromMsDos(buffer + 6);
506 const qint64 compr_size = parseUi32(buffer + 14);
507 const qint64 uncomp_size = parseUi32(buffer + 18);
508 const int namelen = uint(uchar(buffer[22])) | uint(uchar(buffer[23])) << 8;
509 const int extralen = uint(uchar(buffer[24])) | uint(uchar(buffer[25])) << 8;
527 setErrorString(tr(
"Invalid ZIP file. Name not completely read (#2)"));
533 pfi.dataoffset = 30 + extralen + namelen;
537 unsigned int extraFieldEnd = dev->
pos() + extralen;
538 int handledextralen = qMin(extralen, (
int)
sizeof buffer);
543 n = dev->
read(buffer, handledextralen);
545 if (!parseExtraField(buffer, n, pfi)) {
551 dev->
seek(extraFieldEnd);
559 if (!seekPostDataDescriptor(dev, isZip64)) {
566 bool foundSignature =
false;
569 && uncomp_size <= max_path_len
570 && uncomp_size > 0) {
573 pfi.guessed_symlink = dev->
read(uncomp_size);
574 if (pfi.guessed_symlink.
size() < uncomp_size) {
575 setErrorString(tr(
"Invalid ZIP file. Unexpected end of file. (#5)"));
579 if (compr_size > dev->
size()) {
582 if (!seekToNextHeaderToken(dev)) {
586 foundSignature =
true;
589 const bool success = dev->
seek(dev->
pos() + compr_size);
602 if (!foundSignature) {
605 n = dev->
peek(buffer, 4);
607 setErrorString(tr(
"Invalid ZIP file. Unexpected end of file. (#1)"));
628 }
else if (!memcmp(buffer,
"PK\1\2", 4)) {
635 auto offset = dev->
pos() - 4;
638 if (d->m_offset == 0) {
639 d->m_offset = offset;
642 n = dev->
read(buffer + 4, 42);
644 setErrorString(tr(
"Invalid ZIP file, central entry too short (not long enough for valid entry)"));
649 int extralen = (uchar)buffer[31] << 8 | (uchar)buffer[30];
651 int commlen = (uchar)buffer[33] << 8 | (uchar)buffer[32];
653 int cmethod = (uchar)buffer[11] << 8 | (uchar)buffer[10];
657 int namelen = (uchar)buffer[29] << 8 | (uchar)buffer[28];
659 setErrorString(tr(
"Invalid ZIP file, file path name length is zero"));
664 ParseFileInfo extrafi;
666 parseExtraField(varData.
constData() + namelen, extralen, extrafi);
670 if (bufferName.
size() < namelen) {
674 ParseFileInfo pfi = pfi_map.
value(bufferName, ParseFileInfo());
684 uint crc32 = parseUi32(buffer + 16);
687 quint64 ucsize = parseUi32(buffer + 24);
688 if (ucsize == 0xFFFFFFFF) {
689 ucsize = extrafi.uncompressedSize;
692 quint64 csize = parseUi32(buffer + 20);
693 if (csize == 0xFFFFFFFF) {
694 csize = extrafi.compressedSize;
698 quint64 localheaderoffset = parseUi32(buffer + 42);
699 if (localheaderoffset == 0xFFFFFFFF) {
700 localheaderoffset = extrafi.localheaderoffset;
706 qint64 dataoffset = localheaderoffset + pfi.dataoffset;
710 int os_madeby = (uchar)buffer[5];
712 int access = 0100644;
714 if (os_madeby == 3) {
715 access = (uchar)buffer[40] | (uchar)buffer[41] << 8;
723 if (os_madeby != 3) {
724 access = S_IFDIR | 0755;
726 access |= S_IFDIR | 0700;
734 entryName = name.
mid(pos + 1);
749 QDateTime mtime = KArchivePrivate::time_tToDateTime(pfi.mtime);
755 if ((access & QT_STAT_MASK) == QT_STAT_LNK) {
758 QDateTime mtime = KArchivePrivate::time_tToDateTime(pfi.mtime);
760 new KZipFileEntry(
this, entryName, access, mtime,
rootDir()->user(),
rootDir()->group(), symlink, name, dataoffset, ucsize, cmethod, csize);
761 static_cast<KZipFileEntry *
>(entry)->setHeaderStart(localheaderoffset);
779 setErrorString(tr(
"File %1 is in folder %2, but %3 is actually a file.").arg(entryName, path, path));
787 offset += 46 + commlen + extralen + namelen;
788 const bool b = dev->
seek(offset);
793 }
else if (startOfFile) {
798 bool foundSignature =
false;
801 while (seekAnyHeader(dev, header, 4)) {
808 foundSignature =
true;
813 if (!foundSignature) {
818 setErrorString(tr(
"Invalid ZIP file. Unrecognized header at offset %1").arg(dev->
pos() - 4));
838 uLong crc = crc32(0L,
nullptr, 0);
840 qint64 centraldiroffset =
device()->
pos();
842 qint64 atbackup = centraldiroffset;
845 if (!
device()->seek(entry->headerStart() + 14)) {
853 uLong mycrc = entry->
crc32();
854 buffer[0] = char(mycrc);
855 buffer[1] = char(mycrc >> 8);
856 buffer[2] = char(mycrc >> 16);
857 buffer[3] = char(mycrc >> 24);
859 int mysize1 = entry->compressedSize();
860 buffer[4] = char(mysize1);
861 buffer[5] = char(mysize1 >> 8);
862 buffer[6] = char(mysize1 >> 16);
863 buffer[7] = char(mysize1 >> 24);
865 int myusize = entry->
size();
866 buffer[8] = char(myusize);
867 buffer[9] = char(myusize >> 8);
868 buffer[10] = char(myusize >> 16);
869 buffer[11] = char(myusize >> 24);
871 if (
device()->write(buffer, 12) != 12) {
885 const int bufferSize = extra_field_len + path.
length() + 46;
886 char *buffer =
new char[bufferSize];
888 memset(buffer, 0, 46);
891 const char head[] = {
900 memmove(buffer, head,
sizeof(head));
902 buffer[10] = char(entry->encoding());
903 buffer[11] = char(entry->encoding() >> 8);
905 transformToMsDos(entry->
date(), &buffer[12]);
907 uLong mycrc = entry->
crc32();
908 buffer[16] = char(mycrc);
909 buffer[17] = char(mycrc >> 8);
910 buffer[18] = char(mycrc >> 16);
911 buffer[19] = char(mycrc >> 24);
913 int mysize1 = entry->compressedSize();
914 buffer[20] = char(mysize1);
915 buffer[21] = char(mysize1 >> 8);
916 buffer[22] = char(mysize1 >> 16);
917 buffer[23] = char(mysize1 >> 24);
919 int mysize = entry->
size();
920 buffer[24] = char(mysize);
921 buffer[25] = char(mysize >> 8);
922 buffer[26] = char(mysize >> 16);
923 buffer[27] = char(mysize >> 24);
925 buffer[28] = char(path.
length());
926 buffer[29] = char(path.
length() >> 8);
928 buffer[30] = char(extra_field_len);
929 buffer[31] = char(extra_field_len >> 8);
934 int myhst = entry->headerStart();
935 buffer[42] = char(myhst);
936 buffer[43] = char(myhst >> 8);
937 buffer[44] = char(myhst >> 16);
938 buffer[45] = char(myhst >> 24);
946 char *extfield = buffer + 46 + path.
length();
952 extfield[4] = 1 | 2 | 4;
956 extfield[5] = char(time);
957 extfield[6] = char(time >> 8);
958 extfield[7] = char(time >> 16);
959 extfield[8] = char(time >> 24);
962 crc = crc32(crc, (Bytef *)buffer, bufferSize);
963 bool ok = (
device()->
write(buffer, bufferSize) == bufferSize);
970 qint64 centraldirendoffset =
device()->
pos();
986 int count = d->m_fileList.count();
989 buffer[8] = char(count);
990 buffer[9] = char(count >> 8);
992 buffer[10] = buffer[8];
993 buffer[11] = buffer[9];
995 int cdsize = centraldirendoffset - centraldiroffset;
996 buffer[12] = char(cdsize);
997 buffer[13] = char(cdsize >> 8);
998 buffer[14] = char(cdsize >> 16);
999 buffer[15] = char(cdsize >> 24);
1004 buffer[16] = char(centraldiroffset);
1005 buffer[17] = char(centraldiroffset >> 8);
1006 buffer[18] = char(centraldiroffset >> 16);
1007 buffer[19] = char(centraldiroffset >> 24);
1012 if (
device()->write(buffer, 22) != 22) {
1049 setErrorString(tr(
"Application error: ZIP file must be open before being written into"));
1050 qCWarning(KArchiveLog) <<
"doPrepareWriting failed: !isOpen()";
1055 setErrorString(tr(
"Application error: attempted to write into non-writable ZIP file"));
1056 qCWarning(KArchiveLog) <<
"doPrepareWriting failed: !(mode() & QIODevice::WriteOnly)";
1066 if (!
device()->seek(d->m_offset)) {
1091 for (
auto it = d->m_fileList.begin(); it != d->m_fileList.end();) {
1093 if (name == (*it)->path()) {
1100 it = d->m_fileList.erase(it);
1125 d->m_currentFile = e;
1126 d->m_fileList.append(e);
1128 int extra_field_len = 0;
1130 extra_field_len = 17;
1135 int bufferSize = extra_field_len + encodedName.
length() + 30;
1137 char *buffer =
new char[bufferSize];
1150 buffer[8] = char(e->encoding());
1151 buffer[9] = char(e->encoding() >> 8);
1153 transformToMsDos(e->
date(), &buffer[10]);
1170 buffer[26] = (uchar)(encodedName.
length());
1171 buffer[27] = (uchar)(encodedName.
length() >> 8);
1173 buffer[28] = (uchar)(extra_field_len);
1174 buffer[29] = (uchar)(extra_field_len >> 8);
1181 char *extfield = buffer + 30 + encodedName.
length();
1187 extfield[4] = 1 | 2 | 4;
1189 extfield[5] = char(mtime);
1190 extfield[6] = char(mtime >> 8);
1191 extfield[7] = char(mtime >> 16);
1192 extfield[8] = char(mtime >> 24);
1194 extfield[9] = char(atime);
1195 extfield[10] = char(atime >> 8);
1196 extfield[11] = char(atime >> 16);
1197 extfield[12] = char(atime >> 24);
1199 extfield[13] = char(ctime);
1200 extfield[14] = char(ctime >> 8);
1201 extfield[15] = char(ctime >> 16);
1202 extfield[16] = char(ctime >> 24);
1206 bool b = (
device()->
write(buffer, bufferSize) == bufferSize);
1211 setErrorString(tr(
"Could not write to the archive. Disk full?"));
1217 if (d->m_compression == 0) {
1218 d->m_currentDev =
device();
1223 d->m_currentDev = compressionDevice;
1224 compressionDevice->setSkipHeaders();
1230 setErrorString(tr(
"Could not open compression device: %1").arg(d->m_currentDev->errorString()));
1238 if (d->m_currentFile->encoding() == 8) {
1240 (void)d->m_currentDev->write(
nullptr, 0);
1241 delete d->m_currentDev;
1244 d->m_currentDev =
nullptr;
1246 Q_ASSERT(d->m_currentFile);
1249 d->m_currentFile->setSize(size);
1250 int extra_field_len = 0;
1252 extra_field_len = 17;
1256 int csize =
device()->
pos() - d->m_currentFile->headerStart() - 30 - encodedName.
length() - extra_field_len;
1257 d->m_currentFile->setCompressedSize(csize);
1263 d->m_currentFile->setCRC32(d->m_crc);
1265 d->m_currentFile =
nullptr;
1283 perm |= QT_STAT_LNK;
1307void KZip::virtual_hook(
int id,
void *data)
1309 KArchive::virtual_hook(
id, data);
1314 Q_ASSERT(d->m_currentFile);
1315 Q_ASSERT(d->m_currentDev);
1316 if (!d->m_currentFile || !d->m_currentDev) {
1323 d->m_crc = crc32(d->m_crc, (
const Bytef *)data, size);
1325 qint64 written = d->m_currentDev->write(data, size);
1327 const bool ok = written == size;
1330 setErrorString(tr(
"Error writing data: %1").arg(d->m_currentDev->errorString()));
1348 d->m_extraField = ef;
1353 return d->m_extraField;
1362 KZipFileEntryPrivate()
1370 qint64 compressedSize;
1385 qint64 uncompressedSize,
1387 qint64 compressedSize)
1389 , d(new KZipFileEntryPrivate)
1392 d->encoding = encoding;
1393 d->compressedSize = compressedSize;
1401int KZipFileEntry::encoding()
const
1406qint64 KZipFileEntry::compressedSize()
const
1408 return d->compressedSize;
1413 d->compressedSize = compressedSize;
1418 d->headerStart = headerstart;
1421qint64 KZipFileEntry::headerStart()
const
1423 return d->headerStart;
1431void KZipFileEntry::setCRC32(
unsigned long crc32)
1456 KLimitedIODevice *limitedDev =
new KLimitedIODevice(archive()->device(),
position(), compressedSize());
1457 if (encoding() == 0 || compressedSize() == 0) {
1461 if (encoding() == 8) {
1475 qCCritical(KArchiveLog) <<
"This zip file contains files compressed with method" << encoding() <<
", this method is currently not supported by KZip,"
1476 <<
"please use a command-line tool to handle this file.";
Represents a directory entry in a KArchive.
bool removeEntryV2(KArchiveEntry *)
Removes an entry from the directory.
bool addEntryV2(KArchiveEntry *)
const KArchiveEntry * entry(const QString &name) const
Returns the entry in the archive with the given name.
A base class for entries in an KArchive.
mode_t permissions() const
The permissions and mode flags as returned by the stat() function in st_mode.
QString user() const
User who created the file.
virtual bool isDirectory() const
Checks whether the entry is a directory.
QDateTime date() const
Creation date of the file.
QString group() const
Group of the user who created the file.
QString name() const
Name of the file without path.
KArchiveFile(KArchive *archive, const QString &name, int access, const QDateTime &date, const QString &user, const QString &group, const QString &symlink, qint64 pos, qint64 size)
Creates a new file entry.
qint64 size() const
Size of the data.
qint64 position() const
Position of the data in the [uncompressed] archive.
QString errorString() const
Returns a description of the last error.
QIODevice * device() const
The underlying device.
virtual bool close()
Closes the archive.
virtual KArchiveDirectory * rootDir()
Retrieves or create the root directory.
bool finishWriting(qint64 size)
Call finishWriting after writing the data.
KArchive(const QString &fileName)
Base constructor (protected since this is a pure virtual class).
bool writeFile(const QString &name, QByteArrayView data, mode_t perm=0100644, const QString &user=QString(), const QString &group=QString(), const QDateTime &atime=QDateTime(), const QDateTime &mtime=QDateTime(), const QDateTime &ctime=QDateTime())
Writes a new file into the archive.
bool writeData(const char *data, qint64 size)
Write data into the current file - to be called after calling prepareWriting.
KArchiveDirectory * findOrCreate(const QString &path)
Ensures that path exists, create otherwise.
QIODevice::OpenMode mode() const
Returns the mode in which the archive was opened.
QString fileName() const
The name of the archive file, as passed to the constructor that takes a fileName, or an empty string ...
bool isOpen() const
Checks whether the archive is open.
void setErrorString(const QString &errorStr)
Sets error description.
A class for reading and writing compressed data onto a device (e.g.
void setSkipHeaders()
Call this let this device skip the gzip headers when reading/writing.
bool open(QIODevice::OpenMode mode) override
Open for reading or writing.
A KZipFileEntry represents a file in a zip archive.
KZipFileEntry(KZip *zip, const QString &name, int access, const QDateTime &date, const QString &user, const QString &group, const QString &symlink, const QString &path, qint64 start, qint64 uncompressedSize, int encoding, qint64 compressedSize)
Creates a new zip file entry.
~KZipFileEntry() override
Destructor.
void setCompressedSize(qint64 compressedSize)
Only used when writing.
unsigned long crc32() const
CRC: only used when writing.
const QString & path() const
Name with complete path - KArchiveFile::name() is the filename only (no path)
QIODevice * createDevice() const override
This method returns a QIODevice to read the file contents.
QByteArray data() const override
void setHeaderStart(qint64 headerstart)
Header start: only used when writing.
A class for reading / writing zip archives.
bool openArchive(QIODevice::OpenMode mode) override
Opens the archive for reading.
bool doWriteDir(const QString &name, const QString &user, const QString &group, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &ctime) override
Reimplemented from KArchive.
void setCompression(Compression c)
Call this before writeFile or prepareWriting, to define whether the next files to be written should b...
ExtraField
Describes the contents of the "extra field" for a given file in the Zip archive.
@ ModificationTime
Modification time ("extended timestamp" header)
bool doPrepareWriting(const QString &name, const QString &user, const QString &group, qint64 size, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &creationTime) override
Reimplemented from KArchive.
void setExtraField(ExtraField ef)
Call this before writeFile or prepareWriting, to define what the next file to be written should have ...
KZip(const QString &filename)
Creates an instance that operates on the given filename.
bool doFinishWriting(qint64 size) override
Write data to a file that has been created using prepareWriting().
~KZip() override
If the zip file is still opened, then it will be closed automatically by the destructor.
bool closeArchive() override
Closes the archive.
Compression
Describes the compression type for a given file in the Zip archive.
@ DeflateCompression
Deflate compression method.
@ NoCompression
Uncompressed.
Compression compression() const
The current compression mode that will be used for new files.
bool doWriteSymLink(const QString &name, const QString &target, const QString &user, const QString &group, mode_t perm, const QDateTime &atime, const QDateTime &mtime, const QDateTime &ctime) override
Reimplemented from KArchive.
bool doWriteData(const char *data, qint64 size) override
Write data to a file that has been created using prepareWriting().
ExtraField extraField() const
The current type of "extra field" that will be used for new files.
Q_SCRIPTABLE QString start(QString train="")
QString path(const QString &relativePath)
const char * constData() const const
qsizetype indexOf(QByteArrayView bv, qsizetype from) const const
qsizetype length() const const
QByteArray & remove(qsizetype pos, qsizetype len)
void resize(qsizetype newSize, char c)
qsizetype size() const const
bool startsWith(QByteArrayView bv) const const
bool startsWith(QByteArrayView bv) const const
QDateTime currentDateTime()
bool isValid() const const
qint64 toSecsSinceEpoch() const const
QString cleanPath(const QString &path)
QString decodeName(const QByteArray &localFileName)
QByteArray encodeName(const QString &fileName)
iterator insert(const Key &key, const T &value)
T value(const Key &key) const const
QByteArray peek(qint64 maxSize)
virtual qint64 pos() const const
QByteArray read(qint64 maxSize)
virtual bool seek(qint64 pos)
virtual qint64 size() const const
qint64 write(const QByteArray &data)
QString & append(QChar ch)
const QChar * constData() const const
bool endsWith(QChar c, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
qsizetype lastIndexOf(QChar ch, Qt::CaseSensitivity cs) const const
QString left(qsizetype n) const const
qsizetype length() const const
QString mid(qsizetype position, qsizetype n) const const