worker.rs 11.3 KB
Newer Older
R
Ryan Dahl 已提交
1
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
2
use crate::fmt_errors::JSError;
3
use crate::ops;
4
use crate::state::ThreadSafeState;
5
use deno;
6
use deno::ErrBox;
7
use deno::ModuleSpecifier;
B
Bert Belder 已提交
8
use deno::RecursiveLoad;
R
Ryan Dahl 已提交
9
use deno::StartupData;
10
use futures::Async;
R
Ryan Dahl 已提交
11
use futures::Future;
12
use std::env;
13 14
use std::sync::Arc;
use std::sync::Mutex;
15
use url::Url;
B
Bartek Iwańczuk 已提交
16

17
/// Wraps deno::Isolate to provide source maps, ops for the CLI, and
18
/// high-level module loading
19
#[derive(Clone)]
20
pub struct Worker {
21
  isolate: Arc<Mutex<deno::Isolate>>,
22
  pub state: ThreadSafeState,
23 24
}

25 26 27 28
impl Worker {
  pub fn new(
    _name: String,
    startup_data: StartupData,
29
    state: ThreadSafeState,
30
  ) -> Worker {
31 32 33
    let isolate = Arc::new(Mutex::new(deno::Isolate::new(startup_data, false)));
    {
      let mut i = isolate.lock().unwrap();
34

35 36 37 38 39 40 41
      ops::compiler::init(&mut i, &state);
      ops::errors::init(&mut i, &state);
      ops::fetch::init(&mut i, &state);
      ops::files::init(&mut i, &state);
      ops::fs::init(&mut i, &state);
      ops::io::init(&mut i, &state);
      ops::net::init(&mut i, &state);
42
      ops::tls::init(&mut i, &state);
43 44 45 46 47 48 49 50
      ops::os::init(&mut i, &state);
      ops::permissions::init(&mut i, &state);
      ops::process::init(&mut i, &state);
      ops::random::init(&mut i, &state);
      ops::repl::init(&mut i, &state);
      ops::resources::init(&mut i, &state);
      ops::timers::init(&mut i, &state);
      ops::workers::init(&mut i, &state);
B
Bert Belder 已提交
51 52 53 54 55 56 57 58 59 60 61 62 63

      let state_ = state.clone();
      i.set_dyn_import(move |id, specifier, referrer| {
        let load_stream = RecursiveLoad::dynamic_import(
          id,
          specifier,
          referrer,
          state_.clone(),
          state_.modules.clone(),
        );
        Box::new(load_stream)
      });

64 65
      let state_ = state.clone();
      i.set_js_error_create(move |v8_exception| {
B
Bartek Iwańczuk 已提交
66
        JSError::from_v8_exception(v8_exception, &state_.ts_compiler)
67
      })
68
    }
69
    Self { isolate, state }
70 71
  }

72
  /// Same as execute2() but the filename defaults to "$CWD/__anonymous__".
73
  pub fn execute(&mut self, js_source: &str) -> Result<(), ErrBox> {
74 75 76
    let path = env::current_dir().unwrap().join("__anonymous__");
    let url = Url::from_file_path(path).unwrap();
    self.execute2(url.as_str(), js_source)
77 78 79 80 81
  }

  /// Executes the provided JavaScript source code. The js_filename argument is
  /// provided only for debugging purposes.
  pub fn execute2(
82
    &mut self,
83 84
    js_filename: &str,
    js_source: &str,
85
  ) -> Result<(), ErrBox> {
86
    let mut isolate = self.isolate.lock().unwrap();
87
    isolate.execute(js_filename, js_source)
88 89
  }

90
  /// Executes the provided JavaScript module.
91
  pub fn execute_mod_async(
92
    &mut self,
93
    module_specifier: &ModuleSpecifier,
94
    maybe_code: Option<String>,
95
    is_prefetch: bool,
96
  ) -> impl Future<Item = (), Error = ErrBox> {
97 98
    let worker = self.clone();
    let loader = self.state.clone();
99
    let isolate = self.isolate.clone();
100
    let modules = self.state.modules.clone();
101 102 103 104 105 106 107
    let recursive_load = RecursiveLoad::main(
      &module_specifier.to_string(),
      maybe_code,
      loader,
      modules,
    )
    .get_future(isolate);
108 109 110 111 112 113 114 115 116
    recursive_load.and_then(move |id| -> Result<(), ErrBox> {
      worker.state.progress.done();
      if is_prefetch {
        Ok(())
      } else {
        let mut isolate = worker.isolate.lock().unwrap();
        isolate.mod_evaluate(id)
      }
    })
117
  }
118
}
119

120
impl Future for Worker {
121
  type Item = ();
122
  type Error = ErrBox;
123

124
  fn poll(&mut self) -> Result<Async<()>, ErrBox> {
125
    let mut isolate = self.isolate.lock().unwrap();
126
    isolate.poll()
127 128
  }
}
129

R
Ryan Dahl 已提交
130 131 132
#[cfg(test)]
mod tests {
  use super::*;
133
  use crate::flags;
R
Ryan Dahl 已提交
134
  use crate::progress::Progress;
135 136
  use crate::resources;
  use crate::startup_data;
137
  use crate::state::ThreadSafeState;
138
  use crate::tokio_util;
139 140
  use futures::future::lazy;
  use std::sync::atomic::Ordering;
141 142

  #[test]
143
  fn execute_mod_esm_imports_a() {
144 145 146 147 148
    let p = std::path::PathBuf::from(env!("CARGO_MANIFEST_DIR"))
      .parent()
      .unwrap()
      .join("tests/esm_imports_a.js")
      .to_owned();
149
    let module_specifier =
150
      ModuleSpecifier::resolve_url_or_path(&p.to_string_lossy()).unwrap();
151
    let argv = vec![String::from("./deno"), module_specifier.to_string()];
R
Ryan Dahl 已提交
152 153 154 155
    let state = ThreadSafeState::new(
      flags::DenoFlags::default(),
      argv,
      Progress::new(),
156
      true,
157 158
    )
    .unwrap();
159 160
    let state_ = state.clone();
    tokio_util::run(lazy(move || {
161 162
      let mut worker =
        Worker::new("TEST".to_string(), StartupData::None, state);
163
      worker
164
        .execute_mod_async(&module_specifier, None, false)
165 166 167 168 169 170
        .then(|result| {
          if let Err(err) = result {
            eprintln!("execute_mod err {:?}", err);
          }
          tokio_util::panic_on_error(worker)
        })
171 172 173
    }));

    let metrics = &state_.metrics;
174
    assert_eq!(metrics.resolve_count.load(Ordering::SeqCst), 2);
R
Ryan Dahl 已提交
175 176
    // Check that we didn't start the compiler.
    assert_eq!(metrics.compiler_starts.load(Ordering::SeqCst), 0);
177 178 179 180
  }

  #[test]
  fn execute_mod_circular() {
181 182 183 184 185
    let p = std::path::PathBuf::from(env!("CARGO_MANIFEST_DIR"))
      .parent()
      .unwrap()
      .join("tests/circular1.ts")
      .to_owned();
186
    let module_specifier =
187 188
      ModuleSpecifier::resolve_url_or_path(&p.to_string_lossy()).unwrap();
    let argv = vec![String::from("deno"), module_specifier.to_string()];
R
Ryan Dahl 已提交
189 190 191 192
    let state = ThreadSafeState::new(
      flags::DenoFlags::default(),
      argv,
      Progress::new(),
193
      true,
194 195
    )
    .unwrap();
196 197
    let state_ = state.clone();
    tokio_util::run(lazy(move || {
198 199
      let mut worker =
        Worker::new("TEST".to_string(), StartupData::None, state);
200
      worker
201
        .execute_mod_async(&module_specifier, None, false)
202 203 204 205 206 207
        .then(|result| {
          if let Err(err) = result {
            eprintln!("execute_mod err {:?}", err);
          }
          tokio_util::panic_on_error(worker)
        })
208 209 210
    }));

    let metrics = &state_.metrics;
211
    // TODO  assert_eq!(metrics.resolve_count.load(Ordering::SeqCst), 2);
R
Ryan Dahl 已提交
212 213
    // Check that we didn't start the compiler.
    assert_eq!(metrics.compiler_starts.load(Ordering::SeqCst), 0);
214
  }
215

R
Ryan Dahl 已提交
216 217
  #[test]
  fn execute_006_url_imports() {
218
    let http_server_guard = crate::test_util::http_server();
219

220 221 222
    let p = std::path::PathBuf::from(env!("CARGO_MANIFEST_DIR"))
      .parent()
      .unwrap()
223
      .join("cli/tests/006_url_imports.ts")
224
      .to_owned();
225
    let module_specifier =
226
      ModuleSpecifier::resolve_url_or_path(&p.to_string_lossy()).unwrap();
227
    let argv = vec![String::from("deno"), module_specifier.to_string()];
R
Ryan Dahl 已提交
228 229 230
    let mut flags = flags::DenoFlags::default();
    flags.reload = true;
    let state =
R
Ryan Dahl 已提交
231
      ThreadSafeState::new(flags, argv, Progress::new(), true).unwrap();
R
Ryan Dahl 已提交
232 233 234 235 236 237 238
    let state_ = state.clone();
    tokio_util::run(lazy(move || {
      let mut worker = Worker::new(
        "TEST".to_string(),
        startup_data::deno_isolate_init(),
        state,
      );
239
      worker.execute("denoMain()").unwrap();
240
      worker
241
        .execute_mod_async(&module_specifier, None, false)
242 243 244 245 246 247
        .then(|result| {
          if let Err(err) = result {
            eprintln!("execute_mod err {:?}", err);
          }
          tokio_util::panic_on_error(worker)
        })
R
Ryan Dahl 已提交
248 249 250 251
    }));

    let metrics = &state_.metrics;
    assert_eq!(metrics.resolve_count.load(Ordering::SeqCst), 3);
R
Ryan Dahl 已提交
252 253
    // Check that we've only invoked the compiler once.
    assert_eq!(metrics.compiler_starts.load(Ordering::SeqCst), 1);
254
    drop(http_server_guard);
R
Ryan Dahl 已提交
255 256
  }

257
  fn create_test_worker() -> Worker {
K
Kitson Kelly 已提交
258 259 260 261
    let state = ThreadSafeState::mock(vec![
      String::from("./deno"),
      String::from("hello.js"),
    ]);
262
    let mut worker =
263
      Worker::new("TEST".to_string(), startup_data::deno_isolate_init(), state);
264 265
    worker.execute("denoMain()").unwrap();
    worker.execute("workerMain()").unwrap();
266 267 268 269 270
    worker
  }

  #[test]
  fn test_worker_messages() {
271
    tokio_util::run_in_task(|| {
272 273 274 275 276
      let mut worker = create_test_worker();
      let source = r#"
        onmessage = function(e) {
          console.log("msg from main script", e.data);
          if (e.data == "exit") {
277
            delete window.onmessage;
278 279 280 281 282 283 284 285
            return;
          } else {
            console.assert(e.data === "hi");
          }
          postMessage([1, 2, 3]);
          console.log("after postMessage");
        }
        "#;
286
      worker.execute(source).unwrap();
287 288 289 290 291 292 293

      let resource = worker.state.resource.clone();
      let resource_ = resource.clone();

      tokio::spawn(lazy(move || {
        worker.then(move |r| -> Result<(), ()> {
          resource_.close();
294
          r.unwrap();
295 296 297 298 299 300
          Ok(())
        })
      }));

      let msg = json!("hi").to_string().into_boxed_str().into_boxed_bytes();

301 302 303
      let r = resources::post_message_to_worker(resource.rid, msg)
        .expect("Bad resource")
        .wait();
304 305 306 307 308 309 310 311 312 313 314 315 316
      assert!(r.is_ok());

      let maybe_msg = resources::get_message_from_worker(resource.rid)
        .wait()
        .unwrap();
      assert!(maybe_msg.is_some());
      // Check if message received is [1, 2, 3] in json
      assert_eq!(*maybe_msg.unwrap(), *b"[1,2,3]");

      let msg = json!("exit")
        .to_string()
        .into_boxed_str()
        .into_boxed_bytes();
317 318 319
      let r = resources::post_message_to_worker(resource.rid, msg)
        .expect("Bad resource")
        .wait();
320 321 322 323 324 325
      assert!(r.is_ok());
    })
  }

  #[test]
  fn removed_from_resource_table_on_close() {
326
    tokio_util::run_in_task(|| {
327
      let mut worker = create_test_worker();
328 329 330
      worker
        .execute("onmessage = () => { delete window.onmessage; }")
        .unwrap();
331 332 333 334

      let resource = worker.state.resource.clone();
      let rid = resource.rid;

335 336
      let worker_future = worker
        .then(move |r| -> Result<(), ()> {
337 338
          resource.close();
          println!("workers.rs after resource close");
339
          r.unwrap();
340
          Ok(())
341 342
        })
        .shared();
343 344 345

      let worker_future_ = worker_future.clone();
      tokio::spawn(lazy(move || worker_future_.then(|_| Ok(()))));
346 347

      let msg = json!("hi").to_string().into_boxed_str().into_boxed_bytes();
348 349 350
      let r = resources::post_message_to_worker(rid, msg)
        .expect("Bad resource")
        .wait();
351 352 353
      assert!(r.is_ok());
      debug!("rid {:?}", rid);

354
      worker_future.wait().unwrap();
355 356
    })
  }
357 358 359

  #[test]
  fn execute_mod_resolve_error() {
360
    tokio_util::run_in_task(|| {
K
Kitson Kelly 已提交
361
      // "foo" is not a valid module specifier so this should return an error.
362
      let mut worker = create_test_worker();
363
      let module_specifier =
364
        ModuleSpecifier::resolve_url_or_path("does-not-exist").unwrap();
365 366 367
      let result = worker
        .execute_mod_async(&module_specifier, None, false)
        .wait();
R
Ryan Dahl 已提交
368 369
      assert!(result.is_err());
    })
370 371 372 373
  }

  #[test]
  fn execute_mod_002_hello() {
374
    tokio_util::run_in_task(|| {
R
Ryan Dahl 已提交
375 376
      // This assumes cwd is project root (an assumption made throughout the
      // tests).
377
      let mut worker = create_test_worker();
378 379 380 381 382
      let p = std::path::PathBuf::from(env!("CARGO_MANIFEST_DIR"))
        .parent()
        .unwrap()
        .join("tests/002_hello.ts")
        .to_owned();
383
      let module_specifier =
384
        ModuleSpecifier::resolve_url_or_path(&p.to_string_lossy()).unwrap();
385 386 387
      let result = worker
        .execute_mod_async(&module_specifier, None, false)
        .wait();
R
Ryan Dahl 已提交
388 389
      assert!(result.is_ok());
    })
390
  }
391
}