1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
extern crate cfg_if;
#[cfg(windows)]
extern crate reqwest;

use std::process::Command;

use cfg_if::cfg_if;

cfg_if! {
  if #[cfg(windows)] {
    use std::env::var;
    use std::fs::File;
    use std::io::copy;
    use std::path::PathBuf;

    pub fn setup() {
      let node_full_version =
        String::from_utf8(Command::new("node").arg("-v").output().unwrap().stdout).unwrap();
      let node_exec_path = String::from_utf8(
        Command::new("node")
          .arg("-e")
          .arg("console.log(process.execPath)")
          .output()
          .unwrap()
          .stdout,
      )
      .unwrap();
      let node_exec_path = node_exec_path.trim_end();
      let mut node_lib_file_dir = PathBuf::from(node_exec_path)
        .parent()
        .unwrap()
        .to_path_buf();
      let node_lib_dir = PathBuf::from(&node_lib_file_dir);
      node_lib_file_dir.push(format!("node-{}.lib", node_full_version.trim_end()));
      if !node_lib_file_dir.exists() {
        let lib_file_download_url = format!(
          "https://nodejs.org/dist/{}/win-x64/node.lib",
          node_full_version
        );
        let mut resp =
          reqwest::blocking::get(&lib_file_download_url).expect("Download node.lib file failed");
        let mut node_lib_file = File::create(&node_lib_file_dir).unwrap();
        copy(&mut resp, &mut node_lib_file).expect("Save node.lib file failed");
      }
      println!(
        "cargo:rustc-link-lib={}",
        &node_lib_file_dir.file_stem().unwrap().to_str().unwrap()
      );
      println!("cargo:rustc-link-search={}", node_lib_dir.to_str().unwrap());
      // Link `win_delay_load_hook.obj` for windows electron
      let node_runtime_env = "npm_config_runtime";
      println!("cargo:rerun-if-env-changed={}", node_runtime_env);
      if var(node_runtime_env).map(|s| s == "electron") == Ok(true) {
        println!("cargo:rustc-cdylib-link-arg=win_delay_load_hook.obj");
        println!("cargo:rustc-cdylib-link-arg=delayimp.lib");
        println!("cargo:rustc-cdylib-link-arg=/DELAYLOAD:node.exe");
      }
      setup_napi_feature();
    }
  } else if #[cfg(target_os = "macos")] {
    /// Set up the build environment by setting Cargo configuration variables.
    pub fn setup() {
      println!("cargo:rustc-cdylib-link-arg=-undefined");
      println!("cargo:rustc-cdylib-link-arg=dynamic_lookup");
      setup_napi_feature();
    }
  } else {
    pub fn setup() {
      setup_napi_feature();
    }
  }
}

fn setup_napi_feature() {
  let napi_version = String::from_utf8(
    Command::new("node")
      .args(&["-e", "console.log(process.versions.napi)"])
      .output()
      .unwrap()
      .stdout,
  )
  .expect("Get NAPI version failed");

  let napi_version_number = napi_version.trim().parse::<u32>().unwrap();

  if napi_version_number < 2 {
    panic!("current napi version is too low");
  }

  if napi_version_number == 2 {
    println!("cargo:rustc-cfg=napi{}", napi_version_number);
  } else {
    for version in 2..(napi_version_number + 1) {
      println!("cargo:rustc-cfg=napi{}", version);
    }
  }
}