-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbackupper.cpp
138 lines (125 loc) · 5.32 KB
/
backupper.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
#include <QSettings>
#include <QDebug>
#include "task.h"
#include "backupper.h"
Backupper::Backupper() {
// search for backup dir
QString path = QFileInfo(QSettings(QSettings::IniFormat, QSettings::UserScope, "ZanzaSoft", "SAK").fileName()).path() + QDir::separator() + "sakbcks";
m_dir =QDir(path);
m_dir.mkpath(path);
m_dir.cd(path);
QStringList list = m_dir.entryList();
foreach(QString s, list) {
QString fullSuffix = QFileInfo(s).completeSuffix();
QString baseName = QFileInfo(s).baseName();
if (fullSuffix.count() >= 19)
fullSuffix = fullSuffix.mid(fullSuffix.count()-19);
if (fullSuffix.count() == 19 && fullSuffix[14]=='.' && fullSuffix.mid(15) == "cbck") {
QDateTime time = QDateTime::fromString(fullSuffix.left(14), BACKUPDATEFORMAT);
qDebug() << "SAK: found cyclic backup " << s << " on date " << time;
m_cyclicBackups[baseName][time]= s;
}
}
}
void Backupper::doCyclicBackup() {
// build the list of files to be backupped
QStringList backupPaths;
QSettings settings(QSettings::IniFormat, QSettings::UserScope, "ZanzaSoft", "SAK");
backupPaths << m_dir.filePath(QFileInfo(settings.fileName()).baseName());
QDir saveDir(QFileInfo(settings.fileName()).dir());
saveDir.mkdir("SakTasks");
saveDir.cd("SakTasks");
QStringList nameFilters;
nameFilters << "*.xml";
QStringList files = saveDir.entryList( nameFilters, QDir::Files);
foreach(QString file, files) {
backupPaths << m_dir.filePath(file);
}
foreach(QString filePath, backupPaths) {
QDateTime now = QDateTime::currentDateTime();
QString fileName = filePath + "." + now.toString(BACKUPDATEFORMAT) + ".cbck";
QString baseName = QFileInfo(fileName).baseName();
QMap<QDateTime, QString>& map = m_cyclicBackups[baseName];
QList<QDateTime> list = map.keys();
if (list.size() >= m_nCyclicBackups) {
int toBeRemoved = list.size() - map.count() + 1;
for(int i=0; i< toBeRemoved; i++) {
const QDateTime& d = list[i];
qWarning() << "SAK: remove old circular backup file " << map[d];
QFile(m_dir.filePath(map[d])).remove();
map.remove(d);
}
}
QString origFile = saveDir.filePath(QFileInfo(filePath).fileName());
if (!QFile(origFile).copy(fileName) ) {
qWarning() << "SAK: failed to copy " << origFile << " to cyclic backup file " << fileName;
} else {
qWarning() << "SAK: backup " << origFile << " to " << fileName;
}
}
}
//BEGIN: INCREMENTAL
Incremental::Incremental() {
// search for backup dir
QString path = QFileInfo(QSettings(QSettings::IniFormat, QSettings::UserScope, "ZanzaSoft", "SAK").fileName()).path() + QDir::separator() + "sakincr";
m_dir =QDir(path);
m_dir.mkpath(path);
m_dir.cd(path);
QStringList list = m_dir.entryList();
foreach(QString s, list) {
QString fullSuffix = QFileInfo(s).completeSuffix();
QString task, subtask;
int divisor = s.lastIndexOf(":");
if (divisor>=0) {
task = s.mid(0,divisor);
subtask = QFileInfo(s.mid(divisor+1)).baseName();
} else {
task = QFileInfo(s).baseName();
subtask = QString();
}
if (fullSuffix.count() == 19 && fullSuffix[14]=='.' && fullSuffix.mid(15) == "incr") {
QDateTime time = QDateTime::fromString(fullSuffix.left(14), BACKUPDATEFORMAT);
QFile f(m_dir.filePath(s));
if (!f.open(QIODevice::ReadOnly)) {
qDebug() << "SAK: error opening file " << f.fileName() << ": " << f.errorString();
}
QDataStream stream(&f);
stream.setVersion(QDataStream::Qt_4_3);
Hit& p = foundPieces[time];
p.task = task;
p.subtask = subtask;
stream >> p.duration;
qDebug() << "SAK: found piece from task \"" << task << "\" subtask \"" << subtask << "\" timestamp " << time << " duration " << Task::hours(p.duration) << " hours";
foundFiles << m_dir.filePath(s);
}
}
lastTimeStamp = QDateTime::currentDateTime();
}
void Incremental::writePiece(const QString& task, const QString& subTask, const QDateTime& now, int value) {
QString filename= m_dir.filePath(task + ":" + subTask + "." + now.toString(BACKUPDATEFORMAT) + ".incr");
addedFiles << filename;
QFile f(filename);
f.open(QIODevice::ReadWrite);
QDataStream stream(&f);
stream.setVersion(QDataStream::Qt_4_3);
stream << value;
qDebug() << "SAK: write piece of task " << task << " of duration " << Task::hours(value) << " in file " << filename;
addedPieces[now] = Hit(task, subTask, value);
lastTimeStamp = now;
}
void Incremental::addPiece(const QString& task, const QString& subTask, const QDateTime& now, int value) {
foundPieces.insertMulti(now, Hit(task, subTask, value));
lastTimeStamp = now;
}
void Incremental::clearAddedPieces() {
foreach(const QString& file, addedFiles) {
QFile::remove(file);
}
}
void Incremental::clearMergedPieces() {
foreach(const QString& file, foundFiles) {
if (!QFile::remove(file))
qDebug() << "Failed removing file " << file;
}
foundPieces.clear();
}