// Copyright (C) 2022 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0 #include "killappstep.h" #include "abstractremotelinuxdeploystep.h" #include "abstractremotelinuxdeployservice.h" #include "remotelinux_constants.h" #include "remotelinuxtr.h" #include #include #include #include #include using namespace ProjectExplorer; using namespace Utils; namespace RemoteLinux::Internal { class KillAppService : public AbstractRemoteLinuxDeployService { public: ~KillAppService() override; void setRemoteExecutable(const FilePath &filePath); private: void handleStdErr(); void handleProcessFinished(); bool isDeploymentNecessary() const override; void doDeploy() override; void stopDeployment() override; void handleSignalOpFinished(const QString &errorMessage); void cleanup(); void finishDeployment(); FilePath m_remoteExecutable; DeviceProcessSignalOperation::Ptr m_signalOperation; }; KillAppService::~KillAppService() { cleanup(); } void KillAppService::setRemoteExecutable(const FilePath &filePath) { m_remoteExecutable = filePath; } bool KillAppService::isDeploymentNecessary() const { return !m_remoteExecutable.isEmpty(); } void KillAppService::doDeploy() { m_signalOperation = deviceConfiguration()->signalOperation(); if (!m_signalOperation) { handleDeploymentDone(); return; } connect(m_signalOperation.data(), &DeviceProcessSignalOperation::finished, this, &KillAppService::handleSignalOpFinished); emit progressMessage(Tr::tr("Trying to kill \"%1\" on remote device...") .arg(m_remoteExecutable.path())); m_signalOperation->killProcess(m_remoteExecutable.path()); } void KillAppService::cleanup() { if (m_signalOperation) { disconnect(m_signalOperation.data(), nullptr, this, nullptr); m_signalOperation.clear(); } } void KillAppService::finishDeployment() { cleanup(); handleDeploymentDone(); } void KillAppService::stopDeployment() { finishDeployment(); } void KillAppService::handleSignalOpFinished(const QString &errorMessage) { if (errorMessage.isEmpty()) emit progressMessage(Tr::tr("Remote application killed.")); else emit progressMessage(Tr::tr("Failed to kill remote application. Assuming it was not running.")); finishDeployment(); } class KillAppStep : public AbstractRemoteLinuxDeployStep { public: KillAppStep(BuildStepList *bsl, Id id) : AbstractRemoteLinuxDeployStep(bsl, id) { auto service = createDeployService(); setWidgetExpandedByDefault(false); setInternalInitializer([this, service] { Target * const theTarget = target(); QTC_ASSERT(theTarget, return CheckResult::failure()); RunConfiguration * const rc = theTarget->activeRunConfiguration(); const FilePath remoteExe = rc ? rc->runnable().command.executable() : FilePath(); service->setRemoteExecutable(remoteExe); return CheckResult::success(); }); } }; KillAppStepFactory::KillAppStepFactory() { registerStep(Constants::KillAppStepId); setDisplayName(Tr::tr("Kill current application instance")); setSupportedConfiguration(RemoteLinux::Constants::DeployToGenericLinux); setSupportedStepList(ProjectExplorer::Constants::BUILDSTEPS_DEPLOY); } } // RemoteLinux::Internal