@@ -222,9 +222,12 @@ void FSReqBase::MemoryInfo(MemoryTracker* tracker) const {
222222// collection if necessary. If that happens, a process warning will be
223223// emitted (or a fatal exception will occur if the fd cannot be closed.)
224224FileHandle::FileHandle(BindingData* binding_data,
225- Local<Object> obj, int fd)
225+ Local<Object> obj,
226+ int fd,
227+ std::string original_name)
226228 : AsyncWrap(binding_data->env(), obj, AsyncWrap::PROVIDER_FILEHANDLE),
227229 StreamBase(env()),
230+ original_name_(std::move(original_name)),
228231 fd_(fd),
229232 binding_data_(binding_data) {
230233 MakeWeak();
@@ -234,6 +237,7 @@ FileHandle::FileHandle(BindingData* binding_data,
234237FileHandle* FileHandle::New(BindingData* binding_data,
235238 int fd,
236239 Local<Object> obj,
240+ std::string original_name,
237241 std::optional<int64_t> maybeOffset,
238242 std::optional<int64_t> maybeLength) {
239243 Environment* env = binding_data->env();
@@ -242,7 +246,7 @@ FileHandle* FileHandle::New(BindingData* binding_data,
242246 .ToLocal(&obj)) {
243247 return nullptr;
244248 }
245- auto handle = new FileHandle(binding_data, obj, fd);
249+ auto handle = new FileHandle(binding_data, obj, fd, original_name );
246250 if (maybeOffset.has_value()) handle->read_offset_ = maybeOffset.value();
247251 if (maybeLength.has_value()) handle->read_length_ = maybeLength.value();
248252 return handle;
@@ -274,6 +278,7 @@ void FileHandle::New(const FunctionCallbackInfo<Value>& args) {
274278 FileHandle::New(binding_data,
275279 args[0].As<Int32>()->Value(),
276280 args.This(),
281+ {},
277282 maybeOffset,
278283 maybeLength);
279284}
@@ -293,6 +298,7 @@ int FileHandle::DoWrite(WriteWrap* w,
293298
294299void FileHandle::MemoryInfo(MemoryTracker* tracker) const {
295300 tracker->TrackField("current_read", current_read_);
301+ tracker->TrackField("original_name", original_name_);
296302}
297303
298304BaseObject::TransferMode FileHandle::GetTransferMode() const {
@@ -344,9 +350,13 @@ inline void FileHandle::Close() {
344350 FS_SYNC_TRACE_END(close);
345351 uv_fs_req_cleanup(&req);
346352
347- struct err_detail { int ret; int fd; };
353+ struct err_detail {
354+ int ret;
355+ int fd;
356+ std::string name;
357+ };
348358
349- err_detail detail { ret, fd_ };
359+ err_detail detail{ ret, fd_, original_name_ };
350360
351361 AfterClose();
352362
@@ -362,25 +372,30 @@ inline void FileHandle::Close() {
362372 // down the process is the only reasonable thing we can do here.
363373 env()->SetImmediate([detail](Environment* env) {
364374 HandleScope handle_scope(env->isolate());
375+ static constexpr std::string_view unknown_path = "<unknown path>";
376+ std::string_view filename =
377+ detail.name.empty() ? unknown_path : detail.name;
365378
366379 // If there was an error while trying to close the file descriptor,
367380 // we will throw that instead.
368381 if (detail.ret < 0) {
369- char msg[70];
370- snprintf(msg,
371- arraysize(msg),
372- "Closing file descriptor %d on garbage collection failed",
373- detail.fd);
382+ auto formatted = SPrintF(
383+ "Closing file descriptor %d on garbage collection failed (%s)",
384+ detail.fd,
385+ filename);
374386 HandleScope handle_scope(env->isolate());
375- env->ThrowUVException(detail.ret, "close", msg );
387+ env->ThrowUVException(detail.ret, "close", formatted.c_str() );
376388 return;
377389 }
378390
379391 THROW_ERR_INVALID_STATE(
380392 env,
381393 "A FileHandle object was closed during garbage collection. "
382394 "This used to be allowed with a deprecation warning but is now "
383- "considered an error. Please close FileHandle objects explicitly.");
395+ "considered an error. Please close FileHandle objects explicitly. "
396+ "File descriptor: %d (%s)",
397+ detail.fd,
398+ filename);
384399 });
385400}
386401
@@ -824,8 +839,8 @@ void AfterOpenFileHandle(uv_fs_t* req) {
824839 FS_ASYNC_TRACE_END1(
825840 req->fs_type, req_wrap, "result", static_cast<int>(req->result))
826841 if (after.Proceed()) {
827- FileHandle* fd = FileHandle::New(req_wrap->binding_data(),
828- static_cast<int>(req->result));
842+ FileHandle* fd = FileHandle::New(
843+ req_wrap->binding_data(), static_cast<int>(req->result), {}, req->path );
829844 if (fd == nullptr) return;
830845 req_wrap->Resolve(fd->object());
831846 }
@@ -2222,7 +2237,7 @@ static void OpenFileHandle(const FunctionCallbackInfo<Value>& args) {
22222237 if (result < 0) {
22232238 return; // syscall failed, no need to continue, error info is in ctx
22242239 }
2225- FileHandle* fd = FileHandle::New(binding_data, result);
2240+ FileHandle* fd = FileHandle::New(binding_data, result, {}, path.ToString() );
22262241 if (fd == nullptr) return;
22272242 args.GetReturnValue().Set(fd->object());
22282243 }
0 commit comments