1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-05 05:49:20 -05:00

fix(ops): Always close cancel handles for read_async/write_async (#17736)

Fixes https://github.com/denoland/deno/issues/17734
This commit is contained in:
Kamil Ogórek 2023-02-11 13:19:13 +01:00 committed by GitHub
parent 211f49619a
commit 0164959d34
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 98 additions and 31 deletions

View file

@ -140,6 +140,17 @@ Deno.test(
}, },
); );
// Test that AbortController's cancel handle is cleaned-up correctly, and do not leak resources.
Deno.test(
{ permissions: { read: true } },
async function readFileWithAbortSignalNotCalled() {
const ac = new AbortController();
await Deno.readFile("cli/tests/testdata/assets/fixture.json", {
signal: ac.signal,
});
},
);
Deno.test( Deno.test(
{ permissions: { read: true }, ignore: Deno.build.os !== "linux" }, { permissions: { read: true }, ignore: Deno.build.os !== "linux" },
async function readFileProcFs() { async function readFileProcFs() {

View file

@ -95,7 +95,7 @@ Deno.test(
queueMicrotask(() => ac.abort()); queueMicrotask(() => ac.abort());
const error = await assertRejects( const error = await assertRejects(
async () => { async () => {
await Deno.readFile("cli/tests/testdata/assets/fixture.json", { await Deno.readTextFile("cli/tests/testdata/assets/fixture.json", {
signal: ac.signal, signal: ac.signal,
}); });
}, },
@ -113,7 +113,7 @@ Deno.test(
queueMicrotask(() => ac.abort(abortReason)); queueMicrotask(() => ac.abort(abortReason));
const error = await assertRejects( const error = await assertRejects(
async () => { async () => {
await Deno.readFile("cli/tests/testdata/assets/fixture.json", { await Deno.readTextFile("cli/tests/testdata/assets/fixture.json", {
signal: ac.signal, signal: ac.signal,
}); });
}, },
@ -128,7 +128,7 @@ Deno.test(
const ac = new AbortController(); const ac = new AbortController();
queueMicrotask(() => ac.abort("Some string")); queueMicrotask(() => ac.abort("Some string"));
try { try {
await Deno.readFile("cli/tests/testdata/assets/fixture.json", { await Deno.readTextFile("cli/tests/testdata/assets/fixture.json", {
signal: ac.signal, signal: ac.signal,
}); });
unreachable(); unreachable();
@ -138,6 +138,17 @@ Deno.test(
}, },
); );
// Test that AbortController's cancel handle is cleaned-up correctly, and do not leak resources.
Deno.test(
{ permissions: { read: true } },
async function readTextFileWithAbortSignalNotCalled() {
const ac = new AbortController();
await Deno.readTextFile("cli/tests/testdata/assets/fixture.json", {
signal: ac.signal,
});
},
);
Deno.test( Deno.test(
{ permissions: { read: true }, ignore: Deno.build.os !== "linux" }, { permissions: { read: true }, ignore: Deno.build.os !== "linux" },
async function readTextFileProcFs() { async function readTextFileProcFs() {

View file

@ -379,6 +379,22 @@ Deno.test(
}, },
); );
// Test that AbortController's cancel handle is cleaned-up correctly, and do not leak resources.
Deno.test(
{ permissions: { read: true, write: true } },
async function writeFileWithAbortSignalNotCalled() {
const ac = new AbortController();
const enc = new TextEncoder();
const data = enc.encode("Hello");
const filename = Deno.makeTempDirSync() + "/test.txt";
await Deno.writeFile(filename, data, { signal: ac.signal });
const dataRead = Deno.readFileSync(filename);
const dec = new TextDecoder("utf-8");
const actual = dec.decode(dataRead);
assertEquals(actual, "Hello");
},
);
function assertNotExists(filename: string | URL) { function assertNotExists(filename: string | URL) {
if (pathExists(filename)) { if (pathExists(filename)) {
throw new Error(`The file ${filename} exists.`); throw new Error(`The file ${filename} exists.`);

View file

@ -267,14 +267,6 @@ async fn op_write_file_async(
data: ZeroCopyBuf, data: ZeroCopyBuf,
cancel_rid: Option<ResourceId>, cancel_rid: Option<ResourceId>,
) -> Result<(), AnyError> { ) -> Result<(), AnyError> {
let cancel_handle = match cancel_rid {
Some(cancel_rid) => state
.borrow_mut()
.resource_table
.get::<CancelHandle>(cancel_rid)
.ok(),
None => None,
};
let (path, open_options) = open_helper( let (path, open_options) = open_helper(
&mut state.borrow_mut(), &mut state.borrow_mut(),
&path, &path,
@ -282,15 +274,30 @@ async fn op_write_file_async(
Some(&write_open_options(create, append, create_new)), Some(&write_open_options(create, append, create_new)),
"Deno.writeFile()", "Deno.writeFile()",
)?; )?;
let write_future = tokio::task::spawn_blocking(move || { let write_future = tokio::task::spawn_blocking(move || {
write_file(&path, open_options, mode, data) write_file(&path, open_options, mode, data)
}); });
let cancel_handle = cancel_rid.and_then(|rid| {
state
.borrow_mut()
.resource_table
.get::<CancelHandle>(rid)
.ok()
});
if let Some(cancel_handle) = cancel_handle { if let Some(cancel_handle) = cancel_handle {
write_future.or_cancel(cancel_handle).await???; let write_future_rv = write_future.or_cancel(cancel_handle).await;
} else {
write_future.await??; if let Some(cancel_rid) = cancel_rid {
state.borrow_mut().resource_table.close(cancel_rid).ok();
};
return write_future_rv??;
} }
Ok(())
write_future.await?
} }
fn write_file( fn write_file(
@ -2046,20 +2053,31 @@ async fn op_readfile_async(
.borrow_mut::<PermissionsContainer>() .borrow_mut::<PermissionsContainer>()
.check_read(path, "Deno.readFile()")?; .check_read(path, "Deno.readFile()")?;
} }
let fut = tokio::task::spawn_blocking(move || {
let read_future = tokio::task::spawn_blocking(move || {
let path = Path::new(&path); let path = Path::new(&path);
Ok(std::fs::read(path).map(ZeroCopyBuf::from)?) Ok(std::fs::read(path).map(ZeroCopyBuf::from)?)
}); });
if let Some(cancel_rid) = cancel_rid {
let cancel_handle = state let cancel_handle = cancel_rid.and_then(|rid| {
state
.borrow_mut() .borrow_mut()
.resource_table .resource_table
.get::<CancelHandle>(cancel_rid); .get::<CancelHandle>(rid)
if let Ok(cancel_handle) = cancel_handle { .ok()
return fut.or_cancel(cancel_handle).await??; });
if let Some(cancel_handle) = cancel_handle {
let read_future_rv = read_future.or_cancel(cancel_handle).await;
if let Some(cancel_rid) = cancel_rid {
state.borrow_mut().resource_table.close(cancel_rid).ok();
};
return read_future_rv??;
} }
}
fut.await? read_future.await?
} }
#[op] #[op]
@ -2075,20 +2093,31 @@ async fn op_readfile_text_async(
.borrow_mut::<PermissionsContainer>() .borrow_mut::<PermissionsContainer>()
.check_read(path, "Deno.readTextFile()")?; .check_read(path, "Deno.readTextFile()")?;
} }
let fut = tokio::task::spawn_blocking(move || {
let read_future = tokio::task::spawn_blocking(move || {
let path = Path::new(&path); let path = Path::new(&path);
Ok(string_from_utf8_lossy(std::fs::read(path)?)) Ok(string_from_utf8_lossy(std::fs::read(path)?))
}); });
if let Some(cancel_rid) = cancel_rid {
let cancel_handle = state let cancel_handle = cancel_rid.and_then(|rid| {
state
.borrow_mut() .borrow_mut()
.resource_table .resource_table
.get::<CancelHandle>(cancel_rid); .get::<CancelHandle>(rid)
if let Ok(cancel_handle) = cancel_handle { .ok()
return fut.or_cancel(cancel_handle).await??; });
if let Some(cancel_handle) = cancel_handle {
let read_future_rv = read_future.or_cancel(cancel_handle).await;
if let Some(cancel_rid) = cancel_rid {
state.borrow_mut().resource_table.close(cancel_rid).ok();
};
return read_future_rv??;
} }
}
fut.await? read_future.await?
} }
// Like String::from_utf8_lossy but operates on owned values // Like String::from_utf8_lossy but operates on owned values