|
|
|
|
#include <fstream>
|
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <sstream>
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
|
|
#include "filesystem.h"
|
|
|
|
|
|
|
|
|
|
const size_t buffer_size=131072;
|
|
|
|
|
|
|
|
|
|
//Only use on small files
|
|
|
|
|
std::string filesystem::read(const std::string &path) {
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
std::ifstream input(path, std::ofstream::binary);
|
|
|
|
|
if(input) {
|
|
|
|
|
ss << input.rdbuf();
|
|
|
|
|
input.close();
|
|
|
|
|
}
|
|
|
|
|
return ss.str();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int filesystem::read(const std::string &path, Glib::RefPtr<Gtk::TextBuffer> text_buffer) {
|
|
|
|
|
std::ifstream input(path, std::ofstream::binary);
|
|
|
|
|
|
|
|
|
|
if(input) {
|
|
|
|
|
//need to read the whole file to make this work...
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
ss << input.rdbuf();
|
|
|
|
|
Glib::ustring ustr=ss.str();
|
|
|
|
|
|
|
|
|
|
bool valid=true;
|
|
|
|
|
|
|
|
|
|
if(ustr.validate())
|
|
|
|
|
text_buffer->insert_at_cursor(ustr);
|
|
|
|
|
else
|
|
|
|
|
valid=false;
|
|
|
|
|
|
|
|
|
|
//TODO: maybe correct this, emphasis on maybe:
|
|
|
|
|
/*std::vector<char> buffer(buffer_size);
|
|
|
|
|
size_t read_length;
|
|
|
|
|
while((read_length=input.read(&buffer[0], buffer_size).gcount())>0) {
|
|
|
|
|
//auto ustr=Glib::ustring(std::string(&buffer[0], read_length)); //this works...
|
|
|
|
|
|
|
|
|
|
//this does not work:
|
|
|
|
|
Glib::ustring ustr;
|
|
|
|
|
ustr.resize(read_length);
|
|
|
|
|
ustr.replace(0, read_length, &buffer[0]);
|
|
|
|
|
|
|
|
|
|
Glib::ustring::iterator iter;
|
|
|
|
|
while(!ustr.validate(iter)) {
|
|
|
|
|
auto next_char_iter=iter;
|
|
|
|
|
next_char_iter++;
|
|
|
|
|
ustr.replace(iter, next_char_iter, "?");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
text_buffer->insert_at_cursor(ustr); //What if insert happens in the middle of an UTF-8 char???
|
|
|
|
|
}*/
|
|
|
|
|
input.close();
|
|
|
|
|
if(valid)
|
|
|
|
|
return 1;
|
|
|
|
|
else
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int filesystem::read_non_utf8(const std::string &path, Glib::RefPtr<Gtk::TextBuffer> text_buffer) {
|
|
|
|
|
std::ifstream input(path, std::ofstream::binary);
|
|
|
|
|
|
|
|
|
|
if(input) {
|
|
|
|
|
//need to read the whole file to make this work...
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
ss << input.rdbuf();
|
|
|
|
|
Glib::ustring ustr=ss.str();
|
|
|
|
|
|
|
|
|
|
bool valid=true;
|
|
|
|
|
Glib::ustring::iterator iter;
|
|
|
|
|
while(!ustr.validate(iter)) {
|
|
|
|
|
auto next_char_iter=iter;
|
|
|
|
|
next_char_iter++;
|
|
|
|
|
ustr.replace(iter, next_char_iter, "?");
|
|
|
|
|
valid=false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
text_buffer->insert_at_cursor(ustr);
|
|
|
|
|
|
|
|
|
|
input.close();
|
|
|
|
|
if(valid)
|
|
|
|
|
return 1;
|
|
|
|
|
else
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//Only use on small files
|
|
|
|
|
std::vector<std::string> filesystem::read_lines(const std::string &path) {
|
|
|
|
|
std::vector<std::string> res;
|
|
|
|
|
std::ifstream input(path, std::ofstream::binary);
|
|
|
|
|
if (input) {
|
|
|
|
|
do { res.emplace_back(); } while(getline(input, res.back()));
|
|
|
|
|
}
|
|
|
|
|
input.close();
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//Only use on small files
|
|
|
|
|
bool filesystem::write(const std::string &path, const std::string &new_content) {
|
|
|
|
|
std::ofstream output(path, std::ofstream::binary);
|
|
|
|
|
if(output)
|
|
|
|
|
output << new_content;
|
|
|
|
|
else
|
|
|
|
|
return false;
|
|
|
|
|
output.close();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool filesystem::write(const std::string &path, Glib::RefPtr<Gtk::TextBuffer> buffer) {
|
|
|
|
|
std::ofstream output(path, std::ofstream::binary);
|
|
|
|
|
if(output) {
|
|
|
|
|
auto start_iter=buffer->begin();
|
|
|
|
|
auto end_iter=start_iter;
|
|
|
|
|
bool end_reached=false;
|
|
|
|
|
while(!end_reached) {
|
|
|
|
|
for(size_t c=0;c<buffer_size;c++) {
|
|
|
|
|
if(!end_iter.forward_char()) {
|
|
|
|
|
end_reached=true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
output << buffer->get_text(start_iter, end_iter).c_str();
|
|
|
|
|
start_iter=end_iter;
|
|
|
|
|
}
|
|
|
|
|
output.close();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string filesystem::escape_argument(const std::string &argument) {
|
|
|
|
|
auto escaped=argument;
|
|
|
|
|
for(size_t pos=0;pos<escaped.size();++pos) {
|
|
|
|
|
if(escaped[pos]==' ' || escaped[pos]=='(' || escaped[pos]==')' || escaped[pos]=='\'' || escaped[pos]=='"') {
|
|
|
|
|
escaped.insert(pos, "\\");
|
|
|
|
|
++pos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return escaped;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string filesystem::unescape_argument(const std::string &argument) {
|
|
|
|
|
auto unescaped=argument;
|
|
|
|
|
|
|
|
|
|
if(unescaped.size()>=2) {
|
|
|
|
|
if((unescaped[0]=='\'' && unescaped[unescaped.size()-1]=='\'') ||
|
|
|
|
|
(unescaped[0]=='"' && unescaped[unescaped.size()-1]=='"')) {
|
|
|
|
|
char quotation_mark=unescaped[0];
|
|
|
|
|
unescaped=unescaped.substr(1, unescaped.size()-2);
|
|
|
|
|
for(size_t pos=1;pos<unescaped.size();++pos) {
|
|
|
|
|
if(unescaped[pos-1]=='\\' && unescaped[pos]==quotation_mark) {
|
|
|
|
|
unescaped.erase(pos-1, 1);
|
|
|
|
|
--pos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return unescaped;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for(size_t pos=1;pos<unescaped.size();++pos) {
|
|
|
|
|
if(unescaped[pos-1]=='\\' && (unescaped[pos]==' ' || unescaped[pos]=='(' || unescaped[pos]==')' || unescaped[pos]=='\'' || unescaped[pos]=='"')) {
|
|
|
|
|
unescaped.erase(pos-1, 1);
|
|
|
|
|
--pos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return unescaped;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool filesystem::file_in_path(const boost::filesystem::path &file_path, const boost::filesystem::path &path) {
|
|
|
|
|
if(std::distance(file_path.begin(), file_path.end())<std::distance(path.begin(), path.end()))
|
|
|
|
|
return false;
|
|
|
|
|
return std::equal(path.begin(), path.end(), file_path.begin());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boost::filesystem::path filesystem::find_file_in_path_parents(const std::string &file_name, const boost::filesystem::path &path) {
|
|
|
|
|
auto current_path=path;
|
|
|
|
|
while(true) {
|
|
|
|
|
auto test_path=current_path/file_name;
|
|
|
|
|
if(boost::filesystem::exists(test_path))
|
|
|
|
|
return test_path;
|
|
|
|
|
if(current_path==current_path.root_directory())
|
|
|
|
|
return boost::filesystem::path();
|
|
|
|
|
current_path=current_path.parent_path();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boost::filesystem::path filesystem::get_canonical_path(const boost::filesystem::path &path) noexcept {
|
|
|
|
|
if(path.is_absolute()) {
|
|
|
|
|
bool is_canonical=true;
|
|
|
|
|
for(auto &str: path) {
|
|
|
|
|
if(str==".." || str==".") {
|
|
|
|
|
is_canonical=false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if(is_canonical)
|
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
auto canonical_path=boost::filesystem::canonical(path, ec);
|
|
|
|
|
if(ec)
|
|
|
|
|
return path;
|
|
|
|
|
return canonical_path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boost::filesystem::path filesystem::get_normal_path(const boost::filesystem::path &path) noexcept {
|
|
|
|
|
boost::filesystem::path normal_path;
|
|
|
|
|
|
|
|
|
|
for(auto &e: path) {
|
|
|
|
|
if(e==".")
|
|
|
|
|
continue;
|
|
|
|
|
else if(e=="..") {
|
|
|
|
|
auto parent_path=normal_path.parent_path();
|
|
|
|
|
if(!parent_path.empty())
|
|
|
|
|
normal_path=parent_path;
|
|
|
|
|
else
|
|
|
|
|
normal_path/=e;
|
|
|
|
|
}
|
|
|
|
|
else if(e.empty())
|
|
|
|
|
continue;
|
|
|
|
|
else
|
|
|
|
|
normal_path/=e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return normal_path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
boost::filesystem::path filesystem::get_relative_path(const boost::filesystem::path &path, const boost::filesystem::path &base) noexcept {
|
|
|
|
|
boost::filesystem::path relative_path;
|
|
|
|
|
|
|
|
|
|
if(std::distance(path.begin(), path.end())<std::distance(base.begin(), base.end()))
|
|
|
|
|
return boost::filesystem::path();
|
|
|
|
|
|
|
|
|
|
auto base_it=base.begin();
|
|
|
|
|
auto path_it=path.begin();
|
|
|
|
|
while(path_it!=path.end() && base_it!=base.end()) {
|
|
|
|
|
if(*path_it!=*base_it)
|
|
|
|
|
return boost::filesystem::path();
|
|
|
|
|
++path_it;
|
|
|
|
|
++base_it;
|
|
|
|
|
}
|
|
|
|
|
for(;path_it!=path.end();++path_it)
|
|
|
|
|
relative_path/=*path_it;
|
|
|
|
|
|
|
|
|
|
return relative_path;
|
|
|
|
|
}
|