2021-10-24 11:46:30 -04:00
|
|
|
#include "file.js.h"
|
2021-01-02 13:10:00 -05:00
|
|
|
|
2022-06-04 13:04:51 -04:00
|
|
|
#include "mem.h"
|
2021-01-02 13:10:00 -05:00
|
|
|
#include "task.h"
|
2023-07-19 22:20:38 -04:00
|
|
|
#include "trace.h"
|
2021-11-03 18:15:46 -04:00
|
|
|
#include "util.js.h"
|
2021-01-02 13:10:00 -05:00
|
|
|
|
2023-03-08 20:03:35 -05:00
|
|
|
#include "unzip.h"
|
2023-05-21 17:36:51 -04:00
|
|
|
#include "uv.h"
|
2023-03-08 20:03:35 -05:00
|
|
|
|
2021-01-02 13:10:00 -05:00
|
|
|
#include <stdbool.h>
|
2021-10-27 19:27:21 -04:00
|
|
|
#include <string.h>
|
2021-01-02 13:10:00 -05:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
2023-03-08 20:03:35 -05:00
|
|
|
static JSValue _file_read_file_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
2021-01-02 13:10:00 -05:00
|
|
|
static JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
2023-03-08 20:03:35 -05:00
|
|
|
static JSValue _file_stat_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
2023-04-19 20:11:38 -04:00
|
|
|
static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
2021-01-02 13:10:00 -05:00
|
|
|
|
|
|
|
static double _time_spec_to_double(const uv_timespec_t* time_spec);
|
|
|
|
static void _file_on_stat_complete(uv_fs_t* request);
|
|
|
|
|
|
|
|
typedef struct file_stat_t {
|
|
|
|
void* _task;
|
|
|
|
JSContext* _context;
|
|
|
|
promiseid_t _promise;
|
|
|
|
uv_fs_t _request;
|
|
|
|
} file_stat_t;
|
|
|
|
|
2022-05-16 18:30:14 -04:00
|
|
|
typedef struct fs_req_t {
|
|
|
|
uv_fs_t fs;
|
|
|
|
size_t size;
|
2024-01-10 20:38:30 -05:00
|
|
|
uv_file file;
|
2022-05-16 18:30:14 -04:00
|
|
|
char buffer[];
|
|
|
|
} fs_req_t;
|
|
|
|
|
2021-10-24 11:46:30 -04:00
|
|
|
void tf_file_register(JSContext* context)
|
2021-10-10 17:51:38 -04:00
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
JSValue global = JS_GetGlobalObject(context);
|
|
|
|
JSValue file = JS_NewObject(context);
|
2023-03-08 20:03:35 -05:00
|
|
|
void* task = JS_GetContextOpaque(context);
|
|
|
|
const char* zip = tf_task_get_zip_path(task);
|
2021-01-02 13:10:00 -05:00
|
|
|
JS_SetPropertyStr(context, global, "File", file);
|
2023-03-08 20:03:35 -05:00
|
|
|
JS_SetPropertyStr(context, file, "readFile", JS_NewCFunction(context, zip ? _file_read_file_zip : _file_read_file, "readFile", 1));
|
2023-04-19 20:11:38 -04:00
|
|
|
JS_SetPropertyStr(context, file, "writeFile", JS_NewCFunction(context, _file_write_file, "writeFile", 2));
|
2023-03-08 20:03:35 -05:00
|
|
|
JS_SetPropertyStr(context, file, "stat", JS_NewCFunction(context, zip ? _file_stat_zip : _file_stat, "stat", 1));
|
2021-01-02 13:10:00 -05:00
|
|
|
JS_FreeValue(context, global);
|
|
|
|
}
|
|
|
|
|
2022-01-02 14:10:45 -05:00
|
|
|
static const int k_file_read_max = 8 * 1024 * 1024;
|
2021-10-05 21:25:33 -04:00
|
|
|
|
2021-10-27 19:27:21 -04:00
|
|
|
static void _file_async_close_callback(uv_fs_t* req)
|
2021-10-05 21:25:33 -04:00
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(req);
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
|
|
|
|
2023-04-19 20:11:38 -04:00
|
|
|
static void _file_write_write_callback(uv_fs_t* req)
|
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
|
|
|
fs_req_t* fsreq = (fs_req_t*)req;
|
|
|
|
tf_task_t* task = req->loop->data;
|
|
|
|
JSContext* context = tf_task_get_context(task);
|
|
|
|
promiseid_t promise = (promiseid_t)(intptr_t)req->data;
|
|
|
|
if (req->result >= 0)
|
|
|
|
{
|
|
|
|
tf_task_resolve_promise(task, promise, JS_NewInt64(context, req->result));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(req->result)));
|
|
|
|
}
|
|
|
|
int result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback);
|
|
|
|
if (result < 0)
|
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
|
|
|
tf_free(fsreq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _file_write_open_callback(uv_fs_t* req)
|
|
|
|
{
|
|
|
|
fs_req_t* fsreq = (fs_req_t*)req;
|
|
|
|
uv_fs_req_cleanup(req);
|
|
|
|
tf_task_t* task = req->loop->data;
|
|
|
|
JSContext* context = tf_task_get_context(task);
|
|
|
|
promiseid_t promise = (promiseid_t)(intptr_t)req->data;
|
|
|
|
if (req->result >= 0)
|
|
|
|
{
|
|
|
|
uv_buf_t buf = { .base = fsreq->buffer, .len = fsreq->size };
|
|
|
|
fsreq->file = req->result;
|
|
|
|
int result = uv_fs_write(req->loop, req, fsreq->file, &buf, 1, 0, _file_write_write_callback);
|
|
|
|
if (result < 0)
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(result)));
|
|
|
|
result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback);
|
|
|
|
if (result < 0)
|
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
|
|
|
tf_free(fsreq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(req->result)));
|
|
|
|
uv_fs_req_cleanup(req);
|
|
|
|
tf_free(req);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
|
|
{
|
|
|
|
void* task = JS_GetContextOpaque(context);
|
|
|
|
const char* file_name = JS_ToCString(context, argv[0]);
|
|
|
|
|
|
|
|
size_t size;
|
|
|
|
uint8_t* buffer = tf_util_try_get_array_buffer(context, &size, argv[1]);
|
|
|
|
bool is_array_buffer = false;
|
|
|
|
if (buffer)
|
|
|
|
{
|
|
|
|
is_array_buffer = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
buffer = (uint8_t*)JS_ToCStringLen(context, &size, argv[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
promiseid_t promise = -1;
|
|
|
|
JSValue promise_value = tf_task_allocate_promise(task, &promise);
|
|
|
|
fs_req_t* req = tf_malloc(sizeof(fs_req_t) + size);
|
|
|
|
*req = (fs_req_t)
|
|
|
|
{
|
|
|
|
.fs =
|
|
|
|
{
|
|
|
|
.data = (void*)(intptr_t)promise,
|
|
|
|
},
|
|
|
|
.size = size,
|
|
|
|
};
|
|
|
|
memcpy(req->buffer, buffer, size);
|
|
|
|
if (!is_array_buffer)
|
|
|
|
{
|
|
|
|
JS_FreeCString(context, (const char*)buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
int result = uv_fs_open(tf_task_get_loop(task), &req->fs, file_name, UV_FS_O_CREAT | UV_FS_O_WRONLY, 0644, _file_write_open_callback);
|
|
|
|
if (result < 0)
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(result)));
|
|
|
|
}
|
|
|
|
JS_FreeCString(context, file_name);
|
|
|
|
return promise_value;
|
|
|
|
}
|
|
|
|
|
2021-10-05 21:25:33 -04:00
|
|
|
static void _file_read_read_callback(uv_fs_t* req)
|
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
2022-05-16 18:30:14 -04:00
|
|
|
fs_req_t* fsreq = (fs_req_t*)req;
|
2021-10-05 21:25:33 -04:00
|
|
|
tf_task_t* task = req->loop->data;
|
|
|
|
JSContext* context = tf_task_get_context(task);
|
|
|
|
promiseid_t promise = (promiseid_t)(intptr_t)req->data;
|
|
|
|
if (req->result >= 0)
|
|
|
|
{
|
2023-01-28 16:59:36 -05:00
|
|
|
JSValue array = tf_util_new_uint8_array(context, (const uint8_t*)fsreq->buffer, req->result);
|
|
|
|
tf_task_resolve_promise(task, promise, array);
|
|
|
|
JS_FreeValue(context, array);
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
2021-10-05 21:25:33 -04:00
|
|
|
else
|
|
|
|
{
|
2022-09-09 21:42:15 -04:00
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(req->result)));
|
2021-10-05 21:25:33 -04:00
|
|
|
}
|
2022-05-16 18:30:14 -04:00
|
|
|
int result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback);
|
2021-10-05 21:25:33 -04:00
|
|
|
if (result < 0)
|
|
|
|
{
|
2022-06-03 23:01:12 -04:00
|
|
|
uv_fs_req_cleanup(req);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(fsreq);
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
2021-10-05 21:25:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _file_read_open_callback(uv_fs_t* req)
|
|
|
|
{
|
|
|
|
uv_fs_req_cleanup(req);
|
2022-06-02 05:40:19 -04:00
|
|
|
fs_req_t* fsreq = (fs_req_t*)req;
|
2021-10-05 21:25:33 -04:00
|
|
|
tf_task_t* task = req->loop->data;
|
|
|
|
JSContext* context = tf_task_get_context(task);
|
|
|
|
promiseid_t promise = (promiseid_t)(intptr_t)req->data;
|
|
|
|
if (req->result >= 0)
|
|
|
|
{
|
2022-06-02 05:40:19 -04:00
|
|
|
uv_buf_t buf = { .base = fsreq->buffer, .len = fsreq->size };
|
|
|
|
fsreq->file = req->result;
|
|
|
|
int result = uv_fs_read(req->loop, req, fsreq->file, &buf, 1, 0, _file_read_read_callback);
|
2021-10-05 21:25:33 -04:00
|
|
|
if (result < 0)
|
|
|
|
{
|
2022-09-09 21:42:15 -04:00
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(result)));
|
2022-06-02 05:40:19 -04:00
|
|
|
result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback);
|
|
|
|
if (result < 0)
|
|
|
|
{
|
2022-06-03 23:01:12 -04:00
|
|
|
uv_fs_req_cleanup(req);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(fsreq);
|
2022-06-02 05:40:19 -04:00
|
|
|
}
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
|
|
|
}
|
2021-10-05 21:25:33 -04:00
|
|
|
else
|
|
|
|
{
|
2022-09-09 21:42:15 -04:00
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(req->result)));
|
2022-06-03 23:01:12 -04:00
|
|
|
uv_fs_req_cleanup(req);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(req);
|
2021-10-05 21:25:33 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-10 17:51:38 -04:00
|
|
|
static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
|
|
{
|
2021-10-05 21:25:33 -04:00
|
|
|
void* task = JS_GetContextOpaque(context);
|
|
|
|
const char* file_name = JS_ToCString(context, argv[0]);
|
|
|
|
|
2022-01-13 22:05:37 -05:00
|
|
|
promiseid_t promise = -1;
|
|
|
|
JSValue promise_value = tf_task_allocate_promise(task, &promise);
|
2022-06-04 13:04:51 -04:00
|
|
|
fs_req_t* req = tf_malloc(sizeof(fs_req_t) + k_file_read_max);
|
2022-05-16 18:30:14 -04:00
|
|
|
*req = (fs_req_t)
|
2021-10-05 21:25:33 -04:00
|
|
|
{
|
2022-05-16 18:30:14 -04:00
|
|
|
.fs =
|
|
|
|
{
|
|
|
|
.data = (void*)(intptr_t)promise,
|
|
|
|
},
|
2022-06-02 05:40:19 -04:00
|
|
|
.size = k_file_read_max,
|
2021-10-05 21:25:33 -04:00
|
|
|
};
|
2022-05-16 18:30:14 -04:00
|
|
|
int result = uv_fs_open(tf_task_get_loop(task), &req->fs, file_name, UV_FS_O_RDONLY, 0, _file_read_open_callback);
|
2021-10-05 21:25:33 -04:00
|
|
|
if (result < 0)
|
|
|
|
{
|
2022-09-09 21:42:15 -04:00
|
|
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(result)));
|
2022-06-03 23:01:12 -04:00
|
|
|
uv_fs_req_cleanup(&req->fs);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(req);
|
2021-10-05 21:25:33 -04:00
|
|
|
}
|
|
|
|
JS_FreeCString(context, file_name);
|
2022-01-13 22:05:37 -05:00
|
|
|
return promise_value;
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
|
|
|
|
2023-03-08 20:03:35 -05:00
|
|
|
typedef struct _zip_read_work_t
|
|
|
|
{
|
|
|
|
uv_work_t request;
|
|
|
|
JSContext* context;
|
|
|
|
tf_task_t* task;
|
|
|
|
const char* file_path;
|
|
|
|
promiseid_t promise;
|
|
|
|
int result;
|
|
|
|
uint8_t* buffer;
|
|
|
|
size_t size;
|
|
|
|
} zip_read_work_t;
|
|
|
|
|
|
|
|
static void _file_read_file_zip_work(uv_work_t* work)
|
|
|
|
{
|
|
|
|
zip_read_work_t* data = work->data;
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_trace_t* trace = tf_task_get_trace(data->task);
|
|
|
|
tf_trace_begin(trace, "file_read_zip_work");
|
2023-03-08 20:03:35 -05:00
|
|
|
unzFile zip = unzOpen(tf_task_get_zip_path(data->task));
|
|
|
|
bool is_file_open = false;
|
|
|
|
if (!zip)
|
|
|
|
{
|
|
|
|
data->result = errno;
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_trace_end(trace);
|
2023-03-08 20:03:35 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->result = unzLocateFile(zip, data->file_path, 1);
|
|
|
|
if (data->result != UNZ_OK)
|
|
|
|
{
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
unz_file_info64 info = { 0 };
|
|
|
|
data->result = unzGetCurrentFileInfo64(zip, &info, NULL, 0, NULL, 0, NULL, 0);
|
|
|
|
if (data->result != UNZ_OK)
|
|
|
|
{
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->result = unzOpenCurrentFile(zip);
|
|
|
|
if (data->result != UNZ_OK)
|
|
|
|
{
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
is_file_open = true;
|
|
|
|
|
|
|
|
data->buffer = tf_malloc(info.uncompressed_size);
|
|
|
|
data->result = unzReadCurrentFile(zip, data->buffer, info.uncompressed_size);
|
|
|
|
if (data->result <= 0)
|
|
|
|
{
|
|
|
|
tf_free(data->buffer);
|
|
|
|
data->buffer = NULL;
|
|
|
|
}
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_trace_end(trace);
|
2023-03-08 20:03:35 -05:00
|
|
|
|
|
|
|
done:
|
|
|
|
if (is_file_open)
|
|
|
|
{
|
|
|
|
unzCloseCurrentFile(zip);
|
|
|
|
}
|
|
|
|
unzClose(zip);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _file_read_file_zip_after_work(uv_work_t* work, int status)
|
|
|
|
{
|
|
|
|
zip_read_work_t* data = work->data;
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_trace_t* trace = tf_task_get_trace(data->task);
|
|
|
|
tf_trace_begin(trace, "file_read_zip_after_work");
|
2023-03-08 20:03:35 -05:00
|
|
|
if (data->result >= 0)
|
|
|
|
{
|
|
|
|
tf_task_resolve_promise(data->task, data->promise, tf_util_new_uint8_array(data->context, data->buffer, data->result));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(data->task, data->promise, JS_ThrowInternalError(data->context, "Error: %d.", data->result));
|
|
|
|
}
|
|
|
|
tf_free(data->buffer);
|
|
|
|
tf_free((void*)data->file_path);
|
|
|
|
tf_free(data);
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_trace_end(trace);
|
2023-03-08 20:03:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSValue _file_read_file_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
|
|
{
|
2023-07-19 22:20:38 -04:00
|
|
|
tf_task_t* task = JS_GetContextOpaque(context);
|
2023-03-08 20:03:35 -05:00
|
|
|
const char* file_name = JS_ToCString(context, argv[0]);
|
|
|
|
|
|
|
|
zip_read_work_t* work = tf_malloc(sizeof(zip_read_work_t));
|
|
|
|
*work = (zip_read_work_t)
|
|
|
|
{
|
|
|
|
.request =
|
|
|
|
{
|
|
|
|
.data = work,
|
|
|
|
},
|
|
|
|
.context = context,
|
|
|
|
.task = task,
|
|
|
|
.file_path = tf_strdup(file_name),
|
|
|
|
};
|
|
|
|
|
|
|
|
JSValue promise_value = tf_task_allocate_promise(task, &work->promise);
|
|
|
|
|
|
|
|
int r = uv_queue_work(tf_task_get_loop(task), &work->request, _file_read_file_zip_work, _file_read_file_zip_after_work);
|
|
|
|
if (r)
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, work->promise, JS_ThrowInternalError(context, "%s", uv_strerror(r)));
|
|
|
|
tf_free((void*)work->file_path);
|
|
|
|
tf_free(work);
|
|
|
|
}
|
|
|
|
|
|
|
|
JS_FreeCString(context, file_name);
|
|
|
|
return promise_value;
|
|
|
|
}
|
|
|
|
|
2021-10-10 17:51:38 -04:00
|
|
|
JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
void* task = JS_GetContextOpaque(context);
|
|
|
|
const char* path = JS_ToCString(context, argv[0]);
|
2022-01-13 22:05:37 -05:00
|
|
|
promiseid_t promise = -1;
|
|
|
|
JSValue promise_value = tf_task_allocate_promise(task, &promise);
|
2021-01-02 13:10:00 -05:00
|
|
|
|
2022-06-04 13:04:51 -04:00
|
|
|
file_stat_t* data = tf_malloc(sizeof(file_stat_t));
|
2021-01-02 13:10:00 -05:00
|
|
|
data->_task = task;
|
|
|
|
data->_promise = promise;
|
|
|
|
data->_request.data = data;
|
|
|
|
data->_context = context;
|
|
|
|
|
|
|
|
int result = uv_fs_stat(tf_task_get_loop(task), &data->_request, path, _file_on_stat_complete);
|
2021-10-10 17:51:38 -04:00
|
|
|
if (result)
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
2022-06-03 23:01:12 -04:00
|
|
|
uv_fs_req_cleanup(&data->_request);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(data);
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
|
|
|
JS_FreeCString(context, path);
|
2022-01-13 22:05:37 -05:00
|
|
|
return promise_value;
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|
|
|
|
|
2023-03-08 20:03:35 -05:00
|
|
|
JSValue _file_stat_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
|
|
{
|
|
|
|
void* task = JS_GetContextOpaque(context);
|
|
|
|
promiseid_t promise = -1;
|
|
|
|
JSValue promise_value = tf_task_allocate_promise(task, &promise);
|
|
|
|
|
|
|
|
file_stat_t* data = tf_malloc(sizeof(file_stat_t));
|
|
|
|
data->_task = task;
|
|
|
|
data->_promise = promise;
|
|
|
|
data->_request.data = data;
|
|
|
|
data->_context = context;
|
|
|
|
|
|
|
|
/* Ignore the requested path and stat the zip itself. */
|
|
|
|
int result = uv_fs_stat(tf_task_get_loop(task), &data->_request, tf_task_get_zip_path(task), _file_on_stat_complete);
|
|
|
|
if (result)
|
|
|
|
{
|
|
|
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
|
|
|
uv_fs_req_cleanup(&data->_request);
|
|
|
|
tf_free(data);
|
|
|
|
}
|
|
|
|
return promise_value;
|
|
|
|
}
|
|
|
|
|
2021-10-10 17:51:38 -04:00
|
|
|
static double _time_spec_to_double(const uv_timespec_t* time_spec)
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
return time_spec->tv_sec + (double)(time_spec->tv_nsec) / 1e9;
|
|
|
|
}
|
|
|
|
|
2021-10-10 17:51:38 -04:00
|
|
|
static void _file_on_stat_complete(uv_fs_t* request)
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
file_stat_t* data = (file_stat_t*)(request->data);
|
|
|
|
JSContext* context = data->_context;
|
|
|
|
|
2021-10-10 17:51:38 -04:00
|
|
|
if (request->result)
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
tf_task_reject_promise(data->_task, data->_promise, JS_NewInt32(context, request->result));
|
2021-10-10 17:51:38 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-01-02 13:10:00 -05:00
|
|
|
JSValue result = JS_NewObject(context);
|
|
|
|
JS_SetPropertyStr(context, result, "mtime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_mtim)));
|
|
|
|
JS_SetPropertyStr(context, result, "ctime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_ctim)));
|
|
|
|
JS_SetPropertyStr(context, result, "atime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_atim)));
|
|
|
|
JS_SetPropertyStr(context, result, "size", JS_NewFloat64(context, request->statbuf.st_size));
|
|
|
|
tf_task_resolve_promise(data->_task, data->_promise, result);
|
|
|
|
JS_FreeValue(context, result);
|
|
|
|
}
|
|
|
|
uv_fs_req_cleanup(request);
|
2022-06-04 13:04:51 -04:00
|
|
|
tf_free(data);
|
2021-01-02 13:10:00 -05:00
|
|
|
}
|