7429791309
Show times where 0 is the beginning of the process Use QDebug to get QVariant's values Delay constructor print so that we know which class is being constructed Use QJsonObject instead of going back and forth with QVariantMap Reviewed by David Edmundson
142 lines
4.4 KiB
C++
142 lines
4.4 KiB
C++
/*
|
|
* Copyright 2014 by Aleix Pol Gonzalez <aleixpol@blue-systems.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU Library General Public License as
|
|
* published by the Free Software Foundation; either version 2, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this program; if not, write to the
|
|
* Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "timetracker.h"
|
|
#include <containment.h>
|
|
#include <QDateTime>
|
|
#include <qmetaobject.h>
|
|
#include <QFile>
|
|
#include <QDebug>
|
|
#include <qjsondocument.h>
|
|
#include <qjsonarray.h>
|
|
#include <QTimer>
|
|
|
|
using namespace Plasma;
|
|
|
|
Q_GLOBAL_STATIC_WITH_ARGS(const qint64, s_beginning, (QDateTime::currentDateTime().toMSecsSinceEpoch()))
|
|
|
|
struct TimeTrackerWriter : QObject {
|
|
Q_OBJECT
|
|
public:
|
|
TimeTrackerWriter() {
|
|
QObject::connect(QCoreApplication::instance(), &QCoreApplication::aboutToQuit, this, static_cast<void (TimeTrackerWriter::*)()>(&TimeTrackerWriter::print));
|
|
}
|
|
|
|
void print() {
|
|
QJsonArray array;
|
|
|
|
Q_FOREACH(const ObjectHistory& history, m_data) {
|
|
array.append(QJsonObject {
|
|
{ QStringLiteral("events"), serializeEvents(history.events) },
|
|
{ QStringLiteral("initial"), QJsonValue::fromVariant(history.initial) }
|
|
});
|
|
}
|
|
Q_ASSERT(array.count() == m_data.count());
|
|
QJsonDocument doc;
|
|
doc.setArray(array);
|
|
|
|
QFile f(QStringLiteral("/tmp/debug-")+qgetenv("USER")+".json");
|
|
bool b = f.open(QFile::WriteOnly);
|
|
Q_ASSERT(b);
|
|
f.write(doc.toJson());
|
|
}
|
|
|
|
void feed(QObject* obj, const ObjectHistory& tracker)
|
|
{
|
|
m_data[obj] = tracker;
|
|
}
|
|
|
|
QHash<QObject*, ObjectHistory> m_data;
|
|
|
|
private:
|
|
QJsonArray serializeEvents(const QVector<TimeEvent>& events) const {
|
|
QJsonArray ret;
|
|
Q_ASSERT(!events.isEmpty());
|
|
foreach(const TimeEvent& ev, events) {
|
|
ret.append(QJsonObject {
|
|
{ QStringLiteral("comment"), ev.comment },
|
|
{ QStringLiteral("time"), ev.moment.toMSecsSinceEpoch() - *s_beginning }
|
|
});
|
|
}
|
|
Q_ASSERT(ret.count() == events.count());
|
|
return ret;
|
|
}
|
|
};
|
|
Q_GLOBAL_STATIC(TimeTrackerWriter, s_writer)
|
|
|
|
TimeTracker::TimeTracker(QObject* o)
|
|
: QObject(o)
|
|
{
|
|
*s_beginning * 1; // ensure it's initialized
|
|
|
|
QTimer* t = new QTimer(this);
|
|
t->setInterval(2000);
|
|
t->setSingleShot(false);
|
|
connect(t, SIGNAL(timeout()), this, SLOT(sync()));
|
|
t->start();
|
|
|
|
QMetaObject::invokeMethod(this, "init", Qt::QueuedConnection);
|
|
}
|
|
|
|
void TimeTracker::init()
|
|
{
|
|
m_history.events.append(TimeEvent { QDateTime::currentDateTime(), QStringLiteral("constructed %1 %2").arg(parent()->metaObject()->className(), parent()->objectName()) });
|
|
|
|
QMetaMethod propChange = metaObject()->method(metaObject()->indexOfSlot("propertyChanged()"));
|
|
Q_ASSERT(propChange.isValid() && metaObject()->indexOfSlot("propertyChanged()")>=0);
|
|
|
|
QObject* o = parent();
|
|
for (int i = 0, pc = o->metaObject()->propertyCount(); i<pc; ++i) {
|
|
QMetaProperty prop = o->metaObject()->property(i);
|
|
m_history.initial[prop.name()] = prop.read(o);
|
|
|
|
if (prop.hasNotifySignal())
|
|
connect(o, prop.notifySignal(), this, propChange);
|
|
}
|
|
}
|
|
|
|
TimeTracker::~TimeTracker()
|
|
{
|
|
sync();
|
|
}
|
|
|
|
void TimeTracker::sync()
|
|
{
|
|
s_writer->feed(parent(), m_history);
|
|
}
|
|
|
|
void TimeTracker::propertyChanged()
|
|
{
|
|
Q_ASSERT(sender() == parent());
|
|
|
|
const QMetaObject* mo = parent()->metaObject();
|
|
|
|
for (int i = 0, pc = mo->propertyCount(); i<pc; ++i) {
|
|
const QMetaProperty prop = mo->property(i);
|
|
if (prop.notifySignalIndex() == senderSignalIndex()) {
|
|
QString val;
|
|
QDebug d(&val);
|
|
d << prop.read(parent());
|
|
m_history.events.append(TimeEvent { QDateTime::currentDateTime(), QStringLiteral("property %1 changed to %2").arg(prop.name(), val.trimmed())});
|
|
}
|
|
}
|
|
}
|
|
|
|
#include "timetracker.moc"
|