KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
project_archiver.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software: you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation, either version 3 of the License, or (at your
9 * option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include <memory>
21#include <wx/dir.h>
22#include <wx/filedlg.h>
23#include <wx/fs_zip.h>
24#include <wx/uri.h>
25#include <wx/wfstream.h>
26#include <wx/zipstrm.h>
27
28#include <core/arraydim.h>
29#include <macros.h>
31#include <reporter.h>
33#include <wxstream_helper.h>
34#include <wx/log.h>
35#include <kiplatform/io.h>
36
37#include <regex>
38#include <set>
39
40
41#define ZipFileExtension wxT( "zip" )
42
43class PROJECT_ARCHIVER_DIR_ZIP_TRAVERSER : public wxDirTraverser
44{
45public:
46 PROJECT_ARCHIVER_DIR_ZIP_TRAVERSER( const std::string& aExtRegex, const wxString& aPrjDir, wxZipOutputStream& aZipFileOutput,
47 REPORTER& aReporter, bool aVerbose ) :
48 m_zipFile( aZipFileOutput ),
49 m_prjDir( aPrjDir ),
50 m_fileExtRegex( aExtRegex, std::regex_constants::ECMAScript | std::regex_constants::icase ),
51 m_reporter( aReporter ),
52 m_errorOccurred( false ),
53 m_verbose( aVerbose )
54 {}
55
56 virtual wxDirTraverseResult OnFile( const wxString& aFilename ) override
57 {
58 if( std::regex_search( aFilename.ToStdString(), m_fileExtRegex ) )
59 {
60 addFileToZip( aFilename );
61
62 // Special processing for IBIS files to include the corresponding pkg file
63 if( aFilename.EndsWith( FILEEXT::IbisFileExtension ) )
64 {
65 wxFileName package( aFilename );
66 package.MakeRelativeTo( m_prjDir );
67 package.SetExt( wxS( "pkg" ) );
68
69 if( package.Exists() )
70 addFileToZip( package.GetFullPath() );
71 }
72 }
73
74 return wxDIR_CONTINUE;
75 }
76
77 virtual wxDirTraverseResult OnDir( const wxString& aDirname ) override
78 {
79 return wxDIR_CONTINUE;
80 }
81
82 unsigned long GetUncompressedBytes() const
83 {
85 }
86
87 bool GetErrorOccurred() const
88 {
89 return m_errorOccurred;
90 }
91
92private:
93 void addFileToZip( const wxString& aFilename)
94 {
95 wxString msg;
96 wxFileSystem fsfile;
97
98 wxFileName curr_fn( aFilename );
99 curr_fn.MakeRelativeTo( m_prjDir );
100
101 wxString currFilename = curr_fn.GetFullPath();
102
103 // Read input file and add it to the zip file:
104 wxFSFile* infile = fsfile.OpenFile( currFilename );
105
106 if( infile )
107 {
108 m_zipFile.PutNextEntry( currFilename, infile->GetModificationTime() );
109 infile->GetStream()->Read( m_zipFile );
110 m_zipFile.CloseEntry();
111
112 m_uncompressedBytes += infile->GetStream()->GetSize();
113
114 if( m_verbose )
115 {
116 msg.Printf( _( "Archived file '%s'." ), currFilename );
118 }
119
120 delete infile;
121 }
122 else
123 {
124 if( m_verbose )
125 {
126 msg.Printf( _( "Failed to archive file '%s'." ), currFilename );
128 }
129
130 m_errorOccurred = true;
131 }
132 }
133
134private:
135 wxZipOutputStream& m_zipFile;
136
137 wxString m_prjDir;
138 std::regex m_fileExtRegex;
140
141 bool m_errorOccurred; // True if an error archiving the file
142 bool m_verbose; // True to enable verbose logging
143
144 // Keep track of how many bytes would have been used without compression
145 unsigned long m_uncompressedBytes = 0;
146};
147
149{
150}
151
152bool PROJECT_ARCHIVER::AreZipArchivesIdentical( const wxString& aZipFileA,
153 const wxString& aZipFileB, REPORTER& aReporter )
154{
155 wxFFileInputStream streamA( aZipFileA );
156 wxFFileInputStream streamB( aZipFileB );
157
158 if( !streamA.IsOk() || !streamB.IsOk() )
159 {
160 aReporter.Report( _( "Could not open archive file." ), RPT_SEVERITY_ERROR );
161 return false;
162 }
163
164 wxZipInputStream zipStreamA = wxZipInputStream( streamA );
165 wxZipInputStream zipStreamB = wxZipInputStream( streamB );
166
167 std::set<wxUint32> crcsA;
168 std::set<wxUint32> crcsB;
169
170
171 for( wxZipEntry* entry = zipStreamA.GetNextEntry(); entry; entry = zipStreamA.GetNextEntry() )
172 {
173 crcsA.insert( entry->GetCrc() );
174 }
175
176 for( wxZipEntry* entry = zipStreamB.GetNextEntry(); entry; entry = zipStreamB.GetNextEntry() )
177 {
178 crcsB.insert( entry->GetCrc() );
179 }
180
181 return crcsA == crcsB;
182}
183
184
185// Unarchive Files code comes from wxWidgets sample/archive/archive.cpp
186bool PROJECT_ARCHIVER::Unarchive( const wxString& aSrcFile, const wxString& aDestDir,
187 REPORTER& aReporter )
188{
189 wxFFileInputStream stream( aSrcFile );
190
191 if( !stream.IsOk() )
192 {
193 aReporter.Report( _( "Could not open archive file." ), RPT_SEVERITY_ERROR );
194 return false;
195 }
196
197 const wxArchiveClassFactory* archiveClassFactory =
198 wxArchiveClassFactory::Find( aSrcFile, wxSTREAM_FILEEXT );
199
200 if( !archiveClassFactory )
201 {
202 aReporter.Report( _( "Invalid archive file format." ), RPT_SEVERITY_ERROR );
203 return false;
204 }
205
206 std::unique_ptr<wxArchiveInputStream> archiveStream( archiveClassFactory->NewStream( stream ) );
207
208 wxString fileStatus;
209
210 for( wxArchiveEntry* entry = archiveStream->GetNextEntry(); entry;
211 entry = archiveStream->GetNextEntry() )
212 {
213 fileStatus.Printf( _( "Extracting file '%s'." ), entry->GetName() );
214 aReporter.Report( fileStatus, RPT_SEVERITY_INFO );
215
216 wxString fullname = aDestDir + entry->GetName();
217
218 // Ensure the target directory exists and create it if not
219 wxString t_path = wxPathOnly( fullname );
220
221 if( !wxDirExists( t_path ) )
222 {
223 wxFileName::Mkdir( t_path, wxS_DIR_DEFAULT, wxPATH_MKDIR_FULL );
224 }
225
226 // Directory entries need only be created, not extracted (0 size)
227 if( entry->IsDir() )
228 continue;
229
230
231 wxTempFileOutputStream outputFileStream( fullname );
232
233 if( CopyStreamData( *archiveStream, outputFileStream, entry->GetSize() ) )
234 outputFileStream.Commit();
235 else
236 aReporter.Report( _( "Error extracting file!" ), RPT_SEVERITY_ERROR );
237
238 // Now let's set the filetimes based on what's in the zip
239 wxFileName outputFileName( fullname );
240 wxDateTime fileTime = entry->GetDateTime();
241 // For now we set access, mod, create to the same datetime
242 // create (third arg) is only used on Windows
243 outputFileName.SetTimes( &fileTime, &fileTime, &fileTime );
244 }
245
246 aReporter.Report( wxT( "Extracted project." ), RPT_SEVERITY_INFO );
247 return true;
248}
249
250
251bool PROJECT_ARCHIVER::Archive( const wxString& aSrcDir, const wxString& aDestFile,
252 REPORTER& aReporter, bool aVerbose, bool aIncludeExtraFiles )
253{
254
255#define EXT( ext ) "\\." + ext + "|"
256#define NAME( name ) name + "|"
257#define EXT_NO_PIPE( ext ) "\\." + ext
258#define NAME_NO_PIPE( name ) name
259
260 // List of file extensions that are always archived
261 std::string fileExtensionRegex = "("
271 EXT( FILEEXT::JsonFileExtension ) // for design blocks
276
277 // List of additional file extensions that are only archived when aIncludeExtraFiles is true
278 if( aIncludeExtraFiles )
279 {
280 fileExtensionRegex += "|"
289 EXT( FILEEXT::StepFileExtension ) // 3d files
290 EXT( FILEEXT::VrmlFileExtension ) // 3d files
291 EXT( FILEEXT::GerberFileExtensionsRegex ) // Gerber files (g?, g??, .gm12 (from protel export))
292 EXT( FILEEXT::GerberJobFileExtension ) // Gerber job files
293 EXT( FILEEXT::FootprintPlaceFileExtension ) // Our position files
294 EXT( FILEEXT::DrillFileExtension ) // Fab drill files
295 EXT( "nc" ) // Fab drill files
296 EXT( "xnc" ) // Fab drill files
303 EXT( FILEEXT::SpiceFileExtension ) // SPICE files
305 EXT( FILEEXT::SpiceModelFileExtension ) // SPICE files
307 }
308
309 fileExtensionRegex += ")";
310
311#undef EXT
312#undef NAME
313#undef EXT_NO_PIPE
314#undef NAME_NO_PIPE
315
316 bool success = true;
317 wxString msg;
318 wxString oldCwd = wxGetCwd();
319
320 wxFileName sourceDir( aSrcDir );
321
322 wxSetWorkingDirectory( sourceDir.GetFullPath() );
323
324 wxFFileOutputStream ostream( aDestFile );
325
326 if( !ostream.IsOk() ) // issue to create the file. Perhaps not writable dir
327 {
328 msg.Printf( _( "Failed to create file '%s'." ), aDestFile );
329 aReporter.Report( msg, RPT_SEVERITY_ERROR );
330 return false;
331 }
332
333 wxZipOutputStream zipstream( ostream, -1, wxConvUTF8 );
334
335 wxDir projectDir( aSrcDir );
336 wxString currFilename;
337 wxArrayString files;
338
339 if( !projectDir.IsOpened() )
340 {
341 if( aVerbose )
342 {
343 msg.Printf( _( "Error opening directory: '%s'." ), aSrcDir );
344 aReporter.Report( msg, RPT_SEVERITY_ERROR );
345 }
346
347 wxSetWorkingDirectory( oldCwd );
348 return false;
349 }
350
351 try
352 {
353 PROJECT_ARCHIVER_DIR_ZIP_TRAVERSER traverser( fileExtensionRegex, aSrcDir, zipstream,
354 aReporter, aVerbose );
355
356 projectDir.Traverse( traverser );
357
358 success = !traverser.GetErrorOccurred();
359
360 auto reportSize =
361 []( unsigned long aSize ) -> wxString
362 {
363 constexpr float KB = 1024.0;
364 constexpr float MB = KB * 1024.0;
365
366 if( aSize >= MB )
367 return wxString::Format( wxT( "%0.2f MB" ), aSize / MB );
368 else if( aSize >= KB )
369 return wxString::Format( wxT( "%0.2f KB" ), aSize / KB );
370 else
371 return wxString::Format( wxT( "%lu bytes" ), aSize );
372 };
373
374 size_t zipBytesCnt = ostream.GetSize();
375 unsigned long uncompressedBytes = traverser.GetUncompressedBytes();
376
377 if( zipstream.Close() )
378 {
379 msg.Printf( _( "Zip archive '%s' created (%s uncompressed, %s compressed)." ),
380 aDestFile,
381 reportSize( uncompressedBytes ),
382 reportSize( zipBytesCnt ) );
383 aReporter.Report( msg, RPT_SEVERITY_INFO );
384 }
385 else
386 {
387 msg.Printf( wxT( "Failed to create file '%s'." ), aDestFile );
388 aReporter.Report( msg, RPT_SEVERITY_ERROR );
389 success = false;
390 }
391 }
392 catch( const std::regex_error& e )
393 {
394 // Something bad happened here with the regex
395 wxASSERT_MSG( false, e.what() );
396
397 if( aVerbose )
398 {
399 msg.Printf( _( "Error: '%s'." ), e.what() );
400 aReporter.Report( msg, RPT_SEVERITY_ERROR );
401 }
402
403 success = false;
404 }
405
406 wxSetWorkingDirectory( oldCwd );
407 return success;
408}
virtual wxDirTraverseResult OnDir(const wxString &aDirname) override
PROJECT_ARCHIVER_DIR_ZIP_TRAVERSER(const std::string &aExtRegex, const wxString &aPrjDir, wxZipOutputStream &aZipFileOutput, REPORTER &aReporter, bool aVerbose)
virtual wxDirTraverseResult OnFile(const wxString &aFilename) override
void addFileToZip(const wxString &aFilename)
unsigned long GetUncompressedBytes() const
static bool Archive(const wxString &aSrcDir, const wxString &aDestFile, REPORTER &aReporter, bool aVerbose=true, bool aIncludeExtraFiles=false)
Create an archive of the project.
static bool Unarchive(const wxString &aSrcFile, const wxString &aDestDir, REPORTER &aReporter)
Extract an archive of the current project over existing files.
static bool AreZipArchivesIdentical(const wxString &aZipFileA, const wxString &aZipFileB, REPORTER &aReporter)
Compare the CRCs of all the files in zip archive to determine whether the archives are identical.
A pure virtual class used to derive REPORTER objects from.
Definition: reporter.h:73
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED)
Report a string with a given severity.
Definition: reporter.h:102
#define _(s)
static const std::string LegacySchematicFileExtension
static const wxString GerberFileExtensionsRegex
static const std::string NetlistFileExtension
static const std::string SymbolLibraryTableFileName
static const std::string GerberJobFileExtension
static const std::string StepFileAbrvExtension
static const std::string WorkbookFileExtension
static const std::string ReportFileExtension
static const std::string ProjectFileExtension
static const std::string FootprintPlaceFileExtension
static const std::string JsonFileExtension
static const std::string LegacyPcbFileExtension
static const std::string LegacyProjectFileExtension
static const std::string ProjectLocalSettingsFileExtension
static const std::string KiCadSchematicFileExtension
static const std::string LegacySymbolLibFileExtension
static const std::string DesignBlockLibraryTableFileName
static const std::string KiCadSymbolLibFileExtension
static const std::string SpiceFileExtension
static const std::string PdfFileExtension
static const std::string TextFileExtension
static const std::string FootprintLibraryTableFileName
static const std::string DrawingSheetFileExtension
static const std::string IbisFileExtension
static const std::string IpcD356FileExtension
static const std::string KiCadJobSetFileExtension
static const std::string LegacyFootprintLibPathExtension
static const std::string PythonFileExtension
static const std::string StepFileExtension
static const std::string LegacySymbolDocumentFileExtension
static const std::string FootprintAssignmentFileExtension
static const std::string DrillFileExtension
static const std::string SpiceSubcircuitFileExtension
static const std::string SpiceModelFileExtension
static const std::string DesignRulesFileExtension
static const std::string VrmlFileExtension
static const std::string KiCadFootprintFileExtension
static const std::string KiCadPcbFileExtension
This file contains miscellaneous commonly used macros and functions.
STL namespace.
#define EXT(ext)
#define NAME(name)
#define EXT_NO_PIPE(ext)
#define NAME_NO_PIPE(name)
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_INFO
Definition of file extensions used in Kicad.
static bool CopyStreamData(wxInputStream &inputStream, wxOutputStream &outputStream, wxFileOffset size)