ClickHouse/dbms/include/DB/IO/WriteBufferFromFile.h

99 lines
2.7 KiB
C++
Raw Normal View History

2011-10-24 12:10:59 +00:00
#pragma once
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <DB/Common/ProfileEvents.h>
#include <DB/Common/CurrentMetrics.h>
2011-10-24 12:10:59 +00:00
#include <DB/IO/WriteBufferFromFileDescriptor.h>
namespace DB
{
namespace ErrorCodes
{
extern const int FILE_DOESNT_EXIST;
extern const int CANNOT_OPEN_FILE;
extern const int CANNOT_CLOSE_FILE;
}
2011-10-24 12:10:59 +00:00
/** Принимает имя файла. Самостоятельно открывает и закрывает файл.
*/
class WriteBufferFromFile : public WriteBufferFromFileDescriptor
{
private:
std::string file_name;
CurrentMetrics::Increment metric_increment{CurrentMetrics::OpenFileForWrite};
2015-12-13 08:51:28 +00:00
2011-10-24 12:10:59 +00:00
public:
WriteBufferFromFile(const std::string & file_name_, size_t buf_size = DBMS_DEFAULT_BUFFER_SIZE, int flags = -1, mode_t mode = 0666,
2014-04-08 07:31:51 +00:00
char * existing_memory = nullptr, size_t alignment = 0)
: WriteBufferFromFileDescriptor(-1, buf_size, existing_memory, alignment), file_name(file_name_)
2011-10-24 12:10:59 +00:00
{
ProfileEvents::increment(ProfileEvents::FileOpen);
fd = open(file_name.c_str(), flags == -1 ? O_WRONLY | O_TRUNC | O_CREAT : flags, mode);
2015-12-13 08:51:28 +00:00
2011-10-24 12:10:59 +00:00
if (-1 == fd)
throwFromErrno("Cannot open file " + file_name, errno == ENOENT ? ErrorCodes::FILE_DOESNT_EXIST : ErrorCodes::CANNOT_OPEN_FILE);
2011-10-24 12:10:59 +00:00
}
2015-12-13 08:51:28 +00:00
/// Использовать уже открытый файл.
WriteBufferFromFile(int fd, size_t buf_size = DBMS_DEFAULT_BUFFER_SIZE, int flags = -1, mode_t mode = 0666,
char * existing_memory = nullptr, size_t alignment = 0)
: WriteBufferFromFileDescriptor(fd, buf_size, existing_memory, alignment), file_name("(fd = " + toString(fd) + ")")
{
}
2013-09-26 17:42:18 +00:00
~WriteBufferFromFile()
2011-10-24 12:10:59 +00:00
{
2015-12-13 08:51:28 +00:00
if (fd < 0)
return;
2013-11-18 17:17:45 +00:00
try
{
next();
2013-11-18 17:17:45 +00:00
}
catch (...)
{
2013-11-18 19:18:03 +00:00
tryLogCurrentException(__PRETTY_FUNCTION__);
2013-11-18 17:17:45 +00:00
}
2015-12-13 08:51:28 +00:00
::close(fd);
2011-10-24 12:10:59 +00:00
}
2015-12-13 08:51:28 +00:00
/// Закрыть файл раньше вызова деструктора.
void close()
{
next();
if (0 != ::close(fd))
throw Exception("Cannot close file", ErrorCodes::CANNOT_CLOSE_FILE);
fd = -1;
metric_increment.destroy();
2015-12-13 08:51:28 +00:00
}
2013-10-03 12:18:56 +00:00
/** fsync() transfers ("flushes") all modified in-core data of (i.e., modified buffer cache pages for) the file
* referred to by the file descriptor fd to the disk device (or other permanent storage device)
* so that all changed information can be retrieved even after the system crashed or was rebooted.
* This includes writing through or flushing a disk cache if present. The call blocks until the device
* reports that the transfer has completed. It also flushes metadata information associated with the file (see stat(2)).
* - man fsync */
2016-03-07 04:36:54 +00:00
void sync() override
2013-10-03 12:18:56 +00:00
{
fsync(fd);
}
2011-10-24 12:10:59 +00:00
2016-03-07 04:31:10 +00:00
std::string getFileName() const override
2011-10-24 12:10:59 +00:00
{
return file_name;
}
};
}