2011-09-02 18:51:20 +02:00
|
|
|
#include "file.hh"
|
2011-09-09 20:40:59 +02:00
|
|
|
|
2013-04-09 20:05:40 +02:00
|
|
|
#include "assert.hh"
|
2011-09-02 18:51:20 +02:00
|
|
|
#include "buffer.hh"
|
2016-11-29 00:53:50 +01:00
|
|
|
#include "exception.hh"
|
2017-03-15 19:25:59 +01:00
|
|
|
#include "flags.hh"
|
2018-05-26 02:01:26 +02:00
|
|
|
#include "option_types.hh"
|
2016-03-02 14:51:16 +01:00
|
|
|
#include "ranked_match.hh"
|
2014-10-13 14:12:33 +02:00
|
|
|
#include "regex.hh"
|
2014-11-12 22:27:07 +01:00
|
|
|
#include "string.hh"
|
2016-11-29 00:53:50 +01:00
|
|
|
#include "unicode.hh"
|
2012-10-11 00:41:48 +02:00
|
|
|
|
2017-01-08 23:30:15 +01:00
|
|
|
#include <cerrno>
|
2018-01-20 01:19:23 +01:00
|
|
|
#include <cstdlib>
|
2017-10-09 16:12:42 +02:00
|
|
|
#include <cstring>
|
2013-03-13 19:01:59 +01:00
|
|
|
#include <dirent.h>
|
2018-01-20 01:19:23 +01:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <pwd.h>
|
|
|
|
#include <sys/mman.h>
|
2016-08-30 23:56:47 +02:00
|
|
|
#include <sys/select.h>
|
2018-01-20 01:19:23 +01:00
|
|
|
#include <unistd.h>
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2015-11-02 21:22:00 +01:00
|
|
|
#if defined(__FreeBSD__)
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#endif
|
|
|
|
|
2014-10-30 01:50:40 +01:00
|
|
|
#if defined(__APPLE__)
|
|
|
|
#include <mach-o/dyld.h>
|
2015-09-27 19:48:01 +02:00
|
|
|
#define st_mtim st_mtimespec
|
2014-10-30 01:50:40 +01:00
|
|
|
#endif
|
|
|
|
|
2015-09-25 00:36:29 +02:00
|
|
|
#if defined(__HAIKU__)
|
|
|
|
#include <app/Application.h>
|
|
|
|
#include <app/Roster.h>
|
|
|
|
#include <storage/Path.h>
|
|
|
|
#endif
|
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2016-11-29 00:53:50 +01:00
|
|
|
struct file_access_error : runtime_error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
file_access_error(StringView filename,
|
|
|
|
StringView error_desc)
|
|
|
|
: runtime_error(format("{}: {}", filename, error_desc)) {}
|
2017-06-12 06:21:34 +02:00
|
|
|
|
|
|
|
file_access_error(int fd, StringView error_desc)
|
|
|
|
: runtime_error(format("fd {}: {}", fd, error_desc)) {}
|
2016-11-29 00:53:50 +01:00
|
|
|
};
|
|
|
|
|
2018-03-22 22:22:34 +01:00
|
|
|
String parse_filename(StringView filename, StringView buf_dir)
|
2012-01-29 23:24:43 +01:00
|
|
|
{
|
2017-08-29 03:33:00 +02:00
|
|
|
auto prefix = filename.substr(0_byte, 2_byte);
|
|
|
|
if (prefix == "~" or prefix == "~/")
|
2018-01-20 01:19:23 +01:00
|
|
|
return homedir() + filename.substr(1_byte);
|
2018-03-22 22:22:34 +01:00
|
|
|
if ((prefix == "%" or prefix == "%/") and not buf_dir.empty())
|
|
|
|
return buf_dir + filename.substr(1_byte);
|
2017-08-29 03:33:00 +02:00
|
|
|
return filename.str();
|
2012-01-29 23:24:43 +01:00
|
|
|
}
|
|
|
|
|
2015-02-06 14:47:19 +01:00
|
|
|
std::pair<StringView, StringView> split_path(StringView path)
|
|
|
|
{
|
2016-03-08 22:35:56 +01:00
|
|
|
auto it = find(path | reverse(), '/');
|
2015-03-12 21:39:34 +01:00
|
|
|
if (it == path.rend())
|
|
|
|
return { {}, path };
|
|
|
|
const char* slash = it.base()-1;
|
2015-03-13 14:09:54 +01:00
|
|
|
return { {path.begin(), slash+1}, {slash+1, path.end()} };
|
2015-02-06 14:47:19 +01:00
|
|
|
}
|
|
|
|
|
2014-04-18 15:03:08 +02:00
|
|
|
String real_path(StringView filename)
|
2013-03-22 14:27:30 +01:00
|
|
|
{
|
2018-03-19 19:56:20 +01:00
|
|
|
if (filename.empty())
|
|
|
|
return {};
|
|
|
|
|
2014-07-30 01:03:01 +02:00
|
|
|
char buffer[PATH_MAX+1];
|
|
|
|
|
|
|
|
StringView existing = filename;
|
2017-01-29 14:49:45 +01:00
|
|
|
StringView non_existing{};
|
2013-03-25 19:11:26 +01:00
|
|
|
|
2014-07-30 01:03:01 +02:00
|
|
|
while (true)
|
2013-03-25 19:11:26 +01:00
|
|
|
{
|
2018-03-18 13:23:00 +01:00
|
|
|
if (char* res = realpath(existing.zstr(), buffer))
|
2014-07-30 01:03:01 +02:00
|
|
|
{
|
|
|
|
if (non_existing.empty())
|
|
|
|
return res;
|
2018-03-19 10:49:05 +01:00
|
|
|
|
|
|
|
StringView dir = res;
|
|
|
|
if (dir.substr(dir.length()-1_byte, 1_byte) == "/")
|
|
|
|
dir = dir.substr(0_byte, dir.length()-1_byte);
|
|
|
|
return format("{}/{}", dir, non_existing);
|
2014-07-30 01:03:01 +02:00
|
|
|
}
|
2013-03-25 19:11:26 +01:00
|
|
|
|
2018-03-18 13:23:00 +01:00
|
|
|
auto it = find(existing.rbegin() + 1, existing.rend(), '/');
|
2014-07-30 01:03:01 +02:00
|
|
|
if (it == existing.rend())
|
2016-05-15 11:37:01 +02:00
|
|
|
{
|
|
|
|
char cwd[1024];
|
|
|
|
return format("{}/{}", getcwd(cwd, 1024), filename);
|
|
|
|
}
|
2014-07-30 01:03:01 +02:00
|
|
|
|
2018-03-18 13:23:00 +01:00
|
|
|
existing = StringView{existing.begin(), it.base()};
|
2014-07-30 01:03:01 +02:00
|
|
|
non_existing = StringView{it.base(), filename.end()};
|
|
|
|
}
|
2013-03-25 19:11:26 +01:00
|
|
|
}
|
|
|
|
|
2014-04-18 15:03:08 +02:00
|
|
|
String compact_path(StringView filename)
|
2013-03-25 19:11:26 +01:00
|
|
|
{
|
|
|
|
String real_filename = real_path(filename);
|
|
|
|
|
|
|
|
char cwd[1024];
|
2015-12-10 09:00:10 +01:00
|
|
|
if (!::getcwd(cwd, 1024))
|
|
|
|
throw runtime_error(format("unable to get the current working directory (errno: {})", ::strerror(errno)));
|
|
|
|
|
2014-12-08 14:59:29 +01:00
|
|
|
String real_cwd = real_path(cwd) + "/";
|
2013-09-23 21:16:25 +02:00
|
|
|
if (prefix_match(real_filename, real_cwd))
|
2015-03-10 20:33:46 +01:00
|
|
|
return real_filename.substr(real_cwd.length()).str();
|
2013-03-25 19:11:26 +01:00
|
|
|
|
2018-01-20 01:19:23 +01:00
|
|
|
const StringView home = homedir();
|
|
|
|
if (not home.empty())
|
2013-03-25 19:11:26 +01:00
|
|
|
{
|
2018-01-20 01:19:23 +01:00
|
|
|
ByteCount home_len = home.length();
|
2013-03-25 19:11:26 +01:00
|
|
|
if (real_filename.substr(0, home_len) == home)
|
|
|
|
return "~" + real_filename.substr(home_len);
|
|
|
|
}
|
|
|
|
|
2014-04-18 15:03:08 +02:00
|
|
|
return filename.str();
|
2013-03-22 14:27:30 +01:00
|
|
|
}
|
|
|
|
|
2017-01-01 14:10:08 +01:00
|
|
|
StringView tmpdir()
|
|
|
|
{
|
|
|
|
StringView tmpdir = getenv("TMPDIR");
|
|
|
|
if (not tmpdir.empty())
|
|
|
|
return tmpdir.back() == '/' ? tmpdir.substr(0_byte, tmpdir.length()-1)
|
|
|
|
: tmpdir;
|
|
|
|
return "/tmp";
|
|
|
|
}
|
|
|
|
|
2018-01-20 01:19:23 +01:00
|
|
|
StringView homedir()
|
|
|
|
{
|
|
|
|
StringView home = getenv("HOME");
|
|
|
|
if (home.empty())
|
|
|
|
return getpwuid(geteuid())->pw_dir;
|
|
|
|
return home;
|
|
|
|
}
|
|
|
|
|
2016-08-30 23:56:47 +02:00
|
|
|
bool fd_readable(int fd)
|
|
|
|
{
|
|
|
|
fd_set rfds;
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(fd, &rfds);
|
|
|
|
|
|
|
|
timeval tv{0,0};
|
|
|
|
return select(fd+1, &rfds, nullptr, nullptr, &tv) == 1;
|
|
|
|
}
|
|
|
|
|
2016-12-01 21:11:09 +01:00
|
|
|
bool fd_writable(int fd)
|
|
|
|
{
|
2017-08-29 03:32:47 +02:00
|
|
|
fd_set wfds;
|
|
|
|
FD_ZERO(&wfds);
|
|
|
|
FD_SET(fd, &wfds);
|
2016-12-01 21:11:09 +01:00
|
|
|
|
|
|
|
timeval tv{0,0};
|
2017-08-29 03:32:47 +02:00
|
|
|
return select(fd+1, nullptr, &wfds, nullptr, &tv) == 1;
|
2016-12-01 21:11:09 +01:00
|
|
|
}
|
|
|
|
|
2015-06-05 14:52:56 +02:00
|
|
|
String read_fd(int fd, bool text)
|
2011-12-22 14:33:29 +01:00
|
|
|
{
|
2012-04-14 03:17:09 +02:00
|
|
|
String content;
|
2015-06-05 14:52:56 +02:00
|
|
|
constexpr size_t bufsize = 256;
|
|
|
|
char buf[bufsize];
|
2017-06-12 06:21:34 +02:00
|
|
|
while (ssize_t size = read(fd, buf, bufsize))
|
2011-09-02 18:51:20 +02:00
|
|
|
{
|
2017-06-12 06:21:34 +02:00
|
|
|
if (size == -1)
|
|
|
|
throw file_access_error{fd, strerror(errno)};
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2015-06-05 14:52:56 +02:00
|
|
|
if (text)
|
|
|
|
{
|
2018-03-18 13:42:17 +01:00
|
|
|
for (StringView data{buf, buf + size}; not data.empty();)
|
2015-06-05 14:52:56 +02:00
|
|
|
{
|
2018-03-18 13:42:17 +01:00
|
|
|
auto it = find(data, '\r');
|
|
|
|
content += StringView{data.begin(), it};
|
|
|
|
data = StringView{(it != data.end()) ? it+1 : it, data.end()};
|
2015-06-05 14:52:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
content += StringView{buf, buf + size};
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
2011-11-27 13:56:38 +01:00
|
|
|
return content;
|
|
|
|
}
|
|
|
|
|
2015-06-05 14:52:56 +02:00
|
|
|
String read_file(StringView filename, bool text)
|
2014-08-15 14:21:54 +02:00
|
|
|
{
|
2017-08-29 04:42:04 +02:00
|
|
|
int fd = open(filename.zstr(), O_RDONLY);
|
2014-08-15 14:21:54 +02:00
|
|
|
if (fd == -1)
|
|
|
|
throw file_access_error(filename, strerror(errno));
|
|
|
|
|
2016-11-29 00:53:50 +01:00
|
|
|
auto close_fd = on_scope_end([fd]{ close(fd); });
|
2015-06-05 14:52:56 +02:00
|
|
|
return read_fd(fd, text);
|
2014-08-15 14:21:54 +02:00
|
|
|
}
|
|
|
|
|
2015-10-16 02:33:17 +02:00
|
|
|
MappedFile::MappedFile(StringView filename)
|
2017-10-30 07:48:15 +01:00
|
|
|
: data{nullptr}
|
2011-11-27 13:56:38 +01:00
|
|
|
{
|
2017-08-29 04:42:04 +02:00
|
|
|
fd = open(filename.zstr(), O_RDONLY | O_NONBLOCK);
|
2012-08-10 14:24:13 +02:00
|
|
|
if (fd == -1)
|
2017-08-29 04:42:04 +02:00
|
|
|
throw file_access_error(filename, strerror(errno));
|
2014-08-15 14:21:54 +02:00
|
|
|
|
2012-11-23 18:43:10 +01:00
|
|
|
fstat(fd, &st);
|
2012-12-05 13:59:08 +01:00
|
|
|
if (S_ISDIR(st.st_mode))
|
2017-08-29 04:42:04 +02:00
|
|
|
throw file_access_error(filename, "is a directory");
|
2012-11-23 18:43:10 +01:00
|
|
|
|
2017-10-30 07:48:15 +01:00
|
|
|
if (st.st_size == 0)
|
|
|
|
return;
|
|
|
|
|
2015-10-16 02:33:17 +02:00
|
|
|
data = (const char*)mmap(nullptr, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
|
2017-10-30 07:48:15 +01:00
|
|
|
if (data == MAP_FAILED)
|
|
|
|
throw file_access_error{filename, strerror(errno)};
|
|
|
|
|
|
|
|
if (st.st_size > std::numeric_limits<int>::max())
|
|
|
|
throw runtime_error("file is too big");
|
2015-10-16 02:33:17 +02:00
|
|
|
}
|
2012-08-10 14:24:13 +02:00
|
|
|
|
2015-10-16 02:33:17 +02:00
|
|
|
MappedFile::~MappedFile()
|
|
|
|
{
|
|
|
|
if (fd != -1)
|
|
|
|
{
|
2017-10-30 07:48:15 +01:00
|
|
|
if (data != nullptr)
|
|
|
|
munmap((void*)data, st.st_size);
|
2015-10-16 02:33:17 +02:00
|
|
|
close(fd);
|
|
|
|
}
|
2012-08-10 14:24:13 +02:00
|
|
|
}
|
|
|
|
|
2016-11-29 00:53:50 +01:00
|
|
|
MappedFile::operator StringView() const
|
|
|
|
{
|
|
|
|
return { data, (int)st.st_size };
|
|
|
|
}
|
|
|
|
|
2015-10-16 14:58:56 +02:00
|
|
|
bool file_exists(StringView filename)
|
|
|
|
{
|
|
|
|
struct stat st;
|
2017-08-29 04:42:04 +02:00
|
|
|
return stat(filename.zstr(), &st) == 0;
|
2015-10-16 14:58:56 +02:00
|
|
|
}
|
|
|
|
|
2015-11-27 00:40:17 +01:00
|
|
|
void write(int fd, StringView data)
|
2012-08-10 14:24:13 +02:00
|
|
|
{
|
2014-04-18 15:03:08 +02:00
|
|
|
const char* ptr = data.data();
|
|
|
|
ssize_t count = (int)data.length();
|
2012-08-10 14:24:13 +02:00
|
|
|
|
|
|
|
while (count)
|
|
|
|
{
|
|
|
|
ssize_t written = ::write(fd, ptr, count);
|
|
|
|
ptr += written;
|
|
|
|
count -= written;
|
|
|
|
|
|
|
|
if (written == -1)
|
2016-08-09 22:45:06 +02:00
|
|
|
throw file_access_error(format("fd: {}", fd), strerror(errno));
|
2012-08-10 14:24:13 +02:00
|
|
|
}
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
|
|
|
|
2014-08-15 14:21:54 +02:00
|
|
|
void write_buffer_to_fd(Buffer& buffer, int fd)
|
2011-09-02 18:51:20 +02:00
|
|
|
{
|
2015-12-06 13:51:55 +01:00
|
|
|
auto eolformat = buffer.options()["eolformat"].get<EolFormat>();
|
2014-04-18 15:03:08 +02:00
|
|
|
StringView eoldata;
|
2015-12-06 13:51:55 +01:00
|
|
|
if (eolformat == EolFormat::Crlf)
|
2014-04-18 15:03:08 +02:00
|
|
|
eoldata = "\r\n";
|
2012-08-10 14:24:13 +02:00
|
|
|
else
|
2014-04-18 15:03:08 +02:00
|
|
|
eoldata = "\n";
|
2012-08-10 14:24:13 +02:00
|
|
|
|
2015-12-06 13:51:55 +01:00
|
|
|
if (buffer.options()["BOM"].get<ByteOrderMark>() == ByteOrderMark::Utf8)
|
2015-12-10 09:00:10 +01:00
|
|
|
if (::write(fd, "\xEF\xBB\xBF", 3) < 0)
|
|
|
|
throw runtime_error(format("unable to write data to the buffer (fd: {}; errno: {})", fd, ::strerror(errno)));
|
2014-08-15 14:21:54 +02:00
|
|
|
|
|
|
|
for (LineCount i = 0; i < buffer.line_count(); ++i)
|
2014-03-26 20:11:04 +01:00
|
|
|
{
|
2014-08-15 14:21:54 +02:00
|
|
|
// end of lines are written according to eolformat but always
|
|
|
|
// stored as \n
|
|
|
|
StringView linedata = buffer[i];
|
|
|
|
write(fd, linedata.substr(0, linedata.length()-1));
|
|
|
|
write(fd, eoldata);
|
|
|
|
}
|
|
|
|
}
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2017-04-19 17:47:07 +02:00
|
|
|
void write_buffer_to_file(Buffer& buffer, StringView filename, bool force)
|
2014-08-15 14:21:54 +02:00
|
|
|
{
|
2017-04-19 17:47:07 +02:00
|
|
|
struct stat st;
|
|
|
|
auto zfilename = filename.zstr();
|
|
|
|
|
|
|
|
if (force)
|
|
|
|
{
|
|
|
|
if (::stat(zfilename, &st) == 0)
|
|
|
|
{
|
|
|
|
if (::chmod(zfilename, st.st_mode | S_IWUSR) < 0)
|
2018-04-06 16:56:53 +02:00
|
|
|
throw runtime_error("unable to change file permissions");
|
2017-04-19 17:47:07 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
force = false;
|
|
|
|
}
|
|
|
|
auto restore_mode = on_scope_end([&]{
|
|
|
|
if (force and ::chmod(zfilename, st.st_mode) < 0)
|
2018-04-06 16:56:53 +02:00
|
|
|
throw runtime_error("unable to restore file permissions");
|
2017-04-19 17:47:07 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
int fd = open(zfilename, O_CREAT | O_WRONLY | O_TRUNC, 0644);
|
2016-10-01 18:07:50 +02:00
|
|
|
if (fd == -1)
|
|
|
|
throw file_access_error(filename, strerror(errno));
|
2014-08-15 14:21:54 +02:00
|
|
|
|
2016-10-03 21:00:36 +02:00
|
|
|
{
|
|
|
|
auto close_fd = on_scope_end([fd]{ close(fd); });
|
|
|
|
write_buffer_to_fd(buffer, fd);
|
|
|
|
}
|
2012-08-10 18:48:21 +02:00
|
|
|
|
2014-06-29 23:03:59 +02:00
|
|
|
if ((buffer.flags() & Buffer::Flags::File) and
|
|
|
|
real_path(filename) == real_path(buffer.name()))
|
2013-10-17 19:47:09 +02:00
|
|
|
buffer.notify_saved();
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
|
|
|
|
2014-10-13 14:38:28 +02:00
|
|
|
void write_buffer_to_backup_file(Buffer& buffer)
|
|
|
|
{
|
2015-02-06 14:47:19 +01:00
|
|
|
String path = real_path(buffer.name());
|
|
|
|
StringView dir, file;
|
|
|
|
std::tie(dir,file) = split_path(path);
|
|
|
|
|
2015-04-22 14:19:46 +02:00
|
|
|
char pattern[PATH_MAX];
|
|
|
|
if (dir.empty())
|
|
|
|
format_to(pattern, ".{}.kak.XXXXXX", file);
|
|
|
|
else
|
|
|
|
format_to(pattern, "{}/.{}.kak.XXXXXX", dir, file);
|
2015-02-06 14:47:19 +01:00
|
|
|
|
2015-04-22 14:19:46 +02:00
|
|
|
int fd = mkstemp(pattern);
|
2014-10-13 14:38:28 +02:00
|
|
|
if (fd >= 0)
|
|
|
|
{
|
|
|
|
write_buffer_to_fd(buffer, fd);
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-22 22:22:34 +01:00
|
|
|
String find_file(StringView filename, StringView buf_dir, ConstArrayView<String> paths)
|
2013-02-27 19:58:38 +01:00
|
|
|
{
|
2014-01-03 20:32:42 +01:00
|
|
|
struct stat buf;
|
2017-08-29 04:42:04 +02:00
|
|
|
if (filename.substr(0_byte, 1_byte) == "/")
|
2014-01-03 20:32:42 +01:00
|
|
|
{
|
2014-04-20 13:18:40 +02:00
|
|
|
if (stat(filename.zstr(), &buf) == 0 and S_ISREG(buf.st_mode))
|
2014-04-18 15:03:08 +02:00
|
|
|
return filename.str();
|
2014-01-03 20:32:42 +01:00
|
|
|
return "";
|
|
|
|
}
|
2017-08-29 04:42:04 +02:00
|
|
|
if (filename.substr(0_byte, 2_byte) == "~/")
|
2014-01-03 20:32:42 +01:00
|
|
|
{
|
2018-01-20 01:19:23 +01:00
|
|
|
String candidate = homedir() + filename.substr(1_byte);
|
2014-01-03 20:32:42 +01:00
|
|
|
if (stat(candidate.c_str(), &buf) == 0 and S_ISREG(buf.st_mode))
|
|
|
|
return candidate;
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2018-03-22 22:22:34 +01:00
|
|
|
for (auto candidate : paths | transform([&](StringView s) { return parse_filename(s, buf_dir); }))
|
2013-02-27 19:58:38 +01:00
|
|
|
{
|
2013-03-13 18:52:08 +01:00
|
|
|
if (not candidate.empty() and candidate.back() != '/')
|
|
|
|
candidate += '/';
|
|
|
|
candidate += filename;
|
2013-02-27 19:58:38 +01:00
|
|
|
if (stat(candidate.c_str(), &buf) == 0 and S_ISREG(buf.st_mode))
|
|
|
|
return candidate;
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2016-12-16 00:47:34 +01:00
|
|
|
void make_directory(StringView dir, mode_t mode)
|
2015-08-23 15:13:46 +02:00
|
|
|
{
|
|
|
|
auto it = dir.begin(), end = dir.end();
|
|
|
|
while(it != end)
|
|
|
|
{
|
|
|
|
it = std::find(it+1, end, '/');
|
|
|
|
struct stat st;
|
|
|
|
StringView dirname{dir.begin(), it};
|
|
|
|
if (stat(dirname.zstr(), &st) == 0)
|
|
|
|
{
|
|
|
|
if (not S_ISDIR(st.st_mode))
|
2018-04-06 16:56:53 +02:00
|
|
|
throw runtime_error(format("cannot make directory, '{}' exists but is not a directory", dirname));
|
2015-08-23 15:13:46 +02:00
|
|
|
}
|
2015-09-16 21:04:29 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
auto old_mask = umask(0);
|
|
|
|
auto restore_mask = on_scope_end([old_mask]() { umask(old_mask); });
|
|
|
|
|
2016-12-16 00:47:34 +01:00
|
|
|
if (mkdir(dirname.zstr(), mode) != 0)
|
2015-09-16 21:04:29 +02:00
|
|
|
throw runtime_error(format("mkdir failed for directory '{}' errno {}", dirname, errno));
|
|
|
|
}
|
2015-08-23 15:13:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-23 21:43:55 +01:00
|
|
|
template<typename Filter>
|
2016-03-02 14:51:16 +01:00
|
|
|
Vector<String> list_files(StringView dirname, Filter filter)
|
2013-12-23 21:43:55 +01:00
|
|
|
{
|
2015-09-11 14:49:08 +02:00
|
|
|
char buffer[PATH_MAX+1];
|
|
|
|
format_to(buffer, "{}", dirname);
|
|
|
|
DIR* dir = opendir(dirname.empty() ? "./" : buffer);
|
2013-12-23 21:43:55 +01:00
|
|
|
if (not dir)
|
2014-04-06 11:59:51 +02:00
|
|
|
return {};
|
2013-12-23 21:43:55 +01:00
|
|
|
|
2017-05-17 21:17:16 +02:00
|
|
|
auto close_dir = on_scope_end([dir]{ closedir(dir); });
|
2014-04-06 11:59:51 +02:00
|
|
|
|
2015-01-09 14:57:21 +01:00
|
|
|
Vector<String> result;
|
2013-12-23 21:43:55 +01:00
|
|
|
while (dirent* entry = readdir(dir))
|
|
|
|
{
|
2015-09-11 14:49:08 +02:00
|
|
|
StringView filename = entry->d_name;
|
2016-04-07 23:47:41 +02:00
|
|
|
if (filename.empty())
|
2013-12-23 21:43:55 +01:00
|
|
|
continue;
|
|
|
|
|
2014-04-11 19:42:06 +02:00
|
|
|
struct stat st;
|
2016-03-02 14:51:16 +01:00
|
|
|
auto fmt_str = (dirname.empty() or dirname.back() == '/') ? "{}{}" : "{}/{}";
|
|
|
|
format_to(buffer, fmt_str, dirname, filename);
|
2016-04-07 23:47:41 +02:00
|
|
|
if (stat(buffer, &st) != 0 or not filter(*entry, st))
|
2016-03-02 14:51:16 +01:00
|
|
|
continue;
|
2015-09-11 14:49:08 +02:00
|
|
|
|
2016-03-02 14:51:16 +01:00
|
|
|
if (S_ISDIR(st.st_mode))
|
|
|
|
filename = format_to(buffer, "{}/", filename);
|
|
|
|
result.push_back(filename.str());
|
2013-12-23 21:43:55 +01:00
|
|
|
}
|
2016-03-02 14:51:16 +01:00
|
|
|
return result;
|
2013-12-23 21:43:55 +01:00
|
|
|
}
|
|
|
|
|
2015-08-23 15:22:23 +02:00
|
|
|
Vector<String> list_files(StringView directory)
|
|
|
|
{
|
2016-04-07 23:47:41 +02:00
|
|
|
return list_files(directory, [](const dirent& entry, const struct stat&) {
|
2016-03-02 14:51:16 +01:00
|
|
|
return StringView{entry.d_name}.substr(0_byte, 1_byte) != ".";
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static CandidateList candidates(ConstArrayView<RankedMatch> matches, StringView dirname)
|
|
|
|
{
|
|
|
|
CandidateList res;
|
|
|
|
res.reserve(matches.size());
|
|
|
|
for (auto& match : matches)
|
|
|
|
res.push_back(dirname + match.candidate());
|
|
|
|
return res;
|
2015-08-23 15:22:23 +02:00
|
|
|
}
|
|
|
|
|
2016-04-07 23:47:41 +02:00
|
|
|
CandidateList complete_filename(StringView prefix, const Regex& ignored_regex,
|
2016-10-13 20:46:09 +02:00
|
|
|
ByteCount cursor_pos, FilenameFlags flags)
|
2013-03-13 19:01:59 +01:00
|
|
|
{
|
2016-10-13 20:46:09 +02:00
|
|
|
prefix = prefix.substr(0, cursor_pos);
|
2015-02-06 14:47:19 +01:00
|
|
|
StringView dirname, fileprefix;
|
2016-10-13 20:46:09 +02:00
|
|
|
std::tie(dirname, fileprefix) = split_path(prefix);
|
|
|
|
auto parsed_dirname = parse_filename(dirname);
|
2013-03-13 19:01:59 +01:00
|
|
|
|
2013-03-14 13:42:07 +01:00
|
|
|
const bool check_ignored_regex = not ignored_regex.empty() and
|
2015-02-06 14:47:19 +01:00
|
|
|
not regex_match(fileprefix.begin(), fileprefix.end(), ignored_regex);
|
2016-10-13 20:46:09 +02:00
|
|
|
const bool only_dirs = (flags & FilenameFlags::OnlyDirectories);
|
2013-03-13 19:01:59 +01:00
|
|
|
|
2016-11-18 10:37:16 +01:00
|
|
|
auto filter = [&ignored_regex, check_ignored_regex, only_dirs](const dirent& entry, struct stat& st)
|
2013-03-13 19:01:59 +01:00
|
|
|
{
|
2016-05-10 10:12:30 +02:00
|
|
|
StringView name{entry.d_name};
|
2016-11-18 10:37:16 +01:00
|
|
|
return (not check_ignored_regex or not regex_match(name.begin(), name.end(), ignored_regex)) and
|
2016-10-13 20:46:09 +02:00
|
|
|
(not only_dirs or S_ISDIR(st.st_mode));
|
2013-12-23 21:43:55 +01:00
|
|
|
};
|
2016-10-13 20:46:09 +02:00
|
|
|
auto files = list_files(parsed_dirname, filter);
|
2016-03-02 14:51:16 +01:00
|
|
|
Vector<RankedMatch> matches;
|
|
|
|
for (auto& file : files)
|
|
|
|
{
|
|
|
|
if (RankedMatch match{file, fileprefix})
|
|
|
|
matches.push_back(match);
|
|
|
|
}
|
|
|
|
std::sort(matches.begin(), matches.end());
|
2016-10-13 20:46:09 +02:00
|
|
|
const bool expand = (flags & FilenameFlags::Expand);
|
|
|
|
return candidates(matches, expand ? parsed_dirname : dirname);
|
2013-12-23 21:43:55 +01:00
|
|
|
}
|
2013-03-13 19:01:59 +01:00
|
|
|
|
2016-11-28 14:59:55 +01:00
|
|
|
CandidateList complete_command(StringView prefix, ByteCount cursor_pos)
|
2013-12-23 21:43:55 +01:00
|
|
|
{
|
|
|
|
String real_prefix = parse_filename(prefix.substr(0, cursor_pos));
|
2015-02-06 14:47:19 +01:00
|
|
|
StringView dirname, fileprefix;
|
|
|
|
std::tie(dirname, fileprefix) = split_path(real_prefix);
|
2013-03-13 19:01:59 +01:00
|
|
|
|
2015-02-06 14:47:19 +01:00
|
|
|
if (not dirname.empty())
|
2013-12-23 21:43:55 +01:00
|
|
|
{
|
2017-11-22 18:35:40 +01:00
|
|
|
auto filter = [](const dirent& entry, const struct stat& st)
|
2014-12-21 18:08:33 +01:00
|
|
|
{
|
|
|
|
bool executable = (st.st_mode & S_IXUSR)
|
|
|
|
| (st.st_mode & S_IXGRP)
|
|
|
|
| (st.st_mode & S_IXOTH);
|
|
|
|
return S_ISDIR(st.st_mode) or (S_ISREG(st.st_mode) and executable);
|
|
|
|
};
|
2016-03-02 14:51:16 +01:00
|
|
|
auto files = list_files(dirname, filter);
|
|
|
|
Vector<RankedMatch> matches;
|
|
|
|
for (auto& file : files)
|
|
|
|
{
|
|
|
|
if (RankedMatch match{file, real_prefix})
|
|
|
|
matches.push_back(match);
|
|
|
|
}
|
|
|
|
std::sort(matches.begin(), matches.end());
|
|
|
|
return candidates(matches, dirname);
|
2014-12-21 18:08:33 +01:00
|
|
|
}
|
|
|
|
|
2017-01-08 23:30:15 +01:00
|
|
|
using TimeSpec = decltype(stat::st_mtim);
|
2014-05-01 00:03:42 +02:00
|
|
|
|
2014-04-24 20:08:05 +02:00
|
|
|
struct CommandCache
|
|
|
|
{
|
2015-09-27 12:55:34 +02:00
|
|
|
TimeSpec mtim = {};
|
2015-01-09 14:57:21 +01:00
|
|
|
Vector<String> commands;
|
2014-04-24 20:08:05 +02:00
|
|
|
};
|
2017-03-07 01:30:54 +01:00
|
|
|
static HashMap<String, CommandCache, MemoryDomain::Commands> command_cache;
|
2014-04-24 20:08:05 +02:00
|
|
|
|
2016-03-02 14:51:16 +01:00
|
|
|
Vector<RankedMatch> matches;
|
2016-03-25 01:14:56 +01:00
|
|
|
for (auto dir : StringView{getenv("PATH")} | split<StringView>(':'))
|
2013-12-23 21:43:55 +01:00
|
|
|
{
|
2015-09-11 14:49:08 +02:00
|
|
|
auto dirname = ((not dir.empty() and dir.back() == '/') ? dir.substr(0, dir.length()-1) : dir).str();
|
2013-12-23 21:43:55 +01:00
|
|
|
|
2014-04-24 20:08:05 +02:00
|
|
|
struct stat st;
|
2015-09-11 14:49:08 +02:00
|
|
|
if (stat(dirname.c_str(), &st))
|
2014-04-24 20:08:05 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
auto& cache = command_cache[dirname];
|
2015-09-27 12:55:34 +02:00
|
|
|
if (memcmp(&cache.mtim, &st.st_mtim, sizeof(TimeSpec)) != 0)
|
2014-04-24 20:08:05 +02:00
|
|
|
{
|
2017-11-22 18:35:40 +01:00
|
|
|
auto filter = [](const dirent& entry, const struct stat& st) {
|
2015-02-06 14:47:19 +01:00
|
|
|
bool executable = (st.st_mode & S_IXUSR)
|
|
|
|
| (st.st_mode & S_IXGRP)
|
|
|
|
| (st.st_mode & S_IXOTH);
|
|
|
|
return S_ISREG(st.st_mode) and executable;
|
|
|
|
};
|
|
|
|
|
2016-03-02 14:51:16 +01:00
|
|
|
cache.commands = list_files(dirname, filter);
|
2015-09-27 12:55:34 +02:00
|
|
|
memcpy(&cache.mtim, &st.st_mtim, sizeof(TimeSpec));
|
2014-04-24 20:08:05 +02:00
|
|
|
}
|
|
|
|
for (auto& cmd : cache.commands)
|
|
|
|
{
|
2016-03-02 14:51:16 +01:00
|
|
|
if (RankedMatch match{cmd, fileprefix})
|
|
|
|
matches.push_back(match);
|
2014-04-24 20:08:05 +02:00
|
|
|
}
|
2013-12-23 21:43:55 +01:00
|
|
|
}
|
2016-03-02 14:51:16 +01:00
|
|
|
std::sort(matches.begin(), matches.end());
|
|
|
|
auto it = std::unique(matches.begin(), matches.end());
|
|
|
|
matches.erase(it, matches.end());
|
|
|
|
return candidates(matches, "");
|
2013-03-13 19:01:59 +01:00
|
|
|
}
|
|
|
|
|
2015-09-27 12:55:34 +02:00
|
|
|
timespec get_fs_timestamp(StringView filename)
|
2013-10-15 19:51:31 +02:00
|
|
|
{
|
|
|
|
struct stat st;
|
2014-04-20 13:18:40 +02:00
|
|
|
if (stat(filename.zstr(), &st) != 0)
|
2013-10-17 19:47:09 +02:00
|
|
|
return InvalidTime;
|
2015-09-27 12:55:34 +02:00
|
|
|
return st.st_mtim;
|
2013-10-15 19:51:31 +02:00
|
|
|
}
|
|
|
|
|
2014-10-30 01:50:40 +01:00
|
|
|
String get_kak_binary_path()
|
|
|
|
{
|
|
|
|
char buffer[2048];
|
|
|
|
#if defined(__linux__) or defined(__CYGWIN__)
|
|
|
|
ssize_t res = readlink("/proc/self/exe", buffer, 2048);
|
|
|
|
kak_assert(res != -1);
|
|
|
|
buffer[res] = '\0';
|
|
|
|
return buffer;
|
2015-11-02 21:22:00 +01:00
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
int mib[] = {CTL_KERN, KERN_PROC, KERN_PROC_PATHNAME, -1};
|
|
|
|
size_t res = sizeof(buffer);
|
|
|
|
sysctl(mib, 4, buffer, &res, NULL, 0);
|
|
|
|
return buffer;
|
2014-10-30 01:50:40 +01:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
uint32_t bufsize = 2048;
|
|
|
|
_NSGetExecutablePath(buffer, &bufsize);
|
|
|
|
char* canonical_path = realpath(buffer, nullptr);
|
|
|
|
String path = canonical_path;
|
|
|
|
free(canonical_path);
|
|
|
|
return path;
|
2015-09-25 00:36:29 +02:00
|
|
|
#elif defined(__HAIKU__)
|
|
|
|
BApplication app("application/x-vnd.kakoune");
|
|
|
|
app_info info;
|
|
|
|
status_t status = app.GetAppInfo(&info);
|
|
|
|
kak_assert(status == B_OK);
|
|
|
|
BPath path(&info.ref);
|
|
|
|
return path.Path();
|
2015-11-02 19:14:34 +01:00
|
|
|
#elif defined(__DragonFly__)
|
|
|
|
ssize_t res = readlink("/proc/curproc/file", buffer, 2048);
|
|
|
|
kak_assert(res != -1);
|
|
|
|
buffer[res] = '\0';
|
|
|
|
return buffer;
|
2018-04-17 02:31:23 +02:00
|
|
|
#elif defined(__OpenBSD__)
|
2018-06-19 11:24:41 +02:00
|
|
|
return KAK_BIN_PATH;
|
2014-10-30 01:50:40 +01:00
|
|
|
#else
|
|
|
|
# error "finding executable path is not implemented on this platform"
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|