dart_event.rs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. use super::event_template::*;
  2. use crate::ast::EventASTContext;
  3. use crate::flowy_toml::{parse_crate_config_from, CrateConfig};
  4. use crate::util::{is_crate_dir, is_hidden, path_string_with_component, read_file};
  5. use flowy_ast::ASTResult;
  6. use std::fs::File;
  7. use std::io::Write;
  8. use std::path::PathBuf;
  9. use syn::Item;
  10. use walkdir::WalkDir;
  11. pub fn gen(crate_name: &str) {
  12. if std::env::var("CARGO_MAKE_WORKING_DIRECTORY").is_err() {
  13. log::warn!("CARGO_MAKE_WORKING_DIRECTORY was not set, skip generate dart pb");
  14. return;
  15. }
  16. if std::env::var("FLUTTER_FLOWY_SDK_PATH").is_err() {
  17. log::warn!("FLUTTER_FLOWY_SDK_PATH was not set, skip generate dart pb");
  18. return;
  19. }
  20. let crate_path = std::fs::canonicalize(".").unwrap().as_path().display().to_string();
  21. let event_crates = parse_dart_event_files(vec![crate_path]);
  22. let event_ast = event_crates.iter().flat_map(parse_event_crate).collect::<Vec<_>>();
  23. let event_render_ctx = ast_to_event_render_ctx(event_ast.as_ref());
  24. let mut render_result = DART_IMPORTED.to_owned();
  25. for (index, render_ctx) in event_render_ctx.into_iter().enumerate() {
  26. let mut event_template = EventTemplate::new();
  27. if let Some(content) = event_template.render(render_ctx, index) {
  28. render_result.push_str(content.as_ref())
  29. }
  30. }
  31. let dart_event_folder: PathBuf = [
  32. &std::env::var("CARGO_MAKE_WORKING_DIRECTORY").unwrap(),
  33. &std::env::var("FLUTTER_FLOWY_SDK_PATH").unwrap(),
  34. "lib",
  35. "dispatch",
  36. "dart_event",
  37. crate_name,
  38. ]
  39. .iter()
  40. .collect();
  41. if !dart_event_folder.as_path().exists() {
  42. std::fs::create_dir_all(dart_event_folder.as_path()).unwrap();
  43. }
  44. let dart_event_file_path = path_string_with_component(&dart_event_folder, vec!["dart_event.dart"]);
  45. println!("cargo:rerun-if-changed={}", dart_event_file_path);
  46. match std::fs::OpenOptions::new()
  47. .create(true)
  48. .write(true)
  49. .append(false)
  50. .truncate(true)
  51. .open(&dart_event_file_path)
  52. {
  53. Ok(ref mut file) => {
  54. file.write_all(render_result.as_bytes()).unwrap();
  55. File::flush(file).unwrap();
  56. }
  57. Err(err) => {
  58. panic!("Failed to open file: {}, {:?}", dart_event_file_path, err);
  59. }
  60. }
  61. }
  62. const DART_IMPORTED: &str = r#"
  63. /// Auto generate. Do not edit
  64. part of '../../dispatch.dart';
  65. "#;
  66. #[derive(Debug)]
  67. pub struct DartEventCrate {
  68. crate_path: PathBuf,
  69. event_files: Vec<String>,
  70. }
  71. impl DartEventCrate {
  72. pub fn from_config(config: &CrateConfig) -> Self {
  73. DartEventCrate {
  74. crate_path: config.crate_path.clone(),
  75. event_files: config.flowy_config.event_files.clone(),
  76. }
  77. }
  78. }
  79. pub fn parse_dart_event_files(crate_paths: Vec<String>) -> Vec<DartEventCrate> {
  80. let mut dart_event_crates: Vec<DartEventCrate> = vec![];
  81. crate_paths.iter().for_each(|path| {
  82. let crates = WalkDir::new(path)
  83. .into_iter()
  84. .filter_entry(|e| !is_hidden(e))
  85. .filter_map(|e| e.ok())
  86. .filter(is_crate_dir)
  87. .flat_map(|e| parse_crate_config_from(&e))
  88. .map(|crate_config| DartEventCrate::from_config(&crate_config))
  89. .collect::<Vec<DartEventCrate>>();
  90. dart_event_crates.extend(crates);
  91. });
  92. dart_event_crates
  93. }
  94. pub fn parse_event_crate(event_crate: &DartEventCrate) -> Vec<EventASTContext> {
  95. event_crate
  96. .event_files
  97. .iter()
  98. .flat_map(|event_file| {
  99. let file_path = path_string_with_component(&event_crate.crate_path, vec![event_file.as_str()]);
  100. let file_content = read_file(file_path.as_ref()).unwrap();
  101. let ast = syn::parse_file(file_content.as_ref()).expect("Unable to parse file");
  102. ast.items
  103. .iter()
  104. .flat_map(|item| match item {
  105. Item::Enum(item_enum) => {
  106. let ast_result = ASTResult::new();
  107. let attrs = flowy_ast::enum_from_ast(
  108. &ast_result,
  109. &item_enum.ident,
  110. &item_enum.variants,
  111. &item_enum.attrs,
  112. );
  113. ast_result.check().unwrap();
  114. attrs
  115. .iter()
  116. .filter(|attr| !attr.attrs.event_attrs.ignore)
  117. .enumerate()
  118. .map(|(_index, variant)| EventASTContext::from(&variant.attrs))
  119. .collect::<Vec<_>>()
  120. }
  121. _ => vec![],
  122. })
  123. .collect::<Vec<_>>()
  124. })
  125. .collect::<Vec<EventASTContext>>()
  126. }
  127. pub fn ast_to_event_render_ctx(ast: &[EventASTContext]) -> Vec<EventRenderContext> {
  128. ast.iter()
  129. .map(|event_ast| {
  130. let input_deserializer = event_ast
  131. .event_input
  132. .as_ref()
  133. .map(|event_input| event_input.get_ident().unwrap().to_string());
  134. let output_deserializer = event_ast
  135. .event_output
  136. .as_ref()
  137. .map(|event_output| event_output.get_ident().unwrap().to_string());
  138. EventRenderContext {
  139. input_deserializer,
  140. output_deserializer,
  141. error_deserializer: event_ast.event_error.clone(),
  142. event: event_ast.event.to_string(),
  143. event_ty: event_ast.event_ty.to_string(),
  144. }
  145. })
  146. .collect::<Vec<EventRenderContext>>()
  147. }