8#include <private/qqmljsimporter_p.h>
9#include <private/qqmljsimportvisitor_p.h>
10#include <private/qqmljslinterpasses_p.h>
11#include <private/qqmljslintervisitor_p.h>
12#include <private/qqmljsliteralbindingcheck_p.h>
13#include <private/qqmljsloggingutils_p.h>
14#include <private/qqmljsutils_p.h>
15#include <private/qqmlsa_p.h>
17#include <QtCore/qjsonobject.h>
18#include <QtCore/qfileinfo.h>
19#include <QtCore/qloggingcategory.h>
20#include <QtCore/qpluginloader.h>
21#include <QtCore/qlibraryinfo.h>
22#include <QtCore/qdir.h>
23#include <QtCore/private/qduplicatetracker_p.h>
24#include <QtCore/qscopedpointer.h>
28# include <QtCore/qdiriterator.h>
29# include <QtCore/qlibrary.h>
32#if QT_CONFIG(qmlcontextpropertydump)
33# include <QtCore/qsettings.h>
36#include <QtQml/private/qqmljslexer_p.h>
37#include <QtQml/private/qqmljsparser_p.h>
38#include <QtQml/private/qqmljsengine_p.h>
39#include <QtQml/private/qqmljsastvisitor_p.h>
40#include <QtQml/private/qqmljsast_p.h>
41#include <QtQml/private/qqmljsdiagnosticmessage_p.h>
46using namespace Qt::StringLiterals;
48class HasFunctionDefinitionVisitor
final :
public QQmlJS::AST::Visitor
51 bool visit(QQmlJS::AST::FunctionDeclaration *functionDeclaration)
override
53 m_result = !functionDeclaration->name.isEmpty();
58 bool result()
const {
return m_result; }
59 void reset() { m_result =
false; }
62 bool m_result =
false;
65class UnreachableVisitor
final :
public QQmlJS::AST::Visitor
73 node->accept(&m_hasFunctionDefinition);
77 bool visit(QQmlJS::AST::StatementList *unreachable)
override
79 QQmlJS::SourceLocation location;
80 auto report = [
this, &location]() {
81 if (location.isValid()) {
82 m_logger->log(u"Unreachable code"_s, qmlUnreachableCode, location);
84 location = QQmlJS::SourceLocation{};
87 for (
auto it = unreachable; it && it->statement; it = it->next) {
88 if (containsFunctionDeclaration(it->statement)) {
92 location = combine(location,
93 combine(it->statement->firstSourceLocation(),
94 it->statement->lastSourceLocation()));
102 QQmlJSLogger *m_logger =
nullptr;
103 HasFunctionDefinitionVisitor m_hasFunctionDefinition;
106class CodegenWarningInterface
final :
public QV4::Compiler::CodegenWarningInterface
114 QQmlJS::SourceLocation declarationLocation,
115 QQmlJS::SourceLocation accessLocation)
override
119 m_logger->log(
"Identifier '%1' is used here before its declaration."_L1.arg(name),
120 qmlVarUsedBeforeDeclaration, accessLocation);
121 m_logger->log(
"Note: declaration of '%1' here"_L1.arg(name), qmlVarUsedBeforeDeclaration,
122 declarationLocation,
true,
true, {}, accessLocation.startLine);
126 QQmlJS::SourceLocation declarationLocation,
127 QQmlJS::SourceLocation accessLocation)
override
131 m_logger->log(
"Function '%1' is used here before its declaration."_L1.arg(name),
132 qmlFunctionUsedBeforeDeclaration, accessLocation);
133 m_logger->log(
"Note: declaration of '%1' here"_L1.arg(name),
134 qmlFunctionUsedBeforeDeclaration, declarationLocation);
140 QQmlJSLogger *m_logger;
141 UnreachableVisitor m_unreachableVisitor;
145 bool useAbsolutePath)
146 : m_useAbsolutePath(useAbsolutePath),
147 m_enablePlugins(
true),
152 m_plugins = loadPlugins(extraPluginPaths);
161 , m_instance(
std::move(plugin.m_instance))
163 , m_isInternal(
std::move(plugin.m_isInternal))
164 , m_isValid(
std::move(plugin.m_isValid))
167 Q_ASSERT(!plugin.m_loader);
168 plugin.m_instance =
nullptr;
169 plugin.m_isValid =
false;
172#if QT_CONFIG(library)
173QQmlJSLinter::Plugin::Plugin(QString path)
175 m_loader = std::make_unique<QPluginLoader>(path);
176 if (!parseMetaData(m_loader->metaData(), path))
179 QObject *object = m_loader->instance();
183 m_instance = qobject_cast<QQmlSA::LintPlugin *>(object);
193 if (!parseMetaData(staticPlugin.metaData(), u"built-in"_s))
196 m_instance = qobject_cast<
QQmlSA::LintPlugin *>(staticPlugin.instance());
205#if QT_CONFIG(library)
206 if (m_loader !=
nullptr) {
208 m_loader->deleteLater();
213bool QQmlJSLinter::
Plugin::parseMetaData(
const QJsonObject &metaData, QString pluginName)
215 const QString pluginIID = QStringLiteral(QmlLintPluginInterface_iid);
217 if (metaData[u"IID"].toString() != pluginIID)
220 QJsonObject pluginMetaData = metaData[u"MetaData"].toObject();
222 for (
const QString &requiredKey :
223 { u"name"_s, u"version"_s, u"author"_s, u"loggingCategories"_s }) {
224 if (!pluginMetaData.contains(requiredKey)) {
225 qWarning() << pluginName <<
"is missing the required " << requiredKey
226 <<
"metadata, skipping";
231 m_name = pluginMetaData[u"name"].toString();
232 m_author = pluginMetaData[u"author"].toString();
233 m_version = pluginMetaData[u"version"].toString();
234 m_description = pluginMetaData[u"description"].toString(u"-/-"_s);
235 m_isInternal = pluginMetaData[u"isInternal"].toBool(
false);
237 if (!pluginMetaData[u"loggingCategories"].isArray()) {
238 qWarning() << pluginName <<
"has loggingCategories which are not an array, skipping";
242 const QJsonArray categories = pluginMetaData[u"loggingCategories"].toArray();
243 for (
const QJsonValue &value : categories) {
244 if (!value.isObject()) {
245 qWarning() << pluginName <<
"has invalid loggingCategories entries, skipping";
249 const QJsonObject object = value.toObject();
251 for (
const QString &requiredKey : { u"name"_s, u"description"_s }) {
252 if (!object.contains(requiredKey)) {
253 qWarning() << pluginName <<
" logging category is missing the required "
254 << requiredKey <<
"metadata, skipping";
259 const QString prefix = (m_isInternal ? u""_s : u"Plugin."_s).append(m_name).append(u'.');
260 const QString categoryId =
261 prefix + object[u"name"].toString();
262 const auto settingsNameIt = object.constFind(u"settingsName");
263 const QString settingsName = (settingsNameIt == object.constEnd())
265 : prefix + settingsNameIt->toString(categoryId);
266 m_categories << QQmlJS::LoggerCategory{ categoryId, settingsName,
267 object[
"description"_L1].toString(),
268 QQmlJS::WarningSeverity::Warning };
269 const auto itSeverity = object.find(
"defaultSeverity"_L1);
270 if (itSeverity == object.end())
273 const QString severityName = itSeverity->toString();
274 const auto severity = QQmlJS::LoggingUtils::severityFromString(severityName);
275 if (!severity.has_value()) {
276 qWarning() <<
"Invalid logging severity" << severityName <<
"provided for"
277 << m_categories.last().id().name().toString()
278 <<
"(allowed are: disable, info, warning, error) found in plugin metadata.";
282 m_categories.last().setSeverity(severity.value());
288std::vector<QQmlJSLinter::Plugin>
QQmlJSLinter::loadPlugins(QStringList extraPluginPaths)
290 std::vector<Plugin> plugins;
292 QDuplicateTracker<QString> seenPlugins;
294 const auto &staticPlugins = QPluginLoader::staticPlugins();
295 for (
const QStaticPlugin &staticPlugin : staticPlugins) {
296 Plugin plugin(staticPlugin);
297 if (!plugin.isValid())
300 if (seenPlugins.hasSeen(plugin.name().toLower())) {
301 qWarning() <<
"Two plugins named" << plugin.name()
302 <<
"present, make sure no plugins are duplicated. The second plugin will "
307 plugins.push_back(std::move(plugin));
310#if QT_CONFIG(library)
311 const QStringList paths = [&extraPluginPaths]() {
312 QStringList result{ extraPluginPaths };
313 const QStringList libraryPaths = QCoreApplication::libraryPaths();
314 for (
const auto &path : libraryPaths) {
315 result.append(path + u"/qmllint"_s);
319 for (
const QString &pluginDir : paths) {
320 QDirIterator it{ pluginDir, QDir::Files };
322 while (it.hasNext()) {
323 auto potentialPlugin = it.next();
325 if (!QLibrary::isLibrary(potentialPlugin))
328 Plugin plugin(potentialPlugin);
330 if (!plugin.isValid())
333 if (seenPlugins.hasSeen(plugin.name().toLower())) {
334 qWarning() <<
"Two plugins named" << plugin.name()
335 <<
"present, make sure no plugins are duplicated. The second plugin "
336 "will not be loaded.";
340 plugins.push_back(std::move(plugin));
344 Q_UNUSED(extraPluginPaths)
349 const QList<QQmlJS::SourceLocation> &comments)
351 QHash<
int, QSet<QString>> disablesPerLine;
352 QHash<
int, QSet<QString>> enablesPerLine;
353 QHash<
int, QSet<QString>> oneLineDisablesPerLine;
355 struct PostponedWarning
358 QQmlSA::LoggerWarningId category;
359 QQmlJS::SourceLocation location;
362 std::vector<PostponedWarning> postponedWarnings;
363 auto guard = qScopeGuard([&postponedWarnings, &logger]() {
366 for (
const auto &warning : postponedWarnings)
367 logger->log(warning.message, warning.category, warning.location);
370 const QString code = logger->code();
371 const QStringList lines = code.split(u'\n');
372 const auto loggerCategories = logger->categories();
374 for (
const auto &loc : comments) {
375 const QString comment = code.mid(loc.offset, loc.length);
376 if (!comment.startsWith(u" qmllint ") && !comment.startsWith(u"qmllint "))
379 QStringList words = comment.split(u' ', Qt::SkipEmptyParts);
380 if (words.size() < 2)
383 QSet<QString> categories;
384 for (qsizetype i = 2; i < words.size(); i++) {
385 const QString category = words.at(i);
386 const auto categoryExists = std::any_of(
387 loggerCategories.cbegin(), loggerCategories.cend(),
388 [&](
const QQmlJS::LoggerCategory &cat) {
return cat.id().name() == category; });
391 categories << category;
393 postponedWarnings.push_back(
394 { u"qmllint directive on unknown category \"%1\""_s.arg(category),
395 qmlInvalidLintDirective, loc });
399 if (words.size() == 2) {
400 const auto &loggerCategories = logger->categories();
401 for (
const auto &option : loggerCategories)
402 categories << option.id().name().toString();
405 const QString command = words.at(1);
406 if (command == u"disable"_s) {
407 if (
const qsizetype lineIndex = loc.startLine - 1; lineIndex < lines.size()) {
408 const QString line = lines[lineIndex];
409 const QString preComment = line.left(line.indexOf(comment) - 2);
411 bool lineHasContent =
false;
412 for (qsizetype i = 0; i < preComment.size(); i++) {
413 if (!preComment[i].isSpace()) {
414 lineHasContent =
true;
420 oneLineDisablesPerLine[loc.startLine] |= categories;
422 disablesPerLine[loc.startLine] |= categories;
424 }
else if (command == u"enable"_s) {
425 enablesPerLine[loc.startLine + 1] |= categories;
427 postponedWarnings.push_back(
428 { u"Invalid qmllint directive \"%1\" provided"_s.arg(command),
429 qmlInvalidLintDirective, loc });
433 if (disablesPerLine.isEmpty() && oneLineDisablesPerLine.isEmpty())
436 QSet<QString> currentlyDisabled;
437 for (qsizetype i = 1; i <= lines.size(); i++) {
438 currentlyDisabled.unite(disablesPerLine[i]).subtract(enablesPerLine[i]);
440 currentlyDisabled.unite(oneLineDisablesPerLine[i]);
442 if (!currentlyDisabled.isEmpty())
443 logger->ignoreWarnings(i, currentlyDisabled);
445 currentlyDisabled.subtract(oneLineDisablesPerLine[i]);
449static void addJsonWarning(QJsonArray &warnings,
const QQmlJS::DiagnosticMessage &message,
450 QAnyStringView id,
const std::optional<QQmlJSFixSuggestion> &suggestion = {})
452 QJsonObject jsonMessage;
455 switch (message.type) {
463 type = u"critical"_s;
476 jsonMessage[u"type"_s] = type;
477 jsonMessage[u"id"_s] = id.toString();
479 const auto convertLocation = [](
const QQmlJS::SourceLocation &source, QJsonObject *target) {
480 target->insert(
"line"_L1,
int(source.startLine));
481 target->insert(
"column"_L1,
int(source.startColumn));
482 target->insert(
"charOffset"_L1,
int(source.offset));
483 target->insert(
"length"_L1,
int(source.length));
486 if (message.loc.isValid())
487 convertLocation(message.loc, &jsonMessage);
489 jsonMessage[u"message"_s] = message.message;
491 QJsonArray suggestions;
492 if (suggestion.has_value()) {
493 QJsonObject jsonFix {
494 {
"message"_L1, suggestion->description() },
495 {
"replacement"_L1, suggestion->replacement() },
496 {
"isAutoApplicable"_L1, suggestion->isAutoApplicable() }
498 convertLocation(suggestion->location(), &jsonFix);
499 const QString filename = suggestion->filename();
500 if (!filename.isEmpty())
501 jsonFix.insert(
"fileName"_L1, filename);
502 suggestions << jsonFix;
504 jsonMessage[u"suggestions"] = suggestions;
506 warnings << jsonMessage;
511 m_logger->iterateAllMessages([&](
const Message &message) {
512 addJsonWarning(warnings, message, message.id, message.fixSuggestion);
517 const QString &filename, QQmlJSResourceFileMapper *mapper,
518 const QQmlJS::HeuristicContextProperties &heuristicContextProperties)
520 ContextPropertyInfo result;
521 if (m_userContextPropertySettings.search(filename).isValid()) {
522 result.userContextProperties =
523 QQmlJS::UserContextProperties{ m_userContextPropertySettings };
526 if (heuristicContextProperties.isValid()) {
527 result.heuristicContextProperties = heuristicContextProperties;
531#if QT_CONFIG(qmlcontextpropertydump)
532 const QString buildPath = QQmlJSUtils::qmlBuildPathFromSourcePath(mapper, filename);
533 if (
const auto searchResult = m_heuristicContextPropertySearcher.search(buildPath);
534 searchResult.isValid()) {
535 QSettings settings(searchResult.iniFilePath, QSettings::IniFormat);
536 result.heuristicContextProperties =
537 QQmlJS::HeuristicContextProperties::collectFrom(&settings);
547 QJsonArray *json,
const QStringList &qmlImportPaths,
548 const QStringList &qmldirFiles,
const QStringList &resourceFiles,
549 const QList<QQmlJS::LoggerCategory> &categories,
550 const QQmlJS::HeuristicContextProperties &heuristicContextProperties)
553 lintFileImpl(filename, fileContents, silent, json, qmlImportPaths, qmldirFiles,
554 resourceFiles, categories, heuristicContextProperties);
559 processMessages(warnings);
562 result[u"filename"_s] = QFileInfo(filename).absoluteFilePath();
563 result[u"warnings"] = warnings;
566 json->append(result);
570void QQmlJSLinter::setupLoggingCategoriesInLogger(
const QList<QQmlJS::LoggerCategory> &categories)
572 if (m_enablePlugins) {
573 for (
const Plugin &plugin : m_plugins) {
574 for (
const QQmlJS::LoggerCategory &category : plugin.categories())
575 m_logger->registerCategory(category);
579 for (
auto it = categories.cbegin(); it != categories.cend(); ++it) {
580 if (
auto logger = *it; !QQmlJS::LoggerCategoryPrivate::get(&logger)->hasChanged())
583 m_logger->setCategorySeverity(it->id(), it->severity());
588QQmlJSLinter::lintFileImpl(
const QString &filename,
const QString *fileContents,
const bool silent,
589 QJsonArray *json,
const QStringList &qmlImportPaths,
590 const QStringList &qmldirFiles,
const QStringList &resourceFiles,
591 const QList<QQmlJS::LoggerCategory> &categories,
592 const QQmlJS::HeuristicContextProperties &heuristicContextProperties)
596 QFileInfo info(filename);
597 const QString lowerSuffix = info.suffix().toLower();
598 const bool isESModule = lowerSuffix == QLatin1String(
"mjs");
599 const bool isJavaScript = isESModule || lowerSuffix == QLatin1String(
"js");
601 m_logger.reset(
new QQmlJSLogger);
602 m_logger->setFilePath(m_useAbsolutePath ? info.absoluteFilePath() : filename);
603 m_logger->setSilent(silent || json);
604 setupLoggingCategoriesInLogger(categories);
606 if (fileContents ==
nullptr) {
607 QFile file(filename);
608 if (!file.open(QFile::ReadOnly)) {
609 m_logger->log(
"Failed to open file %1: %2"_L1.arg(filename, file.errorString()),
610 qmlImport, QQmlJS::SourceLocation());
614 code = QString::fromUtf8(file.readAll());
617 code = *fileContents;
620 m_fileContents = code;
621 m_logger->setCode(code);
623 QQmlJS::Engine engine;
624 QQmlJS::Lexer lexer(&engine);
626 lexer.setCode(code, 1, !isJavaScript);
627 QQmlJS::Parser parser(&engine);
629 const bool parseSuccess = isJavaScript
630 ? (isESModule ? parser.parseModule() : parser.parseProgram())
632 const auto diagnosticMessages = parser.diagnosticMessages();
633 for (
const QQmlJS::DiagnosticMessage &m : diagnosticMessages)
634 m_logger->log(m.message, qmlSyntax, m.loc);
642 m_importer.setImportPaths(qmlImportPaths);
644 std::optional<QQmlJSResourceFileMapper> mapper;
645 if (!resourceFiles.isEmpty())
646 mapper.emplace(resourceFiles);
647 m_importer.setResourceFileMapper(mapper.has_value() ? &*mapper :
nullptr);
649 QQmlJS::LinterVisitor v{ &m_importer, m_logger.get(),
650 QQmlJSImportVisitor::implicitImportDirectory(
651 m_logger->filePath(), m_importer.resourceFileMapper()),
652 qmldirFiles, &engine };
654 parseComments(m_logger.get(), engine.comments());
656 QQmlJSTypeResolver typeResolver(&m_importer);
663 typeResolver.setParentMode(QQmlJSTypeResolver::UseDocumentParent);
667 typeResolver.setCloneMode(QQmlJSTypeResolver::DoNotCloneTypes);
669 typeResolver.init(&v, parser.rootNode());
671 const QStringList resourcePaths = mapper
672 ? mapper->resourcePaths(QQmlJSResourceFileMapper::localFileFilter(filename))
674 const QString resolvedPath =
675 (resourcePaths.size() == 1) ? u':' + resourcePaths.first() : filename;
678 contextPropertiesFor(filename, mapper ? &*mapper :
nullptr,
679 heuristicContextProperties) };
680 codegen.setTypeResolver(
std::move(typeResolver));
681 codegen.setScopesById(v.addressableScopes());
682 codegen.setRenamedComponents(&v.renamedComponents());
684 using PassManagerPtr =
685 std::unique_ptr<QQmlSA::PassManager,
686 decltype(&QQmlSA::PassManagerPrivate::deletePassManager)>;
687 PassManagerPtr passMan(
688 QQmlSA::PassManagerPrivate::createPassManager(&v, codegen.typeResolver()),
689 &QQmlSA::PassManagerPrivate::deletePassManager);
690 QQmlJSLinterPasses::registerDefaultPasses(passMan.get());
692 if (m_enablePlugins) {
693 for (
const Plugin &plugin : m_plugins) {
694 if (!plugin.isValid() || !plugin.isEnabled())
697 QQmlSA::LintPlugin *instance = plugin.m_instance;
699 instance->registerPasses(passMan.get(), QQmlJSScope::createQQmlSAElement(v.result()));
702 passMan->analyze(QQmlJSScope::createQQmlSAElement(v.result()));
704 if (m_logger->hasErrors())
711 QQmlJSSaveFunction saveFunction = [](
const QV4::CompiledData::SaveableUnitPointer &,
712 const QQmlJSAotFunctionMap &, QString *) {
return true; };
714 QQmlJSCompileError error;
716 QLoggingCategory::setFilterRules(u"qt.qml.compiler=false"_s);
718 CodegenWarningInterface warningInterface(m_logger.get());
719 qCompileQmlFile(filename, saveFunction, &codegen, &error,
true, &warningInterface,
722 QList<QQmlJS::DiagnosticMessage> globalWarnings = m_importer.takeGlobalWarnings();
724 if (!globalWarnings.isEmpty()) {
725 m_logger->log(QStringLiteral(
"Type warnings occurred while evaluating file:"), qmlImport,
726 QQmlJS::SourceLocation());
727 m_logger->processMessages(globalWarnings, qmlImport);
730 if (m_logger->hasErrors())
732 if (m_logger->hasWarnings())
739 const QString &module,
const bool silent, QJsonArray *json,
740 const QStringList &qmlImportPaths,
const QStringList &resourceFiles)
742 const LintResult lintResult = lintModuleImpl(module, silent, json, qmlImportPaths, resourceFiles);
747 processMessages(warnings);
750 result[u"module"_s] = module;
751 result[u"warnings"] = warnings;
754 json->append(result);
759 const QString &module,
const bool silent, QJsonArray *json,
760 const QStringList &qmlImportPaths,
const QStringList &resourceFiles)
766 m_importer.clearCache();
767 m_importer.setImportPaths(qmlImportPaths);
769 QQmlJSResourceFileMapper mapper(resourceFiles);
770 if (!resourceFiles.isEmpty())
771 m_importer.setResourceFileMapper(&mapper);
773 m_importer.setResourceFileMapper(
nullptr);
775 m_logger.reset(
new QQmlJSLogger);
776 m_logger->setFilePath(module);
777 m_logger->setCode(u""_s);
778 m_logger->setSilent(silent || json);
780 const QQmlJSImporter::ImportedTypes types =
781 m_importer.importModule(module, QQmlJS::PrecedenceValues::Default);
783 QList<QQmlJS::DiagnosticMessage> importWarnings =
784 m_importer.takeGlobalWarnings() + types.warnings();
786 if (!importWarnings.isEmpty()) {
787 m_logger->log(QStringLiteral(
"Warnings occurred while importing module:"), qmlImport,
788 QQmlJS::SourceLocation());
789 m_logger->processMessages(importWarnings, qmlImport);
792 QMap<QString, QSet<QString>> missingTypes;
793 QMap<QString, QSet<QString>> partiallyResolvedTypes;
795 const QString modulePrefix = u"$module$."_s;
796 const QString internalPrefix = u"$internal$."_s;
798 for (
auto &&[typeName, importedScope] : types.types().asKeyValueRange()) {
799 QString name = typeName;
800 const QQmlJSScope::ConstPtr scope = importedScope.scope;
802 if (name.startsWith(modulePrefix))
805 if (name.startsWith(internalPrefix)) {
806 name = name.mid(internalPrefix.size());
809 if (scope.isNull()) {
810 if (!missingTypes.contains(name))
811 missingTypes[name] = {};
815 if (!scope->isFullyResolved()) {
816 if (!partiallyResolvedTypes.contains(name))
817 partiallyResolvedTypes[name] = {};
819 const auto &ownProperties = scope->ownProperties();
820 for (
const auto &property : ownProperties) {
821 if (property.typeName().isEmpty()) {
826 if (property.type().isNull()) {
827 missingTypes[property.typeName()]
828 << scope->internalName() + u'.' + property.propertyName();
831 if (!property.type()->isFullyResolved()) {
832 partiallyResolvedTypes[property.typeName()]
833 << scope->internalName() + u'.' + property.propertyName();
836 if (scope->attachedType() && !scope->attachedType()->isFullyResolved()) {
837 m_logger->log(u"Attached type of \"%1\" not fully resolved"_s.arg(name),
838 qmlUnresolvedType, scope->sourceLocation());
841 const auto &ownMethods = scope->ownMethods();
842 for (
const auto &method : ownMethods) {
843 if (method.returnTypeName().isEmpty())
845 if (method.returnType().isNull()) {
846 missingTypes[method.returnTypeName()] << u"return type of "_s
847 + scope->internalName() + u'.' + method.methodName() + u"()"_s;
848 }
else if (!method.returnType()->isFullyResolved()) {
849 partiallyResolvedTypes[method.returnTypeName()] << u"return type of "_s
850 + scope->internalName() + u'.' + method.methodName() + u"()"_s;
853 const auto parameters = method.parameters();
854 for (qsizetype i = 0; i < parameters.size(); i++) {
855 auto ¶meter = parameters[i];
856 const QString typeName = parameter.typeName();
857 const QSharedPointer<
const QQmlJSScope> type = parameter.type();
858 if (typeName.isEmpty())
861 missingTypes[typeName] << u"parameter %1 of "_s.arg(i + 1)
862 + scope->internalName() + u'.' + method.methodName() + u"()"_s;
865 if (!type->isFullyResolved()) {
866 partiallyResolvedTypes[typeName] << u"parameter %1 of "_s.arg(i + 1)
867 + scope->internalName() + u'.' + method.methodName() + u"()"_s;
874 for (
auto &&[name, uses] : missingTypes.asKeyValueRange()) {
875 QString message = u"Type \"%1\" not found"_s.arg(name);
877 if (!uses.isEmpty()) {
878 const QStringList usesList = QStringList(uses.begin(), uses.end());
879 message += u". Used in %1"_s.arg(usesList.join(u", "_s));
882 m_logger->log(message, qmlUnresolvedType, QQmlJS::SourceLocation());
885 for (
auto &&[name, uses] : partiallyResolvedTypes.asKeyValueRange()) {
886 QString message = u"Type \"%1\" is not fully resolved"_s.arg(name);
888 if (!uses.isEmpty()) {
889 const QStringList usesList = QStringList(uses.begin(), uses.end());
890 message += u". Used in %1"_s.arg(usesList.join(u", "_s));
893 m_logger->log(message, qmlUnresolvedType, QQmlJS::SourceLocation());
896 return (m_logger->hasWarnings() || m_logger->hasErrors()) ? HasWarnings : LintSuccess;
901 Q_ASSERT(fixedCode !=
nullptr);
907 if (m_logger ==
nullptr)
910 QString code = m_fileContents;
912 QList<QQmlJSFixSuggestion> fixesToApply;
914 QFileInfo info(m_logger->filePath());
915 const QString currentFileAbsolutePath = info.absoluteFilePath();
917 const QString lowerSuffix = info.suffix().toLower();
918 const bool isESModule = lowerSuffix == QLatin1String(
"mjs");
919 const bool isJavaScript = isESModule || lowerSuffix == QLatin1String(
"js");
921 if (isESModule || isJavaScript)
924 m_logger->iterateAllMessages([&](
const Message &msg) {
925 if (!msg.fixSuggestion.has_value() || !msg.fixSuggestion->isAutoApplicable())
929 const QString filename = msg.fixSuggestion->filename();
930 if (!filename.isEmpty()
931 && QFileInfo(filename).absoluteFilePath() != currentFileAbsolutePath) {
935 fixesToApply << msg.fixSuggestion.value();
938 if (fixesToApply.isEmpty())
941 std::sort(fixesToApply.begin(), fixesToApply.end(),
942 [](
const QQmlJSFixSuggestion &a,
const QQmlJSFixSuggestion &b) {
943 return a.location().offset < b.location().offset;
946 const auto dupes = std::unique(fixesToApply.begin(), fixesToApply.end());
947 fixesToApply.erase(dupes, fixesToApply.end());
949 for (
auto it = fixesToApply.begin(); it + 1 != fixesToApply.end(); it++) {
950 const QQmlJS::SourceLocation srcLocA = it->location();
951 const QQmlJS::SourceLocation srcLocB = (it + 1)->location();
952 if (srcLocA.offset + srcLocA.length > srcLocB.offset) {
954 qWarning() <<
"Fixes for two warnings are overlapping, aborting. Please file a bug "
960 int offsetChange = 0;
962 for (
const auto &fix : std::as_const(fixesToApply)) {
963 const QQmlJS::SourceLocation fixLocation = fix.location();
964 qsizetype cutLocation = fixLocation.offset + offsetChange;
965 const QString before = code.left(cutLocation);
966 const QString after = code.mid(cutLocation + fixLocation.length);
968 const QString replacement = fix.replacement();
969 code = before + replacement + after;
970 offsetChange += replacement.size() - fixLocation.length;
973 QQmlJS::Engine engine;
974 QQmlJS::Lexer lexer(&engine);
976 lexer.setCode(code, 1, !isJavaScript);
977 QQmlJS::Parser parser(&engine);
979 bool success = parser.parse();
982 const auto diagnosticMessages = parser.diagnosticMessages();
985 qDebug() <<
"File became unparseable after suggestions were applied. Please file a bug "
991 for (
const QQmlJS::DiagnosticMessage &m : diagnosticMessages) {
992 qWarning().noquote() << QString::fromLatin1(
"%1:%2:%3: %4")
993 .arg(m_logger->filePath())
994 .arg(m.loc.startLine)
995 .arg(m.loc.startColumn)
void reportVarUsedBeforeDeclaration(const QString &name, const QString &fileName, QQmlJS::SourceLocation declarationLocation, QQmlJS::SourceLocation accessLocation) override
void reportFunctionUsedBeforeDeclaration(const QString &name, const QString &fileName, QQmlJS::SourceLocation declarationLocation, QQmlJS::SourceLocation accessLocation) override
UnreachableVisitor * unreachableVisitor() override
CodegenWarningInterface(QQmlJSLogger *logger)
bool visit(QQmlJS::AST::FunctionDeclaration *functionDeclaration) override
void throwRecursionDepthError() override
void setPassManager(QQmlSA::PassManager *passManager)
Plugin(Plugin &&plugin) noexcept
Plugin(const QStaticPlugin &plugin)
FixResult applyFixes(QString *fixedCode, bool silent)
QQmlJSLinter(const QStringList &importPaths, const QStringList &extraPluginPaths={}, bool useAbsolutePath=false)
LintResult lintModule(const QString &uri, const bool silent, QJsonArray *json, const QStringList &qmlImportPaths, const QStringList &resourceFiles)
LintResult lintFile(const QString &filename, const QString *fileContents, const bool silent, QJsonArray *json, const QStringList &qmlImportPaths, const QStringList &qmldirFiles, const QStringList &resourceFiles, const QList< QQmlJS::LoggerCategory > &categories, const QQmlJS::HeuristicContextProperties &contextProperties={})
UnreachableVisitor(QQmlJSLogger *logger)
void throwRecursionDepthError() override
bool containsFunctionDeclaration(QQmlJS::AST::Node *node)
bool visit(QQmlJS::AST::StatementList *unreachable) override
Combined button and popup list for selecting options.
static void addJsonWarning(QJsonArray &warnings, const QQmlJS::DiagnosticMessage &message, QAnyStringView id, const std::optional< QQmlJSFixSuggestion > &suggestion={})