2010-07-29 14:15:12 +02:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2011-01-11 16:28:15 +01:00
|
|
|
** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
2010-07-29 14:15:12 +02:00
|
|
|
** All rights reserved.
|
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
|
**
|
|
|
|
|
** This file is part of the QtDeclarative module of the Qt Toolkit.
|
|
|
|
|
**
|
|
|
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
|
|
|
** No Commercial Usage
|
|
|
|
|
** This file contains pre-release code and may not be distributed.
|
|
|
|
|
** You may use this file in accordance with the terms and conditions
|
|
|
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
|
|
|
** this package.
|
|
|
|
|
**
|
|
|
|
|
** 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, Nokia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
|
|
|
|
** Nokia at qt-info@nokia.com.
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "qdeclarativedebugclient_p.h"
|
|
|
|
|
|
|
|
|
|
#include "qpacketprotocol_p.h"
|
|
|
|
|
|
|
|
|
|
#include <QtCore/qdebug.h>
|
|
|
|
|
#include <QtCore/qstringlist.h>
|
|
|
|
|
|
|
|
|
|
namespace QmlJsDebugClient {
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
const int protocolVersion = 1;
|
|
|
|
|
const QString serverId = QLatin1String("QDeclarativeDebugServer");
|
|
|
|
|
const QString clientId = QLatin1String("QDeclarativeDebugClient");
|
|
|
|
|
|
|
|
|
|
class QDeclarativeDebugClientPrivate
|
|
|
|
|
{
|
|
|
|
|
// Q_DECLARE_PUBLIC(QDeclarativeDebugClient)
|
|
|
|
|
public:
|
|
|
|
|
QDeclarativeDebugClientPrivate();
|
|
|
|
|
|
|
|
|
|
QString name;
|
2010-11-15 17:09:28 +01:00
|
|
|
QDeclarativeDebugConnection *connection;
|
2010-09-27 12:45:56 +02:00
|
|
|
};
|
|
|
|
|
|
2010-07-29 14:15:12 +02:00
|
|
|
class QDeclarativeDebugConnectionPrivate : public QObject
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
|
|
|
|
QDeclarativeDebugConnectionPrivate(QDeclarativeDebugConnection *c);
|
|
|
|
|
QDeclarativeDebugConnection *q;
|
|
|
|
|
QPacketProtocol *protocol;
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
bool gotHello;
|
|
|
|
|
QStringList serverPlugins;
|
2010-07-29 14:15:12 +02:00
|
|
|
QHash<QString, QDeclarativeDebugClient *> plugins;
|
2010-09-27 12:45:56 +02:00
|
|
|
|
|
|
|
|
void advertisePlugins();
|
|
|
|
|
|
2010-07-29 14:15:12 +02:00
|
|
|
public Q_SLOTS:
|
|
|
|
|
void connected();
|
|
|
|
|
void readyRead();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
QDeclarativeDebugConnectionPrivate::QDeclarativeDebugConnectionPrivate(QDeclarativeDebugConnection *c)
|
2010-09-27 12:45:56 +02:00
|
|
|
: QObject(c), q(c), protocol(0), gotHello(false)
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
|
|
|
|
protocol = new QPacketProtocol(q, this);
|
|
|
|
|
QObject::connect(c, SIGNAL(connected()), this, SLOT(connected()));
|
|
|
|
|
QObject::connect(protocol, SIGNAL(readyRead()), this, SLOT(readyRead()));
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
void QDeclarativeDebugConnectionPrivate::advertisePlugins()
|
|
|
|
|
{
|
|
|
|
|
if (!q->isConnected() || !gotHello)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QPacket pack;
|
|
|
|
|
pack << serverId << 1 << plugins.keys();
|
|
|
|
|
protocol->send(pack);
|
|
|
|
|
q->flush();
|
|
|
|
|
}
|
|
|
|
|
|
2010-07-29 14:15:12 +02:00
|
|
|
void QDeclarativeDebugConnectionPrivate::connected()
|
|
|
|
|
{
|
|
|
|
|
QPacket pack;
|
2010-09-27 12:45:56 +02:00
|
|
|
pack << serverId << 0 << protocolVersion << plugins.keys();
|
2010-07-29 14:15:12 +02:00
|
|
|
protocol->send(pack);
|
2010-09-27 12:45:56 +02:00
|
|
|
q->flush();
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QDeclarativeDebugConnectionPrivate::readyRead()
|
|
|
|
|
{
|
2010-09-27 12:45:56 +02:00
|
|
|
if (!gotHello) {
|
|
|
|
|
QPacket pack = protocol->read();
|
|
|
|
|
QString name;
|
2010-07-29 14:15:12 +02:00
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
pack >> name;
|
|
|
|
|
|
|
|
|
|
bool validHello = false;
|
|
|
|
|
if (name == clientId) {
|
|
|
|
|
int op = -1;
|
|
|
|
|
pack >> op;
|
|
|
|
|
if (op == 0) {
|
|
|
|
|
int version = -1;
|
|
|
|
|
pack >> version;
|
|
|
|
|
if (version == protocolVersion) {
|
|
|
|
|
pack >> serverPlugins;
|
|
|
|
|
validHello = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!validHello) {
|
|
|
|
|
qWarning("QDeclarativeDebugConnection: Invalid hello message");
|
|
|
|
|
QObject::disconnect(protocol, SIGNAL(readyRead()), this, SLOT(readyRead()));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
gotHello = true;
|
|
|
|
|
|
|
|
|
|
QHash<QString, QDeclarativeDebugClient *>::Iterator iter = plugins.begin();
|
|
|
|
|
for (; iter != plugins.end(); ++iter) {
|
|
|
|
|
QDeclarativeDebugClient::Status newStatus = QDeclarativeDebugClient::Unavailable;
|
|
|
|
|
if (serverPlugins.contains(iter.key()))
|
|
|
|
|
newStatus = QDeclarativeDebugClient::Enabled;
|
|
|
|
|
iter.value()->statusChanged(newStatus);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (protocol->packetsAvailable()) {
|
|
|
|
|
QPacket pack = protocol->read();
|
|
|
|
|
QString name;
|
|
|
|
|
pack >> name;
|
|
|
|
|
|
|
|
|
|
if (name == clientId) {
|
|
|
|
|
int op = -1;
|
|
|
|
|
pack >> op;
|
|
|
|
|
|
|
|
|
|
if (op == 1) {
|
|
|
|
|
// Service Discovery
|
|
|
|
|
QStringList oldServerPlugins = serverPlugins;
|
|
|
|
|
pack >> serverPlugins;
|
|
|
|
|
|
|
|
|
|
QHash<QString, QDeclarativeDebugClient *>::Iterator iter = plugins.begin();
|
|
|
|
|
for (; iter != plugins.end(); ++iter) {
|
|
|
|
|
const QString pluginName = iter.key();
|
|
|
|
|
QDeclarativeDebugClient::Status newStatus = QDeclarativeDebugClient::Unavailable;
|
|
|
|
|
if (serverPlugins.contains(pluginName))
|
|
|
|
|
newStatus = QDeclarativeDebugClient::Enabled;
|
|
|
|
|
|
|
|
|
|
if (oldServerPlugins.contains(pluginName)
|
|
|
|
|
!= serverPlugins.contains(pluginName)) {
|
|
|
|
|
iter.value()->statusChanged(newStatus);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
qWarning() << "QDeclarativeDebugConnection: Unknown control message id" << op;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
QByteArray message;
|
|
|
|
|
pack >> message;
|
|
|
|
|
|
|
|
|
|
QHash<QString, QDeclarativeDebugClient *>::Iterator iter =
|
|
|
|
|
plugins.find(name);
|
|
|
|
|
if (iter == plugins.end()) {
|
|
|
|
|
qWarning() << "QDeclarativeDebugConnection: Message received for missing plugin" << name;
|
|
|
|
|
} else {
|
|
|
|
|
(*iter)->messageReceived(message);
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QDeclarativeDebugConnection::QDeclarativeDebugConnection(QObject *parent)
|
|
|
|
|
: QTcpSocket(parent), d(new QDeclarativeDebugConnectionPrivate(this))
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
QDeclarativeDebugConnection::~QDeclarativeDebugConnection()
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
2010-09-27 12:45:56 +02:00
|
|
|
QHash<QString, QDeclarativeDebugClient*>::iterator iter = d->plugins.begin();
|
|
|
|
|
for (; iter != d->plugins.end(); ++iter) {
|
2010-11-15 17:09:28 +01:00
|
|
|
iter.value()->d_func()->connection = 0;
|
2010-09-27 12:45:56 +02:00
|
|
|
iter.value()->statusChanged(QDeclarativeDebugClient::NotConnected);
|
|
|
|
|
}
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
bool QDeclarativeDebugConnection::isConnected() const
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
2010-09-27 12:45:56 +02:00
|
|
|
return state() == ConnectedState;
|
|
|
|
|
}
|
2010-07-29 14:15:12 +02:00
|
|
|
|
|
|
|
|
QDeclarativeDebugClientPrivate::QDeclarativeDebugClientPrivate()
|
2010-11-15 17:09:28 +01:00
|
|
|
: connection(0)
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QDeclarativeDebugClient::QDeclarativeDebugClient(const QString &name,
|
|
|
|
|
QDeclarativeDebugConnection *parent)
|
2010-11-15 17:09:28 +01:00
|
|
|
: QObject(parent), d_ptr(new QDeclarativeDebugClientPrivate())
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
2010-11-15 17:09:28 +01:00
|
|
|
Q_D(QDeclarativeDebugClient);
|
2010-07-29 14:15:12 +02:00
|
|
|
d->name = name;
|
2010-11-15 17:09:28 +01:00
|
|
|
d->connection = parent;
|
2010-07-29 14:15:12 +02:00
|
|
|
|
2010-11-15 17:09:28 +01:00
|
|
|
if (!d->connection)
|
2010-07-29 14:15:12 +02:00
|
|
|
return;
|
|
|
|
|
|
2010-11-15 17:09:28 +01:00
|
|
|
if (d->connection->d->plugins.contains(name)) {
|
2010-07-29 14:15:12 +02:00
|
|
|
qWarning() << "QDeclarativeDebugClient: Conflicting plugin name" << name;
|
2010-11-15 17:09:28 +01:00
|
|
|
d->connection = 0;
|
2010-07-29 14:15:12 +02:00
|
|
|
} else {
|
2010-11-15 17:09:28 +01:00
|
|
|
d->connection->d->plugins.insert(name, this);
|
|
|
|
|
d->connection->d->advertisePlugins();
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
QDeclarativeDebugClient::~QDeclarativeDebugClient()
|
|
|
|
|
{
|
2010-11-15 17:09:28 +01:00
|
|
|
Q_D(QDeclarativeDebugClient);
|
|
|
|
|
if (d->connection && d->connection->d) {
|
|
|
|
|
d->connection->d->plugins.remove(d->name);
|
|
|
|
|
d->connection->d->advertisePlugins();
|
2010-09-27 12:45:56 +02:00
|
|
|
}
|
|
|
|
|
}
|
2010-07-29 14:15:12 +02:00
|
|
|
|
|
|
|
|
QString QDeclarativeDebugClient::name() const
|
|
|
|
|
{
|
2010-11-15 17:09:28 +01:00
|
|
|
Q_D(const QDeclarativeDebugClient);
|
2010-07-29 14:15:12 +02:00
|
|
|
return d->name;
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
QDeclarativeDebugClient::Status QDeclarativeDebugClient::status() const
|
2010-07-29 14:15:12 +02:00
|
|
|
{
|
2010-11-15 17:09:28 +01:00
|
|
|
Q_D(const QDeclarativeDebugClient);
|
|
|
|
|
if (!d->connection
|
|
|
|
|
|| !d->connection->isConnected()
|
|
|
|
|
|| !d->connection->d->gotHello)
|
2010-09-27 12:45:56 +02:00
|
|
|
return NotConnected;
|
2010-07-29 14:15:12 +02:00
|
|
|
|
2010-11-15 17:09:28 +01:00
|
|
|
if (d->connection->d->serverPlugins.contains(d->name))
|
2010-09-27 12:45:56 +02:00
|
|
|
return Enabled;
|
2010-07-29 14:15:12 +02:00
|
|
|
|
2010-09-27 12:45:56 +02:00
|
|
|
return Unavailable;
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QDeclarativeDebugClient::sendMessage(const QByteArray &message)
|
|
|
|
|
{
|
2010-11-15 17:09:28 +01:00
|
|
|
Q_D(QDeclarativeDebugClient);
|
2010-09-27 12:45:56 +02:00
|
|
|
if (status() != Enabled)
|
2010-07-29 14:15:12 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QPacket pack;
|
|
|
|
|
pack << d->name << message;
|
2010-11-15 17:09:28 +01:00
|
|
|
d->connection->d->protocol->send(pack);
|
|
|
|
|
d->connection->flush();
|
2010-09-27 12:45:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QDeclarativeDebugClient::statusChanged(Status)
|
|
|
|
|
{
|
2010-07-29 14:15:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QDeclarativeDebugClient::messageReceived(const QByteArray &)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-16 10:02:30 +01:00
|
|
|
} // namespace QmlJsDebugClient
|
2010-07-29 14:15:12 +02:00
|
|
|
|
|
|
|
|
#include <qdeclarativedebugclient.moc>
|