41 if ( !QFileInfo::exists( zipFilename ) )
46 else if ( zipFilename.isEmpty() )
51 else if ( !QDir( dir ).exists( dir ) )
56 else if ( !QFileInfo( dir ).isDir() )
61 else if ( !QFileInfo( dir ).isWritable() )
68 const QByteArray fileNamePtr = zipFilename.toUtf8();
69 struct zip *z = zip_open( fileNamePtr.constData(), checkConsistency ? ZIP_CHECKCONS : 0, &rc );
71 if ( rc == ZIP_ER_OK && z )
73 const int count = zip_get_num_entries( z, ZIP_FL_UNCHANGED );
78 for (
int i = 0; i < count; i++ )
80 zip_stat_index( z, i, 0, &stat );
81 const size_t len = stat.size;
83 struct zip_file *file = zip_fopen_index( z, i, 0 );
84 const std::unique_ptr< char[] > buf(
new char[len] );
85 if ( zip_fread( file, buf.get(), len ) != -1 )
87 const QString fileName( stat.name );
88 if ( fileName.endsWith(
"/" ) )
93 const QFileInfo newFile( QDir( dir ), fileName );
95 if ( !QString( QDir::cleanPath( newFile.absolutePath() ) + u
"/"_s ).startsWith( QDir( dir ).absolutePath() + u
"/"_s ) )
97 QgsMessageLog::logMessage( QObject::tr(
"Skipped file %1 outside of the directory %2" ).arg( newFile.absoluteFilePath(), QDir( dir ).absolutePath() ) );
102 if ( !newFile.absoluteDir().exists() )
104 if ( !QDir( dir ).mkpath( newFile.absolutePath() ) )
108 QFile outFile( newFile.absoluteFilePath() );
109 if ( !outFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
115 outFile.write( buf.get(), len );
118 files.append( newFile.absoluteFilePath() );
139 QgsMessageLog::logMessage( QObject::tr(
"Error opening zip archive: '%1' (Error code: %2)" ).arg( z ? zip_strerror( z ) : zipFilename ).arg( rc ) );
148 if ( zipFilename.isEmpty() )
155 const QByteArray zipFileNamePtr = zipFilename.toUtf8();
156 struct zip *z = zip_open( zipFileNamePtr.constData(), overwrite ? ( ZIP_CREATE | ZIP_TRUNCATE ) : ZIP_CREATE, &rc );
158 if ( rc == ZIP_ER_OK && z )
160 for (
const auto &file :
files )
162 const QFileInfo fileInfo( file );
163 if ( !fileInfo.exists() )
170 const QByteArray fileNamePtr = file.toUtf8();
171 zip_source *src = zip_source_file( z, fileNamePtr.constData(), 0, 0 );
174 const QByteArray fileInfoPtr = fileInfo.fileName().toUtf8();
175#if LIBZIP_VERSION_MAJOR < 1
176 rc = ( int ) zip_add( z, fileInfoPtr.constData(), src );
178 rc = ( int ) zip_file_add( z, fileInfoPtr.constData(), src, 0 );
199 QgsMessageLog::logMessage( QObject::tr(
"Error creating zip archive '%1': %2" ).arg( zipFilename, z ? zip_strerror( z ) : zipFilename ) );
213 unsigned char *bytesInPtr =
reinterpret_cast<unsigned char *
>(
const_cast<char *
>( bytesIn ) );
214 uint bytesInLeft =
static_cast<uint
>( size );
216 const uint CHUNK = 16384;
217 unsigned char out[CHUNK];
218 const int DEC_MAGIC_NUM_FOR_GZIP = 16;
222 strm.zalloc = Z_NULL;
224 strm.opaque = Z_NULL;
226 strm.next_in = Z_NULL;
228 int ret = inflateInit2( &strm, MAX_WBITS + DEC_MAGIC_NUM_FOR_GZIP );
235 while ( ret != Z_STREAM_END )
238 const uint bytesToProcess = std::min( CHUNK, bytesInLeft );
239 strm.next_in = bytesInPtr;
240 strm.avail_in = bytesToProcess;
241 bytesInPtr += bytesToProcess;
242 bytesInLeft -= bytesToProcess;
244 if ( bytesToProcess == 0 )
250 strm.avail_out = CHUNK;
252 ret = inflate( &strm, Z_NO_FLUSH );
253 Q_ASSERT( ret != Z_STREAM_ERROR );
254 if ( ret == Z_NEED_DICT || ret == Z_DATA_ERROR || ret == Z_MEM_ERROR )
259 const unsigned have = CHUNK - strm.avail_out;
260 bytesOut.append( QByteArray::fromRawData(
reinterpret_cast<const char *
>( out ),
static_cast<int>( have ) ) );
261 }
while ( strm.avail_out == 0 );
265 return ret == Z_STREAM_END;
270 unsigned char *bytesInPtr =
reinterpret_cast<unsigned char *
>(
const_cast<char *
>( bytesIn.constData() ) );
271 const uint bytesInLeft =
static_cast<uint
>( bytesIn.count() );
273 const uint CHUNK = 16384;
274 unsigned char out[CHUNK];
275 const int DEC_MAGIC_NUM_FOR_GZIP = 16;
279 strm.zalloc = Z_NULL;
281 strm.opaque = Z_NULL;
283 int ret = deflateInit2( &strm, Z_DEFAULT_COMPRESSION, Z_DEFLATED, MAX_WBITS + DEC_MAGIC_NUM_FOR_GZIP, 8, Z_DEFAULT_STRATEGY );
287 strm.avail_in = bytesInLeft;
288 strm.next_in = bytesInPtr;
294 strm.avail_out = CHUNK;
296 ret = deflate( &strm, Z_FINISH );
297 Q_ASSERT( ret != Z_STREAM_ERROR );
299 const unsigned have = CHUNK - strm.avail_out;
300 bytesOut.append( QByteArray::fromRawData(
reinterpret_cast<const char *
>( out ),
static_cast<int>( have ) ) );
301 }
while ( strm.avail_out == 0 );
302 Q_ASSERT( ret == Z_STREAM_END );
311 if (
zip.isEmpty() && !QFileInfo::exists(
zip ) )
313 return QStringList();
318 const QByteArray fileNamePtr =
zip.toUtf8();
319 struct zip *z = zip_open( fileNamePtr.constData(), 0, &rc );
321 if ( rc == ZIP_ER_OK && z )
323 const int count = zip_get_num_entries( z, ZIP_FL_UNCHANGED );
326 struct zip_stat stat;
328 for (
int i = 0; i < count; i++ )
330 zip_stat_index( z, i, 0, &stat );
331 files << QString( stat.name );
343 if ( !QFileInfo::exists( zipFilename ) )
349 if ( filenameInZip.isEmpty() )
356 const QByteArray zipFilenamePtr = zipFilename.toUtf8();
357 struct zip *z = zip_open( zipFilenamePtr.constData(), 0, &err );
361 zip_error_init_with_code( &error, err );
362 QgsMessageLog::logMessage( QObject::tr(
"Error opening zip archive '%1': %2" ).arg( zipFilename, zip_error_strerror( &error ) ) );
363 zip_error_fini( &error );
367 const QByteArray filenameInZipPtr = filenameInZip.toUtf8();
369 zip_stat_init( &st );
370 if ( zip_stat( z, filenameInZipPtr.constData(), 0, &st ) != 0 )
372 QgsMessageLog::logMessage( QObject::tr(
"File '%1' not found in zip archive '%2': %3" ).arg( filenameInZip, zipFilename, zip_strerror( z ) ) );
377 zip_file *zf = zip_fopen( z, filenameInZipPtr.constData(), 0 );
380 QgsMessageLog::logMessage( QObject::tr(
"Could not open file '%1' in zip archive '%2': %3" ).arg( filenameInZip, zipFilename, zip_strerror( z ) ) );
385 bytesOut.resize(
static_cast<int>( st.size ) );
386 zip_int64_t readBytes = zip_fread( zf, bytesOut.data(), st.size );
392 if (
static_cast<zip_uint64_t
>( readBytes ) != st.size )
394 QgsMessageLog::logMessage( QObject::tr(
"Error reading file '%1' from zip archive '%2'." ).arg( filenameInZip, zipFilename ) );
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true, const char *file=__builtin_FILE(), const char *function=__builtin_FUNCTION(), int line=__builtin_LINE(), Qgis::StringFormat format=Qgis::StringFormat::PlainText)
Adds a message to the log instance (and creates it if necessary).
static bool extractFileFromZip(const QString &zipFilename, const QString &filenameInZip, QByteArray &bytesOut)
Extracts a file from a zip archive, returns true on success.
static bool unzip(const QString &zip, const QString &dir, QStringList &files, bool checkConsistency=true)
Unzip a zip file in an output directory.