mirror of
https://github.com/pocketpy/pocketpy
synced 2025-10-23 21:10:19 +00:00
189 lines
5.8 KiB
C++
189 lines
5.8 KiB
C++
#pragma once
|
|
|
|
#include "ceval.h"
|
|
#include "cffi.h"
|
|
#include "common.h"
|
|
|
|
#if PK_ENABLE_OS
|
|
|
|
#include <filesystem>
|
|
#include <cstdio>
|
|
|
|
namespace pkpy{
|
|
|
|
inline Bytes _default_import_handler(const Str& name){
|
|
std::filesystem::path path(name.sv());
|
|
bool exists = std::filesystem::exists(path);
|
|
if(!exists) return Bytes();
|
|
std::string cname = name.str();
|
|
FILE* fp = fopen(cname.c_str(), "rb");
|
|
if(!fp) return Bytes();
|
|
fseek(fp, 0, SEEK_END);
|
|
std::vector<char> buffer(ftell(fp));
|
|
fseek(fp, 0, SEEK_SET);
|
|
fread(buffer.data(), 1, buffer.size(), fp);
|
|
fclose(fp);
|
|
return Bytes(std::move(buffer));
|
|
};
|
|
|
|
struct FileIO {
|
|
PY_CLASS(FileIO, io, FileIO)
|
|
|
|
Str file;
|
|
Str mode;
|
|
FILE* fp;
|
|
|
|
bool is_text() const { return mode != "rb" && mode != "wb" && mode != "ab"; }
|
|
|
|
FileIO(VM* vm, std::string file, std::string mode): file(file), mode(mode) {
|
|
fp = fopen(file.c_str(), mode.c_str());
|
|
if(!fp) vm->IOError(strerror(errno));
|
|
}
|
|
|
|
void close(){
|
|
if(fp == nullptr) return;
|
|
fclose(fp);
|
|
fp = nullptr;
|
|
}
|
|
|
|
static void _register(VM* vm, PyObject* mod, PyObject* type){
|
|
vm->bind_constructor<3>(type, [](VM* vm, ArgsView args){
|
|
return VAR_T(FileIO,
|
|
vm, CAST(Str&, args[1]).str(), CAST(Str&, args[2]).str()
|
|
);
|
|
});
|
|
|
|
vm->bind_method<0>(type, "read", [](VM* vm, ArgsView args){
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
fseek(io.fp, 0, SEEK_END);
|
|
std::vector<char> buffer(ftell(io.fp));
|
|
fseek(io.fp, 0, SEEK_SET);
|
|
fread(buffer.data(), 1, buffer.size(), io.fp);
|
|
Bytes b(std::move(buffer));
|
|
if(io.is_text()) return VAR(Str(b.str()));
|
|
return VAR(std::move(b));
|
|
});
|
|
|
|
vm->bind_method<1>(type, "write", [](VM* vm, ArgsView args){
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
if(io.is_text()){
|
|
Str& s = CAST(Str&, args[1]);
|
|
fwrite(s.data, 1, s.length(), io.fp);
|
|
}else{
|
|
Bytes& buffer = CAST(Bytes&, args[1]);
|
|
fwrite(buffer.data(), 1, buffer.size(), io.fp);
|
|
}
|
|
return vm->None;
|
|
});
|
|
|
|
vm->bind_method<0>(type, "close", [](VM* vm, ArgsView args){
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
io.close();
|
|
return vm->None;
|
|
});
|
|
|
|
vm->bind_method<0>(type, "__exit__", [](VM* vm, ArgsView args){
|
|
FileIO& io = CAST(FileIO&, args[0]);
|
|
io.close();
|
|
return vm->None;
|
|
});
|
|
|
|
vm->bind_method<0>(type, "__enter__", PK_LAMBDA(vm->None));
|
|
}
|
|
};
|
|
|
|
inline void add_module_io(VM* vm){
|
|
PyObject* mod = vm->new_module("io");
|
|
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]).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]).sv());
|
|
std::filesystem::directory_iterator di;
|
|
try{
|
|
di = std::filesystem::directory_iterator(path);
|
|
}catch(std::filesystem::filesystem_error& e){
|
|
std::string msg = e.what();
|
|
auto pos = msg.find_last_of(":");
|
|
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()));
|
|
return VAR(ret);
|
|
});
|
|
|
|
vm->bind_func<1>(mod, "remove", [](VM* vm, ArgsView args){
|
|
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]).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]).sv());
|
|
bool ok = std::filesystem::remove(path);
|
|
if(!ok) vm->IOError("operation failed");
|
|
return vm->None;
|
|
});
|
|
|
|
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]).sv();
|
|
}
|
|
return VAR(path.string());
|
|
});
|
|
|
|
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);
|
|
});
|
|
|
|
vm->bind_func<1>(path_obj, "basename", [](VM* vm, ArgsView args){
|
|
std::filesystem::path path(CAST(Str&, args[0]).sv());
|
|
return VAR(path.filename().string());
|
|
});
|
|
}
|
|
|
|
} // namespace pkpy
|
|
|
|
|
|
#else
|
|
|
|
namespace pkpy{
|
|
inline void add_module_io(void* vm){}
|
|
inline void add_module_os(void* vm){}
|
|
inline Bytes _default_import_handler(const Str& name) { return Bytes(); }
|
|
} // namespace pkpy
|
|
|
|
#endif |