2011-09-02 18:51:20 +02:00
|
|
|
#ifndef file_hh_INCLUDED
|
|
|
|
#define file_hh_INCLUDED
|
|
|
|
|
2015-03-09 14:48:41 +01:00
|
|
|
#include "array_view.hh"
|
2019-02-12 10:54:37 +01:00
|
|
|
#include "enum.hh"
|
2017-03-15 19:25:59 +01:00
|
|
|
#include "meta.hh"
|
2018-03-24 21:55:39 +01:00
|
|
|
#include "string.hh"
|
2016-11-29 00:53:50 +01:00
|
|
|
#include "units.hh"
|
|
|
|
#include "vector.hh"
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2015-10-16 02:33:17 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
2021-07-31 01:44:24 +02:00
|
|
|
#include <cstring>
|
2015-10-16 02:33:17 +02:00
|
|
|
|
2011-09-09 20:40:59 +02:00
|
|
|
namespace Kakoune
|
2011-09-02 18:51:20 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
class Buffer;
|
2014-11-12 22:27:07 +01:00
|
|
|
class String;
|
2016-11-29 00:53:50 +01:00
|
|
|
class Regex;
|
|
|
|
|
|
|
|
using CandidateList = Vector<String, MemoryDomain::Completion>;
|
2012-01-29 23:24:43 +01:00
|
|
|
|
2018-03-22 22:22:34 +01:00
|
|
|
// parse ~/ and %/ in filename and returns the translated filename
|
2018-03-24 21:55:39 +01:00
|
|
|
String parse_filename(StringView filename, StringView buf_dir = {});
|
2018-03-22 22:22:34 +01:00
|
|
|
|
2014-04-18 15:03:08 +02:00
|
|
|
String real_path(StringView filename);
|
|
|
|
String compact_path(StringView filename);
|
2012-01-29 23:24:43 +01:00
|
|
|
|
2017-01-01 14:10:08 +01:00
|
|
|
StringView tmpdir();
|
2018-01-20 01:19:23 +01:00
|
|
|
StringView homedir();
|
2017-01-01 14:10:08 +01:00
|
|
|
|
2015-03-12 21:39:34 +01:00
|
|
|
// returns pair { directory, filename }
|
|
|
|
std::pair<StringView, StringView> split_path(StringView path);
|
|
|
|
|
2014-10-30 01:50:40 +01:00
|
|
|
String get_kak_binary_path();
|
|
|
|
|
2016-08-30 23:56:47 +02:00
|
|
|
bool fd_readable(int fd);
|
2016-12-01 21:11:09 +01:00
|
|
|
bool fd_writable(int fd);
|
2015-06-05 14:52:56 +02:00
|
|
|
String read_fd(int fd, bool text = false);
|
|
|
|
String read_file(StringView filename, bool text = false);
|
2015-11-27 00:40:17 +01:00
|
|
|
void write(int fd, StringView data);
|
2019-04-07 01:31:36 +02:00
|
|
|
void write_to_file(StringView filename, StringView data);
|
2014-08-15 14:21:54 +02:00
|
|
|
|
2015-10-16 02:33:17 +02:00
|
|
|
struct MappedFile
|
|
|
|
{
|
|
|
|
MappedFile(StringView filename);
|
|
|
|
~MappedFile();
|
2015-10-16 14:58:56 +02:00
|
|
|
|
2016-11-29 00:53:50 +01:00
|
|
|
operator StringView() const;
|
2015-10-16 14:58:56 +02:00
|
|
|
|
|
|
|
int fd;
|
|
|
|
const char* data;
|
|
|
|
struct stat st {};
|
2015-10-16 02:33:17 +02:00
|
|
|
};
|
2014-08-15 14:21:54 +02:00
|
|
|
|
2019-02-12 10:54:37 +01:00
|
|
|
enum class WriteMethod
|
|
|
|
{
|
|
|
|
Overwrite,
|
|
|
|
Replace
|
|
|
|
};
|
|
|
|
constexpr auto enum_desc(Meta::Type<WriteMethod>)
|
|
|
|
{
|
2019-10-17 13:48:22 +02:00
|
|
|
return make_array<EnumDesc<WriteMethod>>({
|
2019-02-12 10:54:37 +01:00
|
|
|
{ WriteMethod::Overwrite, "overwrite" },
|
|
|
|
{ WriteMethod::Replace, "replace" },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-02-12 10:18:34 +01:00
|
|
|
enum class WriteFlags
|
|
|
|
{
|
|
|
|
None = 0,
|
|
|
|
Force = 0b01,
|
|
|
|
Sync = 0b10
|
|
|
|
};
|
|
|
|
constexpr bool with_bit_ops(Meta::Type<WriteFlags>) { return true; }
|
|
|
|
|
|
|
|
void write_buffer_to_file(Buffer& buffer, StringView filename,
|
2019-02-12 10:54:37 +01:00
|
|
|
WriteMethod method, WriteFlags flags);
|
2019-02-12 10:18:34 +01:00
|
|
|
void write_buffer_to_fd(Buffer& buffer, int fd);
|
2014-10-13 14:38:28 +02:00
|
|
|
void write_buffer_to_backup_file(Buffer& buffer);
|
2014-08-15 14:21:54 +02:00
|
|
|
|
2018-03-22 22:22:34 +01:00
|
|
|
String find_file(StringView filename, StringView buf_dir, ConstArrayView<String> paths);
|
2015-10-16 14:58:56 +02:00
|
|
|
bool file_exists(StringView filename);
|
2021-07-20 14:06:57 +02:00
|
|
|
bool regular_file_exists(StringView filename);
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2015-08-23 15:22:23 +02:00
|
|
|
Vector<String> list_files(StringView directory);
|
|
|
|
|
2016-12-16 00:47:34 +01:00
|
|
|
void make_directory(StringView dir, mode_t mode);
|
2015-08-23 15:13:46 +02:00
|
|
|
|
2019-11-23 15:55:46 +01:00
|
|
|
struct FsStatus
|
|
|
|
{
|
|
|
|
timespec timestamp;
|
|
|
|
ByteCount file_size;
|
|
|
|
size_t hash;
|
|
|
|
};
|
|
|
|
|
2015-09-27 12:55:34 +02:00
|
|
|
timespec get_fs_timestamp(StringView filename);
|
2019-11-23 15:55:46 +01:00
|
|
|
FsStatus get_fs_status(StringView filename);
|
2015-09-27 12:55:34 +02:00
|
|
|
|
|
|
|
constexpr bool operator==(const timespec& lhs, const timespec& rhs)
|
|
|
|
{
|
|
|
|
return lhs.tv_sec == rhs.tv_sec and lhs.tv_nsec == rhs.tv_nsec;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool operator!=(const timespec& lhs, const timespec& rhs)
|
|
|
|
{
|
|
|
|
return not (lhs == rhs);
|
|
|
|
}
|
2013-10-15 19:51:31 +02:00
|
|
|
|
2016-10-13 20:46:09 +02:00
|
|
|
enum class FilenameFlags
|
|
|
|
{
|
|
|
|
None = 0,
|
|
|
|
OnlyDirectories = 1 << 0,
|
|
|
|
Expand = 1 << 1
|
|
|
|
};
|
2017-03-15 18:55:34 +01:00
|
|
|
constexpr bool with_bit_ops(Meta::Type<FilenameFlags>) { return true; }
|
2016-10-13 20:46:09 +02:00
|
|
|
|
2015-01-09 14:57:21 +01:00
|
|
|
CandidateList complete_filename(StringView prefix, const Regex& ignore_regex,
|
2016-10-13 20:46:09 +02:00
|
|
|
ByteCount cursor_pos = -1,
|
|
|
|
FilenameFlags flags = FilenameFlags::None);
|
2013-12-23 21:43:55 +01:00
|
|
|
|
2015-01-09 14:57:21 +01:00
|
|
|
CandidateList complete_command(StringView prefix, ByteCount cursor_pos = -1);
|
2015-06-16 00:00:37 +02:00
|
|
|
|
2021-07-31 01:44:24 +02:00
|
|
|
template<int buffer_size = 4096>
|
|
|
|
struct BufferedWriter
|
|
|
|
{
|
|
|
|
BufferedWriter(int fd)
|
|
|
|
: m_fd{fd}, m_exception_count{std::uncaught_exceptions()} {}
|
|
|
|
|
|
|
|
~BufferedWriter() noexcept(false)
|
|
|
|
{
|
|
|
|
if (m_pos != 0 and m_exception_count == std::uncaught_exceptions())
|
|
|
|
Kakoune::write(m_fd, {m_buffer, m_pos});
|
|
|
|
}
|
|
|
|
|
|
|
|
void write(StringView data)
|
|
|
|
{
|
|
|
|
while (not data.empty())
|
|
|
|
{
|
|
|
|
const ByteCount length = data.length();
|
|
|
|
const ByteCount write_len = std::min(length, size - m_pos);
|
|
|
|
memcpy(m_buffer + (int)m_pos, data.data(), (int)write_len);
|
|
|
|
m_pos += write_len;
|
|
|
|
if (m_pos == size)
|
|
|
|
{
|
|
|
|
Kakoune::write(m_fd, {m_buffer, size});
|
|
|
|
m_pos = 0;
|
|
|
|
}
|
|
|
|
data = data.substr(write_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static constexpr ByteCount size = buffer_size;
|
|
|
|
int m_fd;
|
|
|
|
int m_exception_count;
|
|
|
|
ByteCount m_pos = 0;
|
|
|
|
char m_buffer[(int)size];
|
|
|
|
};
|
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif // file_hh_INCLUDED
|