580 lines
13 KiB
C++
580 lines
13 KiB
C++
#include <stdafx.h>
|
|
|
|
#include "ScriptAPI.h"
|
|
#include <sys/stat.h>
|
|
|
|
#pragma warning(disable : 4702) // disable unreachable code warning
|
|
|
|
enum fsop
|
|
{
|
|
FS_READ,
|
|
FS_WRITE
|
|
};
|
|
static duk_ret_t ReadWriteImpl(duk_context * ctx, fsop op); // (fd, buffer, offset, length, position)
|
|
static duk_ret_t js__FileFinalizer(duk_context * ctx);
|
|
|
|
void ScriptAPI::Define_fs(duk_context * ctx)
|
|
{
|
|
// todo DukPutProps DukClass
|
|
|
|
const duk_function_list_entry funcs[] = {
|
|
{"open", js_fs_open, DUK_VARARGS},
|
|
{"close", js_fs_close, DUK_VARARGS},
|
|
{"write", js_fs_write, DUK_VARARGS},
|
|
{"writefile", js_fs_writefile, DUK_VARARGS},
|
|
{"read", js_fs_read, DUK_VARARGS},
|
|
{"readfile", js_fs_readfile, DUK_VARARGS},
|
|
{"exists", js_fs_exists, DUK_VARARGS},
|
|
{"fstat", js_fs_fstat, DUK_VARARGS},
|
|
{"stat", js_fs_stat, DUK_VARARGS},
|
|
{"unlink", js_fs_unlink, DUK_VARARGS},
|
|
{"mkdir", js_fs_mkdir, DUK_VARARGS},
|
|
{"rmdir", js_fs_rmdir, DUK_VARARGS},
|
|
{"readdir", js_fs_readdir, DUK_VARARGS},
|
|
{"Stats", js_fs_Stats__constructor, DUK_VARARGS},
|
|
{nullptr, nullptr, 0},
|
|
};
|
|
|
|
const duk_function_list_entry Stats_funcs[] = {
|
|
{"isFile", js_fs_Stats_isFile, DUK_VARARGS},
|
|
{"isDirectory", js_fs_Stats_isDirectory, DUK_VARARGS},
|
|
{nullptr, nullptr, 0},
|
|
};
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_push_string(ctx, "fs");
|
|
duk_push_object(ctx);
|
|
duk_put_function_list(ctx, -1, funcs);
|
|
|
|
duk_get_prop_string(ctx, -1, "Stats");
|
|
duk_push_object(ctx);
|
|
duk_put_function_list(ctx, -1, Stats_funcs);
|
|
duk_put_prop_string(ctx, -2, "prototype");
|
|
duk_pop(ctx);
|
|
|
|
duk_freeze(ctx, -1);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_open(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String, Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
const char * mode = duk_get_string(ctx, 1);
|
|
|
|
bool bModeValid = false;
|
|
|
|
// clang-format off
|
|
const char * validModes[] = {
|
|
"r", "rb",
|
|
"w", "wb",
|
|
"a", "ab",
|
|
"r+", "rb+", "r+b",
|
|
"w+", "wb+", "w+b",
|
|
"a+", "ab+", "a+b",
|
|
nullptr,
|
|
};
|
|
// clang-format on
|
|
|
|
for (int i = 0; validModes[i] != nullptr; i++)
|
|
{
|
|
if (strcmp(mode, validModes[i]) == 0)
|
|
{
|
|
bModeValid = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!bModeValid)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "mode '%s' is not valid", mode);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
FILE * fp = fopen(path, mode);
|
|
|
|
if (fp == nullptr)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR, "could not open '%s' (mode: '%s')", path, mode);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
int fd = _fileno(fp);
|
|
|
|
// FILES[fd] = {fp: fp}
|
|
duk_get_global_string(ctx, HS_gOpenFileDescriptors);
|
|
|
|
duk_push_object(ctx);
|
|
duk_push_pointer(ctx, fp);
|
|
duk_put_prop_string(ctx, -2, "fp");
|
|
duk_push_c_function(ctx, js__FileFinalizer, 1);
|
|
duk_set_finalizer(ctx, -2);
|
|
|
|
duk_put_prop_index(ctx, -2, fd);
|
|
duk_pop_n(ctx, 2);
|
|
|
|
duk_push_number(ctx, fd);
|
|
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_close(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_Number});
|
|
|
|
int fd = duk_get_int(ctx, 0);
|
|
int rc = -1;
|
|
|
|
duk_get_global_string(ctx, HS_gOpenFileDescriptors);
|
|
|
|
if (duk_has_prop_index(ctx, -1, fd))
|
|
{
|
|
duk_get_prop_index(ctx, -1, fd);
|
|
duk_get_prop_string(ctx, -1, "fp");
|
|
|
|
FILE * fp = (FILE *)duk_get_pointer(ctx, -1);
|
|
rc = fclose(fp);
|
|
|
|
// unset finalizer before deleting
|
|
duk_push_undefined(ctx);
|
|
duk_set_finalizer(ctx, -3);
|
|
duk_del_prop_index(ctx, -3, fd);
|
|
|
|
duk_pop_n(ctx, 2);
|
|
}
|
|
else
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR, "invalid file descriptor");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_pop(ctx);
|
|
duk_push_number(ctx, rc);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_write(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_Number, Arg_Any, Arg_OptNumber, Arg_OptNumber, Arg_OptNumber});
|
|
return ReadWriteImpl(ctx, FS_WRITE);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_writefile(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String, Arg_Any});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
|
|
void * buffer;
|
|
duk_size_t bufferSize;
|
|
|
|
if (duk_is_string(ctx, 1))
|
|
{
|
|
buffer = (void *)duk_get_lstring(ctx, 1, &bufferSize);
|
|
}
|
|
else if (duk_is_buffer_data(ctx, 1))
|
|
{
|
|
buffer = duk_get_buffer_data(ctx, 1, &bufferSize);
|
|
}
|
|
else
|
|
{
|
|
return ThrowInvalidArgsError(ctx);
|
|
}
|
|
|
|
FILE * fp = fopen(path, "wb");
|
|
|
|
if (fp == nullptr)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR, "could not open '%s' (mode: 'wb')", path);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
if (fwrite(buffer, 1, bufferSize, fp) != bufferSize)
|
|
{
|
|
fclose(fp);
|
|
return DUK_RET_ERROR;
|
|
}
|
|
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_read(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_Number, Arg_Any, Arg_Number, Arg_Number, Arg_Number});
|
|
return ReadWriteImpl(ctx, FS_READ);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_readfile(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
|
|
FILE * fp = fopen(path, "rb");
|
|
|
|
if (fp == nullptr)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR, "could not open '%s' (mode: 'rb')", path);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
struct stat stats;
|
|
if (fstat(_fileno(fp), &stats) != 0)
|
|
{
|
|
fclose(fp);
|
|
return DUK_RET_ERROR;
|
|
}
|
|
|
|
void * data = duk_push_fixed_buffer(ctx, stats.st_size);
|
|
duk_push_buffer_object(ctx, -1, 0, stats.st_size, DUK_BUFOBJ_NODEJS_BUFFER);
|
|
|
|
if (fread(data, 1, stats.st_size, fp) != (size_t)stats.st_size)
|
|
{
|
|
duk_pop(ctx);
|
|
fclose(fp);
|
|
return DUK_RET_ERROR;
|
|
}
|
|
|
|
fclose(fp);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_exists(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
const char * path = duk_get_string(ctx, 0);
|
|
duk_push_boolean(ctx, PathFileExistsA(path) ? 1 : 0);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_fstat(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_Number});
|
|
|
|
int fd = duk_get_int(ctx, 0);
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_get_prop_string(ctx, -1, "fs");
|
|
duk_get_prop_string(ctx, -1, "Stats");
|
|
duk_push_number(ctx, fd);
|
|
duk_new(ctx, 1);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_stat(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_get_prop_string(ctx, -1, "fs");
|
|
duk_get_prop_string(ctx, -1, "Stats");
|
|
duk_push_string(ctx, path);
|
|
duk_new(ctx, 1);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_unlink(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
duk_push_boolean(ctx, DeleteFileA(path) != 0);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_mkdir(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
duk_push_boolean(ctx, CreateDirectoryA(path, nullptr) != 0);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_rmdir(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
duk_push_boolean(ctx, RemoveDirectoryA(path) != 0);
|
|
return 1;
|
|
}
|
|
|
|
// todo make sure failure behavior is similar to nodejs's fs.readdirSync
|
|
duk_ret_t ScriptAPI::js_fs_readdir(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {Arg_String});
|
|
|
|
const char * path = duk_get_string(ctx, 0);
|
|
|
|
char findFileName[MAX_PATH];
|
|
snprintf(findFileName, sizeof(findFileName), "%s%s", path, "\\*");
|
|
|
|
WIN32_FIND_DATAA ffd;
|
|
HANDLE hFind = FindFirstFileA(findFileName, &ffd);
|
|
|
|
if (hFind == INVALID_HANDLE_VALUE)
|
|
{
|
|
return DUK_RET_ERROR;
|
|
}
|
|
|
|
duk_uarridx_t idx = 0;
|
|
duk_push_array(ctx);
|
|
|
|
do
|
|
{
|
|
if (strcmp(ffd.cFileName, ".") == 0 || strcmp(ffd.cFileName, "..") == 0)
|
|
{
|
|
continue;
|
|
}
|
|
duk_push_string(ctx, ffd.cFileName);
|
|
duk_put_prop_index(ctx, -2, idx++);
|
|
} while (FindNextFileA(hFind, &ffd));
|
|
|
|
FindClose(hFind);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_Stats__constructor(duk_context * ctx)
|
|
{
|
|
if (!duk_is_constructor_call(ctx))
|
|
{
|
|
return DUK_RET_TYPE_ERROR;
|
|
}
|
|
|
|
if (duk_get_top(ctx) != 1)
|
|
{
|
|
return ThrowInvalidArgsError(ctx);
|
|
}
|
|
|
|
struct stat stats;
|
|
|
|
if (duk_is_number(ctx, 0))
|
|
{
|
|
int fd = duk_get_int(ctx, 0);
|
|
if (fstat(fd, &stats) != 0)
|
|
{
|
|
return DUK_RET_ERROR;
|
|
}
|
|
}
|
|
else if (duk_is_string(ctx, 0))
|
|
{
|
|
const char * path = duk_get_string(ctx, 0);
|
|
if (stat(path, &stats) != 0)
|
|
{
|
|
return DUK_RET_ERROR;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return ThrowInvalidArgsError(ctx);
|
|
}
|
|
|
|
const duk_number_list_entry numbers[] = {
|
|
{"dev", (double)stats.st_dev},
|
|
{"ino", (double)stats.st_ino},
|
|
{"mode", (double)stats.st_mode},
|
|
{"nlink", (double)stats.st_nlink},
|
|
{"uid", (double)stats.st_uid},
|
|
{"gid", (double)stats.st_gid},
|
|
{"rdev", (double)stats.st_rdev},
|
|
{"size", (double)stats.st_size},
|
|
{"atimeMs", (double)stats.st_atime * 1000},
|
|
{"mtimeMs", (double)stats.st_mtime * 1000},
|
|
{"ctimeMs", (double)stats.st_ctime * 1000},
|
|
{nullptr, 0},
|
|
};
|
|
|
|
struct
|
|
{
|
|
const char * key;
|
|
time_t time;
|
|
} dates[3] = {
|
|
{"atime", stats.st_atime * 1000},
|
|
{"mtime", stats.st_mtime * 1000},
|
|
{"ctime", stats.st_ctime * 1000},
|
|
};
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_get_prop_string(ctx, -1, "Date");
|
|
duk_remove(ctx, -2);
|
|
|
|
duk_push_this(ctx);
|
|
duk_put_number_list(ctx, -1, numbers);
|
|
|
|
for (int i = 0; i < 3; i++)
|
|
{
|
|
duk_dup(ctx, -2);
|
|
duk_push_number(ctx, (duk_double_t)dates[i].time);
|
|
duk_new(ctx, 1);
|
|
duk_put_prop_string(ctx, -2, dates[i].key);
|
|
}
|
|
|
|
duk_remove(ctx, -2);
|
|
duk_freeze(ctx, -1);
|
|
return 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_Stats_isDirectory(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {});
|
|
|
|
duk_push_this(ctx);
|
|
duk_get_prop_string(ctx, -1, "mode");
|
|
duk_uint_t mode = duk_get_uint(ctx, -1);
|
|
duk_push_boolean(ctx, (mode & S_IFDIR) != 0);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_fs_Stats_isFile(duk_context * ctx)
|
|
{
|
|
CheckArgs(ctx, {});
|
|
|
|
duk_push_this(ctx);
|
|
duk_get_prop_string(ctx, -1, "mode");
|
|
duk_uint_t mode = duk_get_uint(ctx, -1);
|
|
duk_push_boolean(ctx, (mode & S_IFREG) != 0);
|
|
return 1;
|
|
}
|
|
|
|
static duk_ret_t ReadWriteImpl(duk_context * ctx, fsop op)
|
|
{
|
|
int fd;
|
|
const char * buffer;
|
|
size_t offset = 0;
|
|
size_t length = 0;
|
|
size_t position = 0;
|
|
|
|
bool bHavePos = false;
|
|
duk_size_t bufferSize;
|
|
FILE * fp;
|
|
size_t rc;
|
|
|
|
duk_idx_t nargs = duk_get_top(ctx);
|
|
|
|
if (nargs < 2 || !duk_is_number(ctx, 0))
|
|
{
|
|
goto err_invalid_args;
|
|
}
|
|
|
|
fd = duk_get_int(ctx, 0);
|
|
|
|
if (duk_is_buffer_data(ctx, 1))
|
|
{
|
|
buffer = (const char *)duk_get_buffer_data(ctx, 1, &bufferSize);
|
|
}
|
|
else if (duk_is_string(ctx, 1) && op == FS_WRITE)
|
|
{
|
|
buffer = duk_get_lstring(ctx, 1, &bufferSize);
|
|
}
|
|
else
|
|
{
|
|
goto err_invalid_args;
|
|
}
|
|
|
|
if (nargs >= 3)
|
|
{
|
|
if (!duk_is_number(ctx, 2))
|
|
{
|
|
goto err_invalid_args;
|
|
}
|
|
|
|
offset = duk_get_uint(ctx, 2);
|
|
|
|
if (offset >= bufferSize)
|
|
{
|
|
goto err_invalid_range;
|
|
}
|
|
}
|
|
|
|
length = bufferSize - offset;
|
|
|
|
if (nargs >= 4)
|
|
{
|
|
if (!duk_is_number(ctx, 3))
|
|
{
|
|
goto err_invalid_args;
|
|
}
|
|
length = duk_get_uint(ctx, 3);
|
|
}
|
|
|
|
if (nargs >= 5)
|
|
{
|
|
if (!duk_is_number(ctx, 4))
|
|
{
|
|
goto err_invalid_args;
|
|
}
|
|
position = duk_get_uint(ctx, 4);
|
|
bHavePos = true;
|
|
}
|
|
|
|
if (offset + length > bufferSize)
|
|
{
|
|
goto err_invalid_range;
|
|
}
|
|
|
|
duk_get_global_string(ctx, HS_gOpenFileDescriptors);
|
|
|
|
if (!duk_has_prop_index(ctx, -1, fd))
|
|
{
|
|
goto err_invalid_fd;
|
|
}
|
|
|
|
duk_get_prop_index(ctx, -1, fd);
|
|
duk_get_prop_string(ctx, -1, "fp");
|
|
fp = (FILE *)duk_get_pointer(ctx, -1);
|
|
|
|
duk_pop_n(ctx, 3);
|
|
|
|
if (bHavePos)
|
|
{
|
|
fseek(fp, (long)((UINT_PTR)position), SEEK_SET);
|
|
}
|
|
|
|
switch (op)
|
|
{
|
|
case FS_READ:
|
|
rc = fread((void *)&buffer[offset], 1, length, fp);
|
|
break;
|
|
case FS_WRITE:
|
|
rc = fwrite((void *)&buffer[offset], 1, length, fp);
|
|
break;
|
|
default:
|
|
return DUK_RET_ERROR;
|
|
}
|
|
|
|
duk_push_number(ctx, (duk_double_t)((UINT_PTR)rc));
|
|
return 1;
|
|
|
|
err_invalid_args:
|
|
return ScriptAPI::ThrowInvalidArgsError(ctx);
|
|
|
|
err_invalid_range:
|
|
duk_push_error_object(ctx, DUK_ERR_RANGE_ERROR, "invalid range");
|
|
return duk_throw(ctx);
|
|
|
|
err_invalid_fd:
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR, "invalid file descriptor");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
static duk_ret_t js__FileFinalizer(duk_context * ctx)
|
|
{
|
|
CScriptInstance * inst = ScriptAPI::GetInstance(ctx);
|
|
|
|
duk_get_prop_string(ctx, 0, "fp");
|
|
FILE * fp = (FILE *)duk_get_pointer(ctx, -1);
|
|
duk_pop(ctx);
|
|
fclose(fp);
|
|
|
|
inst->System()->ConsoleLog("[SCRIPTSYS]: warning ('%s'): gc closed leftover file descriptor",
|
|
inst->Name().c_str());
|
|
|
|
return 0;
|
|
}
|