Files
KDDockWidgets/tests/fuzzer/Fuzzer.cpp
2019-10-11 17:04:26 +01:00

328 lines
9.3 KiB
C++

/*
This file is part of KDDockWidgets.
Copyright (C) 2019 Klarälvdalens Datakonsult AB, a KDAB Group company, info@kdab.com
Author: Sérgio Martins <sergio.martins@kdab.com>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
// We don't care about performance related checks in the tests
// clazy:excludeall=ctor-missing-parent-argument,missing-qobject-macro,range-loop,missing-typeinfo,detaching-member,function-args-by-ref,non-pod-global-static,reserve-candidates,qstring-allocations
#include "Fuzzer.h"
#include "DockRegistry_p.h"
#include "DockWidget.h"
#include "MainWindow.h"
#include <QJsonDocument>
#include <QString>
#include <QTest>
using namespace KDDockWidgets;
using namespace KDDockWidgets::Testing;
using namespace KDDockWidgets::Testing::Operations;
#define OPERATIONS_PER_TEST 200
static MainWindow* createMainWindow(const Fuzzer::MainWindowDescriptor &mwd)
{
static int count = 0;
count++;
auto mainWindow = new MainWindow(QStringLiteral("MainWindow-%1").arg(count), mwd.mainWindowOption);
mainWindow->setGeometry(mwd.geometry);
mainWindow->show();
return mainWindow;
}
static DockWidget* createDockWidget(const Fuzzer::DockWidgetDescriptor &dwd)
{
static int count = 0;
count++;
auto dockWidget = new DockWidget(QStringLiteral("DockWidget-%1").arg(count));
dockWidget->setWidget(new Testing::HostedWidget(dwd.minSize));
if (dwd.isFloating)
dockWidget->setGeometry(dwd.geometry);
if (dwd.isVisible)
dockWidget->show();
return dockWidget;
}
static void createLayout(const Fuzzer::Layout &layout)
{
for (const Fuzzer::MainWindowDescriptor &mwd : layout.mainWindows) {
createMainWindow(mwd);
}
for (const Fuzzer::DockWidgetDescriptor &dwd : layout.dockWidgets) {
createDockWidget(dwd);
}
}
void Fuzzer::runTest(const Test &test)
{
m_currentTest = test;
if (!DockRegistry::self()->isEmpty())
qFatal("There's dock widgets and the start runTest");
createLayout(test.initialLayout);
for (const auto &op : test.operations) {
op->execute();
//QTest::qWait(1000);
}
for (MainWindowBase *mw : DockRegistry::self()->mainwindows())
delete mw;
for (FloatingWindow *fw : DockRegistry::self()->nestedwindows())
delete fw;
for (DockWidgetBase *dw : DockRegistry::self()->dockwidgets())
delete dw;
if (!DockRegistry::self()->isEmpty())
qFatal("There's still dock widgets and the end of runTest");
}
Fuzzer::Fuzzer(Fuzzer::FuzzerConfig config, QObject *parent)
: QObject(parent)
, m_randomEngine(m_randomDevice())
, m_fuzzerConfig(config)
{
Testing::installFatalMessageHandler();
Testing::setWarningObserver(this);
}
Fuzzer::Layout Fuzzer::generateRandomLayout()
{
// for now we only support 1 main window
Fuzzer::Layout layout;
Fuzzer::MainWindowDescriptor mainWindow;
mainWindow.geometry = randomGeometry();
mainWindow.mainWindowOption = MainWindowOption_None; // TODO: Maybe test other options
layout.mainWindows << mainWindow;
std::uniform_int_distribution<> numDocksDistrib(1, 10); // TODO: Increase
const int numDockWidgets = numDocksDistrib(m_randomEngine);
for (int i = 0; i < numDockWidgets; ++i) {
layout.dockWidgets << generateRandomDockWidget();
}
return layout;
}
Fuzzer::DockWidgetDescriptor Fuzzer::generateRandomDockWidget()
{
Fuzzer::DockWidgetDescriptor dwd;
dwd.isFloating = getRandomBool(35);
dwd.isVisible = getRandomBool(70);
std::uniform_int_distribution<> minSizeDistriv(150, 600);
dwd.minSize.setWidth(minSizeDistriv(m_randomEngine));
dwd.minSize.setHeight(minSizeDistriv(m_randomEngine));
QPoint pos = getRandomPos();
std::uniform_int_distribution<> widthDistrib(dwd.minSize.width() + 50, dwd.minSize.width() + 600);
std::uniform_int_distribution<> heightDistrib(dwd.minSize.height() + 50, dwd.minSize.height() + 600);
dwd.geometry = QRect(pos, QSize(widthDistrib(m_randomEngine), heightDistrib(m_randomEngine)));
return dwd;
}
Fuzzer::DockWidgetDescriptor::List Fuzzer::generateRandomDockWidgets(int num)
{
Fuzzer::DockWidgetDescriptor::List dockWidgets;
for (int i = 0; i < num; ++i) {
dockWidgets << generateRandomDockWidget();
}
return dockWidgets;
}
bool Fuzzer::getRandomBool(int truePercentage)
{
std::uniform_int_distribution<> distrib(1, 100);
return distrib(m_randomEngine) < truePercentage;
}
Testing::AddDockWidgetParams Fuzzer::getRandomAddDockWidgetParams()
{
AddDockWidgetParams params;
params.dockWidget = getRandomDockWidget();
if (!params.dockWidget) {
qWarning() << Q_FUNC_INFO << "No dock widgets exist yet!";
return {};
}
params.mainWindow = getRandomMainWindow();
params.relativeTo = getRandomBool() ? getRandomRelativeTo(params.mainWindow, params.dockWidget)
: nullptr;
params.location = getRandomLocation();
params.addingOption = AddingOption_None; // TODO: Test the other ones
return params;
}
MainWindowBase *Fuzzer::getRandomMainWindow()
{
auto windows = DockRegistry::self()->mainwindows();
if (windows.isEmpty()) {
qWarning() << Q_FUNC_INFO << "No MainWindows exist yet!";
return nullptr;
}
return windows.first();
}
DockWidgetBase *Fuzzer::getRandomDockWidget(DockWidgetBase *excluding)
{
auto docks = DockRegistry::self()->dockwidgets();
if (excluding)
docks.removeOne(excluding);
if (docks.isEmpty())
return nullptr;
std::uniform_int_distribution<> locationDistrib(0, docks.size() - 1);
return docks[locationDistrib(m_randomEngine)];
}
DockWidgetBase *Fuzzer::getRandomRelativeTo(MainWindowBase *mainWindow, DockWidgetBase *excluding)
{
auto docks = DockRegistry::self()->dockwidgets();
DockWidgetBase::List candidates;
for (DockWidgetBase *dw : docks) {
if (dw != excluding && dw->window() == mainWindow)
candidates << dw;
}
if (candidates.isEmpty())
return nullptr;
std::uniform_int_distribution<> locationDistrib(0, candidates.size() - 1);
return candidates[locationDistrib(m_randomEngine)];
}
Location Fuzzer::getRandomLocation()
{
std::uniform_int_distribution<> locationDistrib(1, 4);
return Location(locationDistrib(m_randomEngine));
}
QPoint Fuzzer::getRandomPos()
{
std::uniform_int_distribution<> posDistrib(0, 500);
const int x = posDistrib(m_randomEngine);
const int y = posDistrib(m_randomEngine);
return {x, y};
}
OperationBase::Ptr Fuzzer::getRandomOperation()
{
Testing::Operations::OperationBase::Ptr operation;
std::uniform_int_distribution<> operationDistrib(OperationType_None + 1, OperationType_Count - 1);
auto operationType = OperationType(operationDistrib(m_randomEngine));
switch (operationType) {
case OperationType_CloseViaDockWidgetAPI:
operation = OperationBase::Ptr(new CloseViaDockWidgetAPI(this));
break;
case OperationType_HideViaDockWidgetAPI:
operation = OperationBase::Ptr(new HideViaDockWidgetAPI(this));
break;
case OperationType_ShowViaDockWidgetAPI:
operation = OperationBase::Ptr(new ShowViaDockWidgetAPI(this));
break;
case OperationType_AddDockWidget:
operation = OperationBase::Ptr(new AddDockWidget(this));
break;
default:
qFatal("Doesn't happen");
}
return operation;
}
Fuzzer::Test Fuzzer::generateRandomTest()
{
Fuzzer::Test test;
test.initialLayout = generateRandomLayout();
const int numOperationsPerTest = OPERATIONS_PER_TEST;
test.operations.reserve(numOperationsPerTest);
for (int i = 0; i < numOperationsPerTest; ++i)
test.operations << getRandomOperation();
return test;
}
Fuzzer::Test::List Fuzzer::generateRandomTests(int num)
{
Fuzzer::Test::List tests;
for (int i = 0; i < num; ++i) {
tests << generateRandomTest();
}
return tests;
}
void Fuzzer::fuzz()
{
const Fuzzer::Test::List tests = generateRandomTests(m_fuzzerConfig.numTests);
qDebug().noquote() << "Running" << QString("%1 tests...").arg(tests.size());
for (const auto &test : tests) {
runTest(test);
}
}
QRect Fuzzer::randomGeometry()
{
std::uniform_int_distribution<> posDistrib(0, 500);
std::uniform_int_distribution<> sizeDistrib(700, 1500);
const int width = posDistrib(m_randomEngine);
const int height = sizeDistrib(m_randomEngine);
QPoint pos = getRandomPos();
return QRect(pos, QSize(width, height));
}
void Fuzzer::onFatal()
{
// Tests failed! Let's dump
QVariantMap map = m_currentTest.toVariantMap();
QJsonDocument jsonDoc = QJsonDocument::fromVariant(map);
QFile file("fuzzer_dump.json");
if (file.open(QIODevice::WriteOnly)) {
file.write(jsonDoc.toJson());
}
file.close();
}