阶段性提交

This commit is contained in:
zhaominyong 2021-08-24 18:08:18 +08:00
parent ac0d321be6
commit 46d4c5c38a
33 changed files with 493 additions and 77 deletions

2
backup-daemon/backup-daemon.pro Normal file → Executable file
View File

@ -31,6 +31,7 @@ HEADERS += \
mymountproxy.h \
myprocess/calcbackupsize.h \
myprocess/mountbackupprocess.h \
myprocess/rsyncpathtodirprocess.h \
parsebackuplist.h \
systembackupproxy.h \
workerfactory.h
@ -47,6 +48,7 @@ SOURCES += \
mymountproxy.cpp \
myprocess/calcbackupsize.cpp \
myprocess/mountbackupprocess.cpp \
myprocess/rsyncpathtodirprocess.cpp \
parsebackuplist.cpp \
systembackupproxy.cpp \
workerfactory.cpp

4
backup-daemon/backupmanager_adaptor.cpp Normal file → Executable file
View File

@ -105,11 +105,11 @@ int ManagerAdaptor::goRestore(BackupWrapper backupWrapper)
return out0;
}
int ManagerAdaptor::cancel()
int ManagerAdaptor::cancel(BackupWrapper backupWrapper)
{
// handle method call com.kylin.backup.manager.goRestore
int out0;
QMetaObject::invokeMethod(parent(), "cancel", Q_RETURN_ARG(int, out0));
QMetaObject::invokeMethod(parent(), "cancel", Q_RETURN_ARG(int, out0), Q_ARG(BackupWrapper, backupWrapper));
return out0;
}

4
backup-daemon/backupmanager_adaptor.h Normal file → Executable file
View File

@ -100,6 +100,8 @@ class ManagerAdaptor: public QDBusAbstractAdaptor
" </method>\n"
" <method name=\"cancel\">\n"
" <arg direction=\"out\" type=\"i\"/>\n"
" <annotation value=\"BackupWrapper\" name=\"org.qtproject.QtDBus.QtTypeName.In0\"/>\n"
" <arg direction=\"in\" type=\"a(i)\" name=\"backupWrapper\"/>\n"
" </method>\n"
" </interface>\n"
"")
@ -118,7 +120,7 @@ public Q_SLOTS: // METHODS
int ghostBackup(BackupWrapper backupWrapper);
int goBackup(BackupWrapper backupWrapper);
int goRestore(BackupWrapper backupWrapper);
int cancel();
int cancel(BackupWrapper backupWrapper);
Q_SIGNALS: // SIGNALS
void sendBackupResult(bool result);
void sendDeleteResult(bool result);

0
backup-daemon/com.kylin.backup.manager.xml Normal file → Executable file
View File

1
backup-daemon/main.cpp Normal file → Executable file
View File

@ -30,6 +30,7 @@ int main(int argc, char *argv[])
qDebug() << QString("测试 begin");
BackupWrapper backupWrapper;
backupWrapper.m_backupExcludePaths = Utils::getFromExcludePathsFile();
MyBackupManager manager;
manager.checkEnv(backupWrapper);

43
backup-daemon/mybackupmanager.cpp Normal file → Executable file
View File

@ -41,7 +41,7 @@ int MyBackupManager::Mount_backup_partition()
/**
* @brief
* @param backupWrapper
* @return
* @return 00
*/
int MyBackupManager::checkEnv(const BackupWrapper& backupWrapper)
{
@ -57,8 +57,10 @@ int MyBackupManager::checkEnv(const BackupWrapper& backupWrapper)
}
worker->setParam(backupWrapper);
connect(worker, &Worker::checkResult, this, &MyBackupManager::sendEnvCheckResult);
connect(worker, &Worker::checkResult, this, &MyBackupManager::finished);
connect(worker, &Worker::checkResult, this, [&] (int result) {
emit this->sendEnvCheckResult(result);
this->finished();
});
worker->moveToThread(&workerThread);
connect(&workerThread, &QThread::started, worker, &Worker::checkEnv);
connect(&workerThread, &QThread::finished, worker, &Worker::deleteLater);
@ -71,12 +73,39 @@ int MyBackupManager::checkEnv(const BackupWrapper& backupWrapper)
/**
* @brief
* @param backupWrapper
* @return
* @return 00
*/
int MyBackupManager::goBackup(const BackupWrapper& backupWrapper)
{
Q_UNUSED(backupWrapper)
return 0;
if (m_isActive || !lock(backupWrapper.m_frontUid)) {
emit sendEnvCheckResult(int(BackupResult::LOCK_PROGRAM_FAIL));
return int(BackupResult::LOCK_PROGRAM_FAIL);
}
Worker* worker = WorkerFactory::createWorker(backupWrapper.m_type, backupWrapper.m_iPosition);
if (nullptr == worker) {
emit sendEnvCheckResult(int(BackupResult::NO_FOUND_DEALCLASS));
return int(BackupResult::NO_FOUND_DEALCLASS);
}
worker->setParam(backupWrapper);
connect(worker, &Worker::checkResult, this, [&](int result) {
emit this->sendEnvCheckResult(result);
if (result != int(BackupResult::CHECK_ENV_SUCCESS)) {
this->finished();
}
});
connect(worker, &Worker::workResult, this, [&] (bool result) {
emit this->sendBackupResult(result);
this->finished();
});
worker->moveToThread(&workerThread);
connect(&workerThread, &QThread::started, worker, &Worker::doWork);
connect(&workerThread, &QThread::finished, worker, &Worker::deleteLater);
workerThread.start();
return int(BackupResult::BACKUP_RESULT_INIT);
}
/**
@ -156,7 +185,7 @@ int MyBackupManager::getBackupState(bool& isActive)
/**
* @brief
*/
int MyBackupManager::cancel()
int MyBackupManager::cancel(const BackupWrapper& backupWrapper)
{
return 0;
}

2
backup-daemon/mybackupmanager.h Normal file → Executable file
View File

@ -52,7 +52,7 @@ public slots:
// 获取备份状态
int getBackupState(bool& isActive);
// 取消操作
int cancel();
int cancel(const BackupWrapper& backupWrapper);
// 任务结束
void finished();

0
backup-daemon/mymountproxy.cpp Normal file → Executable file
View File

0
backup-daemon/mymountproxy.h Normal file → Executable file
View File

81
backup-daemon/myprocess/calcbackupsize.cpp Normal file → Executable file
View File

@ -5,6 +5,10 @@ CalcBackupSize::CalcBackupSize(QObject* parent) :
QObject(parent),
m_process(new QProcess(this))
{
connect(m_process, &QProcess::readyRead, this, [&]() {
parseResult();
});
connect(m_process, &QProcess::readyReadStandardError, this, [&]() {
QByteArray err = m_process->readAllStandardError();
qCritical("backup process error: %s", err.data());
@ -14,12 +18,14 @@ CalcBackupSize::CalcBackupSize(QObject* parent) :
m_process->kill();
});
connect(m_process, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(getCalcResult(int, QProcess::ExitStatus)));
connect(m_process, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(processFinish(int, QProcess::ExitStatus)));
}
CalcBackupSize::~CalcBackupSize()
{
if (!m_process && m_process->state() != QProcess::NotRunning) {
m_process->kill();
}
}
/**
@ -28,8 +34,10 @@ CalcBackupSize::~CalcBackupSize()
* @param block true
* @return block为true时返回值为待备份数据大小block为false时0;
*/
qint64 CalcBackupSize::calcBackupSize(QStringList args, bool block)
qint64 CalcBackupSize::start(QStringList args, bool block)
{
qDebug() << "CalcBackupSize::calcBackupSize invoke begin";
QString cmd("rsync ");
for (const QString& item : args) {
cmd += " ";
@ -49,53 +57,66 @@ qint64 CalcBackupSize::calcBackupSize(QStringList args, bool block)
m_process->waitForFinished();
}
qDebug() << "CalcBackupSize::calcBackupSize invoke end";
return m_size;
}
/**
* @brief
* @param exitCode
* @brief process的输出结果
*/
void CalcBackupSize::getCalcResult(int exitCode, QProcess::ExitStatus)
void CalcBackupSize::parseResult()
{
// 解析结果,结果内容样例如下:
/*
Number of files: 8 (reg: 4, dir: 4)
Number of created files: 7 (reg: 4, dir: 3)
Number of files: 256,274 (reg: 207,101, dir: 23,764, link: 25,407, special: 2)
Number of created files: 256,274 (reg: 207,101, dir: 23,764, link: 25,407, special: 2)
Number of deleted files: 0
Number of regular files transferred: 4
Total file size: 20 bytes
Total transferred file size: 20 bytes
Number of regular files transferred: 207,101
Total file size: 12,160,363,663 bytes
Total transferred file size: 12,159,780,794 bytes
Literal data: 0 bytes
Matched data: 0 bytes
File list size: 0
File list size: 786,254
File list generation time: 0.001 seconds
File list transfer time: 0.000 seconds
Total bytes sent: 248
Total bytes received: 43
Total bytes sent: 8,273,515
Total bytes received: 794,093
sent 8,273,515 bytes received 794,093 bytes 1,209,014.40 bytes/sec
total size is 12,160,363,663 speedup is 1,341.08 (DRY RUN)
sent 248 bytes received 43 bytes 582.00 bytes/sec
total size is 20 speedup is 0.07 (DRY RUN)
*/
QString out(m_process->readAll());
QStringList lines = out.split("\n");
for (QString& line : lines) {
line.trimmed();
// 获取文件夹数目
if (line.startsWith("Number of files:")) {
int indexOfLastColon = line.lastIndexOf(":");
int indexOfLastLable = line.lastIndexOf(")");
int numOfDirs = line.mid(indexOfLastColon+1, indexOfLastLable-indexOfLastColon).trimmed().toInt();
// 每个目录下还都有.和..,故总数还要*3
numOfDirs *= 3;
// ext4格式的目录本身要占用4K空间4096
m_size += numOfDirs * 4096;
}
if (line.startsWith("Total file size:")) {
m_size += line.replace("Total file size:", "").replace("bytes", "").trimmed().toLongLong();
break ;
// if (line.startsWith("Number of files:")) {
// int indexOfDir = line.indexOf("dir: ");
// indexOfDir += 5;
// int indexOfDirEnd = line.indexOf(QRegularExpression("[ )]"), indexOfDir);
// int numOfDirs = line.mid(indexOfDir, indexOfDirEnd-indexOfDir).replace(",","").trimmed().toInt();
// // 每个目录下还都有.和..,故总数还要*3
// numOfDirs *= 3;
// // ext4格式的目录本身要占用4K空间4096
// m_size += numOfDirs * 4096;
// }
if (line.startsWith("Total transferred file size: ")) {
m_size += line.replace("Total file size:", "").replace("bytes", "").replace(",","").trimmed().toLongLong();
}
}
emit calcFinished(m_size);
}
/**
* @brief process结束
* @param exitCode
*/
void CalcBackupSize::processFinish(int exitCode, QProcess::ExitStatus)
{
qDebug() << "CalcBackupSize::getCalcResult invoke begin";
parseResult();
emit finished(m_size);
qDebug() << "CalcBackupSize::getCalcResult invoke end";
}

13
backup-daemon/myprocess/calcbackupsize.h Normal file → Executable file
View File

@ -21,15 +21,22 @@ public:
* @param block true
* @return block为true时返回值为待备份数据大小block为false时0
*/
qint64 calcBackupSize(QStringList args, bool block = true);
qint64 start(QStringList args, bool block = true);
signals:
void calcFinished(qint64 size);
// 计算结束信号
void finished(qint64 size);
private slots:
void getCalcResult(int exitCode, QProcess::ExitStatus);
// process结束槽
void processFinish(int exitCode, QProcess::ExitStatus);
private:
/**
* @brief process的输出结果
*/
void parseResult();
QProcess * m_process;
qint64 m_size = 0; // 备份大小,单位字节

0
backup-daemon/myprocess/mountbackupprocess.cpp Normal file → Executable file
View File

0
backup-daemon/myprocess/mountbackupprocess.h Normal file → Executable file
View File

View File

@ -0,0 +1,88 @@
#include "rsyncpathtodirprocess.h"
#include <unistd.h>
#include <QDebug>
RsyncPathToDirProcess::RsyncPathToDirProcess(QObject *parent) :
QObject(parent),
m_p(new QProcess(this))
{
m_currentRate = 0;
m_bSuccess = false;
connect(m_p, &QProcess::readyRead, this, [&]() {
QString str = QString(m_p->readAll());
if (str.contains("B\/s") && str.contains("%")) {
if (str.split("%").at(0).length() < 3)
return;
int tmpRate = str.split("%").at(0).right(3).toInt();
if (m_currentRate == tmpRate)
return;
m_currentRate = tmpRate;
sync();
emit progress(m_currentRate);
}
});
connect(m_p, &QProcess::readyReadStandardError, this, [&]() {
QByteArray err = m_p->readAllStandardError();
qCritical("backup process error: %s", err.data());
});
connect(m_p, &QProcess::errorOccurred, this, [&](QProcess::ProcessError error) {
m_p->kill();
});
connect(m_p, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(processFinished(int, QProcess::ExitStatus)));
}
RsyncPathToDirProcess::~RsyncPathToDirProcess()
{
if (!m_p && m_p->state() != QProcess::NotRunning) {
m_p->kill();
}
}
bool RsyncPathToDirProcess::start(QStringList args, bool block)
{
qDebug() << "RsyncPathToDirProcess::start invoke begin";
m_currentRate = 0;
m_bSuccess = false;
QString cmd("rsync ");
for (const QString& item : args) {
cmd += " ";
cmd += item;
}
qDebug() << cmd;
m_p->start("rsync", args);
if (!m_p->waitForStarted()) {
qCritical("rsync started failed");
return false;
}
if (block) {
if (!m_p->waitForFinished()) {
qCritical("rsync finished failed");
return false;
}
}
qDebug() << "RsyncPathToDirProcess::start invoke end";
return m_bSuccess;
}
void RsyncPathToDirProcess::processFinished(int exitCode, QProcess::ExitStatus)
{
qDebug() << "RsyncPathToDirProcess::processFinished invoke begin";
if (exitCode == QProcess::NormalExit || exitCode == 24 || exitCode == 23) {
sync();
m_bSuccess = true;
emit finished(true);
} else {
emit finished(false);
}
qDebug() << "RsyncPathToDirProcess::processFinished invoke end";
}

View File

@ -0,0 +1,31 @@
#ifndef RSYNCPATHTODIRPROCESS_H
#define RSYNCPATHTODIRPROCESS_H
#include <QProcess>
class RsyncPathToDirProcess : public QObject
{
Q_OBJECT
public:
RsyncPathToDirProcess(QObject *parent = nullptr);
~RsyncPathToDirProcess();
bool start(QStringList args, bool block = true);
signals:
// 结果信号
void finished(bool result);
// 进度百分比
void progress(int currentRate);
private slots:
// m_p执行结束
void processFinished(int exitCode, QProcess::ExitStatus);
private:
QProcess * m_p;
int m_currentRate;
bool m_bSuccess;
};
#endif // RSYNCPATHTODIRPROCESS_H

0
backup-daemon/parsebackuplist.cpp Normal file → Executable file
View File

0
backup-daemon/parsebackuplist.h Normal file → Executable file
View File

171
backup-daemon/systembackupproxy.cpp Normal file → Executable file
View File

@ -10,15 +10,21 @@
IMPLEMENT_DYNCREATE(SystemBackupProxy)
SystemBackupProxy::SystemBackupProxy()
{}
{
m_bSuccess = false;
m_p = nullptr;
}
SystemBackupProxy::~SystemBackupProxy()
{}
{
delete m_p;
}
/**
* @brief
* @return false,;true,
*/
void SystemBackupProxy::checkEnvEx()
bool SystemBackupProxy::checkEnvEx()
{
qDebug() << "SystemBackupProxy::checkEnv invoke begin";
@ -26,29 +32,40 @@ void SystemBackupProxy::checkEnvEx()
MyMountProxy mountProxy;
if ( MyMountProxy::MountResult::MOUNTED != mountProxy.mountBackupPartition() ) {
emit checkResult(int(BackupResult::BACKUP_PARTITION_MOUNT_FAIL));
return ;
return false;
}
// 2、检测备份是否增量备份
checkIsIncBackup();
// 2、判断备份是否增量备份
isIncBackup();
// 3、检测空间是否满足备份
checkFreeCapacity();
bool result = checkFreeCapacity();
qDebug() << "SystemBackupProxy::checkEnv invoke end";
return result;
}
/**
* @brief
*/
void SystemBackupProxy::doWorkEx()
{}
{
// 环境检测
if (!checkEnvEx())
return ;
// 开始备份
doBackup();
}
void SystemBackupProxy::cancelEx()
{}
/**
* @brief
* @brief
* @return true, false
*/
bool SystemBackupProxy::checkIsIncBackup()
bool SystemBackupProxy::isIncBackup()
{
QString backupPath;
ParseBackupList::BackupPoint point;
@ -76,8 +93,10 @@ bool SystemBackupProxy::checkIsIncBackup()
/**
* @brief
*/
void SystemBackupProxy::checkFreeCapacity()
bool SystemBackupProxy::checkFreeCapacity()
{
qDebug() << "SystemBackupProxy::checkFreeCapacity invoke begin";
// 1、计算待备份数据的大小
qint64 itotalSize = calcSizeForBackup();
// 备份过程中会有一些临时文件产生会占用一部分空间故我们预留500M的空间
@ -90,10 +109,16 @@ void SystemBackupProxy::checkFreeCapacity()
qint64 freeSize = backupDisk.bytesFree();
// 3、校验空间是否足够
bool result = true;
if (itotalSize > freeSize) {
emit checkResult(int(BackupResult::BACKUP_CAPACITY_IS_NOT_ENOUGH));
return ;
result = false;
} else {
emit checkResult(int(BackupResult::CHECK_ENV_SUCCESS));
}
qDebug() << "SystemBackupProxy::checkFreeCapacity invoke end";
return result;
}
/**
@ -102,18 +127,26 @@ void SystemBackupProxy::checkFreeCapacity()
*/
qint64 SystemBackupProxy::calcSizeForBackup()
{
QString destPath = Utils::getSysRootPath();
QStringList args;
if (m_backupWrapper.m_bIncrement) {
if (m_backupWrapper.m_uuid.isEmpty()) {
// 新增增量备份点场景
args = getRsyncArgs(SystemBackupScene::TRY_INC_SYSTEM_BACKUP);
destPath += CHECK_PATH;
} else {
// 在原来的备份点上增量备份场景
args = getRsyncArgs(SystemBackupScene::TRY_INC_SYSTEM_BACKUP_AT_BASE);
destPath += BACKUP_SNAPSHOTS_PATH;
destPath += "/";
destPath += m_backupWrapper.m_uuid;
destPath += "/data/";
}
} else {
// 全量备份场景
args = getRsyncArgs(SystemBackupScene::TRY_SYSTEM_BACKUP);
destPath += CHECK_PATH;
}
// 拼接备份源路径和目标路径
@ -121,14 +154,12 @@ qint64 SystemBackupProxy::calcSizeForBackup()
srcPath += "/";
srcPath.replace("//", "/");
args << srcPath;
QString destPath = Utils::getSysRootPath();
destPath += CHECK_PATH;
destPath.replace("//", "/");
args << destPath;
Utils::mkpath(destPath);
CalcBackupSize calcator;
return calcator.calcBackupSize(args);
return calcator.start(args);
}
/**
@ -150,7 +181,7 @@ QStringList SystemBackupProxy::getRsyncArgs(SystemBackupScene scene)
args << "-avAXr";
args << "--progress";
args << "--ignore-missing-args";
args << QString("--link-dest=../%1/data").arg(m_baseUuid);
args << QString("--link-dest=../../%1/data").arg(m_backupWrapper.m_baseUuid);
break ;
case SystemBackupScene::INC_SYSTEM_BACKUP_AT_BASE :
args << "-avAXr";
@ -167,7 +198,7 @@ QStringList SystemBackupProxy::getRsyncArgs(SystemBackupScene scene)
args << "-aAXrn";
args << "--stats";
args << "--ignore-missing-args";
args << QString("--link-dest=../%1/data").arg(m_baseUuid);
args << QString("--link-dest=../../%1/data").arg(m_backupWrapper.m_baseUuid);
break ;
case SystemBackupScene::TRY_INC_SYSTEM_BACKUP_AT_BASE :
args << "-aAXrn";
@ -175,15 +206,121 @@ QStringList SystemBackupProxy::getRsyncArgs(SystemBackupScene scene)
args << "--ignore-missing-args";
args << "--delete";
break ;
case SystemBackupScene::EFI_BACKUP :
args << "-avAXr";
args << "--progress";
args << "--ignore-missing-args";
break ;
default:
return args;
}
if (SystemBackupScene::EFI_BACKUP != scene) {
// --exclude=排除路径设置
for (const QString & item : m_backupWrapper.m_backupExcludePaths) {
args << QString("--exclude=%1").arg(item);
}
}
return args;
}
/**
* @brief
*/
void SystemBackupProxy::doBackup()
{
qDebug() << "SystemBackupProxy::doBackup invoke begin";
// 准备
if (!doPrepare())
return ;
// 启动备份efi
if (!backupEfi()) {
emit checkResult(int(BackupResult::EFI_RSYNC_FAIL));
return ;
}
// 启动系统备份
qDebug() << "SystemBackupProxy::doBackup invoke end";
}
/**
* @brief
* @return true,false
*/
bool SystemBackupProxy::doPrepare()
{
qDebug() << "SystemBackupProxy::doPrepare invoke begin";
// 1、设置当前备份的Uuid
if (m_backupWrapper.m_uuid.isEmpty()) {
// 新增备份点不指定uuid的场景
m_curUuid = "{";
m_curUuid += Utils::createUuid();
m_curUuid += "}";
} else {
// 指定uuid备份的场景
m_curUuid = m_backupWrapper.m_uuid;
}
// 2、准备备份目录及文件
m_destPath = Utils::getSysRootPath() + BACKUP_SNAPSHOTS_PATH + "/" + m_curUuid + "/data";
m_destPath.replace("//", "/");
if (!Utils::mkpath(m_destPath)) {
emit checkResult(int(BackupResult::WRITE_BACKUP_PATHS_TO_USER_FAILED));
qCritical() << QString("mkdir %1 failed !").arg(m_destPath) ;
return false;
}
QString userFile = Utils::getSysRootPath() + BACKUP_SNAPSHOTS_PATH + "/" + m_curUuid + "/" + PATHS_USER_FILE;
userFile.replace("//", "/");
if (Utils::writeFileByLines(userFile, m_backupWrapper.m_backupPaths)) {
emit checkResult(int(BackupResult::WRITE_BACKUP_PATHS_TO_USER_FAILED));
qCritical() << QString("create file %1 failed !").arg(userFile) ;
return false;
}
QString excludeUserFile = Utils::getSysRootPath() + BACKUP_SNAPSHOTS_PATH + "/" + m_curUuid + "/" + EXCLUDE_PATHS_USER_FILE;
excludeUserFile.replace("//", "/");
if (Utils::writeFileByLines(userFile, m_backupWrapper.m_backupPaths)) {
emit checkResult(int(BackupResult::WRITE_EXCLUDE_BACKUP_PATHS_TO_USER_FAILED));
qCritical() << QString("create file %1 failed !").arg(excludeUserFile) ;
return false;
}
qDebug() << "SystemBackupProxy::doPrepare invoke end";
return true;
}
/**
* @brief /boot/efi
* @return truefalse
*/
bool SystemBackupProxy::backupEfi()
{
qDebug() << "SystemBackupProxy::backupEfi invoke begin";
// /boot/efi分区是自动挂载的这里不用去挂载
QString efiPath = Utils::getSysRootPath() + "/boot/efi";
efiPath.replace("//", "/");
QStringList args;
if (Utils::isDirExist(efiPath)) {
args = getRsyncArgs(SystemBackupScene::EFI_BACKUP);
}
args << efiPath;
args << m_destPath;
m_p = new RsyncPathToDirProcess(this);
bool result = m_p->start(args);
delete m_p;
m_p = nullptr;
qDebug() << "SystemBackupProxy::backupEfi invoke end";
return result;
}

32
backup-daemon/systembackupproxy.h Normal file → Executable file
View File

@ -2,6 +2,7 @@
#define SYSTEMBACKUPPROXY_H
#include "workerfactory.h"
#include "myprocess/rsyncpathtodirprocess.h"
class SystemBackupProxy : public Worker
{
@ -16,6 +17,7 @@ public:
TRY_SYSTEM_BACKUP, // 测试系统备份,可用于计算备份传输数据大小
TRY_INC_SYSTEM_BACKUP, // 测试增量系统备份,可用于计算备份传输数据大小
TRY_INC_SYSTEM_BACKUP_AT_BASE, // 测试增量系统备份,在原备份点里增量备份,可用于计算备份传输数据大小
EFI_BACKUP, // 备份/boot/efi
};
explicit SystemBackupProxy();
@ -23,7 +25,7 @@ public:
public:
// 环境检测
virtual void checkEnvEx();
virtual bool checkEnvEx();
// 任务处理
virtual void doWorkEx();
@ -32,11 +34,11 @@ public:
virtual void cancelEx();
private:
// 校验是否增量备份
bool checkIsIncBackup();
// 判断是否增量备份
bool isIncBackup();
// 校验剩余空间是否满足备份
void checkFreeCapacity();
bool checkFreeCapacity();
// 计算备份所需空间大小
qint64 calcSizeForBackup();
@ -48,10 +50,24 @@ private:
*/
QStringList getRsyncArgs(SystemBackupScene scene);
// 是否增量备份
bool m_bIncrement = false;
// 基准备份点,新增增量备份点时用
QString m_baseUuid;
// 备份准备
bool doPrepare();
// 备份
void doBackup();
// 备份/boot/efi
bool backupEfi();
// 是否备份成功
bool m_bSuccess;
// 当前备份uuid
QString m_curUuid;
// 当前备份目标目录
QString m_destPath;
// 备份进程
RsyncPathToDirProcess *m_p;
};
#endif // SYSTEMBACKUPPROXY_H

2
backup-daemon/workerfactory.cpp Normal file → Executable file
View File

@ -34,7 +34,7 @@ void Worker::cancel()
}
// 环境检测,个性化部分派生类去实现
void Worker::checkEnvEx()
bool Worker::checkEnvEx()
{}
// 任务处理,个性化部分派生类去实现

2
backup-daemon/workerfactory.h Normal file → Executable file
View File

@ -33,7 +33,7 @@ public slots:
protected:
// 环境检测,个性化部分派生类去实现
virtual void checkEnvEx();
virtual bool checkEnvEx();
// 任务处理,个性化部分派生类去实现
virtual void doWorkEx();

0
common/mydefine.cpp Normal file → Executable file
View File

9
common/mydefine.h Normal file → Executable file
View File

@ -22,6 +22,12 @@
#define BACKUP_CLI_NAME "kybackup"
#define AUTO_BACKUP_UUID "{01234567-0123-0123-0123-0123456789ab}"
#define FACTORY_BACKUP_UUID "{00000000-0000-0000-0000-000000000000}"
#define PATHS_USER_FILE ".user.txt"
#define EXCLUDE_PATHS_USER_FILE ".exclude.user.txt"
#define PID_STRING_LEN 1024
/**
@ -73,6 +79,8 @@ struct BackupWrapper {
QStringList m_backupPaths;
// 备份需要排除的路径
QStringList m_backupExcludePaths;
// 备份目标路径(前缀),在向移动设备备份时使用它来指定对应的移动设备路径
QString m_prefixDestPath;
// 备注信息
QString m_note;
// 备份用户名
@ -213,6 +221,7 @@ enum class BackupResult {
// 根据操作类型动态创建处理类失败
NO_FOUND_DEALCLASS,
CHECK_ENV_SUCCESS,
};
#endif // MYDEFINE_H

5
common/mydusizetool.cpp Normal file → Executable file
View File

@ -2,9 +2,10 @@
#include <QDebug>
#include "utils.h"
MyDuSizeTool::MyDuSizeTool(QObject* parent)
MyDuSizeTool::MyDuSizeTool(QObject* parent) :
QObject(parent),
m_p(new QProcess(this))
{
m_p = new QProcess(this);
connect(m_p, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(finished(int, QProcess::ExitStatus)));
connect(m_p, &QProcess::readyReadStandardError, this, [&]() {
QByteArray err = m_p->readAllStandardError();

0
common/mydusizetool.h Normal file → Executable file
View File

0
common/mylittleparse.cpp Normal file → Executable file
View File

0
common/mylittleparse.h Normal file → Executable file
View File

0
common/reflect.cpp Normal file → Executable file
View File

0
common/reflect.h Normal file → Executable file
View File

0
common/singleton.h Normal file → Executable file
View File

0
common/spinlock_mutex.h Normal file → Executable file
View File

60
common/utils.cpp Normal file → Executable file
View File

@ -6,6 +6,7 @@
#include <QDir>
#include <QRegularExpression>
#include <QThread>
#include <QUuid>
#include <iostream>
#include <unistd.h>
#include <fcntl.h>
@ -13,6 +14,7 @@
#include <sys/stat.h>
#include <sys/types.h>
#include "mylittleparse.h"
#include "mydefine.h"
QString Utils::m_sysRootPath = "/";
@ -79,7 +81,7 @@ void Utils::customMessageHandler(QtMsgType type, const QMessageLogContext& conte
file.open(QIODevice::ReadWrite | QIODevice::Append);
QTextStream stream(&file);
stream << strMessage << endl;
file.flush();
stream.flush();
file.close();
}
@ -181,6 +183,28 @@ QString Utils::getBackupPartitionUuid()
QString restoreUuid;
parse.find("RECOVERY_DEV_UUID", restoreUuid);
if (restoreUuid.isEmpty()) {
QString fstab = Utils::m_sysRootPath + FSTAB_PATH;
fstab.replace("//", "/");
QFile file(fstab);
if (file.exists()) {
if (file.open(QIODevice::ReadOnly | QIODevice::Text)) {
QTextStream in(&file);
while (!in.atEnd()) {
QString line = in.readLine();
if (line.startsWith("UUID=") && line.contains("/backup")) {
int indexOfSpace = line.indexOf(QRegularExpression("[ \t]"), 0);
QString uuid = line.mid(0, indexOfSpace);
uuid.replace("UUID=", "");
restoreUuid = uuid.trimmed();
break ;
}
}
}
}
}
return restoreUuid;
}
@ -368,3 +392,37 @@ QStringList Utils::getFromExcludePathsFile()
return list;
}
/**
* @brief Uuid
* @return UUID
*/
QString Utils::createUuid()
{
QString uuid;
do {
uuid = QUuid::createUuid().toString();
} while (uuid == AUTO_BACKUP_UUID || uuid == FACTORY_BACKUP_UUID);
return uuid;
}
/**
* @brief
* @param fileName
* @param lines
* @return truefalse
*/
bool Utils::writeFileByLines(const QString& fileName, const QStringList& lines)
{
QFile file(fileName);
if (!file.open(QIODevice::ReadWrite | QIODevice::Truncate)) {
return false;
}
QTextStream out(&file);
for (const QString& line : lines) {
out << line << endl;
}
file.close();
return true;
}

14
common/utils.h Normal file → Executable file
View File

@ -105,6 +105,20 @@ public:
*/
static bool isDirExist(const QString& fullDirName);
/**
* @brief Uuid
* @return UUID
*/
static QString createUuid();
/**
* @brief
* @param fileName
* @param lines
* @return truefalse
*/
static bool writeFileByLines(const QString& fileName, const QStringList& lines);
private:
// 系统根目录,默认"/"
static QString m_sysRootPath;