2011-09-02 18:51:20 +02:00
|
|
|
#include "file.hh"
|
2011-09-09 20:40:59 +02:00
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
#include "buffer.hh"
|
2011-09-09 20:40:59 +02:00
|
|
|
#include "buffer_manager.hh"
|
2011-09-09 21:24:18 +02:00
|
|
|
#include "assert.hh"
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2012-10-11 00:41:48 +02:00
|
|
|
#include "unicode.hh"
|
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
2012-05-30 14:19:53 +02:00
|
|
|
#include <unistd.h>
|
2011-09-02 18:51:20 +02:00
|
|
|
#include <cstring>
|
2011-09-09 21:24:18 +02:00
|
|
|
|
2011-09-02 18:51:20 +02:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
String parse_filename(const String& filename)
|
2012-01-29 23:24:43 +01:00
|
|
|
{
|
2012-09-12 19:42:12 +02:00
|
|
|
if (filename.length() >= 2 and filename[0] == '~' and filename[1] == '/')
|
2012-10-11 00:41:48 +02:00
|
|
|
return parse_filename("$HOME/" + filename.substr(2_byte));
|
2012-01-29 23:24:43 +01:00
|
|
|
|
2012-10-11 00:41:48 +02:00
|
|
|
ByteCount pos = 0;
|
2012-04-14 03:17:09 +02:00
|
|
|
String result;
|
2012-10-11 00:41:48 +02:00
|
|
|
for (ByteCount i = 0; i < filename.length(); ++i)
|
2012-01-29 23:24:43 +01:00
|
|
|
{
|
|
|
|
if (filename[i] == '$' and (i == 0 or filename[i-1] != '\\'))
|
|
|
|
{
|
|
|
|
result += filename.substr(pos, i - pos);
|
2012-10-11 00:41:48 +02:00
|
|
|
ByteCount end = i+1;
|
|
|
|
while (end != filename.length() and is_word(filename[end]))
|
2012-01-29 23:24:43 +01:00
|
|
|
++end;
|
2012-04-14 03:17:09 +02:00
|
|
|
String var_name = filename.substr(i+1, end - i - 1);
|
2012-01-29 23:24:43 +01:00
|
|
|
const char* var_value = getenv(var_name.c_str());
|
|
|
|
if (var_value)
|
|
|
|
result += var_value;
|
|
|
|
|
|
|
|
pos = end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (pos != filename.length())
|
|
|
|
result += filename.substr(pos);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
String read_file(const String& filename)
|
2011-12-22 14:33:29 +01:00
|
|
|
{
|
2012-09-12 19:42:12 +02:00
|
|
|
int fd = open(parse_filename(filename).c_str(), O_RDONLY);
|
2011-09-02 18:51:20 +02:00
|
|
|
if (fd == -1)
|
2011-09-02 20:01:20 +02:00
|
|
|
{
|
|
|
|
if (errno == ENOENT)
|
2011-09-09 20:40:59 +02:00
|
|
|
throw file_not_found(filename);
|
2011-09-02 20:01:20 +02:00
|
|
|
|
2011-09-09 20:40:59 +02:00
|
|
|
throw file_access_error(filename, strerror(errno));
|
2011-09-02 20:01:20 +02:00
|
|
|
}
|
2012-11-20 18:52:36 +01:00
|
|
|
auto close_fd = on_scope_end([fd]{ close(fd); });
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
String content;
|
2011-09-02 18:51:20 +02:00
|
|
|
char buf[256];
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
ssize_t size = read(fd, buf, 256);
|
|
|
|
if (size == -1 or size == 0)
|
|
|
|
break;
|
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
content += String(buf, buf + size);
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
2011-11-27 13:56:38 +01:00
|
|
|
return content;
|
|
|
|
}
|
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
Buffer* create_buffer_from_file(const String& filename)
|
2011-11-27 13:56:38 +01:00
|
|
|
{
|
2012-09-12 19:42:12 +02:00
|
|
|
int fd = open(parse_filename(filename).c_str(), O_RDONLY);
|
2012-08-10 14:24:13 +02:00
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
if (errno == ENOENT)
|
2012-10-16 14:59:39 +02:00
|
|
|
return nullptr;
|
2012-08-10 14:24:13 +02:00
|
|
|
|
|
|
|
throw file_access_error(filename, strerror(errno));
|
|
|
|
}
|
2012-11-20 18:52:36 +01:00
|
|
|
auto close_fd = on_scope_end([fd]{ close(fd); });
|
2012-08-10 14:24:13 +02:00
|
|
|
|
2012-08-17 18:52:04 +02:00
|
|
|
if (Buffer* buffer = BufferManager::instance().get_buffer(filename))
|
|
|
|
delete buffer;
|
|
|
|
|
|
|
|
Buffer* buffer = new Buffer(filename, Buffer::Type::File, "");
|
|
|
|
|
2012-08-10 14:24:13 +02:00
|
|
|
String content;
|
|
|
|
char buf[256];
|
|
|
|
bool crlf = false;
|
2012-08-10 18:48:21 +02:00
|
|
|
bool bom = false;
|
|
|
|
bool at_file_begin = true;
|
2012-08-10 14:24:13 +02:00
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
ssize_t size = read(fd, buf, 256);
|
|
|
|
if (size == -1 or size == 0)
|
|
|
|
break;
|
|
|
|
|
2012-08-10 18:48:21 +02:00
|
|
|
ssize_t pos = 0;
|
|
|
|
// detect utf-8 byte order mark
|
|
|
|
if (at_file_begin and size >= 3 and
|
|
|
|
buf[0] == '\xEF' and buf[1] == '\xBB' and buf[2] == '\xBF')
|
|
|
|
{
|
|
|
|
bom = true;
|
|
|
|
pos = 3;
|
|
|
|
}
|
|
|
|
ssize_t start = pos;
|
|
|
|
|
|
|
|
while (pos < size+1)
|
2012-08-10 14:24:13 +02:00
|
|
|
{
|
|
|
|
if (buf[pos] == '\r' or pos == size)
|
|
|
|
{
|
|
|
|
if (buf[pos] == '\r')
|
|
|
|
crlf = true;
|
|
|
|
|
2012-09-10 19:26:17 +02:00
|
|
|
buffer->insert(buffer->end()-1, String(buf+start, buf+pos));
|
2012-08-10 14:24:13 +02:00
|
|
|
start = pos+1;
|
|
|
|
}
|
2012-08-10 18:48:21 +02:00
|
|
|
++pos;
|
2012-08-10 14:24:13 +02:00
|
|
|
}
|
2012-08-10 18:48:21 +02:00
|
|
|
at_file_begin = false;
|
2012-08-10 14:24:13 +02:00
|
|
|
}
|
|
|
|
|
2012-08-10 18:48:21 +02:00
|
|
|
OptionManager& option_manager = buffer->option_manager();
|
|
|
|
option_manager.set_option("eolformat", Option(crlf ? "crlf" : "lf"));
|
|
|
|
option_manager.set_option("BOM", Option(bom ? "utf-8" : "no"));
|
2012-08-10 14:24:13 +02:00
|
|
|
|
2012-08-14 14:13:10 +02:00
|
|
|
// if the file ended with a \n, remove the \n added by the buffer
|
|
|
|
if (*(buffer->end() - 2) == '\n')
|
|
|
|
buffer->erase(buffer->end() - 1, buffer->end());
|
|
|
|
|
2012-08-10 14:24:13 +02:00
|
|
|
// it never happened, buffer always was like that
|
|
|
|
buffer->reset_undo_data();
|
|
|
|
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void write(int fd, const memoryview<char>& data, const String& filename)
|
|
|
|
{
|
|
|
|
const char* ptr = data.pointer();
|
|
|
|
ssize_t count = data.size();
|
|
|
|
|
|
|
|
while (count)
|
|
|
|
{
|
|
|
|
ssize_t written = ::write(fd, ptr, count);
|
|
|
|
ptr += written;
|
|
|
|
count -= written;
|
|
|
|
|
|
|
|
if (written == -1)
|
|
|
|
throw file_access_error(filename, strerror(errno));
|
|
|
|
}
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
|
|
|
|
2012-04-14 03:17:09 +02:00
|
|
|
void write_buffer_to_file(const Buffer& buffer, const String& filename)
|
2011-09-02 18:51:20 +02:00
|
|
|
{
|
2012-08-10 14:24:13 +02:00
|
|
|
String eolformat = buffer.option_manager()["eolformat"].as_string();
|
|
|
|
if (eolformat == "crlf")
|
|
|
|
eolformat = "\r\n";
|
|
|
|
else
|
|
|
|
eolformat = "\n";
|
|
|
|
auto eoldata = eolformat.data();
|
|
|
|
|
2012-09-12 19:42:12 +02:00
|
|
|
int fd = open(parse_filename(filename).c_str(),
|
|
|
|
O_CREAT | O_WRONLY | O_TRUNC, 0644);
|
2011-09-02 18:51:20 +02:00
|
|
|
if (fd == -1)
|
2011-09-09 20:40:59 +02:00
|
|
|
throw file_access_error(filename, strerror(errno));
|
2012-11-20 18:52:36 +01:00
|
|
|
auto close_fd = on_scope_end([fd]{ close(fd); });
|
2011-09-02 18:51:20 +02:00
|
|
|
|
2012-08-10 18:48:21 +02:00
|
|
|
if (buffer.option_manager()["BOM"].as_string() == "utf-8")
|
|
|
|
::write(fd, "\xEF\xBB\xBF", 3);
|
|
|
|
|
2012-08-22 23:33:52 +02:00
|
|
|
for (LineCount i = 0; i < buffer.line_count(); ++i)
|
2011-09-02 18:51:20 +02:00
|
|
|
{
|
2012-08-10 14:24:13 +02:00
|
|
|
// end of lines are written according to eolformat but always
|
|
|
|
// stored as \n
|
|
|
|
memoryview<char> linedata = buffer.line_content(i).data();
|
|
|
|
write(fd, linedata.subrange(0, linedata.size()-1), filename);
|
|
|
|
write(fd, eoldata, filename);
|
2011-09-02 18:51:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|