forked from cory/tildefriends
Make all of the File.* operations async.
git-svn-id: https://www.unprompted.com/svn/projects/tildefriends/trunk@3675 ed5197a5-7fde-0310-b194-c3ffbd925b24
This commit is contained in:
parent
77ec1a0b2e
commit
21ba7cb02c
139
src/file.js.c
139
src/file.js.c
@ -15,7 +15,7 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
static JSValue _file_make_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
static JSValue _file_make_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
||||||
static JSValue _file_read_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
static JSValue _file_remove_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
||||||
static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
||||||
static JSValue _file_rename_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
static JSValue _file_rename_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
||||||
static JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
static JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv);
|
||||||
@ -39,13 +39,11 @@ void tf_file_register(JSContext* context)
|
|||||||
JS_SetPropertyStr(context, global, "File", file);
|
JS_SetPropertyStr(context, global, "File", file);
|
||||||
JS_SetPropertyStr(context, file, "readFile", JS_NewCFunction(context, _file_read_file, "readFile", 1));
|
JS_SetPropertyStr(context, file, "readFile", JS_NewCFunction(context, _file_read_file, "readFile", 1));
|
||||||
JS_SetPropertyStr(context, file, "writeFile", JS_NewCFunction(context, _file_write_file, "writeFile", 2));
|
JS_SetPropertyStr(context, file, "writeFile", JS_NewCFunction(context, _file_write_file, "writeFile", 2));
|
||||||
|
|
||||||
/* TODO: async */
|
|
||||||
JS_SetPropertyStr(context, file, "makeDirectory", JS_NewCFunction(context, _file_make_directory, "makeDirectory", 1));
|
JS_SetPropertyStr(context, file, "makeDirectory", JS_NewCFunction(context, _file_make_directory, "makeDirectory", 1));
|
||||||
JS_SetPropertyStr(context, file, "readDirectory", JS_NewCFunction(context, _file_read_directory, "readDirectory", 1));
|
JS_SetPropertyStr(context, file, "removeDirectory", JS_NewCFunction(context, _file_remove_directory, "removeDirectory", 1));
|
||||||
|
JS_SetPropertyStr(context, file, "unlinkFile", JS_NewCFunction(context, _file_unlink_file, "unlinkFile", 1));
|
||||||
JS_SetPropertyStr(context, file, "renameFile", JS_NewCFunction(context, _file_rename_file, "renameFile", 2));
|
JS_SetPropertyStr(context, file, "renameFile", JS_NewCFunction(context, _file_rename_file, "renameFile", 2));
|
||||||
JS_SetPropertyStr(context, file, "stat", JS_NewCFunction(context, _file_stat, "stat", 1));
|
JS_SetPropertyStr(context, file, "stat", JS_NewCFunction(context, _file_stat, "stat", 1));
|
||||||
JS_SetPropertyStr(context, file, "unlinkFile", JS_NewCFunction(context, _file_unlink_file, "unlinkFile", 1));
|
|
||||||
JS_FreeValue(context, global);
|
JS_FreeValue(context, global);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -204,7 +202,7 @@ static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int a
|
|||||||
JS_FreeCString(context, (const char*)buffer);
|
JS_FreeCString(context, (const char*)buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
int result = uv_fs_open(tf_task_get_loop(task), req, file_name, UV_FS_O_CREAT | UV_FS_O_WRONLY, 0755, _file_write_open_callback);
|
int result = uv_fs_open(tf_task_get_loop(task), req, file_name, UV_FS_O_CREAT | UV_FS_O_WRONLY, 0644, _file_write_open_callback);
|
||||||
if (result < 0)
|
if (result < 0)
|
||||||
{
|
{
|
||||||
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, uv_strerror(result)));
|
tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, uv_strerror(result)));
|
||||||
@ -213,63 +211,63 @@ static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int a
|
|||||||
return promise_value;
|
return promise_value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void _file_async_callback(uv_fs_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)
|
||||||
|
{
|
||||||
|
tf_task_resolve_promise(task, promise, JS_NewInt32(context, req->result));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, req->result));
|
||||||
|
}
|
||||||
|
free(req);
|
||||||
|
}
|
||||||
|
|
||||||
static JSValue _file_rename_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
static JSValue _file_rename_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
||||||
{
|
{
|
||||||
void* task = JS_GetContextOpaque(context);
|
void* task = JS_GetContextOpaque(context);
|
||||||
const char* oldName = JS_ToCString(context, argv[0]);
|
const char* old_name = JS_ToCString(context, argv[0]);
|
||||||
const char* newName = JS_ToCString(context, argv[1]);
|
const char* new_name = JS_ToCString(context, argv[1]);
|
||||||
uv_fs_t req;
|
promiseid_t promise = tf_task_allocate_promise(task);
|
||||||
int result = uv_fs_rename(tf_task_get_loop(task), &req, oldName, newName, 0);
|
JSValue promise_value = tf_task_get_promise(task, promise);
|
||||||
JS_FreeCString(context, oldName);
|
uv_fs_t* req = malloc(sizeof(uv_fs_t));
|
||||||
JS_FreeCString(context, newName);
|
*req = (uv_fs_t)
|
||||||
return JS_NewInt32(context, result);
|
{
|
||||||
|
.data = (void*)(intptr_t)promise,
|
||||||
|
};
|
||||||
|
int result = uv_fs_rename(tf_task_get_loop(task), req, old_name, new_name,_file_async_callback);
|
||||||
|
JS_FreeCString(context, old_name);
|
||||||
|
JS_FreeCString(context, new_name);
|
||||||
|
if (result < 0)
|
||||||
|
{
|
||||||
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
||||||
|
}
|
||||||
|
return promise_value;
|
||||||
}
|
}
|
||||||
|
|
||||||
static JSValue _file_unlink_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
static JSValue _file_unlink_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
||||||
{
|
{
|
||||||
void* task = JS_GetContextOpaque(context);
|
void* task = JS_GetContextOpaque(context);
|
||||||
const char* fileName = JS_ToCString(context, argv[0]);
|
const char* file_name = JS_ToCString(context, argv[0]);
|
||||||
uv_fs_t req;
|
promiseid_t promise = tf_task_allocate_promise(task);
|
||||||
int result = uv_fs_unlink(tf_task_get_loop(task), &req, fileName, 0);
|
JSValue promise_value = tf_task_get_promise(task, promise);
|
||||||
JS_FreeCString(context, fileName);
|
uv_fs_t* req = malloc(sizeof(uv_fs_t));
|
||||||
return JS_NewInt32(context, result);
|
*req = (uv_fs_t)
|
||||||
}
|
|
||||||
|
|
||||||
static JSValue _file_read_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
|
||||||
{
|
{
|
||||||
const char* directory = JS_ToCString(context, argv[0]);
|
.data = (void*)(intptr_t)promise,
|
||||||
JSValue array = JS_NewArray(context);
|
};
|
||||||
|
int result = uv_fs_unlink(tf_task_get_loop(task), req, file_name, _file_async_callback);
|
||||||
#ifdef _WIN32
|
JS_FreeCString(context, file_name);
|
||||||
WIN32_FIND_DATA find;
|
if (result < 0)
|
||||||
std::string pattern = directory;
|
|
||||||
pattern += "\\*";
|
|
||||||
HANDLE handle = FindFirstFile(pattern.c_str(), &find);
|
|
||||||
if (handle != INVALID_HANDLE_VALUE)
|
|
||||||
{
|
{
|
||||||
int index = 0;
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
||||||
do {
|
|
||||||
JS_SetPropertyUint32(context, array, index++, JS_NewString(context, find.cFileName));
|
|
||||||
} while (FindNextFile(handle, &find) != 0);
|
|
||||||
FindClose(handle);
|
|
||||||
}
|
}
|
||||||
#else
|
return promise_value;
|
||||||
DIR* dir = opendir(directory);
|
|
||||||
if (dir)
|
|
||||||
{
|
|
||||||
uint32_t index = 0;
|
|
||||||
struct dirent* entry = readdir(dir);
|
|
||||||
while (entry)
|
|
||||||
{
|
|
||||||
JS_SetPropertyUint32(context, array, index++, JS_NewString(context, entry->d_name));
|
|
||||||
entry = readdir(dir);
|
|
||||||
}
|
|
||||||
closedir(dir);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
JS_FreeCString(context, directory);
|
|
||||||
return array;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JSValue _file_make_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
JSValue _file_make_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
||||||
@ -277,10 +275,41 @@ JSValue _file_make_directory(JSContext* context, JSValueConst this_val, int argc
|
|||||||
void* task = JS_GetContextOpaque(context);
|
void* task = JS_GetContextOpaque(context);
|
||||||
const char* directory = JS_ToCString(context, argv[0]);
|
const char* directory = JS_ToCString(context, argv[0]);
|
||||||
|
|
||||||
uv_fs_t req;
|
promiseid_t promise = tf_task_allocate_promise(task);
|
||||||
int result = uv_fs_mkdir(tf_task_get_loop(task), &req, directory, 0755, 0);
|
JSValue promise_value = tf_task_get_promise(task, promise);
|
||||||
|
uv_fs_t* req = malloc(sizeof(uv_fs_t));
|
||||||
|
*req = (uv_fs_t)
|
||||||
|
{
|
||||||
|
.data = (void*)(intptr_t)promise,
|
||||||
|
};
|
||||||
|
int result = uv_fs_mkdir(tf_task_get_loop(task), req, directory, 0755, _file_async_callback);
|
||||||
JS_FreeCString(context, directory);
|
JS_FreeCString(context, directory);
|
||||||
return JS_NewInt32(context, result);
|
if (result < 0)
|
||||||
|
{
|
||||||
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
||||||
|
}
|
||||||
|
return promise_value;
|
||||||
|
}
|
||||||
|
|
||||||
|
JSValue _file_remove_directory(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
||||||
|
{
|
||||||
|
void* task = JS_GetContextOpaque(context);
|
||||||
|
const char* directory = JS_ToCString(context, argv[0]);
|
||||||
|
|
||||||
|
promiseid_t promise = tf_task_allocate_promise(task);
|
||||||
|
JSValue promise_value = tf_task_get_promise(task, promise);
|
||||||
|
uv_fs_t* req = malloc(sizeof(uv_fs_t));
|
||||||
|
*req = (uv_fs_t)
|
||||||
|
{
|
||||||
|
.data = (void*)(intptr_t)promise,
|
||||||
|
};
|
||||||
|
int result = uv_fs_rmdir(tf_task_get_loop(task), req, directory, _file_async_callback);
|
||||||
|
JS_FreeCString(context, directory);
|
||||||
|
if (result < 0)
|
||||||
|
{
|
||||||
|
tf_task_reject_promise(task, promise, JS_NewInt32(context, result));
|
||||||
|
}
|
||||||
|
return promise_value;
|
||||||
}
|
}
|
||||||
|
|
||||||
JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv)
|
||||||
|
23
src/tests.c
23
src/tests.c
@ -496,7 +496,6 @@ static void _test_file(const tf_test_options_t* options)
|
|||||||
fprintf(file,
|
fprintf(file,
|
||||||
"'use strict';\n"
|
"'use strict';\n"
|
||||||
"File.readFile('out/test.js').then(function(data) {\n"
|
"File.readFile('out/test.js').then(function(data) {\n"
|
||||||
" print('READ', utf8Decode(data));\n"
|
|
||||||
"}).catch(function(error) {\n"
|
"}).catch(function(error) {\n"
|
||||||
" print('ERROR', error);\n"
|
" print('ERROR', error);\n"
|
||||||
" exit(1);\n"
|
" exit(1);\n"
|
||||||
@ -507,10 +506,14 @@ static void _test_file(const tf_test_options_t* options)
|
|||||||
"}).catch(function(error) {\n"
|
"}).catch(function(error) {\n"
|
||||||
" print('expected error', error);\n"
|
" print('expected error', error);\n"
|
||||||
"});\n"
|
"});\n"
|
||||||
"File.writeFile('out/new.txt', 'hello').then(function(result) {\n"
|
"File.unlinkFile('out/test/new.txt').finally(function() {\n"
|
||||||
" File.readFile('out/new.txt').then(function(data) {\n"
|
" return File.removeDirectory('out/test');\n"
|
||||||
" print('READ', utf8Decode(data));\n"
|
"}).finally(function() {\n"
|
||||||
|
" return File.makeDirectory('out/test').then(function() {\n"
|
||||||
|
" return File.writeFile('out/test/new.txt', 'hello').then(function(result) {\n"
|
||||||
|
" return File.readFile('out/test/new.txt').then(function(data) {\n"
|
||||||
" if (utf8Decode(data) != 'hello') {\n"
|
" if (utf8Decode(data) != 'hello') {\n"
|
||||||
|
" print('READ', utf8Decode(data));\n"
|
||||||
" exit(1);\n"
|
" exit(1);\n"
|
||||||
" }\n"
|
" }\n"
|
||||||
" }).catch(function(error) {\n"
|
" }).catch(function(error) {\n"
|
||||||
@ -520,6 +523,18 @@ static void _test_file(const tf_test_options_t* options)
|
|||||||
" }).catch(function(error) {\n"
|
" }).catch(function(error) {\n"
|
||||||
" print('unexpected write error', error);\n"
|
" print('unexpected write error', error);\n"
|
||||||
" exit(1);\n"
|
" exit(1);\n"
|
||||||
|
" });\n"
|
||||||
|
" }).catch(function(error) {\n"
|
||||||
|
" print('unexpected make directory error', error);\n"
|
||||||
|
" exit(1);\n"
|
||||||
|
" });\n"
|
||||||
|
"}).finally(function() {\n"
|
||||||
|
" return File.renameFile('out/test/new.txt', 'out/test/renamed.txt').catch(x => print(x)).finally(function() {\n"
|
||||||
|
" return File.unlinkFile('out/test/renamed.txt').catch(x => print(x)).finally(function() {\n"
|
||||||
|
" print('removing directory');\n"
|
||||||
|
" return File.removeDirectory('out/test').catch(x => print(x));\n"
|
||||||
|
" });\n"
|
||||||
|
" });\n"
|
||||||
"});\n");
|
"});\n");
|
||||||
fclose(file);
|
fclose(file);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user