2023-08-02 10:36:17 +02:00
|
|
|
/*
|
|
|
|
* Cppcheck - A tool for static C/C++ code analysis
|
|
|
|
* Copyright (C) 2007-2023 Cppcheck team.
|
|
|
|
*
|
|
|
|
* 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 3 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/>.
|
|
|
|
*/
|
|
|
|
|
2023-01-30 14:59:45 +01:00
|
|
|
#include "compliancereportdialog.h"
|
2023-04-08 16:08:47 +02:00
|
|
|
|
2023-01-30 14:59:45 +01:00
|
|
|
#include "ui_compliancereportdialog.h"
|
|
|
|
|
2023-04-08 16:08:47 +02:00
|
|
|
#include "errortypes.h"
|
2023-01-30 14:59:45 +01:00
|
|
|
#include "filelist.h"
|
2023-11-02 17:42:41 +01:00
|
|
|
#include "filesettings.h"
|
2023-04-08 16:08:47 +02:00
|
|
|
#include "importproject.h"
|
2023-01-30 14:59:45 +01:00
|
|
|
#include "projectfile.h"
|
|
|
|
|
2023-04-08 16:08:47 +02:00
|
|
|
#include <algorithm>
|
|
|
|
#include <iterator>
|
|
|
|
#include <list>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <QByteArray>
|
|
|
|
#include <QCheckBox>
|
2023-08-02 10:36:17 +02:00
|
|
|
#include <QComboBox>
|
2023-04-08 16:08:47 +02:00
|
|
|
#include <QCoreApplication>
|
2023-12-27 10:36:18 +01:00
|
|
|
#include <QCryptographicHash>
|
2023-04-08 16:08:47 +02:00
|
|
|
#include <QDialogButtonBox>
|
|
|
|
#include <QDir>
|
2023-01-30 14:59:45 +01:00
|
|
|
#include <QFile>
|
|
|
|
#include <QFileDialog>
|
|
|
|
#include <QFileInfo>
|
2023-04-08 16:08:47 +02:00
|
|
|
#include <QIODevice>
|
|
|
|
#include <QLineEdit>
|
2023-10-09 10:07:20 +02:00
|
|
|
#include <QList>
|
2023-01-30 14:59:45 +01:00
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QProcess>
|
|
|
|
#include <QRegularExpression>
|
|
|
|
#include <QSet>
|
2023-04-08 16:08:47 +02:00
|
|
|
#include <QStringList>
|
2023-01-30 14:59:45 +01:00
|
|
|
#include <QTemporaryFile>
|
|
|
|
#include <QTextStream>
|
|
|
|
|
|
|
|
static void addHeaders(const QString& file1, QSet<QString> &allFiles) {
|
|
|
|
if (allFiles.contains(file1))
|
|
|
|
return;
|
|
|
|
QFile file(file1);
|
|
|
|
if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
|
|
|
|
return;
|
|
|
|
allFiles << file1;
|
|
|
|
const QRegularExpression re("^#include[ ]*\"([^\">]+)\".*");
|
|
|
|
QTextStream in(&file);
|
|
|
|
QString line = in.readLine();
|
|
|
|
while (!in.atEnd()) {
|
|
|
|
if (line.startsWith("#include")) {
|
|
|
|
const QRegularExpressionMatch match = re.match(line);
|
|
|
|
if (match.hasMatch()) {
|
|
|
|
QString hfile = match.captured(1);
|
|
|
|
if (file1.contains("/"))
|
|
|
|
hfile = file1.mid(0,file1.lastIndexOf("/") + 1) + hfile;
|
|
|
|
addHeaders(hfile, allFiles);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
line = in.readLine();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::vector<std::string> toStdStringList(const QStringList& from) {
|
|
|
|
std::vector<std::string> ret;
|
|
|
|
std::transform(from.cbegin(), from.cend(), std::back_inserter(ret), [](const QString& e) {
|
|
|
|
return e.toStdString();
|
|
|
|
});
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ComplianceReportDialog::ComplianceReportDialog(ProjectFile* projectFile, QString resultsFile) :
|
|
|
|
QDialog(nullptr),
|
|
|
|
mUI(new Ui::ComplianceReportDialog),
|
|
|
|
mProjectFile(projectFile),
|
|
|
|
mResultsFile(std::move(resultsFile))
|
|
|
|
{
|
|
|
|
mUI->setupUi(this);
|
|
|
|
mUI->mEditProjectName->setText(projectFile->getProjectName());
|
|
|
|
connect(mUI->buttonBox, &QDialogButtonBox::clicked, this, &ComplianceReportDialog::buttonClicked);
|
2023-11-29 16:13:01 +01:00
|
|
|
mUI->mCodingStandard->clear();
|
|
|
|
if (projectFile->getCodingStandards().contains("misra-c-2023"))
|
|
|
|
mUI->mCodingStandard->addItem("Misra C 2023");
|
|
|
|
else if (projectFile->getAddons().contains("misra"))
|
|
|
|
mUI->mCodingStandard->addItem("Misra C 2012");
|
|
|
|
if (projectFile->getCodingStandards().contains("misra-c++-2008"))
|
|
|
|
mUI->mCodingStandard->addItem("Misra C++ 2008");
|
|
|
|
if (projectFile->getCodingStandards().contains("cert-c-2016"))
|
|
|
|
mUI->mCodingStandard->addItem("Cert C");
|
|
|
|
if (projectFile->getCodingStandards().contains("cert-c++-2016"))
|
|
|
|
mUI->mCodingStandard->addItem("Cert C++");
|
|
|
|
mUI->mCodingStandard->addItems(projectFile->getCodingStandards());
|
2023-01-30 14:59:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ComplianceReportDialog::~ComplianceReportDialog()
|
|
|
|
{
|
|
|
|
delete mUI;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ComplianceReportDialog::buttonClicked(QAbstractButton* button)
|
|
|
|
{
|
|
|
|
switch (mUI->buttonBox->standardButton(button)) {
|
|
|
|
case QDialogButtonBox::StandardButton::Save:
|
|
|
|
save();
|
|
|
|
break;
|
|
|
|
case QDialogButtonBox::StandardButton::Close:
|
|
|
|
close();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void ComplianceReportDialog::save()
|
|
|
|
{
|
2023-06-27 13:34:58 +02:00
|
|
|
const QString std(mUI->mCodingStandard->currentText().toLower().replace(" ", "-"));
|
|
|
|
|
2023-01-30 14:59:45 +01:00
|
|
|
const QString outFile = QFileDialog::getSaveFileName(this,
|
|
|
|
tr("Compliance report"),
|
2023-06-27 13:34:58 +02:00
|
|
|
QDir::homePath() + "/" + std + "-compliance-report.html",
|
2023-01-30 14:59:45 +01:00
|
|
|
tr("HTML files (*.html)"));
|
|
|
|
if (outFile.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
close();
|
|
|
|
|
|
|
|
const QString& projectName = mUI->mEditProjectName->text();
|
|
|
|
const QString& projectVersion = mUI->mEditProjectVersion->text();
|
|
|
|
const bool files = mUI->mCheckFiles->isChecked();
|
|
|
|
|
|
|
|
if (projectName != mProjectFile->getProjectName()) {
|
|
|
|
mProjectFile->setProjectName(projectName);
|
|
|
|
mProjectFile->write();
|
|
|
|
}
|
|
|
|
|
|
|
|
QTemporaryFile tempFiles;
|
|
|
|
if (files && tempFiles.open()) {
|
|
|
|
QTextStream out(&tempFiles);
|
|
|
|
FileList fileList;
|
|
|
|
fileList.addPathList(mProjectFile->getCheckPaths());
|
|
|
|
if (!mProjectFile->getImportProject().isEmpty()) {
|
|
|
|
QFileInfo inf(mProjectFile->getFilename());
|
|
|
|
|
|
|
|
QString prjfile;
|
|
|
|
if (QFileInfo(mProjectFile->getImportProject()).isAbsolute())
|
|
|
|
prjfile = mProjectFile->getImportProject();
|
|
|
|
else
|
|
|
|
prjfile = inf.canonicalPath() + '/' + mProjectFile->getImportProject();
|
|
|
|
|
|
|
|
ImportProject p;
|
|
|
|
try {
|
|
|
|
p.import(prjfile.toStdString());
|
|
|
|
} catch (InternalError &e) {
|
|
|
|
QMessageBox msg(QMessageBox::Critical,
|
|
|
|
tr("Save compliance report"),
|
2023-12-27 10:36:18 +01:00
|
|
|
tr("Failed to import '%1' (%2), can not show files in compliance report").arg(prjfile, QString::fromStdString(e.errorMessage)),
|
2023-01-30 14:59:45 +01:00
|
|
|
QMessageBox::Ok,
|
|
|
|
this);
|
|
|
|
msg.exec();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
p.ignorePaths(toStdStringList(mProjectFile->getExcludedPaths()));
|
|
|
|
|
|
|
|
QDir dir(inf.absoluteDir());
|
2023-11-02 17:42:41 +01:00
|
|
|
for (const FileSettings& fs: p.fileSettings)
|
2023-01-30 14:59:45 +01:00
|
|
|
fileList.addFile(dir.relativeFilePath(QString::fromStdString(fs.filename)));
|
|
|
|
}
|
|
|
|
|
|
|
|
QSet<QString> allFiles;
|
|
|
|
for (const QString &sourcefile: fileList.getFileList())
|
|
|
|
addHeaders(sourcefile, allFiles);
|
|
|
|
for (const QString& fileName: allFiles) {
|
|
|
|
QFile f(fileName);
|
|
|
|
if (f.open(QFile::ReadOnly)) {
|
|
|
|
QCryptographicHash hash(QCryptographicHash::Algorithm::Md5);
|
|
|
|
if (hash.addData(&f)) {
|
|
|
|
for (auto b: hash.result())
|
|
|
|
out << QString::number((unsigned char)b,16);
|
|
|
|
out << " " << fileName << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tempFiles.close();
|
|
|
|
}
|
|
|
|
|
2023-09-04 10:58:48 +02:00
|
|
|
QStringList suppressions;
|
|
|
|
for (const auto& suppression: mProjectFile->getSuppressions()) {
|
|
|
|
if (!suppression.errorId.empty())
|
|
|
|
suppressions.append(QString::fromStdString(suppression.errorId));
|
|
|
|
}
|
|
|
|
|
2023-06-27 13:34:58 +02:00
|
|
|
QStringList args{"--project-name=" + projectName,
|
2023-01-30 14:59:45 +01:00
|
|
|
"--project-version=" + projectVersion,
|
2023-11-29 16:13:01 +01:00
|
|
|
"--output-file=" + outFile};
|
|
|
|
if (!suppressions.isEmpty())
|
|
|
|
args << "--suppressions=" + suppressions.join(",");
|
2023-06-27 13:34:58 +02:00
|
|
|
|
2023-08-25 13:38:27 +02:00
|
|
|
args << ("--" + std);
|
2023-06-27 13:34:58 +02:00
|
|
|
|
2023-01-30 14:59:45 +01:00
|
|
|
if (files)
|
|
|
|
args << "--files=" + tempFiles.fileName();
|
|
|
|
args << mResultsFile;
|
|
|
|
|
|
|
|
const QString appPath = QFileInfo(QCoreApplication::applicationFilePath()).canonicalPath();
|
|
|
|
|
|
|
|
QProcess process;
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
process.start(appPath + "/compliance-report.exe", args);
|
|
|
|
#else
|
|
|
|
process.start(appPath + "/compliance-report", args);
|
|
|
|
#endif
|
|
|
|
process.waitForFinished();
|
2023-11-29 16:13:01 +01:00
|
|
|
const QString output = process.readAll();
|
|
|
|
if (!output.isEmpty()) {
|
|
|
|
QMessageBox msg(QMessageBox::Critical,
|
|
|
|
tr("Save compliance report"),
|
|
|
|
output,
|
|
|
|
QMessageBox::Ok,
|
|
|
|
this);
|
|
|
|
msg.exec();
|
|
|
|
}
|
2023-01-30 14:59:45 +01:00
|
|
|
}
|