#include "file.hh" #include "assert.hh" #include "buffer.hh" #include "buffer_manager.hh" #include "completion.hh" #include "debug.hh" #include "unicode.hh" #include #include #include #include #include #include #include namespace Kakoune { String parse_filename(StringView filename) { if (filename.length() >= 1 and filename[0] == '~' and (filename.length() == 1 or filename[1] == '/')) return parse_filename("$HOME"_str + filename.substr(1_byte)); ByteCount pos = 0; String result; for (ByteCount i = 0; i < filename.length(); ++i) { if (filename[i] == '$' and (i == 0 or filename[i-1] != '\\')) { result += filename.substr(pos, i - pos); ByteCount end = i+1; while (end != filename.length() and is_word(filename[end])) ++end; StringView var_name = filename.substr(i+1, end - i - 1); const char* var_value = getenv(var_name.zstr()); if (var_value) result += var_value; pos = end; } } if (pos != filename.length()) result += filename.substr(pos); return result; } String real_path(StringView filename) { StringView dirname = "."; StringView basename = filename; auto it = find(filename.rbegin(), filename.rend(), '/'); if (it != filename.rend()) { dirname = StringView{filename.begin(), it.base()}; basename = StringView{it.base(), filename.end()}; } char buffer[PATH_MAX+1]; char* res = realpath(dirname.zstr(), buffer); if (not res) throw file_not_found{dirname}; return res + "/"_str + basename; } String compact_path(StringView filename) { String real_filename = real_path(filename); char cwd[1024]; getcwd(cwd, 1024); String real_cwd = real_path(cwd) + '/'; if (prefix_match(real_filename, real_cwd)) return real_filename.substr(real_cwd.length()); const char* home = getenv("HOME"); if (home) { ByteCount home_len = (int)strlen(home); if (real_filename.substr(0, home_len) == home) return "~" + real_filename.substr(home_len); } return filename.str(); } String read_file(StringView filename) { int fd = open(parse_filename(filename).c_str(), O_RDONLY); if (fd == -1) { if (errno == ENOENT) throw file_not_found(filename); throw file_access_error(filename, strerror(errno)); } auto close_fd = on_scope_end([fd]{ close(fd); }); String content; char buf[256]; while (true) { ssize_t size = read(fd, buf, 256); if (size == -1 or size == 0) break; content += String(buf, buf + size); } return content; } Buffer* create_buffer_from_file(String filename) { filename = real_path(parse_filename(filename)); int fd = open(filename.c_str(), O_RDONLY); if (fd == -1) { if (errno == ENOENT) return nullptr; throw file_access_error(filename, strerror(errno)); } struct stat st; fstat(fd, &st); if (S_ISDIR(st.st_mode)) { close(fd); throw file_access_error(filename, "is a directory"); } const char* data = (const char*)mmap(nullptr, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0); auto cleanup = on_scope_end([&]{ munmap((void*)data, st.st_size); close(fd); }); const char* pos = data; bool crlf = false; bool bom = false; if (st.st_size >= 3 and data[0] == '\xEF' and data[1] == '\xBB' and data[2] == '\xBF') { bom = true; pos = data + 3; } std::vector lines; const char* end = data + st.st_size; while (pos < end) { const char* line_end = pos; while (line_end < end and *line_end != '\r' and *line_end != '\n') ++line_end; // this should happen only when opening a file which has no // end of line as last character. if (line_end == end) { lines.emplace_back(pos, line_end); lines.back() += '\n'; break; } lines.emplace_back(pos, line_end + 1); lines.back().back() = '\n'; if (line_end+1 != end and *line_end == '\r' and *(line_end+1) == '\n') { crlf = true; pos = line_end + 2; } else pos = line_end + 1; } Buffer* buffer = BufferManager::instance().get_buffer_ifp(filename); if (buffer) buffer->reload(std::move(lines), st.st_mtime); else buffer = new Buffer{filename, Buffer::Flags::File, std::move(lines), st.st_mtime}; OptionManager& options = buffer->options(); options.get_local_option("eolformat").set(crlf ? "crlf" : "lf"); options.get_local_option("BOM").set(bom ? "utf-8" : "no"); return buffer; } static void write(int fd, StringView data, StringView filename) { const char* ptr = data.data(); ssize_t count = (int)data.length(); while (count) { ssize_t written = ::write(fd, ptr, count); ptr += written; count -= written; if (written == -1) throw file_access_error(filename, strerror(errno)); } } void write_buffer_to_file(Buffer& buffer, StringView filename) { buffer.run_hook_in_own_context("BufWritePre", buffer.name()); const String& eolformat = buffer.options()["eolformat"].get(); StringView eoldata; if (eolformat == "crlf") eoldata = "\r\n"; else eoldata = "\n"; { int fd = open(parse_filename(filename).c_str(), O_CREAT | O_WRONLY | O_TRUNC, 0644); if (fd == -1) throw file_access_error(filename, strerror(errno)); auto close_fd = on_scope_end([fd]{ close(fd); }); if (buffer.options()["BOM"].get() == "utf-8") ::write(fd, "\xEF\xBB\xBF", 3); for (LineCount i = 0; i < buffer.line_count(); ++i) { // 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), filename); write(fd, eoldata, filename); } } if ((buffer.flags() & Buffer::Flags::File) and filename == buffer.name()) buffer.notify_saved(); buffer.run_hook_in_own_context("BufWritePost", buffer.name()); } String find_file(StringView filename, memoryview paths) { struct stat buf; if (filename.length() > 1 and filename[0] == '/') { if (stat(filename.zstr(), &buf) == 0 and S_ISREG(buf.st_mode)) return filename.str(); return ""; } if (filename.length() > 2 and filename[0] == '~' and filename[1] == '/') { String candidate = getenv("HOME") + filename.substr(1_byte).str(); if (stat(candidate.c_str(), &buf) == 0 and S_ISREG(buf.st_mode)) return candidate; return ""; } for (auto candidate : paths) { if (not candidate.empty() and candidate.back() != '/') candidate += '/'; candidate += filename; if (stat(candidate.c_str(), &buf) == 0 and S_ISREG(buf.st_mode)) return candidate; } return ""; } template std::vector list_files(StringView prefix, StringView dirname, Filter filter) { kak_assert(dirname.empty() or dirname.back() == '/'); DIR* dir = opendir(dirname.empty() ? "./" : dirname.zstr()); if (not dir) return {}; auto closeDir = on_scope_end([=]{ closedir(dir); }); std::vector result; std::vector subseq_result; while (dirent* entry = readdir(dir)) { if (not filter(*entry)) continue; String filename = entry->d_name; if (filename.empty()) continue; const bool match_prefix = prefix_match(filename, prefix); const bool match_subseq = subsequence_match(filename, prefix); struct stat st; if ((match_prefix or match_subseq) and stat((dirname + filename).c_str(), &st) == 0) { if (S_ISDIR(st.st_mode)) filename += '/'; if (prefix.length() != 0 or filename[0] != '.') { if (match_prefix) result.push_back(filename); if (match_subseq) subseq_result.push_back(filename); } } } return result.empty() ? subseq_result : result; } std::vector complete_filename(StringView prefix, const Regex& ignored_regex, ByteCount cursor_pos) { String real_prefix = parse_filename(prefix.substr(0, cursor_pos)); String dirname; String fileprefix = real_prefix; ByteCount dir_end = -1; for (ByteCount i = 0; i < real_prefix.length(); ++i) { if (real_prefix[i] == '/') dir_end = i; } if (dir_end != -1) { dirname = real_prefix.substr(0, dir_end + 1); fileprefix = real_prefix.substr(dir_end + 1); } const bool check_ignored_regex = not ignored_regex.empty() and not boost::regex_match(fileprefix.c_str(), ignored_regex); auto filter = [&](const dirent& entry) { return not check_ignored_regex or not boost::regex_match(entry.d_name, ignored_regex); }; std::vector res = list_files(fileprefix, dirname, filter); for (auto& file : res) file = escape(dirname + file); std::sort(res.begin(), res.end()); return res; } std::vector complete_command(StringView prefix, ByteCount cursor_pos) { String real_prefix = parse_filename(prefix.substr(0, cursor_pos)); String dirname; String fileprefix = real_prefix; ByteCount dir_end = -1; for (ByteCount i = 0; i < real_prefix.length(); ++i) { if (real_prefix[i] == '/') dir_end = i; } typedef decltype(stat::st_mtime) TimeSpec; struct CommandCache { TimeSpec mtime = {}; std::vector commands; }; static std::unordered_map command_cache; std::vector path; if (dir_end != -1) { path.emplace_back(real_prefix.substr(0, dir_end + 1)); fileprefix = real_prefix.substr(dir_end + 1); } else path = split(getenv("PATH"), ':'); std::vector res; for (auto dirname : path) { if (not dirname.empty() and dirname.back() != '/') dirname += '/'; struct stat st; if (stat(dirname.substr(0_byte, dirname.length() - 1).zstr(), &st)) continue; auto filter = [&](const dirent& entry) { struct stat st; if (stat((dirname + entry.d_name).c_str(), &st)) return false; 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; }; auto& cache = command_cache[dirname]; if (memcmp(&cache.mtime, &st.st_mtime, sizeof(TimeSpec)) != 0) { memcpy(&cache.mtime, &st.st_mtime, sizeof(TimeSpec)); cache.commands = list_files("", dirname, filter); } for (auto& cmd : cache.commands) { if (prefix_match(cmd, fileprefix)) res.push_back(cmd); } } std::sort(res.begin(), res.end()); auto it = std::unique(res.begin(), res.end()); res.erase(it, res.end()); return res; } time_t get_fs_timestamp(StringView filename) { struct stat st; if (stat(filename.zstr(), &st) != 0) return InvalidTime; return st.st_mtime; } }