mirror of
https://github.com/DrKLO/Telegram.git
synced 2025-01-10 20:12:39 +01:00
57ada21a2d
On Linux unistd.h is needed for fsync(). The native library does not compile otherwise.
125 lines
4.4 KiB
C++
125 lines
4.4 KiB
C++
/*
|
|
* This is the source code of tgnet library v. 1.0
|
|
* It is licensed under GNU GPL v. 2 or later.
|
|
* You should have received a copy of the license in this archive (see LICENSE).
|
|
*
|
|
* Copyright Nikolai Kudashov, 2015.
|
|
*/
|
|
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include "Config.h"
|
|
#include "ConnectionsManager.h"
|
|
#include "FileLog.h"
|
|
#include "BuffersStorage.h"
|
|
|
|
Config::Config(std::string fileName) {
|
|
configPath = ConnectionsManager::getInstance().currentConfigPath + fileName;
|
|
backupPath = configPath + ".bak";
|
|
FILE *backup = fopen(backupPath.c_str(), "rb");
|
|
if (backup != nullptr) {
|
|
DEBUG_D("Config(%p, %s) backup file found %s", this, configPath.c_str(), backupPath.c_str());
|
|
fclose(backup);
|
|
remove(configPath.c_str());
|
|
rename(backupPath.c_str(), configPath.c_str());
|
|
}
|
|
}
|
|
|
|
NativeByteBuffer *Config::readConfig() {
|
|
NativeByteBuffer *buffer = nullptr;
|
|
FILE *file = fopen(configPath.c_str(), "rb");
|
|
if (file != nullptr) {
|
|
fseek(file, 0, SEEK_END);
|
|
long fileSize = ftell(file);
|
|
if (fseek(file, 0, SEEK_SET)) {
|
|
DEBUG_E("Config(%p, %s) failed fseek to begin, reopen it", this, configPath.c_str());
|
|
fclose(file);
|
|
file = fopen(configPath.c_str(), "rb");
|
|
}
|
|
uint32_t size = 0;
|
|
int bytesRead = fread(&size, sizeof(uint32_t), 1, file);
|
|
DEBUG_D("Config(%p, %s) load, size = %u, fileSize = %u", this, configPath.c_str(), size, (uint32_t) fileSize);
|
|
if (bytesRead > 0 && size > 0 && (int32_t) size < fileSize) {
|
|
buffer = BuffersStorage::getInstance().getFreeBuffer(size);
|
|
if (fread(buffer->bytes(), sizeof(uint8_t), size, file) != size) {
|
|
buffer->reuse();
|
|
buffer = nullptr;
|
|
}
|
|
}
|
|
fclose(file);
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
void Config::writeConfig(NativeByteBuffer *buffer) {
|
|
DEBUG_D("Config(%p, %s) start write config", this, configPath.c_str());
|
|
FILE *file = fopen(configPath.c_str(), "rb");
|
|
FILE *backup = fopen(backupPath.c_str(), "rb");
|
|
bool error = false;
|
|
if (file != nullptr) {
|
|
if (backup == nullptr) {
|
|
fclose(file);
|
|
if (rename(configPath.c_str(), backupPath.c_str()) != 0) {
|
|
DEBUG_E("Config(%p) unable to rename file %s to backup file %s", this, configPath.c_str(), backupPath.c_str());
|
|
error = true;
|
|
}
|
|
} else {
|
|
fclose(file);
|
|
fclose(backup);
|
|
remove(configPath.c_str());
|
|
}
|
|
}
|
|
if (error) {
|
|
return;
|
|
}
|
|
file = fopen(configPath.c_str(), "wb");
|
|
if (chmod(configPath.c_str(), 0660)) {
|
|
DEBUG_E("Config(%p, %s) chmod failed", this, configPath.c_str());
|
|
}
|
|
if (file == nullptr) {
|
|
DEBUG_E("Config(%p, %s) unable to open file for writing", this, configPath.c_str());
|
|
return;
|
|
}
|
|
uint32_t size = buffer->position();
|
|
if (fwrite(&size, sizeof(uint32_t), 1, file) == 1) {
|
|
if (fwrite(buffer->bytes(), sizeof(uint8_t), size, file) != size) {
|
|
DEBUG_E("Config(%p, %s) failed to write config data to file", this, configPath.c_str());
|
|
error = true;
|
|
}
|
|
} else {
|
|
DEBUG_E("Config(%p, %s) failed to write config size to file", this, configPath.c_str());
|
|
error = true;
|
|
}
|
|
if (fflush(file)) {
|
|
DEBUG_E("Config(%p, %s) fflush failed", this, configPath.c_str());
|
|
error = true;
|
|
}
|
|
int fd = fileno(file);
|
|
if (fd == -1) {
|
|
DEBUG_E("Config(%p, %s) fileno failed", this, configPath.c_str());
|
|
error = true;
|
|
} else {
|
|
DEBUG_D("Config(%p, %s) fileno = %d", this, configPath.c_str(), fd);
|
|
}
|
|
if (fd != -1 && fsync(fd) == -1) {
|
|
DEBUG_E("Config(%p, %s) fsync failed", this, configPath.c_str());
|
|
error = true;
|
|
}
|
|
if (fclose(file)) {
|
|
DEBUG_E("Config(%p, %s) fclose failed", this, configPath.c_str());
|
|
error = true;
|
|
}
|
|
if (error) {
|
|
DEBUG_E("Config(%p, %s) failed to write config", this, configPath.c_str());
|
|
if (remove(configPath.c_str())) {
|
|
DEBUG_E("Config(%p, %s) remove config failed", this, configPath.c_str());
|
|
}
|
|
} else {
|
|
if (remove(backupPath.c_str())) {
|
|
DEBUG_E("Config(%p, %s) remove backup failed failed", this, configPath.c_str());
|
|
}
|
|
}
|
|
if (!error) {
|
|
DEBUG_D("Config(%p, %s) config write ok", this, configPath.c_str());
|
|
}
|
|
}
|