Aleix Pol 7429791309 Improve timetracker output
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
2016-09-29 12:32:11 +02:00

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"