|
@@ -31,93 +31,103 @@ async fn setup_file_watcher<F: Fn() -> Result<BuildResult> + Send + 'static>(
|
|
|
let mut watcher = notify::recommended_watcher(move |info: notify::Result<notify::Event>| {
|
|
|
let config = watcher_config.clone();
|
|
|
if let Ok(e) = info {
|
|
|
- if chrono::Local::now().timestamp() > last_update_time {
|
|
|
- let mut needs_full_rebuild;
|
|
|
- if let Some(hot_reload) = &hot_reload {
|
|
|
- // find changes to the rsx in the file
|
|
|
- let mut rsx_file_map = hot_reload.file_map.lock().unwrap();
|
|
|
- let mut messages: Vec<Template<'static>> = Vec::new();
|
|
|
-
|
|
|
- // In hot reload mode, we only need to rebuild if non-rsx code is changed
|
|
|
- needs_full_rebuild = false;
|
|
|
-
|
|
|
- for path in &e.paths {
|
|
|
- // if this is not a rust file, rebuild the whole project
|
|
|
- if path.extension().and_then(|p| p.to_str()) != Some("rs") {
|
|
|
- needs_full_rebuild = true;
|
|
|
- break;
|
|
|
- }
|
|
|
-
|
|
|
- // Workaround for notify and vscode-like editor:
|
|
|
- // when edit & save a file in vscode, there will be two notifications,
|
|
|
- // the first one is a file with empty content.
|
|
|
- // filter the empty file notification to avoid false rebuild during hot-reload
|
|
|
- if let Ok(metadata) = fs::metadata(path) {
|
|
|
- if metadata.len() == 0 {
|
|
|
- continue;
|
|
|
+ match e.kind {
|
|
|
+ notify::EventKind::Create(_)
|
|
|
+ | notify::EventKind::Remove(_)
|
|
|
+ | notify::EventKind::Modify(_) => {
|
|
|
+ if chrono::Local::now().timestamp() > last_update_time {
|
|
|
+ let mut needs_full_rebuild;
|
|
|
+ if let Some(hot_reload) = &hot_reload {
|
|
|
+ // find changes to the rsx in the file
|
|
|
+ let mut rsx_file_map = hot_reload.file_map.lock().unwrap();
|
|
|
+ let mut messages: Vec<Template<'static>> = Vec::new();
|
|
|
+
|
|
|
+ // In hot reload mode, we only need to rebuild if non-rsx code is changed
|
|
|
+ needs_full_rebuild = false;
|
|
|
+
|
|
|
+ for path in &e.paths {
|
|
|
+ // if this is not a rust file, rebuild the whole project
|
|
|
+ if path.extension().and_then(|p| p.to_str()) != Some("rs") {
|
|
|
+ needs_full_rebuild = true;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Workaround for notify and vscode-like editor:
|
|
|
+ // when edit & save a file in vscode, there will be two notifications,
|
|
|
+ // the first one is a file with empty content.
|
|
|
+ // filter the empty file notification to avoid false rebuild during hot-reload
|
|
|
+ if let Ok(metadata) = fs::metadata(path) {
|
|
|
+ if metadata.len() == 0 {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ match rsx_file_map.update_rsx(path, &config.crate_dir) {
|
|
|
+ Ok(UpdateResult::UpdatedRsx(msgs)) => {
|
|
|
+ messages.extend(msgs);
|
|
|
+ needs_full_rebuild = false;
|
|
|
+ }
|
|
|
+ Ok(UpdateResult::NeedsRebuild) => {
|
|
|
+ needs_full_rebuild = true;
|
|
|
+ }
|
|
|
+ Err(err) => {
|
|
|
+ log::error!("{}", err);
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- match rsx_file_map.update_rsx(path, &config.crate_dir) {
|
|
|
- Ok(UpdateResult::UpdatedRsx(msgs)) => {
|
|
|
- messages.extend(msgs);
|
|
|
- needs_full_rebuild = false;
|
|
|
+ if needs_full_rebuild {
|
|
|
+ // Reset the file map to the new state of the project
|
|
|
+ let FileMapBuildResult {
|
|
|
+ map: new_file_map,
|
|
|
+ errors,
|
|
|
+ } = FileMap::<HtmlCtx>::create(config.crate_dir.clone()).unwrap();
|
|
|
+
|
|
|
+ for err in errors {
|
|
|
+ log::error!("{}", err);
|
|
|
+ }
|
|
|
+
|
|
|
+ *rsx_file_map = new_file_map;
|
|
|
+ } else {
|
|
|
+ for msg in messages {
|
|
|
+ let _ = hot_reload.messages.send(msg);
|
|
|
+ }
|
|
|
}
|
|
|
- Ok(UpdateResult::NeedsRebuild) => {
|
|
|
- needs_full_rebuild = true;
|
|
|
- }
|
|
|
- Err(err) => {
|
|
|
- log::error!("{}", err);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if needs_full_rebuild {
|
|
|
- // Reset the file map to the new state of the project
|
|
|
- let FileMapBuildResult {
|
|
|
- map: new_file_map,
|
|
|
- errors,
|
|
|
- } = FileMap::<HtmlCtx>::create(config.crate_dir.clone()).unwrap();
|
|
|
-
|
|
|
- for err in errors {
|
|
|
- log::error!("{}", err);
|
|
|
- }
|
|
|
-
|
|
|
- *rsx_file_map = new_file_map;
|
|
|
- } else {
|
|
|
- for msg in messages {
|
|
|
- let _ = hot_reload.messages.send(msg);
|
|
|
+ } else {
|
|
|
+ needs_full_rebuild = true;
|
|
|
}
|
|
|
- }
|
|
|
- } else {
|
|
|
- needs_full_rebuild = true;
|
|
|
- }
|
|
|
|
|
|
- if needs_full_rebuild {
|
|
|
- match build_with() {
|
|
|
- Ok(res) => {
|
|
|
- last_update_time = chrono::Local::now().timestamp();
|
|
|
-
|
|
|
- #[allow(clippy::redundant_clone)]
|
|
|
- print_console_info(
|
|
|
- &config,
|
|
|
- PrettierOptions {
|
|
|
- changed: e.paths.clone(),
|
|
|
- warnings: res.warnings,
|
|
|
- elapsed_time: res.elapsed_time,
|
|
|
- },
|
|
|
- web_info.clone(),
|
|
|
- );
|
|
|
-
|
|
|
- #[cfg(feature = "plugin")]
|
|
|
- let _ = PluginManager::on_serve_rebuild(
|
|
|
- chrono::Local::now().timestamp(),
|
|
|
- e.paths,
|
|
|
- );
|
|
|
+ if needs_full_rebuild {
|
|
|
+ match build_with() {
|
|
|
+ Ok(res) => {
|
|
|
+ last_update_time = chrono::Local::now().timestamp();
|
|
|
+
|
|
|
+ #[allow(clippy::redundant_clone)]
|
|
|
+ print_console_info(
|
|
|
+ &config,
|
|
|
+ PrettierOptions {
|
|
|
+ changed: e.paths.clone(),
|
|
|
+ warnings: res.warnings,
|
|
|
+ elapsed_time: res.elapsed_time,
|
|
|
+ },
|
|
|
+ web_info.clone(),
|
|
|
+ );
|
|
|
+
|
|
|
+ #[cfg(feature = "plugin")]
|
|
|
+ let _ = PluginManager::on_serve_rebuild(
|
|
|
+ chrono::Local::now().timestamp(),
|
|
|
+ e.paths,
|
|
|
+ );
|
|
|
+ }
|
|
|
+ Err(e) => {
|
|
|
+ last_update_time = chrono::Local::now().timestamp();
|
|
|
+ log::error!("{:?}", e);
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
- Err(e) => log::error!("{}", e),
|
|
|
}
|
|
|
}
|
|
|
+ _ => {}
|
|
|
}
|
|
|
}
|
|
|
})
|