mirror of
https://github.com/denoland/deno.git
synced 2025-01-13 01:22:20 -05:00
feat: Deno.fsEvents() (#3452)
This commit is contained in:
parent
754b8c65ad
commit
bd640bc7e6
12 changed files with 1355 additions and 943 deletions
2036
Cargo.lock
generated
2036
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -43,6 +43,7 @@ indexmap = "1.3.0"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
libc = "0.2.66"
|
libc = "0.2.66"
|
||||||
log = "0.4.8"
|
log = "0.4.8"
|
||||||
|
notify = { version = "5.0.0-pre.2" }
|
||||||
rand = "0.7.2"
|
rand = "0.7.2"
|
||||||
regex = "1.3.1"
|
regex = "1.3.1"
|
||||||
remove_dir_all = "0.5.2"
|
remove_dir_all = "0.5.2"
|
||||||
|
@ -53,7 +54,7 @@ serde = { version = "1.0.104", features = ["derive"] }
|
||||||
serde_derive = "1.0.104"
|
serde_derive = "1.0.104"
|
||||||
serde_json = { version = "1.0.44", features = [ "preserve_order" ] }
|
serde_json = { version = "1.0.44", features = [ "preserve_order" ] }
|
||||||
source-map-mappings = "0.5.0"
|
source-map-mappings = "0.5.0"
|
||||||
sys-info = "0.5.8"
|
sys-info = "=0.5.8" # 0.5.9 seems to be broken on windows.
|
||||||
tempfile = "3.1.0"
|
tempfile = "3.1.0"
|
||||||
termcolor = "1.0.5"
|
termcolor = "1.0.5"
|
||||||
tokio = { version = "0.2", features = ["rt-core", "tcp", "udp", "process", "fs", "blocking", "sync", "io-std", "macros", "time"] }
|
tokio = { version = "0.2", features = ["rt-core", "tcp", "udp", "process", "fs", "blocking", "sync", "io-std", "macros", "time"] }
|
||||||
|
|
|
@ -43,6 +43,7 @@ export {
|
||||||
OpenOptions,
|
OpenOptions,
|
||||||
OpenMode
|
OpenMode
|
||||||
} from "./files.ts";
|
} from "./files.ts";
|
||||||
|
export { FsEvent, fsEvents } from "./fs_events.ts";
|
||||||
export {
|
export {
|
||||||
EOF,
|
EOF,
|
||||||
copy,
|
copy,
|
||||||
|
|
|
@ -73,6 +73,8 @@ export let OP_CWD: number;
|
||||||
export let OP_CONNECT_TLS: number;
|
export let OP_CONNECT_TLS: number;
|
||||||
export let OP_HOSTNAME: number;
|
export let OP_HOSTNAME: number;
|
||||||
export let OP_OPEN_PLUGIN: number;
|
export let OP_OPEN_PLUGIN: number;
|
||||||
|
export let OP_FS_EVENTS_OPEN: number;
|
||||||
|
export let OP_FS_EVENTS_POLL: number;
|
||||||
export let OP_COMPILE: number;
|
export let OP_COMPILE: number;
|
||||||
export let OP_TRANSPILE: number;
|
export let OP_TRANSPILE: number;
|
||||||
export let OP_SIGNAL_BIND: number;
|
export let OP_SIGNAL_BIND: number;
|
||||||
|
|
40
cli/js/fs_events.ts
Normal file
40
cli/js/fs_events.ts
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
// Copyright 2019 the Deno authors. All rights reserved. MIT license.
|
||||||
|
import { sendSync, sendAsync } from "./dispatch_json.ts";
|
||||||
|
import * as dispatch from "./dispatch.ts";
|
||||||
|
import { close } from "./files.ts";
|
||||||
|
|
||||||
|
export interface FsEvent {
|
||||||
|
kind: "any" | "access" | "create" | "modify" | "remove";
|
||||||
|
paths: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
class FsEvents implements AsyncIterableIterator<FsEvent> {
|
||||||
|
readonly rid: number;
|
||||||
|
|
||||||
|
constructor(paths: string[], options: { recursive: boolean }) {
|
||||||
|
const { recursive } = options;
|
||||||
|
this.rid = sendSync(dispatch.OP_FS_EVENTS_OPEN, { recursive, paths });
|
||||||
|
}
|
||||||
|
|
||||||
|
async next(): Promise<IteratorResult<FsEvent>> {
|
||||||
|
return await sendAsync(dispatch.OP_FS_EVENTS_POLL, {
|
||||||
|
rid: this.rid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async return(value?: FsEvent): Promise<IteratorResult<FsEvent>> {
|
||||||
|
close(this.rid);
|
||||||
|
return { value, done: true };
|
||||||
|
}
|
||||||
|
|
||||||
|
[Symbol.asyncIterator](): AsyncIterableIterator<FsEvent> {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fsEvents(
|
||||||
|
paths: string | string[],
|
||||||
|
options = { recursive: true }
|
||||||
|
): AsyncIterableIterator<FsEvent> {
|
||||||
|
return new FsEvents(Array.isArray(paths) ? paths : [paths], options);
|
||||||
|
}
|
52
cli/js/fs_events_test.ts
Normal file
52
cli/js/fs_events_test.ts
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
||||||
|
import { testPerm, assert } from "./test_util.ts";
|
||||||
|
|
||||||
|
// TODO(ry) Add more tests to specify format.
|
||||||
|
|
||||||
|
testPerm({ read: false }, function fsEventsPermissions() {
|
||||||
|
let thrown = false;
|
||||||
|
try {
|
||||||
|
Deno.fsEvents(".");
|
||||||
|
} catch (err) {
|
||||||
|
assert(err instanceof Deno.Err.PermissionDenied);
|
||||||
|
thrown = true;
|
||||||
|
}
|
||||||
|
assert(thrown);
|
||||||
|
});
|
||||||
|
|
||||||
|
async function getTwoEvents(
|
||||||
|
iter: AsyncIterableIterator<Deno.FsEvent>
|
||||||
|
): Promise<Deno.FsEvent[]> {
|
||||||
|
const events = [];
|
||||||
|
for await (const event of iter) {
|
||||||
|
console.log(">>>> event", event);
|
||||||
|
events.push(event);
|
||||||
|
if (events.length > 2) break;
|
||||||
|
}
|
||||||
|
return events;
|
||||||
|
}
|
||||||
|
|
||||||
|
testPerm({ read: true, write: true }, async function fsEventsBasic(): Promise<
|
||||||
|
void
|
||||||
|
> {
|
||||||
|
const testDir = await Deno.makeTempDir();
|
||||||
|
const iter = Deno.fsEvents(testDir);
|
||||||
|
|
||||||
|
// Asynchornously capture two fs events.
|
||||||
|
const eventsPromise = getTwoEvents(iter);
|
||||||
|
|
||||||
|
// Make some random file system activity.
|
||||||
|
const file1 = testDir + "/file1.txt";
|
||||||
|
const file2 = testDir + "/file2.txt";
|
||||||
|
Deno.writeFileSync(file1, new Uint8Array([0, 1, 2]));
|
||||||
|
Deno.writeFileSync(file2, new Uint8Array([0, 1, 2]));
|
||||||
|
|
||||||
|
// We should have gotten two fs events.
|
||||||
|
const events = await eventsPromise;
|
||||||
|
console.log("events", events);
|
||||||
|
assert(events.length >= 2);
|
||||||
|
assert(events[0].kind == "create");
|
||||||
|
assert(events[0].paths[0].includes(testDir));
|
||||||
|
assert(events[1].kind == "create" || events[1].kind == "modify");
|
||||||
|
assert(events[1].paths[0].includes(testDir));
|
||||||
|
});
|
15
cli/js/lib.deno.ns.d.ts
vendored
15
cli/js/lib.deno.ns.d.ts
vendored
|
@ -1620,6 +1620,21 @@ declare namespace Deno {
|
||||||
*/
|
*/
|
||||||
export function resources(): ResourceMap;
|
export function resources(): ResourceMap;
|
||||||
|
|
||||||
|
/** UNSTABLE: new API. Needs docs. */
|
||||||
|
export interface FsEvent {
|
||||||
|
kind: "any" | "access" | "create" | "modify" | "remove";
|
||||||
|
paths: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/** UNSTABLE: new API. Needs docs.
|
||||||
|
*
|
||||||
|
* recursive option is true by default.
|
||||||
|
*/
|
||||||
|
export function fsEvents(
|
||||||
|
paths: string | string[],
|
||||||
|
options?: { recursive: boolean }
|
||||||
|
): AsyncIterableIterator<FsEvent>;
|
||||||
|
|
||||||
/** How to handle subprocess stdio.
|
/** How to handle subprocess stdio.
|
||||||
*
|
*
|
||||||
* "inherit" The default if unspecified. The child inherits from the
|
* "inherit" The default if unspecified. The child inherits from the
|
||||||
|
|
|
@ -23,6 +23,7 @@ import "./fetch_test.ts";
|
||||||
import "./file_test.ts";
|
import "./file_test.ts";
|
||||||
import "./files_test.ts";
|
import "./files_test.ts";
|
||||||
import "./form_data_test.ts";
|
import "./form_data_test.ts";
|
||||||
|
import "./fs_events_test.ts";
|
||||||
import "./get_random_values_test.ts";
|
import "./get_random_values_test.ts";
|
||||||
import "./globals_test.ts";
|
import "./globals_test.ts";
|
||||||
import "./headers_test.ts";
|
import "./headers_test.ts";
|
||||||
|
|
129
cli/ops/fs_events.rs
Normal file
129
cli/ops/fs_events.rs
Normal file
|
@ -0,0 +1,129 @@
|
||||||
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
||||||
|
use super::dispatch_json::{Deserialize, JsonOp, Value};
|
||||||
|
use crate::deno_error::bad_resource;
|
||||||
|
use crate::ops::json_op;
|
||||||
|
use crate::state::State;
|
||||||
|
use deno_core::*;
|
||||||
|
use futures::future::poll_fn;
|
||||||
|
use futures::future::FutureExt;
|
||||||
|
use notify::event::Event as NotifyEvent;
|
||||||
|
use notify::Error as NotifyError;
|
||||||
|
use notify::EventKind;
|
||||||
|
use notify::RecommendedWatcher;
|
||||||
|
use notify::RecursiveMode;
|
||||||
|
use notify::Watcher;
|
||||||
|
use serde::Serialize;
|
||||||
|
use std::convert::From;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
use tokio::sync::mpsc;
|
||||||
|
|
||||||
|
pub fn init(i: &mut Isolate, s: &State) {
|
||||||
|
i.register_op(
|
||||||
|
"fs_events_open",
|
||||||
|
s.core_op(json_op(s.stateful_op(op_fs_events_open))),
|
||||||
|
);
|
||||||
|
i.register_op(
|
||||||
|
"fs_events_poll",
|
||||||
|
s.core_op(json_op(s.stateful_op(op_fs_events_poll))),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct FsEventsResource {
|
||||||
|
#[allow(unused)]
|
||||||
|
watcher: RecommendedWatcher,
|
||||||
|
receiver: mpsc::Receiver<Result<FsEvent, ErrBox>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Represents a file system event.
|
||||||
|
///
|
||||||
|
/// We do not use the event directly from the notify crate. We flatten
|
||||||
|
/// the structure into this simpler structure. We want to only make it more
|
||||||
|
/// complex as needed.
|
||||||
|
///
|
||||||
|
/// Feel free to expand this struct as long as you can add tests to demonstrate
|
||||||
|
/// the complexity.
|
||||||
|
#[derive(Serialize, Debug)]
|
||||||
|
struct FsEvent {
|
||||||
|
kind: String,
|
||||||
|
paths: Vec<PathBuf>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<NotifyEvent> for FsEvent {
|
||||||
|
fn from(e: NotifyEvent) -> Self {
|
||||||
|
let kind = match e.kind {
|
||||||
|
EventKind::Any => "any",
|
||||||
|
EventKind::Access(_) => "access",
|
||||||
|
EventKind::Create(_) => "create",
|
||||||
|
EventKind::Modify(_) => "modify",
|
||||||
|
EventKind::Remove(_) => "remove",
|
||||||
|
EventKind::Other => todo!(), // What's this for? Leaving it out for now.
|
||||||
|
}
|
||||||
|
.to_string();
|
||||||
|
FsEvent {
|
||||||
|
kind,
|
||||||
|
paths: e.paths,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn op_fs_events_open(
|
||||||
|
state: &State,
|
||||||
|
args: Value,
|
||||||
|
_zero_copy: Option<ZeroCopyBuf>,
|
||||||
|
) -> Result<JsonOp, ErrBox> {
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct OpenArgs {
|
||||||
|
recursive: bool,
|
||||||
|
paths: Vec<String>,
|
||||||
|
}
|
||||||
|
let args: OpenArgs = serde_json::from_value(args)?;
|
||||||
|
let (sender, receiver) = mpsc::channel::<Result<FsEvent, ErrBox>>(16);
|
||||||
|
let sender = std::sync::Mutex::new(sender);
|
||||||
|
let mut watcher: RecommendedWatcher =
|
||||||
|
Watcher::new_immediate(move |res: Result<NotifyEvent, NotifyError>| {
|
||||||
|
let res2 = res.map(FsEvent::from).map_err(ErrBox::from);
|
||||||
|
let mut sender = sender.lock().unwrap();
|
||||||
|
futures::executor::block_on(sender.send(res2)).expect("fs events error");
|
||||||
|
})?;
|
||||||
|
let recursive_mode = if args.recursive {
|
||||||
|
RecursiveMode::Recursive
|
||||||
|
} else {
|
||||||
|
RecursiveMode::NonRecursive
|
||||||
|
};
|
||||||
|
for path in &args.paths {
|
||||||
|
state.check_read(&PathBuf::from(path))?;
|
||||||
|
watcher.watch(path, recursive_mode)?;
|
||||||
|
}
|
||||||
|
let resource = FsEventsResource { watcher, receiver };
|
||||||
|
let table = &mut state.borrow_mut().resource_table;
|
||||||
|
let rid = table.add("fsEvents", Box::new(resource));
|
||||||
|
Ok(JsonOp::Sync(json!(rid)))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn op_fs_events_poll(
|
||||||
|
state: &State,
|
||||||
|
args: Value,
|
||||||
|
_zero_copy: Option<ZeroCopyBuf>,
|
||||||
|
) -> Result<JsonOp, ErrBox> {
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct PollArgs {
|
||||||
|
rid: u32,
|
||||||
|
}
|
||||||
|
let PollArgs { rid } = serde_json::from_value(args)?;
|
||||||
|
let state = state.clone();
|
||||||
|
let f = poll_fn(move |cx| {
|
||||||
|
let resource_table = &mut state.borrow_mut().resource_table;
|
||||||
|
let watcher = resource_table
|
||||||
|
.get_mut::<FsEventsResource>(rid)
|
||||||
|
.ok_or_else(bad_resource)?;
|
||||||
|
watcher
|
||||||
|
.receiver
|
||||||
|
.poll_recv(cx)
|
||||||
|
.map(|maybe_result| match maybe_result {
|
||||||
|
Some(Ok(value)) => Ok(json!({ "value": value, "done": false })),
|
||||||
|
Some(Err(err)) => Err(err),
|
||||||
|
None => Ok(json!({ "done": true })),
|
||||||
|
})
|
||||||
|
});
|
||||||
|
Ok(JsonOp::Async(f.boxed_local()))
|
||||||
|
}
|
|
@ -13,6 +13,7 @@ pub mod errors;
|
||||||
pub mod fetch;
|
pub mod fetch;
|
||||||
pub mod files;
|
pub mod files;
|
||||||
pub mod fs;
|
pub mod fs;
|
||||||
|
pub mod fs_events;
|
||||||
pub mod io;
|
pub mod io;
|
||||||
pub mod net;
|
pub mod net;
|
||||||
pub mod os;
|
pub mod os;
|
||||||
|
|
|
@ -206,6 +206,7 @@ impl MainWorker {
|
||||||
ops::fetch::init(isolate, &state);
|
ops::fetch::init(isolate, &state);
|
||||||
ops::files::init(isolate, &state);
|
ops::files::init(isolate, &state);
|
||||||
ops::fs::init(isolate, &state);
|
ops::fs::init(isolate, &state);
|
||||||
|
ops::fs_events::init(isolate, &state);
|
||||||
ops::io::init(isolate, &state);
|
ops::io::init(isolate, &state);
|
||||||
ops::plugins::init(isolate, &state, op_registry);
|
ops::plugins::init(isolate, &state, op_registry);
|
||||||
ops::net::init(isolate, &state);
|
ops::net::init(isolate, &state);
|
||||||
|
|
|
@ -467,6 +467,23 @@ for await (const _ of sig) {
|
||||||
|
|
||||||
The above for-await loop exits after 5 seconds when sig.dispose() is called.
|
The above for-await loop exits after 5 seconds when sig.dispose() is called.
|
||||||
|
|
||||||
|
### File system events
|
||||||
|
|
||||||
|
To poll for file system events:
|
||||||
|
|
||||||
|
```ts
|
||||||
|
const iter = Deno.fsEvents("/");
|
||||||
|
for await (const event of iter) {
|
||||||
|
console.log(">>>> event", event);
|
||||||
|
// { kind: "create", paths: [ "/foo.txt" ] }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that the exact ordering of the events can vary between operating systems.
|
||||||
|
This feature uses different syscalls depending on the platform:
|
||||||
|
|
||||||
|
Linux: inotify macOS: FSEvents Windows: ReadDirectoryChangesW
|
||||||
|
|
||||||
### Linking to third party code
|
### Linking to third party code
|
||||||
|
|
||||||
In the above examples, we saw that Deno could execute scripts from URLs. Like
|
In the above examples, we saw that Deno could execute scripts from URLs. Like
|
||||||
|
|
Loading…
Reference in a new issue