2013-04-25 16:02:17 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2013-04-25 16:02:17 +02:00
|
|
|
**
|
|
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-04-25 16:02:17 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-04-25 16:02:17 +02:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "iosdeploystepwidget.h"
|
|
|
|
|
#include "iosdeploystep.h"
|
|
|
|
|
#include "iosbuildstep.h"
|
|
|
|
|
#include "iosconstants.h"
|
|
|
|
|
#include "iosrunconfiguration.h"
|
|
|
|
|
#include "iostoolhandler.h"
|
|
|
|
|
|
|
|
|
|
#include <coreplugin/messagemanager.h>
|
|
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
|
|
|
#include <projectexplorer/target.h>
|
2013-11-04 22:45:52 +01:00
|
|
|
#include <projectexplorer/taskhub.h>
|
2014-05-13 16:09:11 +02:00
|
|
|
#include <projectexplorer/kitmanager.h>
|
|
|
|
|
#include <projectexplorer/kitinformation.h>
|
|
|
|
|
#include <projectexplorer/devicesupport/devicemanager.h>
|
2013-04-25 16:02:17 +02:00
|
|
|
|
|
|
|
|
#include <qtsupport/qtkitinformation.h>
|
|
|
|
|
|
2017-01-19 16:44:22 +01:00
|
|
|
#include <utils/temporaryfile.h>
|
|
|
|
|
|
2013-04-25 16:02:17 +02:00
|
|
|
#include <QDir>
|
2014-05-08 13:43:25 +02:00
|
|
|
#include <QFile>
|
|
|
|
|
#include <QSettings>
|
2013-04-25 16:02:17 +02:00
|
|
|
|
|
|
|
|
#define ASSERT_STATE(state) ASSERT_STATE_GENERIC(State, state, m_state)
|
|
|
|
|
|
|
|
|
|
using namespace ProjectExplorer;
|
|
|
|
|
|
|
|
|
|
namespace Ios {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
const Core::Id IosDeployStep::Id("Qt4ProjectManager.IosDeployStep");
|
|
|
|
|
|
2015-02-03 23:49:46 +02:00
|
|
|
IosDeployStep::IosDeployStep(BuildStepList *parent)
|
2013-04-25 16:02:17 +02:00
|
|
|
: BuildStep(parent, Id)
|
|
|
|
|
{
|
2018-10-22 17:56:04 +02:00
|
|
|
setImmutable(true);
|
2018-10-22 19:11:59 +02:00
|
|
|
setRunInGuiThread(true);
|
2014-05-13 16:09:11 +02:00
|
|
|
updateDisplayNames();
|
2016-06-29 19:35:23 +03:00
|
|
|
connect(DeviceManager::instance(), &DeviceManager::updated,
|
|
|
|
|
this, &IosDeployStep::updateDisplayNames);
|
|
|
|
|
connect(target(), &Target::kitChanged,
|
|
|
|
|
this, &IosDeployStep::updateDisplayNames);
|
2014-05-13 16:09:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::updateDisplayNames()
|
|
|
|
|
{
|
2015-02-03 23:49:46 +02:00
|
|
|
IDevice::ConstPtr dev =
|
|
|
|
|
DeviceKitInformation::device(target()->kit());
|
2014-05-13 16:09:11 +02:00
|
|
|
const QString devName = dev.isNull() ? IosDevice::name() : dev->displayName();
|
2013-04-25 16:02:17 +02:00
|
|
|
setDefaultDisplayName(tr("Deploy to %1").arg(devName));
|
2014-05-13 16:09:11 +02:00
|
|
|
setDisplayName(tr("Deploy to %1").arg(devName));
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
2015-11-13 12:19:35 +01:00
|
|
|
bool IosDeployStep::init(QList<const BuildStep *> &earlierSteps)
|
2013-04-25 16:02:17 +02:00
|
|
|
{
|
2015-11-13 12:19:35 +01:00
|
|
|
Q_UNUSED(earlierSteps);
|
2014-05-13 16:09:11 +02:00
|
|
|
QTC_ASSERT(m_transferStatus == NoTransfer, return false);
|
2015-02-03 23:49:46 +02:00
|
|
|
m_device = DeviceKitInformation::device(target()->kit());
|
2018-11-12 19:55:59 +01:00
|
|
|
auto runConfig = qobject_cast<const IosRunConfiguration *>(
|
|
|
|
|
this->target()->activeRunConfiguration());
|
2014-05-17 22:53:00 +02:00
|
|
|
QTC_ASSERT(runConfig, return false);
|
2014-04-10 11:35:36 +02:00
|
|
|
m_bundlePath = runConfig->bundleDirectory().toString();
|
2016-09-26 12:40:09 +02:00
|
|
|
|
|
|
|
|
if (iosdevice()) {
|
|
|
|
|
m_deviceType = IosDeviceType(IosDeviceType::IosDevice, deviceId());
|
|
|
|
|
} else if (iossimulator()) {
|
|
|
|
|
m_deviceType = runConfig->deviceType();
|
|
|
|
|
} else {
|
2013-10-07 16:07:16 +02:00
|
|
|
emit addOutput(tr("Error: no device available, deploy failed."),
|
2017-01-12 10:59:12 +01:00
|
|
|
BuildStep::OutputFormat::ErrorMessage);
|
2013-10-07 16:07:16 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
2013-04-25 16:02:17 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::run(QFutureInterface<bool> &fi)
|
|
|
|
|
{
|
|
|
|
|
m_futureInterface = fi;
|
|
|
|
|
QTC_CHECK(m_transferStatus == NoTransfer);
|
2016-09-26 12:40:09 +02:00
|
|
|
if (device().isNull()) {
|
|
|
|
|
TaskHub::addTask(Task::Error, tr("Deployment failed. No iOS device found."),
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2016-04-20 12:49:25 +02:00
|
|
|
reportRunResult(m_futureInterface, !iossimulator().isNull());
|
2013-04-25 16:02:17 +02:00
|
|
|
cleanup();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-09-26 12:40:09 +02:00
|
|
|
m_toolHandler = new IosToolHandler(m_deviceType, this);
|
2013-04-25 16:02:17 +02:00
|
|
|
m_transferStatus = TransferInProgress;
|
|
|
|
|
m_futureInterface.setProgressRange(0, 200);
|
|
|
|
|
m_futureInterface.setProgressValueAndText(0, QLatin1String("Transferring application"));
|
|
|
|
|
m_futureInterface.reportStarted();
|
2016-06-29 19:35:23 +03:00
|
|
|
connect(m_toolHandler, &IosToolHandler::isTransferringApp,
|
|
|
|
|
this, &IosDeployStep::handleIsTransferringApp);
|
|
|
|
|
connect(m_toolHandler, &IosToolHandler::didTransferApp,
|
|
|
|
|
this, &IosDeployStep::handleDidTransferApp);
|
|
|
|
|
connect(m_toolHandler, &IosToolHandler::finished,
|
|
|
|
|
this, &IosDeployStep::handleFinished);
|
|
|
|
|
connect(m_toolHandler, &IosToolHandler::errorMsg,
|
|
|
|
|
this, &IosDeployStep::handleErrorMsg);
|
2014-05-08 13:43:25 +02:00
|
|
|
checkProvisioningProfile();
|
2016-09-26 12:40:09 +02:00
|
|
|
m_toolHandler->requestTransferApp(appBundle(), m_deviceType.identifier);
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::cancel()
|
|
|
|
|
{
|
|
|
|
|
if (m_toolHandler)
|
|
|
|
|
m_toolHandler->stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::cleanup()
|
|
|
|
|
{
|
|
|
|
|
QTC_CHECK(m_transferStatus != TransferInProgress);
|
|
|
|
|
m_transferStatus = NoTransfer;
|
|
|
|
|
m_device.clear();
|
2016-09-26 12:40:09 +02:00
|
|
|
m_toolHandler = nullptr;
|
2014-05-08 13:43:25 +02:00
|
|
|
m_expectFail = false;
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::handleIsTransferringApp(IosToolHandler *handler, const QString &bundlePath,
|
|
|
|
|
const QString &deviceId, int progress, int maxProgress,
|
|
|
|
|
const QString &info)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(handler); Q_UNUSED(bundlePath); Q_UNUSED(deviceId);
|
|
|
|
|
QTC_CHECK(m_transferStatus == TransferInProgress);
|
|
|
|
|
m_futureInterface.setProgressRange(0, maxProgress);
|
|
|
|
|
m_futureInterface.setProgressValueAndText(progress, info);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::handleDidTransferApp(IosToolHandler *handler, const QString &bundlePath,
|
|
|
|
|
const QString &deviceId, IosToolHandler::OpStatus status)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(handler); Q_UNUSED(bundlePath); Q_UNUSED(deviceId);
|
|
|
|
|
QTC_CHECK(m_transferStatus == TransferInProgress);
|
2013-11-04 22:45:52 +01:00
|
|
|
if (status == IosToolHandler::Success) {
|
2013-04-25 16:02:17 +02:00
|
|
|
m_transferStatus = TransferOk;
|
2013-11-04 22:45:52 +01:00
|
|
|
} else {
|
2013-04-25 16:02:17 +02:00
|
|
|
m_transferStatus = TransferFailed;
|
2014-05-08 13:43:25 +02:00
|
|
|
if (!m_expectFail)
|
|
|
|
|
TaskHub::addTask(Task::Error,
|
2016-04-13 17:06:37 +02:00
|
|
|
tr("Deployment failed. The settings in the Devices window of Xcode might be incorrect."),
|
2014-05-08 13:43:25 +02:00
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2013-11-04 22:45:52 +01:00
|
|
|
}
|
2016-04-20 12:49:25 +02:00
|
|
|
reportRunResult(m_futureInterface, status == IosToolHandler::Success);
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::handleFinished(IosToolHandler *handler)
|
|
|
|
|
{
|
|
|
|
|
switch (m_transferStatus) {
|
|
|
|
|
case TransferInProgress:
|
|
|
|
|
m_transferStatus = TransferFailed;
|
2013-11-04 22:45:52 +01:00
|
|
|
TaskHub::addTask(Task::Error, tr("Deployment failed."),
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2016-04-20 12:49:25 +02:00
|
|
|
reportRunResult(m_futureInterface, false);
|
2013-04-25 16:02:17 +02:00
|
|
|
break;
|
|
|
|
|
case NoTransfer:
|
|
|
|
|
case TransferOk:
|
|
|
|
|
case TransferFailed:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
cleanup();
|
|
|
|
|
handler->deleteLater();
|
|
|
|
|
// move it when result is reported? (would need care to avoid problems with concurrent runs)
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-07 16:07:16 +02:00
|
|
|
void IosDeployStep::handleErrorMsg(IosToolHandler *handler, const QString &msg)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(handler);
|
2013-11-04 22:45:52 +01:00
|
|
|
if (msg.contains(QLatin1String("AMDeviceInstallApplication returned -402653103")))
|
|
|
|
|
TaskHub::addTask(Task::Warning,
|
|
|
|
|
tr("The Info.plist might be incorrect."),
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2017-01-12 10:59:12 +01:00
|
|
|
emit addOutput(msg, BuildStep::OutputFormat::ErrorMessage);
|
2013-10-07 16:07:16 +02:00
|
|
|
}
|
|
|
|
|
|
2013-04-25 16:02:17 +02:00
|
|
|
BuildStepConfigWidget *IosDeployStep::createConfigWidget()
|
|
|
|
|
{
|
|
|
|
|
return new IosDeployStepWidget(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IosDeployStep::fromMap(const QVariantMap &map)
|
|
|
|
|
{
|
2015-02-03 23:49:46 +02:00
|
|
|
return BuildStep::fromMap(map);
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariantMap IosDeployStep::toMap() const
|
|
|
|
|
{
|
2015-02-03 23:49:46 +02:00
|
|
|
QVariantMap map = BuildStep::toMap();
|
2013-04-25 16:02:17 +02:00
|
|
|
return map;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString IosDeployStep::deviceId() const
|
|
|
|
|
{
|
|
|
|
|
if (iosdevice().isNull())
|
|
|
|
|
return QString();
|
|
|
|
|
return iosdevice()->uniqueDeviceID();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString IosDeployStep::appBundle() const
|
|
|
|
|
{
|
2014-02-14 00:48:40 +01:00
|
|
|
return m_bundlePath;
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::raiseError(const QString &errorString)
|
|
|
|
|
{
|
|
|
|
|
emit addTask(Task(Task::Error, errorString, Utils::FileName::fromString(QString()), -1,
|
2013-11-04 22:45:52 +01:00
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT));
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::writeOutput(const QString &text, OutputFormat format)
|
|
|
|
|
{
|
|
|
|
|
emit addOutput(text, format);
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 13:43:25 +02:00
|
|
|
void IosDeployStep::checkProvisioningProfile()
|
|
|
|
|
{
|
|
|
|
|
IosDevice::ConstPtr device = iosdevice();
|
|
|
|
|
if (device.isNull())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
Utils::FileName provisioningFilePath = Utils::FileName::fromString(appBundle());
|
|
|
|
|
provisioningFilePath.appendPath(QLatin1String("embedded.mobileprovision"));
|
|
|
|
|
|
|
|
|
|
// the file is a signed plist stored in DER format
|
|
|
|
|
// we simply search for start and end of the plist instead of decoding the DER payload
|
2014-10-24 13:15:54 +02:00
|
|
|
if (!provisioningFilePath.exists())
|
2014-05-08 13:43:25 +02:00
|
|
|
return;
|
|
|
|
|
QFile provisionFile(provisioningFilePath.toString());
|
|
|
|
|
if (!provisionFile.open(QIODevice::ReadOnly))
|
|
|
|
|
return;
|
|
|
|
|
QByteArray provisionData = provisionFile.readAll();
|
|
|
|
|
int start = provisionData.indexOf("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
|
|
|
|
|
int end = provisionData.indexOf("</plist>");
|
|
|
|
|
if (start == -1 || end == -1)
|
|
|
|
|
return;
|
|
|
|
|
end += 8;
|
|
|
|
|
|
2017-01-19 16:44:22 +01:00
|
|
|
Utils::TemporaryFile f("iosdeploy");
|
2014-05-08 13:43:25 +02:00
|
|
|
if (!f.open())
|
|
|
|
|
return;
|
|
|
|
|
f.write(provisionData.mid(start, end - start));
|
|
|
|
|
f.flush();
|
|
|
|
|
QSettings provisionPlist(f.fileName(), QSettings::NativeFormat);
|
|
|
|
|
|
|
|
|
|
if (!provisionPlist.contains(QLatin1String("ProvisionedDevices")))
|
|
|
|
|
return;
|
|
|
|
|
QStringList deviceIds = provisionPlist.value(QLatin1String("ProvisionedDevices")).toStringList();
|
|
|
|
|
QString targetId = device->uniqueDeviceID();
|
|
|
|
|
foreach (const QString &deviceId, deviceIds) {
|
|
|
|
|
if (deviceId == targetId)
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_expectFail = true;
|
|
|
|
|
QString provisioningProfile = provisionPlist.value(QLatin1String("Name")).toString();
|
|
|
|
|
QString provisioningUid = provisionPlist.value(QLatin1String("UUID")).toString();
|
|
|
|
|
Task task(Task::Warning,
|
|
|
|
|
tr("The provisioning profile \"%1\" (%2) used to sign the application "
|
|
|
|
|
"does not cover the device %3 (%4). Deployment to it will fail.")
|
|
|
|
|
.arg(provisioningProfile, provisioningUid, device->displayName(),
|
|
|
|
|
targetId),
|
|
|
|
|
Utils::FileName(), /* filename */
|
|
|
|
|
-1, /* line */
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_COMPILE);
|
|
|
|
|
emit addTask(task);
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-25 16:02:17 +02:00
|
|
|
IDevice::ConstPtr IosDeployStep::device() const
|
|
|
|
|
{
|
|
|
|
|
return m_device;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IosDevice::ConstPtr IosDeployStep::iosdevice() const
|
|
|
|
|
{
|
|
|
|
|
return m_device.dynamicCast<const IosDevice>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IosSimulator::ConstPtr IosDeployStep::iossimulator() const
|
|
|
|
|
{
|
|
|
|
|
return m_device.dynamicCast<const IosSimulator>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
2013-10-16 11:02:37 +02:00
|
|
|
} // namespace Ios
|