// Copyright (C) 2022 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial #include "axivionplugin.h" #include "axivionoutputpane.h" #include "axivionprojectsettings.h" #include "axivionquery.h" #include "axivionresultparser.h" #include "axivionsettings.h" #include "axivionsettingspage.h" #include "axiviontr.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef LICENSECHECKER # include #endif #include #include constexpr char AxivionTextMarkId[] = "AxivionTextMark"; namespace Axivion::Internal { class AxivionPluginPrivate : public QObject { public: AxivionProjectSettings *projectSettings(ProjectExplorer::Project *project); void onStartupProjectChanged(); void fetchProjectInfo(const QString &projectName); void handleProjectInfo(const ProjectInfo &info); void handleOpenedDocs(ProjectExplorer::Project *project); void onDocumentOpened(Core::IDocument *doc); void onDocumentClosed(Core::IDocument * doc); void clearAllMarks(); void handleIssuesForFile(const IssuesList &issues); AxivionSettings m_axivionSettings; AxivionSettingsPage m_axivionSettingsPage{&m_axivionSettings}; AxivionOutputPane m_axivionOutputPane; QHash m_axivionProjectSettings; ProjectInfo m_currentProjectInfo; bool m_runningQuery = false; }; static AxivionPlugin *s_instance = nullptr; static AxivionPluginPrivate *dd = nullptr; AxivionPlugin::AxivionPlugin() { s_instance = this; } AxivionPlugin::~AxivionPlugin() { if (!dd->m_axivionProjectSettings.isEmpty()) { qDeleteAll(dd->m_axivionProjectSettings); dd->m_axivionProjectSettings.clear(); } delete dd; dd = nullptr; } AxivionPlugin *AxivionPlugin::instance() { return s_instance; } bool AxivionPlugin::initialize(const QStringList &arguments, QString *errorMessage) { Q_UNUSED(arguments) Q_UNUSED(errorMessage) #ifdef LICENSECHECKER LicenseChecker::LicenseCheckerPlugin *licenseChecker = ExtensionSystem::PluginManager::getObject(); if (!licenseChecker || !licenseChecker->hasValidLicense() || !licenseChecker->enterpriseFeatures()) return true; #endif // LICENSECHECKER dd = new AxivionPluginPrivate; dd->m_axivionSettings.fromSettings(Core::ICore::settings()); auto panelFactory = new ProjectExplorer::ProjectPanelFactory; panelFactory->setPriority(250); panelFactory->setDisplayName(Tr::tr("Axivion")); panelFactory->setCreateWidgetFunction([](ProjectExplorer::Project *project){ return new AxivionProjectSettingsWidget(project); }); ProjectExplorer::ProjectPanelFactory::registerFactory(panelFactory); connect(ProjectExplorer::SessionManager::instance(), &ProjectExplorer::SessionManager::startupProjectChanged, dd, &AxivionPluginPrivate::onStartupProjectChanged); connect(Core::EditorManager::instance(), &Core::EditorManager::documentOpened, dd, &AxivionPluginPrivate::onDocumentOpened); connect(Core::EditorManager::instance(), &Core::EditorManager::documentClosed, dd, &AxivionPluginPrivate::onDocumentClosed); return true; } AxivionSettings *AxivionPlugin::settings() { QTC_ASSERT(dd, return nullptr); return &dd->m_axivionSettings; } AxivionProjectSettings *AxivionPlugin::projectSettings(ProjectExplorer::Project *project) { QTC_ASSERT(project, return nullptr); QTC_ASSERT(dd, return nullptr); return dd->projectSettings(project); } bool AxivionPlugin::handleCertificateIssue() { QTC_ASSERT(dd, return false); const QString serverHost = QUrl(dd->m_axivionSettings.server.dashboard).host(); if (QMessageBox::question(Core::ICore::dialogParent(), Tr::tr("Certificate Error"), Tr::tr("Server certificate for %1 cannot be authenticated.\n" "Do you want to disable SSL verification for this server?\n" "Note: This can expose you to man-in-the-middle attack.") .arg(serverHost)) != QMessageBox::Yes) { return false; } dd->m_axivionSettings.server.validateCert = false; emit s_instance->settingsChanged(); return true; } void AxivionPlugin::fetchProjectInfo(const QString &projectName) { QTC_ASSERT(dd, return); dd->fetchProjectInfo(projectName); } ProjectInfo AxivionPlugin::projectInfo() { QTC_ASSERT(dd, return {}); return dd->m_currentProjectInfo; } AxivionProjectSettings *AxivionPluginPrivate::projectSettings(ProjectExplorer::Project *project) { auto &settings = m_axivionProjectSettings[project]; if (!settings) settings = new AxivionProjectSettings(project); return settings; } void AxivionPluginPrivate::onStartupProjectChanged() { ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject(); if (!project) { clearAllMarks(); m_currentProjectInfo = ProjectInfo(); m_axivionOutputPane.updateDashboard(); return; } const AxivionProjectSettings *projSettings = projectSettings(project); fetchProjectInfo(projSettings->dashboardProjectName()); } void AxivionPluginPrivate::fetchProjectInfo(const QString &projectName) { if (m_runningQuery) { // re-schedule QTimer::singleShot(3000, [this, projectName]{ fetchProjectInfo(projectName); }); return; } clearAllMarks(); if (projectName.isEmpty()) { m_currentProjectInfo = ProjectInfo(); m_axivionOutputPane.updateDashboard(); return; } m_runningQuery = true; AxivionQuery query(AxivionQuery::ProjectInfo, {projectName}); AxivionQueryRunner *runner = new AxivionQueryRunner(query, this); connect(runner, &AxivionQueryRunner::resultRetrieved, this, [this](const QByteArray &result){ handleProjectInfo(ResultParser::parseProjectInfo(result)); }); connect(runner, &AxivionQueryRunner::finished, [runner]{ runner->deleteLater(); }); runner->start(); } void AxivionPluginPrivate::handleOpenedDocs(ProjectExplorer::Project *project) { if (project && ProjectExplorer::SessionManager::startupProject() != project) return; const QList openDocuments = Core::DocumentModel::openedDocuments(); for (Core::IDocument *doc : openDocuments) onDocumentOpened(doc); if (project) disconnect(ProjectExplorer::SessionManager::instance(), &ProjectExplorer::SessionManager::projectFinishedParsing, this, &AxivionPluginPrivate::handleOpenedDocs); } void AxivionPluginPrivate::clearAllMarks() { const QList openDocuments = Core::DocumentModel::openedDocuments(); for (Core::IDocument *doc : openDocuments) onDocumentClosed(doc); } void AxivionPluginPrivate::handleProjectInfo(const ProjectInfo &info) { m_runningQuery = false; if (!info.error.isEmpty()) { Core::MessageManager::writeFlashing("Axivion: " + info.error); return; } m_currentProjectInfo = info; m_axivionOutputPane.updateDashboard(); if (m_currentProjectInfo.name.isEmpty()) return; // handle already opened documents if (!ProjectExplorer::SessionManager::startupBuildSystem()->isParsing()) { handleOpenedDocs(nullptr); } else { connect(ProjectExplorer::SessionManager::instance(), &ProjectExplorer::SessionManager::projectFinishedParsing, this, &AxivionPluginPrivate::handleOpenedDocs); } } void AxivionPluginPrivate::onDocumentOpened(Core::IDocument *doc) { if (m_currentProjectInfo.name.isEmpty()) // we do not have a project info (yet) return; ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject(); if (!doc || !project->isKnownFile(doc->filePath())) return; Utils::FilePath relative = doc->filePath().relativeChildPath(project->projectDirectory()); // for now only style violations AxivionQuery query(AxivionQuery::IssuesForFileList, {m_currentProjectInfo.name, "SV", relative.path() } ); AxivionQueryRunner *runner = new AxivionQueryRunner(query, this); connect(runner, &AxivionQueryRunner::resultRetrieved, this, [this](const QByteArray &result){ handleIssuesForFile(ResultParser::parseIssuesList(result)); }); connect(runner, &AxivionQueryRunner::finished, [runner]{ runner->deleteLater(); }); runner->start(); } void AxivionPluginPrivate::onDocumentClosed(Core::IDocument *doc) { const auto document = qobject_cast(doc); if (!document) return; const TextEditor::TextMarks marks = document->marks(); for (auto m : marks) { if (m->category() == AxivionTextMarkId) delete m; } } void AxivionPluginPrivate::handleIssuesForFile(const IssuesList &issues) { if (issues.issues.isEmpty()) return; ProjectExplorer::Project *project = ProjectExplorer::SessionManager::startupProject(); if (!project) return; const Utils::FilePath filePath = project->projectDirectory() .pathAppended(issues.issues.first().filePath); const Utils::Id axivionId(AxivionTextMarkId); for (const ShortIssue &issue : std::as_const(issues.issues)) { // FIXME the line location can be wrong (even the whole issue could be wrong) // depending on whether this line has been changed since the last axivion run and the // current state of the file - some magic has to happen here auto mark = new TextEditor::TextMark(filePath, issue.lineNumber, axivionId); const QString markText = issue.entity.isEmpty() ? issue.message : issue.entity + ": " + issue.message; mark->setToolTip(issue.errorNumber + " " + markText); mark->setPriority(TextEditor::TextMark::NormalPriority); mark->setLineAnnotation(markText); } } } // Axivion::Internal