|
|
@@ -2,6 +2,7 @@
|
|
|
|
|
|
#include "ceval.h"
|
|
|
#include "cffi.h"
|
|
|
+#include "common.h"
|
|
|
|
|
|
#if PK_ENABLE_FILEIO
|
|
|
|
|
|
@@ -11,7 +12,7 @@
|
|
|
namespace pkpy{
|
|
|
|
|
|
inline Str _read_file_cwd(const Str& name, bool* ok){
|
|
|
- std::filesystem::path path(name.c_str());
|
|
|
+ std::filesystem::path path(name.sv());
|
|
|
bool exists = std::filesystem::exists(path);
|
|
|
if(!exists){
|
|
|
*ok = false;
|
|
|
@@ -31,13 +32,17 @@ struct FileIO {
|
|
|
Str mode;
|
|
|
std::fstream _fs;
|
|
|
|
|
|
+ bool is_text() const { return mode != "rb" && mode != "wb" && mode != "ab"; }
|
|
|
+
|
|
|
FileIO(VM* vm, Str file, Str mode): file(file), mode(mode) {
|
|
|
- if(mode == "rt" || mode == "r"){
|
|
|
- _fs.open(file, std::ios::in);
|
|
|
- }else if(mode == "wt" || mode == "w"){
|
|
|
- _fs.open(file, std::ios::out);
|
|
|
- }else if(mode == "at" || mode == "a"){
|
|
|
- _fs.open(file, std::ios::app);
|
|
|
+ if(mode == "rt" || mode == "r" || mode == "rb"){
|
|
|
+ _fs.open(file.sv(), std::ios::in);
|
|
|
+ }else if(mode == "wt" || mode == "w" || mode == "wb"){
|
|
|
+ _fs.open(file.sv(), std::ios::out);
|
|
|
+ }else if(mode == "at" || mode == "a" || mode == "ab"){
|
|
|
+ _fs.open(file.sv(), std::ios::app);
|
|
|
+ }else{
|
|
|
+ vm->ValueError("invalid mode");
|
|
|
}
|
|
|
if(!_fs.is_open()) vm->IOError(strerror(errno));
|
|
|
}
|
|
|
@@ -51,14 +56,19 @@ struct FileIO {
|
|
|
|
|
|
vm->bind_method<0>(type, "read", [](VM* vm, ArgsView args){
|
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
|
- std::string buffer;
|
|
|
- io._fs >> buffer;
|
|
|
+ Bytes buffer;
|
|
|
+ io._fs >> buffer._data;
|
|
|
+ if(io.is_text()) return VAR(Str(buffer._data));
|
|
|
return VAR(buffer);
|
|
|
});
|
|
|
|
|
|
vm->bind_method<1>(type, "write", [](VM* vm, ArgsView args){
|
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
|
- io._fs << CAST(Str&, args[1]);
|
|
|
+ if(io.is_text()) io._fs << CAST(Str&, args[1]);
|
|
|
+ else{
|
|
|
+ Bytes& buffer = CAST(Bytes&, args[1]);
|
|
|
+ io._fs << buffer._data;
|
|
|
+ }
|
|
|
return vm->None;
|
|
|
});
|
|
|
|
|
|
@@ -80,35 +90,40 @@ struct FileIO {
|
|
|
|
|
|
inline void add_module_io(VM* vm){
|
|
|
PyObject* mod = vm->new_module("io");
|
|
|
- PyObject* type = FileIO::register_class(vm, mod);
|
|
|
- vm->bind_builtin_func<2>("open", [type](VM* vm, ArgsView args){
|
|
|
- return vm->call(type, args);
|
|
|
+ FileIO::register_class(vm, mod);
|
|
|
+ vm->bind_builtin_func<2>("open", [](VM* vm, ArgsView args){
|
|
|
+ static StrName m_io("io");
|
|
|
+ static StrName m_FileIO("FileIO");
|
|
|
+ return vm->call(vm->_modules[m_io]->attr(m_FileIO), args[0], args[1]);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
inline void add_module_os(VM* vm){
|
|
|
PyObject* mod = vm->new_module("os");
|
|
|
+ PyObject* path_obj = vm->heap.gcnew<DummyInstance>(vm->tp_object, {});
|
|
|
+ mod->attr().set("path", path_obj);
|
|
|
+
|
|
|
// Working directory is shared by all VMs!!
|
|
|
vm->bind_func<0>(mod, "getcwd", [](VM* vm, ArgsView args){
|
|
|
return VAR(std::filesystem::current_path().string());
|
|
|
});
|
|
|
|
|
|
vm->bind_func<1>(mod, "chdir", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
std::filesystem::current_path(path);
|
|
|
return vm->None;
|
|
|
});
|
|
|
|
|
|
vm->bind_func<1>(mod, "listdir", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
std::filesystem::directory_iterator di;
|
|
|
try{
|
|
|
di = std::filesystem::directory_iterator(path);
|
|
|
}catch(std::filesystem::filesystem_error& e){
|
|
|
- Str msg = e.what();
|
|
|
+ std::string msg = e.what();
|
|
|
auto pos = msg.find_last_of(":");
|
|
|
- if(pos != Str::npos) msg = msg.substr(pos + 1);
|
|
|
- vm->IOError(msg.lstrip());
|
|
|
+ if(pos != std::string::npos) msg = msg.substr(pos + 1);
|
|
|
+ vm->IOError(Str(msg).lstrip());
|
|
|
}
|
|
|
List ret;
|
|
|
for(auto& p: di) ret.push_back(VAR(p.path().filename().string()));
|
|
|
@@ -116,36 +131,36 @@ inline void add_module_os(VM* vm){
|
|
|
});
|
|
|
|
|
|
vm->bind_func<1>(mod, "remove", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
bool ok = std::filesystem::remove(path);
|
|
|
if(!ok) vm->IOError("operation failed");
|
|
|
return vm->None;
|
|
|
});
|
|
|
|
|
|
vm->bind_func<1>(mod, "mkdir", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
bool ok = std::filesystem::create_directory(path);
|
|
|
if(!ok) vm->IOError("operation failed");
|
|
|
return vm->None;
|
|
|
});
|
|
|
|
|
|
vm->bind_func<1>(mod, "rmdir", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
bool ok = std::filesystem::remove(path);
|
|
|
if(!ok) vm->IOError("operation failed");
|
|
|
return vm->None;
|
|
|
});
|
|
|
|
|
|
- vm->bind_func<-1>(mod, "path_join", [](VM* vm, ArgsView args){
|
|
|
+ vm->bind_func<-1>(path_obj, "join", [](VM* vm, ArgsView args){
|
|
|
std::filesystem::path path;
|
|
|
for(int i=0; i<args.size(); i++){
|
|
|
- path /= CAST(Str&, args[i]).c_str();
|
|
|
+ path /= CAST(Str&, args[i]).sv();
|
|
|
}
|
|
|
return VAR(path.string());
|
|
|
});
|
|
|
|
|
|
- vm->bind_func<1>(mod, "path_exists", [](VM* vm, ArgsView args){
|
|
|
- std::filesystem::path path(CAST(Str&, args[0]).c_str());
|
|
|
+ vm->bind_func<1>(path_obj, "exists", [](VM* vm, ArgsView args){
|
|
|
+ std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
|
bool exists = std::filesystem::exists(path);
|
|
|
return VAR(exists);
|
|
|
});
|