forked from qt-creator/qt-creator
The flexibility here is getting in the way later when trying to remove the dependency on aspects. Change-Id: I7221e80f2067292c7c80aead8f6d739fb7878f7e Reviewed-by: <github-actions-qt-creator@cristianadam.eu> Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Eike Ziller <eike.ziller@qt.io>
676 lines
26 KiB
C++
676 lines
26 KiB
C++
// Copyright (C) 2016 BogDan Vatra <bog_dan_ro@yahoo.com>
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "androiddeployqtstep.h"
|
|
|
|
#include "androidavdmanager.h"
|
|
#include "androidbuildapkstep.h"
|
|
#include "androidconstants.h"
|
|
#include "androiddevice.h"
|
|
#include "androidmanager.h"
|
|
#include "androidqtversion.h"
|
|
#include "androidtr.h"
|
|
#include "androidtr.h"
|
|
|
|
#include <coreplugin/fileutils.h>
|
|
#include <coreplugin/icore.h>
|
|
#include <coreplugin/messagemanager.h>
|
|
|
|
#include <projectexplorer/abstractprocessstep.h>
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
#include <projectexplorer/buildsystem.h>
|
|
#include <projectexplorer/buildsteplist.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectnodes.h>
|
|
#include <projectexplorer/runconfiguration.h>
|
|
#include <projectexplorer/target.h>
|
|
#include <projectexplorer/taskhub.h>
|
|
#include <projectexplorer/toolchain.h>
|
|
|
|
#include <qtsupport/baseqtversion.h>
|
|
#include <qtsupport/qtkitinformation.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/asynctask.h>
|
|
#include <utils/commandline.h>
|
|
#include <utils/environment.h>
|
|
#include <utils/futuresynchronizer.h>
|
|
#include <utils/layoutbuilder.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/qtcprocess.h>
|
|
|
|
#include <QCheckBox>
|
|
#include <QFileDialog>
|
|
#include <QGroupBox>
|
|
#include <QInputDialog>
|
|
#include <QLoggingCategory>
|
|
#include <QMessageBox>
|
|
#include <QPushButton>
|
|
#include <QRegularExpression>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Utils;
|
|
|
|
namespace Android::Internal {
|
|
|
|
static Q_LOGGING_CATEGORY(deployStepLog, "qtc.android.build.androiddeployqtstep", QtWarningMsg)
|
|
|
|
const QLatin1String UninstallPreviousPackageKey("UninstallPreviousPackage");
|
|
const QLatin1String InstallFailedInconsistentCertificatesString("INSTALL_PARSE_FAILED_INCONSISTENT_CERTIFICATES");
|
|
const QLatin1String InstallFailedUpdateIncompatible("INSTALL_FAILED_UPDATE_INCOMPATIBLE");
|
|
const QLatin1String InstallFailedPermissionModelDowngrade("INSTALL_FAILED_PERMISSION_MODEL_DOWNGRADE");
|
|
const QLatin1String InstallFailedVersionDowngrade("INSTALL_FAILED_VERSION_DOWNGRADE");
|
|
|
|
// AndroidDeployQtStep
|
|
|
|
class AndroidDeployQtStep : public BuildStep
|
|
{
|
|
Q_OBJECT
|
|
|
|
enum DeployErrorCode
|
|
{
|
|
NoError = 0,
|
|
InconsistentCertificates = 0x0001,
|
|
UpdateIncompatible = 0x0002,
|
|
PermissionModelDowngrade = 0x0004,
|
|
VersionDowngrade = 0x0008,
|
|
Failure = 0x0010
|
|
};
|
|
|
|
public:
|
|
AndroidDeployQtStep(BuildStepList *bc, Id id);
|
|
|
|
signals:
|
|
void askForUninstall(DeployErrorCode errorCode);
|
|
|
|
private:
|
|
void runCommand(const CommandLine &command);
|
|
|
|
bool init() override;
|
|
void doRun() override;
|
|
void doCancel() override;
|
|
void gatherFilesToPull();
|
|
DeployErrorCode runDeploy();
|
|
void slotAskForUninstall(DeployErrorCode errorCode);
|
|
|
|
void runImpl(QPromise<bool> &promise);
|
|
|
|
QWidget *createConfigWidget() override;
|
|
|
|
void processReadyReadStdOutput(DeployErrorCode &errorCode);
|
|
void stdOutput(const QString &line);
|
|
void processReadyReadStdError(DeployErrorCode &errorCode);
|
|
void stdError(const QString &line);
|
|
DeployErrorCode parseDeployErrors(const QString &deployOutputLine) const;
|
|
|
|
friend void operator|=(DeployErrorCode &e1, const DeployErrorCode &e2) {
|
|
e1 = static_cast<AndroidDeployQtStep::DeployErrorCode>((int)e1 | (int)e2);
|
|
}
|
|
|
|
friend DeployErrorCode operator|(const DeployErrorCode &e1, const DeployErrorCode &e2) {
|
|
return static_cast<AndroidDeployQtStep::DeployErrorCode>((int)e1 | (int)e2);
|
|
}
|
|
|
|
void reportWarningOrError(const QString &message, Task::TaskType type);
|
|
|
|
FilePath m_manifestName;
|
|
QString m_serialNumber;
|
|
QString m_avdName;
|
|
FilePath m_apkPath;
|
|
QMap<QString, FilePath> m_filesToPull;
|
|
|
|
QStringList m_androidABIs;
|
|
BoolAspect *m_uninstallPreviousPackage = nullptr;
|
|
bool m_uninstallPreviousPackageRun = false;
|
|
bool m_useAndroiddeployqt = false;
|
|
bool m_askForUninstall = false;
|
|
CommandLine m_androiddeployqtArgs;
|
|
FilePath m_adbPath;
|
|
FilePath m_command;
|
|
FilePath m_workingDirectory;
|
|
Environment m_environment;
|
|
AndroidDeviceInfo m_deviceInfo;
|
|
|
|
FutureSynchronizer m_synchronizer;
|
|
};
|
|
|
|
AndroidDeployQtStep::AndroidDeployQtStep(BuildStepList *parent, Id id)
|
|
: BuildStep(parent, id)
|
|
{
|
|
setImmutable(true);
|
|
setUserExpanded(true);
|
|
|
|
m_uninstallPreviousPackage = addAspect<BoolAspect>();
|
|
m_uninstallPreviousPackage->setSettingsKey(UninstallPreviousPackageKey);
|
|
m_uninstallPreviousPackage->setLabel(Tr::tr("Uninstall the existing app before deployment"),
|
|
BoolAspect::LabelPlacement::AtCheckBox);
|
|
m_uninstallPreviousPackage->setValue(false);
|
|
|
|
const QtSupport::QtVersion * const qt = QtSupport::QtKitAspect::qtVersion(kit());
|
|
const bool forced = qt && qt->qtVersion() < QVersionNumber(5, 4, 0);
|
|
if (forced) {
|
|
m_uninstallPreviousPackage->setValue(true);
|
|
m_uninstallPreviousPackage->setEnabled(false);
|
|
}
|
|
|
|
connect(this, &AndroidDeployQtStep::askForUninstall,
|
|
this, &AndroidDeployQtStep::slotAskForUninstall,
|
|
Qt::BlockingQueuedConnection);
|
|
}
|
|
|
|
bool AndroidDeployQtStep::init()
|
|
{
|
|
QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(kit());
|
|
if (!version) {
|
|
reportWarningOrError(Tr::tr("The Qt version for kit %1 is invalid.").arg(kit()->displayName()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
|
|
m_androiddeployqtArgs = CommandLine();
|
|
|
|
m_androidABIs = AndroidManager::applicationAbis(target());
|
|
if (m_androidABIs.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("No Android architecture (ABI) is set by the project."),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
|
|
emit addOutput(Tr::tr("Initializing deployment to Android device/simulator"),
|
|
OutputFormat::NormalMessage);
|
|
|
|
RunConfiguration *rc = target()->activeRunConfiguration();
|
|
QTC_ASSERT(rc, reportWarningOrError(Tr::tr("The kit's run configuration is invalid."), Task::Error);
|
|
return false);
|
|
BuildConfiguration *bc = target()->activeBuildConfiguration();
|
|
QTC_ASSERT(bc, reportWarningOrError(Tr::tr("The kit's build configuration is invalid."),
|
|
Task::Error);
|
|
return false);
|
|
|
|
auto androidBuildApkStep = bc->buildSteps()->firstOfType<AndroidBuildApkStep>();
|
|
const int minTargetApi = AndroidManager::minimumSDK(target());
|
|
qCDebug(deployStepLog) << "Target architecture:" << m_androidABIs
|
|
<< "Min target API" << minTargetApi;
|
|
|
|
// Try to re-use user-provided information from an earlier step of the same type.
|
|
BuildStepList *bsl = stepList();
|
|
QTC_ASSERT(bsl, reportWarningOrError(Tr::tr("The kit's build steps list is invalid."), Task::Error);
|
|
return false);
|
|
auto androidDeployQtStep = bsl->firstOfType<AndroidDeployQtStep>();
|
|
QTC_ASSERT(androidDeployQtStep,
|
|
reportWarningOrError(Tr::tr("The kit's deploy configuration is invalid."), Task::Error);
|
|
return false);
|
|
AndroidDeviceInfo info;
|
|
if (androidDeployQtStep != this)
|
|
info = androidDeployQtStep->m_deviceInfo;
|
|
|
|
const BuildSystem *bs = buildSystem();
|
|
auto selectedAbis = bs->property(Constants::AndroidAbis).toStringList();
|
|
|
|
const QString buildKey = target()->activeBuildKey();
|
|
if (selectedAbis.isEmpty())
|
|
selectedAbis = bs->extraData(buildKey, Constants::AndroidAbis).toStringList();
|
|
|
|
if (selectedAbis.isEmpty())
|
|
selectedAbis.append(bs->extraData(buildKey, Constants::AndroidAbi).toString());
|
|
|
|
if (!info.isValid()) {
|
|
const auto dev =
|
|
static_cast<const AndroidDevice *>(DeviceKitAspect::device(kit()).data());
|
|
if (!dev) {
|
|
reportWarningOrError(Tr::tr("No valid deployment device is set."), Task::Error);
|
|
return false;
|
|
}
|
|
|
|
// TODO: use AndroidDevice directly instead of AndroidDeviceInfo.
|
|
info = AndroidDevice::androidDeviceInfoFromIDevice(dev);
|
|
m_deviceInfo = info; // Keep around for later steps
|
|
|
|
if (!info.isValid()) {
|
|
reportWarningOrError(Tr::tr("The deployment device \"%1\" is invalid.")
|
|
.arg(dev->displayName()), Task::Error);
|
|
return false;
|
|
}
|
|
|
|
const bool abiListNotEmpty = !selectedAbis.isEmpty() && !dev->supportedAbis().isEmpty();
|
|
if (abiListNotEmpty && !dev->canSupportAbis(selectedAbis)) {
|
|
const QString error = Tr::tr("The deployment device \"%1\" does not support the "
|
|
"architectures used by the kit.\n"
|
|
"The kit supports \"%2\", but the device uses \"%3\".")
|
|
.arg(dev->displayName()).arg(selectedAbis.join(", "))
|
|
.arg(dev->supportedAbis().join(", "));
|
|
reportWarningOrError(error, Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (!dev->canHandleDeployments()) {
|
|
reportWarningOrError(Tr::tr("The deployment device \"%1\" is disconnected.")
|
|
.arg(dev->displayName()), Task::Error);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
const QtSupport::QtVersion * const qt = QtSupport::QtKitAspect::qtVersion(kit());
|
|
if (qt && qt->supportsMultipleQtAbis() && !info.cpuAbi.isEmpty() &&
|
|
!selectedAbis.contains(info.cpuAbi.first())) {
|
|
TaskHub::addTask(DeploymentTask(Task::Warning,
|
|
Tr::tr("Android: The main ABI of the deployment device (%1) is not selected. The app "
|
|
"execution or debugging might not work properly. Add it from Projects > Build > "
|
|
"Build Steps > qmake > ABIs.")
|
|
.arg(info.cpuAbi.first())));
|
|
}
|
|
|
|
m_avdName = info.avdName;
|
|
m_serialNumber = info.serialNumber;
|
|
qCDebug(deployStepLog) << "Selected device info:" << info;
|
|
|
|
AndroidManager::setDeviceSerialNumber(target(), m_serialNumber);
|
|
AndroidManager::setDeviceApiLevel(target(), info.sdk);
|
|
AndroidManager::setDeviceAbis(target(), info.cpuAbi);
|
|
|
|
gatherFilesToPull();
|
|
|
|
emit addOutput(Tr::tr("Deploying to %1").arg(m_serialNumber), OutputFormat::NormalMessage);
|
|
|
|
m_uninstallPreviousPackageRun = m_uninstallPreviousPackage->value();
|
|
if (m_uninstallPreviousPackageRun)
|
|
m_manifestName = AndroidManager::manifestPath(target());
|
|
|
|
m_useAndroiddeployqt = version->qtVersion() >= QVersionNumber(5, 4, 0);
|
|
if (m_useAndroiddeployqt) {
|
|
const QString buildKey = target()->activeBuildKey();
|
|
const ProjectNode *node = target()->project()->findNodeForBuildKey(buildKey);
|
|
if (!node) {
|
|
reportWarningOrError(Tr::tr("The deployment step's project node is invalid."), Task::Error);
|
|
return false;
|
|
}
|
|
m_apkPath = FilePath::fromString(node->data(Constants::AndroidApk).toString());
|
|
if (!m_apkPath.isEmpty()) {
|
|
m_manifestName = FilePath::fromString(node->data(Constants::AndroidManifest).toString());
|
|
m_command = AndroidConfigurations::currentConfig().adbToolPath();
|
|
AndroidManager::setManifestPath(target(), m_manifestName);
|
|
} else {
|
|
QString jsonFile = AndroidQtVersion::androidDeploymentSettings(target()).toString();
|
|
if (jsonFile.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("Cannot find the androiddeployqt input JSON file."),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
m_command = version->hostBinPath();
|
|
if (m_command.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("Cannot find the androiddeployqt tool."), Task::Error);
|
|
return false;
|
|
}
|
|
m_command = m_command.pathAppended("androiddeployqt").withExecutableSuffix();
|
|
|
|
m_workingDirectory = AndroidManager::androidBuildDirectory(target());
|
|
|
|
m_androiddeployqtArgs.addArgs({"--verbose",
|
|
"--output", m_workingDirectory.toString(),
|
|
"--no-build",
|
|
"--input", jsonFile});
|
|
|
|
m_androiddeployqtArgs.addArg("--gradle");
|
|
|
|
if (buildType() == BuildConfiguration::Release)
|
|
m_androiddeployqtArgs.addArgs({"--release"});
|
|
|
|
if (androidBuildApkStep && androidBuildApkStep->signPackage()) {
|
|
// The androiddeployqt tool is not really written to do stand-alone installations.
|
|
// This hack forces it to use the correct filename for the apk file when installing
|
|
// as a temporary fix until androiddeployqt gets the support. Since the --sign is
|
|
// only used to get the correct file name of the apk, its parameters are ignored.
|
|
m_androiddeployqtArgs.addArgs({"--sign", "foo", "bar"});
|
|
}
|
|
}
|
|
} else {
|
|
m_uninstallPreviousPackageRun = true;
|
|
m_command = AndroidConfigurations::currentConfig().adbToolPath();
|
|
m_apkPath = AndroidManager::packagePath(target());
|
|
m_workingDirectory = bc ? AndroidManager::buildDirectory(target()): FilePath();
|
|
}
|
|
m_environment = bc ? bc->environment() : Environment();
|
|
|
|
m_adbPath = AndroidConfigurations::currentConfig().adbToolPath();
|
|
|
|
AndroidAvdManager avdManager;
|
|
// Start the AVD if not running.
|
|
if (!m_avdName.isEmpty() && avdManager.findAvd(m_avdName).isEmpty())
|
|
avdManager.startAvdAsync(m_avdName);
|
|
return true;
|
|
}
|
|
|
|
AndroidDeployQtStep::DeployErrorCode AndroidDeployQtStep::runDeploy()
|
|
{
|
|
CommandLine cmd(m_command);
|
|
if (m_useAndroiddeployqt && m_apkPath.isEmpty()) {
|
|
cmd.addArgs(m_androiddeployqtArgs.arguments(), CommandLine::Raw);
|
|
if (m_uninstallPreviousPackageRun)
|
|
cmd.addArg("--install");
|
|
else
|
|
cmd.addArg("--reinstall");
|
|
|
|
if (!m_serialNumber.isEmpty() && !m_serialNumber.startsWith("????"))
|
|
cmd.addArgs({"--device", m_serialNumber});
|
|
|
|
} else {
|
|
RunConfiguration *rc = target()->activeRunConfiguration();
|
|
QTC_ASSERT(rc, return DeployErrorCode::Failure);
|
|
QString packageName;
|
|
|
|
if (m_uninstallPreviousPackageRun) {
|
|
packageName = AndroidManager::packageName(m_manifestName);
|
|
if (packageName.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("Cannot find the package name from the Android Manifest "
|
|
"file \"%1\".").arg(m_manifestName.toUserOutput()),
|
|
Task::Error);
|
|
return Failure;
|
|
}
|
|
const QString msg = Tr::tr("Uninstalling the previous package \"%1\".").arg(packageName);
|
|
qCDebug(deployStepLog) << msg;
|
|
emit addOutput(msg, OutputFormat::NormalMessage);
|
|
runCommand({m_adbPath,
|
|
AndroidDeviceInfo::adbSelector(m_serialNumber)
|
|
<< "uninstall" << packageName});
|
|
}
|
|
|
|
cmd.addArgs(AndroidDeviceInfo::adbSelector(m_serialNumber));
|
|
cmd.addArgs({"install", "-r", m_apkPath.toString()});
|
|
}
|
|
|
|
QtcProcess process;
|
|
process.setCommand(cmd);
|
|
process.setWorkingDirectory(m_workingDirectory);
|
|
process.setEnvironment(m_environment);
|
|
process.setUseCtrlCStub(true);
|
|
|
|
DeployErrorCode deployError = NoError;
|
|
|
|
process.setStdOutLineCallback([this, &deployError](const QString &line) {
|
|
deployError |= parseDeployErrors(line);
|
|
stdOutput(line);
|
|
});
|
|
process.setStdErrLineCallback([this, &deployError](const QString &line) {
|
|
deployError |= parseDeployErrors(line);
|
|
stdError(line);
|
|
});
|
|
|
|
process.start();
|
|
|
|
emit addOutput(Tr::tr("Starting: \"%1\"").arg(cmd.toUserOutput()), OutputFormat::NormalMessage);
|
|
|
|
while (!process.waitForFinished(200)) {
|
|
if (process.state() == QProcess::NotRunning)
|
|
break;
|
|
|
|
if (isCanceled()) {
|
|
process.kill();
|
|
process.waitForFinished();
|
|
}
|
|
}
|
|
|
|
const QProcess::ExitStatus exitStatus = process.exitStatus();
|
|
const int exitCode = process.exitCode();
|
|
|
|
if (exitStatus == QProcess::NormalExit && exitCode == 0) {
|
|
emit addOutput(Tr::tr("The process \"%1\" exited normally.").arg(m_command.toUserOutput()),
|
|
OutputFormat::NormalMessage);
|
|
} else if (exitStatus == QProcess::NormalExit) {
|
|
const QString error = Tr::tr("The process \"%1\" exited with code %2.")
|
|
.arg(m_command.toUserOutput(), QString::number(exitCode));
|
|
reportWarningOrError(error, Task::Error);
|
|
} else {
|
|
const QString error = Tr::tr("The process \"%1\" crashed.").arg(m_command.toUserOutput());
|
|
reportWarningOrError(error, Task::Error);
|
|
}
|
|
|
|
if (deployError != NoError) {
|
|
if (m_uninstallPreviousPackageRun) {
|
|
deployError = Failure; // Even re-install failed. Set to Failure.
|
|
reportWarningOrError(
|
|
Tr::tr("Installing the app failed even after uninstalling the previous one."),
|
|
Task::Error);
|
|
}
|
|
} else if (exitCode != 0 || exitStatus != QProcess::NormalExit) {
|
|
// Set the deployError to Failure when no deployError code was detected
|
|
// but the adb tool failed otherwise relay the detected deployError.
|
|
reportWarningOrError(Tr::tr("Installing the app failed with an unknown error."), Task::Error);
|
|
deployError = Failure;
|
|
}
|
|
|
|
return deployError;
|
|
}
|
|
|
|
void AndroidDeployQtStep::slotAskForUninstall(DeployErrorCode errorCode)
|
|
{
|
|
Q_ASSERT(errorCode > 0);
|
|
|
|
QString uninstallMsg = Tr::tr("Deployment failed with the following errors:\n\n");
|
|
uint errorCodeFlags = errorCode;
|
|
uint mask = 1;
|
|
while (errorCodeFlags) {
|
|
switch (errorCodeFlags & mask) {
|
|
case DeployErrorCode::PermissionModelDowngrade:
|
|
uninstallMsg += InstallFailedPermissionModelDowngrade+"\n";
|
|
break;
|
|
case InconsistentCertificates:
|
|
uninstallMsg += InstallFailedInconsistentCertificatesString+"\n";
|
|
break;
|
|
case UpdateIncompatible:
|
|
uninstallMsg += InstallFailedUpdateIncompatible+"\n";
|
|
break;
|
|
case VersionDowngrade:
|
|
uninstallMsg += InstallFailedVersionDowngrade+"\n";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
errorCodeFlags &= ~mask;
|
|
mask <<= 1;
|
|
}
|
|
|
|
uninstallMsg.append(Tr::tr("\nUninstalling the installed package may solve the issue.\n"
|
|
"Do you want to uninstall the existing package?"));
|
|
int button = QMessageBox::critical(nullptr, Tr::tr("Install failed"), uninstallMsg,
|
|
QMessageBox::Yes, QMessageBox::No);
|
|
m_askForUninstall = button == QMessageBox::Yes;
|
|
}
|
|
|
|
void AndroidDeployQtStep::runImpl(QPromise<bool> &promise)
|
|
{
|
|
if (!m_avdName.isEmpty()) {
|
|
const QString serialNumber = AndroidAvdManager().waitForAvd(m_avdName,
|
|
QFuture<void>(promise.future()));
|
|
qCDebug(deployStepLog) << "Deploying to AVD:" << m_avdName << serialNumber;
|
|
if (serialNumber.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("The deployment AVD \"%1\" cannot be started.")
|
|
.arg(m_avdName), Task::Error);
|
|
promise.addResult(false);
|
|
return;
|
|
}
|
|
m_serialNumber = serialNumber;
|
|
qCDebug(deployStepLog) << "Deployment device serial number changed:" << serialNumber;
|
|
AndroidManager::setDeviceSerialNumber(target(), serialNumber);
|
|
}
|
|
|
|
DeployErrorCode returnValue = runDeploy();
|
|
if (returnValue > DeployErrorCode::NoError && returnValue < DeployErrorCode::Failure) {
|
|
emit askForUninstall(returnValue);
|
|
if (m_askForUninstall) {
|
|
m_uninstallPreviousPackageRun = true;
|
|
returnValue = runDeploy();
|
|
}
|
|
}
|
|
|
|
if (!m_filesToPull.isEmpty())
|
|
emit addOutput(Tr::tr("Pulling files necessary for debugging."), OutputFormat::NormalMessage);
|
|
|
|
// Note that values are not necessarily unique, e.g. app_process is looked up in several
|
|
// directories
|
|
for (auto itr = m_filesToPull.constBegin(); itr != m_filesToPull.constEnd(); ++itr)
|
|
itr.value().removeFile();
|
|
|
|
for (auto itr = m_filesToPull.constBegin(); itr != m_filesToPull.constEnd(); ++itr) {
|
|
runCommand({m_adbPath,
|
|
AndroidDeviceInfo::adbSelector(m_serialNumber)
|
|
<< "pull" << itr.key() << itr.value().nativePath()});
|
|
if (!itr.value().exists()) {
|
|
const QString error = Tr::tr("Package deploy: Failed to pull \"%1\" to \"%2\".")
|
|
.arg(itr.key())
|
|
.arg(itr.value().nativePath());
|
|
reportWarningOrError(error, Task::Error);
|
|
}
|
|
}
|
|
promise.addResult(returnValue == NoError);
|
|
}
|
|
|
|
void AndroidDeployQtStep::gatherFilesToPull()
|
|
{
|
|
m_filesToPull.clear();
|
|
const FilePath buildDir = AndroidManager::buildDirectory(target());
|
|
|
|
if (!m_deviceInfo.isValid())
|
|
return;
|
|
|
|
QString linkerName("linker");
|
|
QString libDirName("lib");
|
|
const QString preferredAbi = AndroidManager::apkDevicePreferredAbi(target());
|
|
if (preferredAbi == ProjectExplorer::Constants::ANDROID_ABI_ARM64_V8A
|
|
|| preferredAbi == ProjectExplorer::Constants::ANDROID_ABI_X86_64) {
|
|
m_filesToPull["/system/bin/app_process64"] = buildDir / "app_process";
|
|
libDirName = "lib64";
|
|
linkerName = "linker64";
|
|
} else {
|
|
m_filesToPull["/system/bin/app_process32"] = buildDir / "app_process";
|
|
m_filesToPull["/system/bin/app_process"] = buildDir / "app_process";
|
|
}
|
|
|
|
m_filesToPull["/system/bin/" + linkerName] = buildDir / linkerName;
|
|
m_filesToPull["/system/" + libDirName + "/libc.so"] = buildDir / "libc.so";
|
|
|
|
for (auto itr = m_filesToPull.constBegin(); itr != m_filesToPull.constEnd(); ++itr)
|
|
qCDebug(deployStepLog).noquote() << "Pulling file from device:" << itr.key()
|
|
<< "to:" << itr.value();
|
|
}
|
|
|
|
void AndroidDeployQtStep::doRun()
|
|
{
|
|
auto * const watcher = new QFutureWatcher<bool>(this);
|
|
connect(watcher, &QFutureWatcher<bool>::finished, this, [this, watcher] {
|
|
const bool success = !watcher->isCanceled() && watcher->result();
|
|
emit finished(success);
|
|
watcher->deleteLater();
|
|
});
|
|
auto future = Utils::asyncRun(&AndroidDeployQtStep::runImpl, this);
|
|
watcher->setFuture(future);
|
|
m_synchronizer.addFuture(future);
|
|
}
|
|
|
|
void AndroidDeployQtStep::doCancel()
|
|
{
|
|
m_synchronizer.cancelAllFutures();
|
|
}
|
|
|
|
void AndroidDeployQtStep::runCommand(const CommandLine &command)
|
|
{
|
|
QtcProcess buildProc;
|
|
buildProc.setTimeoutS(2 * 60);
|
|
emit addOutput(Tr::tr("Package deploy: Running command \"%1\".").arg(command.toUserOutput()),
|
|
OutputFormat::NormalMessage);
|
|
|
|
buildProc.setCommand(command);
|
|
buildProc.runBlocking(EventLoopMode::On);
|
|
if (buildProc.result() != ProcessResult::FinishedWithSuccess)
|
|
reportWarningOrError(buildProc.exitMessage(), Task::Error);
|
|
}
|
|
|
|
QWidget *AndroidDeployQtStep::createConfigWidget()
|
|
{
|
|
auto widget = new QWidget;
|
|
auto installCustomApkButton = new QPushButton(widget);
|
|
installCustomApkButton->setText(Tr::tr("Install an APK File"));
|
|
|
|
connect(installCustomApkButton, &QAbstractButton::clicked, this, [this, widget] {
|
|
const FilePath packagePath
|
|
= FileUtils::getOpenFilePath(widget,
|
|
Tr::tr("Qt Android Installer"),
|
|
FileUtils::homePath(),
|
|
Tr::tr("Android package (*.apk)"));
|
|
if (!packagePath.isEmpty())
|
|
AndroidManager::installQASIPackage(target(), packagePath);
|
|
});
|
|
|
|
Layouting::Form builder;
|
|
builder.addRow({m_uninstallPreviousPackage});
|
|
builder.addRow({installCustomApkButton});
|
|
builder.attachTo(widget, Layouting::WithoutMargins);
|
|
|
|
return widget;
|
|
}
|
|
|
|
void AndroidDeployQtStep::stdOutput(const QString &line)
|
|
{
|
|
emit addOutput(line, BuildStep::OutputFormat::Stdout, BuildStep::DontAppendNewline);
|
|
}
|
|
|
|
void AndroidDeployQtStep::stdError(const QString &line)
|
|
{
|
|
emit addOutput(line, BuildStep::OutputFormat::Stderr, BuildStep::DontAppendNewline);
|
|
|
|
QString newOutput = line;
|
|
static const QRegularExpression re("^(\\n)+");
|
|
newOutput.remove(re);
|
|
|
|
if (newOutput.isEmpty())
|
|
return;
|
|
|
|
if (newOutput.startsWith("warning", Qt::CaseInsensitive)
|
|
|| newOutput.startsWith("note", Qt::CaseInsensitive)
|
|
|| newOutput.startsWith(QLatin1String("All files should be loaded."))) {
|
|
TaskHub::addTask(DeploymentTask(Task::Warning, newOutput));
|
|
} else {
|
|
TaskHub::addTask(DeploymentTask(Task::Error, newOutput));
|
|
}
|
|
}
|
|
|
|
AndroidDeployQtStep::DeployErrorCode AndroidDeployQtStep::parseDeployErrors(
|
|
const QString &deployOutputLine) const
|
|
{
|
|
DeployErrorCode errorCode = NoError;
|
|
|
|
if (deployOutputLine.contains(InstallFailedInconsistentCertificatesString))
|
|
errorCode |= InconsistentCertificates;
|
|
if (deployOutputLine.contains(InstallFailedUpdateIncompatible))
|
|
errorCode |= UpdateIncompatible;
|
|
if (deployOutputLine.contains(InstallFailedPermissionModelDowngrade))
|
|
errorCode |= PermissionModelDowngrade;
|
|
if (deployOutputLine.contains(InstallFailedVersionDowngrade))
|
|
errorCode |= VersionDowngrade;
|
|
|
|
return errorCode;
|
|
}
|
|
|
|
void AndroidDeployQtStep::reportWarningOrError(const QString &message, Task::TaskType type)
|
|
{
|
|
qCDebug(deployStepLog).noquote() << message;
|
|
emit addOutput(message, OutputFormat::ErrorMessage);
|
|
TaskHub::addTask(DeploymentTask(type, message));
|
|
}
|
|
|
|
// AndroidDeployQtStepFactory
|
|
|
|
AndroidDeployQtStepFactory::AndroidDeployQtStepFactory()
|
|
{
|
|
registerStep<AndroidDeployQtStep>(Constants::ANDROID_DEPLOY_QT_ID);
|
|
setSupportedStepList(ProjectExplorer::Constants::BUILDSTEPS_DEPLOY);
|
|
setSupportedDeviceType(Constants::ANDROID_DEVICE_TYPE);
|
|
setRepeatable(false);
|
|
setDisplayName(Tr::tr("Deploy to Android device"));
|
|
}
|
|
|
|
} // Android::Internal
|
|
|
|
#include "androiddeployqtstep.moc"
|