2013-04-25 16:02:17 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2013-04-25 16:02:17 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
|
|
|
|
**
|
|
|
|
|
** 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 Digia. For licensing terms and
|
|
|
|
|
** conditions see http://qt.digia.com/licensing. For further information
|
|
|
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
|
|
|
**
|
|
|
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "iosdeploystepwidget.h"
|
|
|
|
|
#include "iosdeploystep.h"
|
|
|
|
|
#include "iosbuildstep.h"
|
|
|
|
|
#include "iosconstants.h"
|
|
|
|
|
#include "iosrunconfiguration.h"
|
|
|
|
|
#include "iosmanager.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>
|
2013-10-29 16:19:24 +01:00
|
|
|
#include <qmakeprojectmanager/qmakebuildconfiguration.h>
|
|
|
|
|
#include <qmakeprojectmanager/qmakeproject.h>
|
|
|
|
|
#include <qmakeprojectmanager/qmakenodes.h>
|
2013-04-25 16:02:17 +02:00
|
|
|
|
|
|
|
|
#include <qtsupport/qtkitinformation.h>
|
|
|
|
|
|
|
|
|
|
#include <QDir>
|
2014-05-08 13:43:25 +02:00
|
|
|
#include <QTemporaryFile>
|
|
|
|
|
#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;
|
2013-10-16 11:02:37 +02:00
|
|
|
using namespace QmakeProjectManager;
|
2013-04-25 16:02:17 +02:00
|
|
|
|
|
|
|
|
namespace Ios {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
const Core::Id IosDeployStep::Id("Qt4ProjectManager.IosDeployStep");
|
|
|
|
|
|
|
|
|
|
IosDeployStep::IosDeployStep(ProjectExplorer::BuildStepList *parent)
|
|
|
|
|
: BuildStep(parent, Id)
|
2014-05-08 13:43:25 +02:00
|
|
|
, m_expectFail(false)
|
2013-04-25 16:02:17 +02:00
|
|
|
{
|
|
|
|
|
ctor();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IosDeployStep::IosDeployStep(ProjectExplorer::BuildStepList *parent,
|
|
|
|
|
IosDeployStep *other)
|
|
|
|
|
: BuildStep(parent, other)
|
2014-05-08 13:43:25 +02:00
|
|
|
, m_expectFail(false)
|
2013-04-25 16:02:17 +02:00
|
|
|
{
|
|
|
|
|
ctor();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IosDeployStep::~IosDeployStep() { }
|
|
|
|
|
|
|
|
|
|
void IosDeployStep::ctor()
|
|
|
|
|
{
|
2013-11-28 14:32:43 +01:00
|
|
|
m_toolHandler = 0;
|
2013-04-25 16:02:17 +02:00
|
|
|
m_transferStatus = NoTransfer;
|
|
|
|
|
m_device = ProjectExplorer::DeviceKitInformation::device(target()->kit());
|
2013-11-04 15:08:41 +01:00
|
|
|
const QString devName = m_device.isNull() ? IosDevice::name() : m_device->displayName();
|
2013-04-25 16:02:17 +02:00
|
|
|
setDefaultDisplayName(tr("Deploy to %1").arg(devName));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IosDeployStep::init()
|
|
|
|
|
{
|
|
|
|
|
QTC_CHECK(m_transferStatus == NoTransfer);
|
|
|
|
|
m_device = ProjectExplorer::DeviceKitInformation::device(target()->kit());
|
2014-02-14 00:48:40 +01:00
|
|
|
IosRunConfiguration * runConfig = qobject_cast<IosRunConfiguration *>(
|
|
|
|
|
this->target()->activeRunConfiguration());
|
|
|
|
|
QTC_CHECK(runConfig);
|
|
|
|
|
m_bundlePath = runConfig->bundleDir().toString();
|
2013-10-07 16:07:16 +02:00
|
|
|
if (m_device.isNull()) {
|
|
|
|
|
emit addOutput(tr("Error: no device available, deploy failed."),
|
|
|
|
|
BuildStep::ErrorMessageOutput);
|
|
|
|
|
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);
|
|
|
|
|
if (iosdevice().isNull()) {
|
2013-11-04 22:45:52 +01:00
|
|
|
if (iossimulator().isNull())
|
|
|
|
|
TaskHub::addTask(Task::Error, tr("Deployment failed. No iOS device found."),
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2013-04-25 16:02:17 +02:00
|
|
|
m_futureInterface.reportResult(!iossimulator().isNull());
|
|
|
|
|
cleanup();
|
2013-11-28 14:32:43 +01:00
|
|
|
emit finished();
|
2013-04-25 16:02:17 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
m_transferStatus = TransferInProgress;
|
2013-11-28 14:32:43 +01:00
|
|
|
QTC_CHECK(m_toolHandler == 0);
|
2014-03-28 19:05:35 +01:00
|
|
|
m_toolHandler = new IosToolHandler(IosDeviceType::IosDevice, this);
|
2013-04-25 16:02:17 +02:00
|
|
|
m_futureInterface.setProgressRange(0, 200);
|
|
|
|
|
m_futureInterface.setProgressValueAndText(0, QLatin1String("Transferring application"));
|
|
|
|
|
m_futureInterface.reportStarted();
|
2013-11-28 14:32:43 +01:00
|
|
|
connect(m_toolHandler, SIGNAL(isTransferringApp(Ios::IosToolHandler*,QString,QString,int,int,QString)),
|
2013-04-25 16:02:17 +02:00
|
|
|
SLOT(handleIsTransferringApp(Ios::IosToolHandler*,QString,QString,int,int,QString)));
|
2013-11-28 14:32:43 +01:00
|
|
|
connect(m_toolHandler, SIGNAL(didTransferApp(Ios::IosToolHandler*,QString,QString,Ios::IosToolHandler::OpStatus)),
|
2013-04-25 16:02:17 +02:00
|
|
|
SLOT(handleDidTransferApp(Ios::IosToolHandler*,QString,QString,Ios::IosToolHandler::OpStatus)));
|
2013-11-28 14:32:43 +01:00
|
|
|
connect(m_toolHandler, SIGNAL(finished(Ios::IosToolHandler*)),
|
2013-04-25 16:02:17 +02:00
|
|
|
SLOT(handleFinished(Ios::IosToolHandler*)));
|
2013-11-28 14:32:43 +01:00
|
|
|
connect(m_toolHandler, SIGNAL(errorMsg(Ios::IosToolHandler*,QString)),
|
2013-10-07 16:07:16 +02:00
|
|
|
SLOT(handleErrorMsg(Ios::IosToolHandler*,QString)));
|
2014-05-08 13:43:25 +02:00
|
|
|
checkProvisioningProfile();
|
2013-11-28 14:32:43 +01:00
|
|
|
m_toolHandler->requestTransferApp(appBundle(), deviceId());
|
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();
|
|
|
|
|
m_toolHandler = 0;
|
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,
|
|
|
|
|
tr("Deployment failed. The settings in the Organizer window of Xcode might be incorrect."),
|
|
|
|
|
ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT);
|
2013-11-04 22:45:52 +01:00
|
|
|
}
|
2013-04-25 16:02:17 +02:00
|
|
|
m_futureInterface.reportResult(status == IosToolHandler::Success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
2013-04-25 16:02:17 +02:00
|
|
|
m_futureInterface.reportResult(false);
|
|
|
|
|
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-11-28 14:32:43 +01:00
|
|
|
emit finished();
|
2013-04-25 16:02:17 +02:00
|
|
|
}
|
|
|
|
|
|
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);
|
2013-10-07 16:07:16 +02:00
|
|
|
emit addOutput(msg, BuildStep::ErrorMessageOutput);
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-25 16:02:17 +02:00
|
|
|
BuildStepConfigWidget *IosDeployStep::createConfigWidget()
|
|
|
|
|
{
|
|
|
|
|
return new IosDeployStepWidget(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IosDeployStep::fromMap(const QVariantMap &map)
|
|
|
|
|
{
|
|
|
|
|
return ProjectExplorer::BuildStep::fromMap(map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariantMap IosDeployStep::toMap() const
|
|
|
|
|
{
|
|
|
|
|
QVariantMap map = ProjectExplorer::BuildStep::toMap();
|
|
|
|
|
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
|
|
|
|
|
if (!provisioningFilePath.toFileInfo().exists())
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
QTemporaryFile f;
|
|
|
|
|
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
|