/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "projectimporter.h" #include "buildinfo.h" #include "kit.h" #include "kitinformation.h" #include "kitmanager.h" #include "project.h" #include "projectexplorerconstants.h" #include "target.h" #include "toolchain.h" #include "toolchainmanager.h" #include #include #include #include #include #include namespace ProjectExplorer { static const Utils::Id KIT_IS_TEMPORARY("PE.tmp.isTemporary"); static const Utils::Id KIT_TEMPORARY_NAME("PE.tmp.Name"); static const Utils::Id KIT_FINAL_NAME("PE.tmp.FinalName"); static const Utils::Id TEMPORARY_OF_PROJECTS("PE.tmp.ForProjects"); static Utils::Id fullId(Utils::Id id) { const QString prefix = "PE.tmp."; const QString idStr = id.toString(); QTC_ASSERT(!idStr.startsWith(prefix), return Utils::Id::fromString(idStr)); return Utils::Id::fromString(prefix + idStr); } static bool hasOtherUsers(Utils::Id id, const QVariant &v, Kit *k) { return Utils::contains(KitManager::kits(), [id, v, k](Kit *in) -> bool { if (in == k) return false; const QVariantList tmp = in->value(id).toList(); return tmp.contains(v); }); } ProjectImporter::ProjectImporter(const Utils::FilePath &path) : m_projectPath(path) { useTemporaryKitAspect(ToolChainKitAspect::id(), [this](Kit *k, const QVariantList &vl) { cleanupTemporaryToolChains(k, vl); }, [this](Kit *k, const QVariantList &vl) { persistTemporaryToolChains(k, vl); }); } ProjectImporter::~ProjectImporter() { foreach (Kit *k, KitManager::kits()) removeProject(k); } const QList ProjectImporter::import(const Utils::FilePath &importPath, bool silent) { QList result; const QLoggingCategory log("qtc.projectexplorer.import", QtWarningMsg); qCDebug(log) << "ProjectImporter::import" << importPath << silent; QFileInfo fi = importPath.toFileInfo(); if (!fi.exists() && !fi.isDir()) { qCDebug(log) << "**doesn't exist"; return result; } const Utils::FilePath absoluteImportPath = Utils::FilePath::fromString(fi.absoluteFilePath()); const auto handleFailure = [this, importPath, silent] { if (silent) return; QMessageBox::critical(Core::ICore::dialogParent(), tr("No Build Found"), tr("No build found in %1 matching project %2.") .arg(importPath.toUserOutput(), projectFilePath().toUserOutput())); }; qCDebug(log) << "Examining directory" << absoluteImportPath.toString(); QList dataList = examineDirectory(absoluteImportPath); if (dataList.isEmpty()) { qCDebug(log) << "Nothing to import found in" << absoluteImportPath.toString(); handleFailure(); return result; } qCDebug(log) << "Looking for kits"; foreach (void *data, dataList) { QTC_ASSERT(data, continue); QList kitList; const QList tmp = Utils::filtered(KitManager::kits(), [this, data](Kit *k) { return matchKit(data, k); }); if (tmp.isEmpty()) { Kit *k = createKit(data); if (k) kitList.append(k); qCDebug(log) << " no matching kit found, temporary kit created."; } else { kitList += tmp; qCDebug(log) << " " << tmp.count() << "matching kits found."; } foreach (Kit *k, kitList) { qCDebug(log) << "Creating buildinfos for kit" << k->displayName(); const QList infoList = buildInfoList(data); if (infoList.isEmpty()) { qCDebug(log) << "No build infos for kit" << k->displayName(); continue; } auto factory = BuildConfigurationFactory::find(k, projectFilePath()); for (BuildInfo i : infoList) { i.kitId = k->id(); i.factory = factory; if (!result.contains(i)) result += i; } } } foreach (auto *dd, dataList) deleteDirectoryData(dd); dataList.clear(); if (result.isEmpty()) handleFailure(); return result; } Target *ProjectImporter::preferredTarget(const QList &possibleTargets) { // Select active target // a) The default target // c) Desktop target // d) the first target Target *activeTarget = nullptr; if (possibleTargets.isEmpty()) return activeTarget; activeTarget = possibleTargets.at(0); bool pickedFallback = false; foreach (Target *t, possibleTargets) { if (t->kit() == KitManager::defaultKit()) return t; if (pickedFallback) continue; if (DeviceTypeKitAspect::deviceTypeId(t->kit()) == Constants::DESKTOP_DEVICE_TYPE) { activeTarget = t; pickedFallback = true; } } return activeTarget; } void ProjectImporter::markKitAsTemporary(Kit *k) const { QTC_ASSERT(!k->hasValue(KIT_IS_TEMPORARY), return); UpdateGuard guard(*this); const QString name = k->displayName(); k->setUnexpandedDisplayName(QCoreApplication::translate("ProjectExplorer::ProjectImporter", "%1 - temporary").arg(name)); k->setValue(KIT_TEMPORARY_NAME, k->displayName()); k->setValue(KIT_FINAL_NAME, name); k->setValue(KIT_IS_TEMPORARY, true); } void ProjectImporter::makePersistent(Kit *k) const { QTC_ASSERT(k, return); if (!k->hasValue(KIT_IS_TEMPORARY)) return; UpdateGuard guard(*this); KitGuard kitGuard(k); k->removeKey(KIT_IS_TEMPORARY); k->removeKey(TEMPORARY_OF_PROJECTS); const QString tempName = k->value(KIT_TEMPORARY_NAME).toString(); if (!tempName.isNull() && k->displayName() == tempName) k->setUnexpandedDisplayName(k->value(KIT_FINAL_NAME).toString()); k->removeKey(KIT_TEMPORARY_NAME); k->removeKey(KIT_FINAL_NAME); foreach (const TemporaryInformationHandler &tih, m_temporaryHandlers) { const Utils::Id fid = fullId(tih.id); const QVariantList temporaryValues = k->value(fid).toList(); // Mark permanent in all other kits: foreach (Kit *ok, KitManager::kits()) { if (ok == k || !ok->hasValue(fid)) continue; const QVariantList otherTemporaryValues = Utils::filtered(ok->value(fid).toList(), [&temporaryValues](const QVariant &v) { return !temporaryValues.contains(v); }); ok->setValueSilently(fid, otherTemporaryValues); } // persist: tih.persist(k, temporaryValues); k->removeKeySilently(fid); } } void ProjectImporter::cleanupKit(Kit *k) const { QTC_ASSERT(k, return); foreach (const TemporaryInformationHandler &tih, m_temporaryHandlers) { const Utils::Id fid = fullId(tih.id); const QVariantList temporaryValues = Utils::filtered(k->value(fid).toList(), [fid, k](const QVariant &v) { return !hasOtherUsers(fid, v, k); }); tih.cleanup(k, temporaryValues); k->removeKeySilently(fid); } // remove keys to manage temporary state of kit: k->removeKeySilently(KIT_IS_TEMPORARY); k->removeKeySilently(TEMPORARY_OF_PROJECTS); k->removeKeySilently(KIT_FINAL_NAME); k->removeKeySilently(KIT_TEMPORARY_NAME); } void ProjectImporter::addProject(Kit *k) const { QTC_ASSERT(k, return); if (!k->hasValue(KIT_IS_TEMPORARY)) return; UpdateGuard guard(*this); QStringList projects = k->value(TEMPORARY_OF_PROJECTS, QStringList()).toStringList(); projects.append(m_projectPath.toString()); // note: There can be more than one instance of the project added! k->setValueSilently(TEMPORARY_OF_PROJECTS, projects); } void ProjectImporter::removeProject(Kit *k) const { QTC_ASSERT(k, return); if (!k->hasValue(KIT_IS_TEMPORARY)) return; UpdateGuard guard(*this); QStringList projects = k->value(TEMPORARY_OF_PROJECTS, QStringList()).toStringList(); projects.removeOne(m_projectPath.toString()); if (projects.isEmpty()) { cleanupKit(k); KitManager::deregisterKit(k); } else { k->setValueSilently(TEMPORARY_OF_PROJECTS, projects); } } bool ProjectImporter::isTemporaryKit(Kit *k) const { QTC_ASSERT(k, return false); return k->hasValue(KIT_IS_TEMPORARY); } Kit *ProjectImporter::createTemporaryKit(const KitSetupFunction &setup) const { UpdateGuard guard(*this); const auto init = [&](Kit *k) { KitGuard kitGuard(k); k->setUnexpandedDisplayName(QCoreApplication::translate("ProjectExplorer::ProjectImporter", "Imported Kit")); k->setup(); setup(k); k->fix(); markKitAsTemporary(k); addProject(k); }; // ~KitGuard, sending kitUpdated return KitManager::registerKit(init); // potentially adds kits to other targetsetuppages } bool ProjectImporter::findTemporaryHandler(Utils::Id id) const { return Utils::contains(m_temporaryHandlers, [id](const TemporaryInformationHandler &ch) { return ch.id == id; }); } static ToolChain *toolChainFromVariant(const QVariant &v) { const QByteArray tcId = v.toByteArray(); return ToolChainManager::findToolChain(tcId); } void ProjectImporter::cleanupTemporaryToolChains(Kit *k, const QVariantList &vl) { for (const QVariant &v : vl) { ToolChain *tc = toolChainFromVariant(v); QTC_ASSERT(tc, continue); ToolChainManager::deregisterToolChain(tc); ToolChainKitAspect::setToolChain(k, nullptr); } } void ProjectImporter::persistTemporaryToolChains(Kit *k, const QVariantList &vl) { for (const QVariant &v : vl) { ToolChain *tmpTc = toolChainFromVariant(v); QTC_ASSERT(tmpTc, continue); ToolChain *actualTc = ToolChainKitAspect::toolChain(k, tmpTc->language()); if (tmpTc && actualTc != tmpTc) ToolChainManager::deregisterToolChain(tmpTc); } } void ProjectImporter::useTemporaryKitAspect(Utils::Id id, ProjectImporter::CleanupFunction cleanup, ProjectImporter::PersistFunction persist) { QTC_ASSERT(!findTemporaryHandler(id), return); m_temporaryHandlers.append({id, cleanup, persist}); } void ProjectImporter::addTemporaryData(Utils::Id id, const QVariant &cleanupData, Kit *k) const { QTC_ASSERT(k, return); QTC_ASSERT(findTemporaryHandler(id), return); const Utils::Id fid = fullId(id); KitGuard guard(k); QVariantList tmp = k->value(fid).toList(); QTC_ASSERT(!tmp.contains(cleanupData), return); tmp.append(cleanupData); k->setValue(fid, tmp); } bool ProjectImporter::hasKitWithTemporaryData(Utils::Id id, const QVariant &data) const { Utils::Id fid = fullId(id); return Utils::contains(KitManager::kits(), [data, fid](Kit *k) { return k->value(fid).toList().contains(data); }); } static ProjectImporter::ToolChainData createToolChains(const ToolChainDescription &tcd) { ProjectImporter::ToolChainData data; for (ToolChainFactory *factory : ToolChainFactory::allToolChainFactories()) { data.tcs = factory->detectForImport(tcd); if (data.tcs.isEmpty()) continue; for (ToolChain *tc : data.tcs) ToolChainManager::registerToolChain(tc); data.areTemporary = true; break; } return data; } ProjectImporter::ToolChainData ProjectImporter::findOrCreateToolChains(const ToolChainDescription &tcd) const { ToolChainData result; result.tcs = ToolChainManager::toolChains([&tcd](const ToolChain *tc) { return tc->language() == tcd.language && tc->compilerCommand() == tcd.compilerPath; }); for (const ToolChain *tc : result.tcs) { const QByteArray tcId = tc->id(); result.areTemporary = result.areTemporary ? true : hasKitWithTemporaryData(ToolChainKitAspect::id(), tcId); } if (!result.tcs.isEmpty()) return result; // Create a new toolchain: UpdateGuard guard(*this); return createToolChains(tcd); } } // namespace ProjectExplorer