Skip to content

Commit

Permalink
src,fs: refactor duplicated code in fs.readdir
Browse files Browse the repository at this point in the history
`AfterScanDirWithTypes` is almost same as `AfterScanDir` except for
handling the `with file types` option. This merges the two functions
into one.

Signed-off-by: Daeyeon Jeong daeyeon.dev@gmail.com

PR-URL: #43204
Reviewed-By: Feng Yu <F3n67u@outlook.com>
  • Loading branch information
daeyeon authored and juanarbol committed Oct 11, 2022
1 parent 8f5fd2f commit 37010cb
Show file tree
Hide file tree
Showing 2 changed files with 28 additions and 57 deletions.
82 changes: 25 additions & 57 deletions src/node_file.cc
Expand Up @@ -777,43 +777,6 @@ void AfterScanDir(uv_fs_t* req) {
FSReqBase* req_wrap = FSReqBase::from_req(req);
FSReqAfterScope after(req_wrap, req);

if (!after.Proceed()) {
return;
}
Environment* env = req_wrap->env();
Local<Value> error;
int r;
std::vector<Local<Value>> name_v;

for (;;) {
uv_dirent_t ent;

r = uv_fs_scandir_next(req, &ent);
if (r == UV_EOF)
break;
if (r != 0) {
return req_wrap->Reject(UVException(
env->isolate(), r, nullptr, req_wrap->syscall(), req->path));
}

MaybeLocal<Value> filename =
StringBytes::Encode(env->isolate(),
ent.name,
req_wrap->encoding(),
&error);
if (filename.IsEmpty())
return req_wrap->Reject(error);

name_v.push_back(filename.ToLocalChecked());
}

req_wrap->Resolve(Array::New(env->isolate(), name_v.data(), name_v.size()));
}

void AfterScanDirWithTypes(uv_fs_t* req) {
FSReqBase* req_wrap = FSReqBase::from_req(req);
FSReqAfterScope after(req_wrap, req);

if (!after.Proceed()) {
return;
}
Expand All @@ -826,6 +789,8 @@ void AfterScanDirWithTypes(uv_fs_t* req) {
std::vector<Local<Value>> name_v;
std::vector<Local<Value>> type_v;

const bool with_file_types = req_wrap->with_file_types();

for (;;) {
uv_dirent_t ent;

Expand All @@ -837,23 +802,23 @@ void AfterScanDirWithTypes(uv_fs_t* req) {
UVException(isolate, r, nullptr, req_wrap->syscall(), req->path));
}

MaybeLocal<Value> filename =
StringBytes::Encode(isolate,
ent.name,
req_wrap->encoding(),
&error);
if (filename.IsEmpty())
Local<Value> filename;
if (!StringBytes::Encode(isolate, ent.name, req_wrap->encoding(), &error)
.ToLocal(&filename)) {
return req_wrap->Reject(error);
}
name_v.push_back(filename);

name_v.push_back(filename.ToLocalChecked());
type_v.emplace_back(Integer::New(isolate, ent.type));
if (with_file_types) type_v.emplace_back(Integer::New(isolate, ent.type));
}

Local<Value> result[] = {
Array::New(isolate, name_v.data(), name_v.size()),
Array::New(isolate, type_v.data(), type_v.size())
};
req_wrap->Resolve(Array::New(isolate, result, arraysize(result)));
if (with_file_types) {
Local<Value> result[] = {Array::New(isolate, name_v.data(), name_v.size()),
Array::New(isolate, type_v.data(), type_v.size())};
req_wrap->Resolve(Array::New(isolate, result, arraysize(result)));
} else {
req_wrap->Resolve(Array::New(isolate, name_v.data(), name_v.size()));
}
}

void Access(const FunctionCallbackInfo<Value>& args) {
Expand Down Expand Up @@ -1650,13 +1615,16 @@ static void ReadDir(const FunctionCallbackInfo<Value>& args) {

FSReqBase* req_wrap_async = GetReqWrap(args, 3);
if (req_wrap_async != nullptr) { // readdir(path, encoding, withTypes, req)
if (with_types) {
AsyncCall(env, req_wrap_async, args, "scandir", encoding,
AfterScanDirWithTypes, uv_fs_scandir, *path, 0 /*flags*/);
} else {
AsyncCall(env, req_wrap_async, args, "scandir", encoding,
AfterScanDir, uv_fs_scandir, *path, 0 /*flags*/);
}
req_wrap_async->set_with_file_types(with_types);
AsyncCall(env,
req_wrap_async,
args,
"scandir",
encoding,
AfterScanDir,
uv_fs_scandir,
*path,
0 /*flags*/);
} else { // readdir(path, encoding, withTypes, undefined, ctx)
CHECK_EQ(argc, 5);
FSReqWrapSync req_wrap_sync;
Expand Down
3 changes: 3 additions & 0 deletions src/node_file.h
Expand Up @@ -89,8 +89,10 @@ class FSReqBase : public ReqWrap<uv_fs_t> {
enum encoding encoding() const { return encoding_; }
bool use_bigint() const { return use_bigint_; }
bool is_plain_open() const { return is_plain_open_; }
bool with_file_types() const { return with_file_types_; }

void set_is_plain_open(bool value) { is_plain_open_ = value; }
void set_with_file_types(bool value) { with_file_types_ = value; }

FSContinuationData* continuation_data() const {
return continuation_data_.get();
Expand All @@ -116,6 +118,7 @@ class FSReqBase : public ReqWrap<uv_fs_t> {
bool has_data_ = false;
bool use_bigint_ = false;
bool is_plain_open_ = false;
bool with_file_types_ = false;
const char* syscall_ = nullptr;

BaseObjectPtr<BindingData> binding_data_;
Expand Down

0 comments on commit 37010cb

Please sign in to comment.