KiCad PCB EDA Suite
Loading...
Searching...
No Matches
board_test_utils.h
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
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24
25#pragma once
26
27#include <filesystem>
28#include <map>
29#include <memory>
30#include <mutex>
31#include <functional>
32#include <optional>
33#include <string>
34
35#include <wx/string.h>
36
37#include <boost/test/unit_test.hpp>
38
39#include <reporter.h>
40#include <core/typeinfo.h>
41#include <tool/tool_manager.h>
42#include <pcb_io/pcb_io.h>
43#include <pcb_track_types.h>
44
45class BOARD;
46class BOARD_ITEM;
47class FOOTPRINT;
48class KIID;
49class PCB_TEXT;
50class PCB_SHAPE;
51class ZONE;
52class PAD;
53class SHAPE_POLY_SET;
55
56
57/*
58 * Boost test printers
59 */
60std::ostream& boost_test_print_type( std::ostream& os, const VIATYPE& aViaType );
61
62
63namespace KI_TEST
64{
65class DUMMY_TOOL : public TOOL_BASE
66{
67public:
69 TOOL_BASE( BATCH, TOOL_MANAGER::MakeToolId( "" ), "testframework.dummytool" )
70 {};
71
72 void Reset( RESET_REASON aReason ) override {}
73};
74
75
85{
86public:
91 TEMPORARY_DIRECTORY( const std::string& aNamePrefix, const std::string aSuffix )
92 {
93 int i = 0;
94
95 // Find a unique directory name
96 while( true )
97 {
98 m_path = std::filesystem::temp_directory_path()
99 / ( aNamePrefix + std::to_string( i ) + aSuffix );
100
101 if( !std::filesystem::exists( m_path ) )
102 break;
103
104 i++;
105 }
106
107 wxASSERT( !std::filesystem::exists( m_path ) );
108 std::filesystem::create_directories( m_path );
109 }
110
111 ~TEMPORARY_DIRECTORY() { std::filesystem::remove_all( m_path ); }
112
113 const std::filesystem::path& GetPath() const { return m_path; }
114
115private:
116 std::filesystem::path m_path;
117};
118
119
132{
133public:
134 BOARD_DUMPER();
135
136 void DumpBoardToFile( BOARD& aBoard, const std::string& aName ) const;
137
138 const bool m_dump_boards;
139};
140
141
143{
144public:
145 enum COLOR
146 {
147 RED = 0,
150 };
151
153
154 const wxString& GetLogContents() const
155 {
156 return m_logContents;
157 }
158
159 void PrintProgress( const wxString& aMessage )
160 {
162 return;
163
166
167 printf( "%s", (const char*) aMessage.c_str() );
168 fflush( stdout );
169
171 }
172
173
174 void Print( const wxString& aMessage )
175 {
178
180 {
181 printf( "%s", (const char*) aMessage.c_str() );
182 fflush( stdout );
183 }
184
185 m_logContents.append( aMessage );
187 }
188
189
190 void SetColor( COLOR color )
191 {
193 return;
194
195 std::map<COLOR, wxString> colorMap =
196 {
197 { RED, "\033[0;31m" },
198 { GREEN, "\033[0;32m" },
199 { DEFAULT, "\033[0;37m" }
200 };
201
202 printf( "%s", (const char*) colorMap[color].c_str() );
203 fflush( stdout );
204 }
205
206
207private:
209 {
211 return;
212
213 printf( "\r\033[K" );
214 fflush( stdout );
215 }
216
219 std::mutex m_lock;
221};
222
223
225{
226public:
229
230
231 virtual REPORTER& Report( const wxString& aText,
232 SEVERITY aSeverity = RPT_SEVERITY_UNDEFINED ) override
233 {
234 switch( aSeverity )
235 {
237 m_log->SetColor( CONSOLE_LOG::RED );
238 m_log->Print( "ERROR | " );
239 break;
240
241 default: m_log->SetColor( CONSOLE_LOG::DEFAULT ); m_log->Print( " | " );
242 }
243
244 m_log->SetColor( CONSOLE_LOG::DEFAULT );
245 m_log->Print( aText + "\n" );
246 return *this;
247 }
248
249 virtual bool HasMessage() const override { return true; }
250
251private:
253};
254
255
257{
259
260 // These tests may well test specific versions of the board file format,
261 // so don't let it change accidentally in the files (e.g. by resaving in newer KiCad)!
262 std::optional<int> m_ExpectedBoardVersion;
263
264 // Be default, printing the board file is sufficent to identify the test case
265 friend std::ostream& operator<<( std::ostream& os, const BOARD_LOAD_TEST_CASE& aTestCase )
266 {
267 os << aTestCase.m_BoardFileRelativePath;
268 return os;
269 }
270};
271
272
273void LoadBoard( SETTINGS_MANAGER& aSettingsManager, const wxString& aRelPath,
274 std::unique_ptr<BOARD>& aBoard );
275
286BOARD_ITEM& RequireBoardItemWithTypeAndId( const BOARD& aBoard, KICAD_T aItemType,
287 const KIID& aID );
288
300void LoadAndTestBoardFile( const wxString aRelativePath, bool aRoundtrip,
301 std::function<void( BOARD& )> aBoardTestFunction,
302 std::optional<int> aExpectedBoardVersion = std::nullopt );
303
307void LoadAndTestFootprintFile( const wxString& aLibRelativePath, const wxString& aFpName,
308 bool aRoundtrip,
309 std::function<void( FOOTPRINT& )> aFootprintTestFunction,
310 std::optional<int> aExpectedFootprintVersion );
311
312
313void FillZones( BOARD* m_board );
314
315
319void CheckFootprint( const FOOTPRINT* expected, const FOOTPRINT* fp );
320
321void CheckFpPad( const PAD* expected, const PAD* pad );
322
323void CheckFpText( const PCB_TEXT* expected, const PCB_TEXT* text );
324
325void CheckFpShape( const PCB_SHAPE* expected, const PCB_SHAPE* shape );
326
327void CheckFpZone( const ZONE* expected, const ZONE* zone );
328
329void CheckShapePolySet( const SHAPE_POLY_SET* expected, const SHAPE_POLY_SET* polyset );
330
334void PrintBoardStats( const BOARD* aBoard, const std::string& aBoardName );
335
336
341{
342public:
348
350 m_errorCount( 0 ),
351 m_warningCount( 0 ),
352 m_infoCount( 0 )
353 {
354 }
355
356 REPORTER& Report( const wxString& aText, SEVERITY aSeverity = RPT_SEVERITY_UNDEFINED ) override;
357
358 bool HasMessage() const override { return !m_messages.empty(); }
359
360 EDA_UNITS GetUnits() const override { return EDA_UNITS::MM; }
361
362 void Clear() override
363 {
364 m_messages.clear();
365 m_errorCount = 0;
366 m_warningCount = 0;
367 m_infoCount = 0;
368 }
369
370 void PrintAllMessages( const std::string& aContext ) const;
371
372 int GetErrorCount() const { return m_errorCount; }
373 int GetWarningCount() const { return m_warningCount; }
374 const std::vector<MESSAGE>& GetMessages() const { return m_messages; }
375
376private:
377 std::vector<MESSAGE> m_messages;
381};
382
383
388std::unique_ptr<BOARD> LoadBoardWithCapture( PCB_IO& aIoPlugin, const std::string& aFilePath, REPORTER* aReporter );
389
390
399{
400public:
408 BOARD* LoadAndCache( const std::string& aFilePath, REPORTER* aReporter );
409
414 BOARD* GetCachedBoard( const std::string& aFilePath );
415
416protected:
417 BOARD* getCachedBoard( PCB_IO& aIoPlugin, const std::string& aFilePath, bool aForceReload, REPORTER* aReporter );
418
422 virtual BOARD* getCachedBoard( const std::string& aFilePath, bool aForceReload, REPORTER* aReporter ) = 0;
423
424private:
425 std::map<std::string, std::unique_ptr<BOARD>> m_boardCache;
426};
427
428} // namespace KI_TEST
std::ostream & boost_test_print_type(std::ostream &os, const VIATYPE &aViaType)
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:84
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:323
Definition kiid.h:48
void DumpBoardToFile(BOARD &aBoard, const std::string &aName) const
Manager for caching loaded boards in memory, to avoid repeatedly loading and parsing the same board.
std::map< std::string, std::unique_ptr< BOARD > > m_boardCache
BOARD * getCachedBoard(PCB_IO &aIoPlugin, const std::string &aFilePath, bool aForceReload, REPORTER *aReporter)
BOARD * LoadAndCache(const std::string &aFilePath, REPORTER *aReporter)
Load (or reload) board for the given file path and send the load messages to the given reporter.
BOARD * GetCachedBoard(const std::string &aFilePath)
Get a cached board for the given file path, or load it if not already cached, without forcing a reloa...
virtual BOARD * getCachedBoard(const std::string &aFilePath, bool aForceReload, REPORTER *aReporter)=0
Implementation of the board loader (probably plug in the PCB_IO plugin here)
bool HasMessage() const override
Returns true if any messages were reported.
void PrintAllMessages(const std::string &aContext) const
const std::vector< MESSAGE > & GetMessages() const
REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED) override
Report a string with a given severity.
EDA_UNITS GetUnits() const override
std::vector< MESSAGE > m_messages
void PrintProgress(const wxString &aMessage)
void SetColor(COLOR color)
void Print(const wxString &aMessage)
const wxString & GetLogContents() const
virtual bool HasMessage() const override
Returns true if any messages were reported.
CONSOLE_MSG_REPORTER(CONSOLE_LOG *log)
virtual REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED) override
Report a string with a given severity.
void Reset(RESET_REASON aReason) override
Bring the tool to a known, initial state.
const std::filesystem::path & GetPath() const
TEMPORARY_DIRECTORY(const std::string &aNamePrefix, const std::string aSuffix)
Create a temporary directory with a given prefix and suffix.
std::filesystem::path m_path
Definition pad.h:65
A base class that BOARD loading and saving plugins should derive from.
Definition pcb_io.h:79
A pure virtual class used to derive REPORTER objects from.
Definition reporter.h:75
REPORTER()
Definition reporter.h:77
Represent a set of closed polygons.
friend class TOOL_MANAGER
Definition tool_base.h:156
RESET_REASON
Determine the reason of reset for a tool.
Definition tool_base.h:78
TOOL_BASE(TOOL_TYPE aType, TOOL_ID aId, const std::string &aName=std::string(""))
Definition tool_base.h:68
Handle a list of polygons defining a copper zone.
Definition zone.h:74
EDA_UNITS
Definition eda_units.h:48
void LoadBoard(SETTINGS_MANAGER &aSettingsManager, const wxString &aRelPath, std::unique_ptr< BOARD > &aBoard)
void CheckFootprint(const FOOTPRINT *expected, const FOOTPRINT *fp)
Helper method to check if two footprints are semantically the same.
void PrintBoardStats(const BOARD *aBoard, const std::string &aBoardName)
Print detailed board statistics for debugging using test-framework logging.
void FillZones(BOARD *m_board)
std::unique_ptr< BOARD > LoadBoardWithCapture(PCB_IO &aIoPlugin, const std::string &aFilePath, REPORTER *aReporter)
Attempt to load an board with a given IO plugin, capturing all reporter messages.
void CheckFpShape(const PCB_SHAPE *expected, const PCB_SHAPE *shape)
void LoadAndTestFootprintFile(const wxString &aLibRelativePath, const wxString &aFpName, bool aRoundtrip, std::function< void(FOOTPRINT &)> aFootprintTestFunction, std::optional< int > aExpectedFootprintVersion)
Same as LoadAndTestBoardFile, but for footprints.
void CheckFpPad(const PAD *expected, const PAD *pad)
void CheckFpZone(const ZONE *expected, const ZONE *zone)
void CheckFpText(const PCB_TEXT *expected, const PCB_TEXT *text)
void LoadAndTestBoardFile(const wxString aRelativePath, bool aRoundtrip, std::function< void(BOARD &)> aBoardTestFunction, std::optional< int > aExpectedBoardVersion)
Perform "some test" on a board file loaded from the path, then optionally save and reload and run the...
void CheckShapePolySet(const SHAPE_POLY_SET *expected, const SHAPE_POLY_SET *polyset)
BOARD_ITEM & RequireBoardItemWithTypeAndId(const BOARD &aBoard, KICAD_T aItemType, const KIID &aID)
Get an item from the given board with a certain type and UUID.
Declarations of types for tracks and vias.
VIATYPE
SEVERITY
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_UNDEFINED
std::optional< int > m_ExpectedBoardVersion
friend std::ostream & operator<<(std::ostream &os, const BOARD_LOAD_TEST_CASE &aTestCase)
VECTOR3I expected(15, 30, 45)
@ BATCH
Tool that runs in the background without any user intervention.
Definition tool_base.h:52
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition typeinfo.h:75