hot_reload_diff.rs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671
  1. use proc_macro2::TokenStream;
  2. use quote::ToTokens;
  3. use syn::{File, Macro};
  4. pub enum DiffResult {
  5. /// Non-rsx was changed in the file
  6. CodeChanged,
  7. /// Rsx was changed in the file
  8. ///
  9. /// Contains a list of macro invocations that were changed
  10. RsxChanged { rsx_calls: Vec<ChangedRsx> },
  11. }
  12. #[derive(Debug)]
  13. pub struct ChangedRsx {
  14. /// The macro that was changed
  15. pub old: Macro,
  16. /// The new tokens for the macro
  17. pub new: TokenStream,
  18. }
  19. /// Find any rsx calls in the given file and return a list of all the rsx calls that have changed.
  20. pub fn find_rsx(new: &File, old: &File) -> DiffResult {
  21. let mut rsx_calls = Vec::new();
  22. if new.items.len() != old.items.len() {
  23. tracing::trace!(
  24. "found not hot reload-able change {:#?} != {:#?}",
  25. new.items
  26. .iter()
  27. .map(|i| i.to_token_stream().to_string())
  28. .collect::<Vec<_>>(),
  29. old.items
  30. .iter()
  31. .map(|i| i.to_token_stream().to_string())
  32. .collect::<Vec<_>>()
  33. );
  34. return DiffResult::CodeChanged;
  35. }
  36. for (new, old) in new.items.iter().zip(old.items.iter()) {
  37. if find_rsx_item(new, old, &mut rsx_calls) {
  38. tracing::trace!(
  39. "found not hot reload-able change {:#?} != {:#?}",
  40. new.to_token_stream().to_string(),
  41. old.to_token_stream().to_string()
  42. );
  43. return DiffResult::CodeChanged;
  44. }
  45. }
  46. tracing::trace!("found hot reload-able changes {:#?}", rsx_calls);
  47. DiffResult::RsxChanged { rsx_calls }
  48. }
  49. fn find_rsx_item(new: &syn::Item, old: &syn::Item, rsx_calls: &mut Vec<ChangedRsx>) -> bool {
  50. match (new, old) {
  51. (syn::Item::Const(new_item), syn::Item::Const(old_item)) => {
  52. find_rsx_expr(&new_item.expr, &old_item.expr, rsx_calls)
  53. || new_item.attrs != old_item.attrs
  54. || new_item.vis != old_item.vis
  55. || new_item.const_token != old_item.const_token
  56. || new_item.ident != old_item.ident
  57. || new_item.colon_token != old_item.colon_token
  58. || new_item.ty != old_item.ty
  59. || new_item.eq_token != old_item.eq_token
  60. || new_item.semi_token != old_item.semi_token
  61. }
  62. (syn::Item::Enum(new_item), syn::Item::Enum(old_item)) => {
  63. if new_item.variants.len() != old_item.variants.len() {
  64. return true;
  65. }
  66. for (new_variant, old_variant) in new_item.variants.iter().zip(old_item.variants.iter())
  67. {
  68. match (&new_variant.discriminant, &old_variant.discriminant) {
  69. (Some((new_eq, new_expr)), Some((old_eq, old_expr))) => {
  70. if find_rsx_expr(new_expr, old_expr, rsx_calls) || new_eq != old_eq {
  71. return true;
  72. }
  73. }
  74. (None, None) => (),
  75. _ => return true,
  76. }
  77. if new_variant.attrs != old_variant.attrs
  78. || new_variant.ident != old_variant.ident
  79. || new_variant.fields != old_variant.fields
  80. {
  81. return true;
  82. }
  83. }
  84. new_item.attrs != old_item.attrs
  85. || new_item.vis != old_item.vis
  86. || new_item.enum_token != old_item.enum_token
  87. || new_item.ident != old_item.ident
  88. || new_item.generics != old_item.generics
  89. || new_item.brace_token != old_item.brace_token
  90. }
  91. (syn::Item::ExternCrate(new_item), syn::Item::ExternCrate(old_item)) => {
  92. old_item != new_item
  93. }
  94. (syn::Item::Fn(new_item), syn::Item::Fn(old_item)) => {
  95. find_rsx_block(&new_item.block, &old_item.block, rsx_calls)
  96. || new_item.attrs != old_item.attrs
  97. || new_item.vis != old_item.vis
  98. || new_item.sig != old_item.sig
  99. }
  100. (syn::Item::ForeignMod(new_item), syn::Item::ForeignMod(old_item)) => old_item != new_item,
  101. (syn::Item::Impl(new_item), syn::Item::Impl(old_item)) => {
  102. if new_item.items.len() != old_item.items.len() {
  103. return true;
  104. }
  105. for (new_item, old_item) in new_item.items.iter().zip(old_item.items.iter()) {
  106. if match (new_item, old_item) {
  107. (syn::ImplItem::Const(new_item), syn::ImplItem::Const(old_item)) => {
  108. find_rsx_expr(&new_item.expr, &old_item.expr, rsx_calls)
  109. }
  110. (syn::ImplItem::Fn(new_item), syn::ImplItem::Fn(old_item)) => {
  111. find_rsx_block(&new_item.block, &old_item.block, rsx_calls)
  112. }
  113. (syn::ImplItem::Type(new_item), syn::ImplItem::Type(old_item)) => {
  114. old_item != new_item
  115. }
  116. (syn::ImplItem::Macro(new_item), syn::ImplItem::Macro(old_item)) => {
  117. old_item != new_item
  118. }
  119. (syn::ImplItem::Verbatim(stream), syn::ImplItem::Verbatim(stream2)) => {
  120. stream.to_string() != stream2.to_string()
  121. }
  122. _ => true,
  123. } {
  124. return true;
  125. }
  126. }
  127. new_item.attrs != old_item.attrs
  128. || new_item.defaultness != old_item.defaultness
  129. || new_item.unsafety != old_item.unsafety
  130. || new_item.impl_token != old_item.impl_token
  131. || new_item.generics != old_item.generics
  132. || new_item.trait_ != old_item.trait_
  133. || new_item.self_ty != old_item.self_ty
  134. || new_item.brace_token != old_item.brace_token
  135. }
  136. (syn::Item::Macro(new_item), syn::Item::Macro(old_item)) => {
  137. find_rsx_macro(&new_item.mac, &old_item.mac, rsx_calls)
  138. || new_item.attrs != old_item.attrs
  139. || new_item.semi_token != old_item.semi_token
  140. || new_item.ident != old_item.ident
  141. }
  142. (syn::Item::Mod(new_item), syn::Item::Mod(old_item)) => {
  143. match (&new_item.content, &old_item.content) {
  144. (Some((_, new_items)), Some((_, old_items))) => {
  145. if new_items.len() != old_items.len() {
  146. return true;
  147. }
  148. for (new_item, old_item) in new_items.iter().zip(old_items.iter()) {
  149. if find_rsx_item(new_item, old_item, rsx_calls) {
  150. return true;
  151. }
  152. }
  153. new_item.attrs != old_item.attrs
  154. || new_item.vis != old_item.vis
  155. || new_item.mod_token != old_item.mod_token
  156. || new_item.ident != old_item.ident
  157. || new_item.semi != old_item.semi
  158. }
  159. (None, None) => {
  160. new_item.attrs != old_item.attrs
  161. || new_item.vis != old_item.vis
  162. || new_item.mod_token != old_item.mod_token
  163. || new_item.ident != old_item.ident
  164. || new_item.semi != old_item.semi
  165. }
  166. _ => true,
  167. }
  168. }
  169. (syn::Item::Static(new_item), syn::Item::Static(old_item)) => {
  170. find_rsx_expr(&new_item.expr, &old_item.expr, rsx_calls)
  171. || new_item.attrs != old_item.attrs
  172. || new_item.vis != old_item.vis
  173. || new_item.static_token != old_item.static_token
  174. || new_item.mutability != old_item.mutability
  175. || new_item.ident != old_item.ident
  176. || new_item.colon_token != old_item.colon_token
  177. || new_item.ty != old_item.ty
  178. || new_item.eq_token != old_item.eq_token
  179. || new_item.semi_token != old_item.semi_token
  180. }
  181. (syn::Item::Struct(new_item), syn::Item::Struct(old_item)) => old_item != new_item,
  182. (syn::Item::Trait(new_item), syn::Item::Trait(old_item)) => {
  183. find_rsx_trait(new_item, old_item, rsx_calls)
  184. }
  185. (syn::Item::TraitAlias(new_item), syn::Item::TraitAlias(old_item)) => old_item != new_item,
  186. (syn::Item::Type(new_item), syn::Item::Type(old_item)) => old_item != new_item,
  187. (syn::Item::Union(new_item), syn::Item::Union(old_item)) => old_item != new_item,
  188. (syn::Item::Use(new_item), syn::Item::Use(old_item)) => old_item != new_item,
  189. (syn::Item::Verbatim(_), syn::Item::Verbatim(_)) => false,
  190. _ => true,
  191. }
  192. }
  193. fn find_rsx_trait(
  194. new_item: &syn::ItemTrait,
  195. old_item: &syn::ItemTrait,
  196. rsx_calls: &mut Vec<ChangedRsx>,
  197. ) -> bool {
  198. if new_item.items.len() != old_item.items.len() {
  199. return true;
  200. }
  201. for (new_item, old_item) in new_item.items.iter().zip(old_item.items.iter()) {
  202. if match (new_item, old_item) {
  203. (syn::TraitItem::Const(new_item), syn::TraitItem::Const(old_item)) => {
  204. if let (Some((_, new_expr)), Some((_, old_expr))) =
  205. (&new_item.default, &old_item.default)
  206. {
  207. find_rsx_expr(new_expr, old_expr, rsx_calls)
  208. } else {
  209. true
  210. }
  211. }
  212. (syn::TraitItem::Fn(new_item), syn::TraitItem::Fn(old_item)) => {
  213. match (&new_item.default, &old_item.default) {
  214. (Some(new_block), Some(old_block)) => {
  215. find_rsx_block(new_block, old_block, rsx_calls)
  216. }
  217. (None, None) => false,
  218. _ => true,
  219. }
  220. }
  221. (syn::TraitItem::Type(new_item), syn::TraitItem::Type(old_item)) => {
  222. old_item != new_item
  223. }
  224. (syn::TraitItem::Macro(new_item), syn::TraitItem::Macro(old_item)) => {
  225. old_item != new_item
  226. }
  227. (syn::TraitItem::Verbatim(stream), syn::TraitItem::Verbatim(stream2)) => {
  228. stream.to_string() != stream2.to_string()
  229. }
  230. _ => true,
  231. } {
  232. return true;
  233. }
  234. }
  235. new_item.attrs != old_item.attrs
  236. || new_item.vis != old_item.vis
  237. || new_item.unsafety != old_item.unsafety
  238. || new_item.auto_token != old_item.auto_token
  239. || new_item.ident != old_item.ident
  240. || new_item.generics != old_item.generics
  241. || new_item.colon_token != old_item.colon_token
  242. || new_item.supertraits != old_item.supertraits
  243. || new_item.brace_token != old_item.brace_token
  244. }
  245. fn find_rsx_block(
  246. new_block: &syn::Block,
  247. old_block: &syn::Block,
  248. rsx_calls: &mut Vec<ChangedRsx>,
  249. ) -> bool {
  250. if new_block.stmts.len() != old_block.stmts.len() {
  251. return true;
  252. }
  253. for (new_stmt, old_stmt) in new_block.stmts.iter().zip(old_block.stmts.iter()) {
  254. if find_rsx_stmt(new_stmt, old_stmt, rsx_calls) {
  255. return true;
  256. }
  257. }
  258. new_block.brace_token != old_block.brace_token
  259. }
  260. fn find_rsx_stmt(
  261. new_stmt: &syn::Stmt,
  262. old_stmt: &syn::Stmt,
  263. rsx_calls: &mut Vec<ChangedRsx>,
  264. ) -> bool {
  265. match (new_stmt, old_stmt) {
  266. (syn::Stmt::Local(new_local), syn::Stmt::Local(old_local)) => {
  267. (match (&new_local.init, &old_local.init) {
  268. (Some(new_local), Some(old_local)) => {
  269. find_rsx_expr(&new_local.expr, &old_local.expr, rsx_calls)
  270. || new_local != old_local
  271. }
  272. (None, None) => false,
  273. _ => true,
  274. } || new_local.attrs != old_local.attrs
  275. || new_local.let_token != old_local.let_token
  276. || new_local.pat != old_local.pat
  277. || new_local.semi_token != old_local.semi_token)
  278. }
  279. (syn::Stmt::Item(new_item), syn::Stmt::Item(old_item)) => {
  280. find_rsx_item(new_item, old_item, rsx_calls)
  281. }
  282. (syn::Stmt::Expr(new_expr, _), syn::Stmt::Expr(old_expr, _)) => {
  283. find_rsx_expr(new_expr, old_expr, rsx_calls)
  284. }
  285. (syn::Stmt::Macro(new_macro), syn::Stmt::Macro(old_macro)) => {
  286. find_rsx_macro(&new_macro.mac, &old_macro.mac, rsx_calls)
  287. || new_macro.attrs != old_macro.attrs
  288. || new_macro.semi_token != old_macro.semi_token
  289. }
  290. _ => true,
  291. }
  292. }
  293. fn find_rsx_expr(
  294. new_expr: &syn::Expr,
  295. old_expr: &syn::Expr,
  296. rsx_calls: &mut Vec<ChangedRsx>,
  297. ) -> bool {
  298. match (new_expr, old_expr) {
  299. (syn::Expr::Array(new_expr), syn::Expr::Array(old_expr)) => {
  300. if new_expr.elems.len() != old_expr.elems.len() {
  301. return true;
  302. }
  303. for (new_el, old_el) in new_expr.elems.iter().zip(old_expr.elems.iter()) {
  304. if find_rsx_expr(new_el, old_el, rsx_calls) {
  305. return true;
  306. }
  307. }
  308. new_expr.attrs != old_expr.attrs || new_expr.bracket_token != old_expr.bracket_token
  309. }
  310. (syn::Expr::Assign(new_expr), syn::Expr::Assign(old_expr)) => {
  311. find_rsx_expr(&new_expr.left, &old_expr.left, rsx_calls)
  312. || find_rsx_expr(&new_expr.right, &old_expr.right, rsx_calls)
  313. || new_expr.attrs != old_expr.attrs
  314. || new_expr.eq_token != old_expr.eq_token
  315. }
  316. (syn::Expr::Async(new_expr), syn::Expr::Async(old_expr)) => {
  317. find_rsx_block(&new_expr.block, &old_expr.block, rsx_calls)
  318. || new_expr.attrs != old_expr.attrs
  319. || new_expr.async_token != old_expr.async_token
  320. || new_expr.capture != old_expr.capture
  321. }
  322. (syn::Expr::Await(new_expr), syn::Expr::Await(old_expr)) => {
  323. find_rsx_expr(&new_expr.base, &old_expr.base, rsx_calls)
  324. || new_expr.attrs != old_expr.attrs
  325. || new_expr.dot_token != old_expr.dot_token
  326. || new_expr.await_token != old_expr.await_token
  327. }
  328. (syn::Expr::Binary(new_expr), syn::Expr::Binary(old_expr)) => {
  329. find_rsx_expr(&new_expr.left, &old_expr.left, rsx_calls)
  330. || find_rsx_expr(&new_expr.right, &old_expr.right, rsx_calls)
  331. || new_expr.attrs != old_expr.attrs
  332. || new_expr.op != old_expr.op
  333. }
  334. (syn::Expr::Block(new_expr), syn::Expr::Block(old_expr)) => {
  335. find_rsx_block(&new_expr.block, &old_expr.block, rsx_calls)
  336. || new_expr.attrs != old_expr.attrs
  337. || new_expr.label != old_expr.label
  338. }
  339. (syn::Expr::Break(new_expr), syn::Expr::Break(old_expr)) => {
  340. match (&new_expr.expr, &old_expr.expr) {
  341. (Some(new_inner), Some(old_inner)) => {
  342. find_rsx_expr(new_inner, old_inner, rsx_calls)
  343. || new_expr.attrs != old_expr.attrs
  344. || new_expr.break_token != old_expr.break_token
  345. || new_expr.label != old_expr.label
  346. }
  347. (None, None) => {
  348. new_expr.attrs != old_expr.attrs
  349. || new_expr.break_token != old_expr.break_token
  350. || new_expr.label != old_expr.label
  351. }
  352. _ => true,
  353. }
  354. }
  355. (syn::Expr::Call(new_expr), syn::Expr::Call(old_expr)) => {
  356. find_rsx_expr(&new_expr.func, &old_expr.func, rsx_calls);
  357. if new_expr.args.len() != old_expr.args.len() {
  358. return true;
  359. }
  360. for (new_arg, old_arg) in new_expr.args.iter().zip(old_expr.args.iter()) {
  361. if find_rsx_expr(new_arg, old_arg, rsx_calls) {
  362. return true;
  363. }
  364. }
  365. new_expr.attrs != old_expr.attrs || new_expr.paren_token != old_expr.paren_token
  366. }
  367. (syn::Expr::Cast(new_expr), syn::Expr::Cast(old_expr)) => {
  368. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  369. || new_expr.attrs != old_expr.attrs
  370. || new_expr.as_token != old_expr.as_token
  371. || new_expr.ty != old_expr.ty
  372. }
  373. (syn::Expr::Closure(new_expr), syn::Expr::Closure(old_expr)) => {
  374. find_rsx_expr(&new_expr.body, &old_expr.body, rsx_calls)
  375. || new_expr.attrs != old_expr.attrs
  376. || new_expr.movability != old_expr.movability
  377. || new_expr.asyncness != old_expr.asyncness
  378. || new_expr.capture != old_expr.capture
  379. || new_expr.or1_token != old_expr.or1_token
  380. || new_expr.inputs != old_expr.inputs
  381. || new_expr.or2_token != old_expr.or2_token
  382. || new_expr.output != old_expr.output
  383. }
  384. (syn::Expr::Const(new_expr), syn::Expr::Const(old_expr)) => {
  385. find_rsx_block(&new_expr.block, &old_expr.block, rsx_calls)
  386. || new_expr.attrs != old_expr.attrs
  387. || new_expr.const_token != old_expr.const_token
  388. }
  389. (syn::Expr::Continue(new_expr), syn::Expr::Continue(old_expr)) => old_expr != new_expr,
  390. (syn::Expr::Field(new_expr), syn::Expr::Field(old_expr)) => {
  391. find_rsx_expr(&new_expr.base, &old_expr.base, rsx_calls)
  392. || new_expr.attrs != old_expr.attrs
  393. || new_expr.dot_token != old_expr.dot_token
  394. || new_expr.member != old_expr.member
  395. }
  396. (syn::Expr::ForLoop(new_expr), syn::Expr::ForLoop(old_expr)) => {
  397. find_rsx_block(&new_expr.body, &old_expr.body, rsx_calls)
  398. || find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  399. || new_expr.attrs != old_expr.attrs
  400. || new_expr.label != old_expr.label
  401. || new_expr.for_token != old_expr.for_token
  402. || new_expr.pat != old_expr.pat
  403. || new_expr.in_token != old_expr.in_token
  404. }
  405. (syn::Expr::Group(new_expr), syn::Expr::Group(old_expr)) => {
  406. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  407. }
  408. (syn::Expr::If(new_expr), syn::Expr::If(old_expr)) => {
  409. if find_rsx_expr(&new_expr.cond, &old_expr.cond, rsx_calls)
  410. || find_rsx_block(&new_expr.then_branch, &old_expr.then_branch, rsx_calls)
  411. {
  412. return true;
  413. }
  414. match (&new_expr.else_branch, &old_expr.else_branch) {
  415. (Some((new_tok, new_else)), Some((old_tok, old_else))) => {
  416. find_rsx_expr(new_else, old_else, rsx_calls)
  417. || new_expr.attrs != old_expr.attrs
  418. || new_expr.if_token != old_expr.if_token
  419. || new_expr.cond != old_expr.cond
  420. || new_tok != old_tok
  421. }
  422. (None, None) => {
  423. new_expr.attrs != old_expr.attrs
  424. || new_expr.if_token != old_expr.if_token
  425. || new_expr.cond != old_expr.cond
  426. }
  427. _ => true,
  428. }
  429. }
  430. (syn::Expr::Index(new_expr), syn::Expr::Index(old_expr)) => {
  431. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  432. || find_rsx_expr(&new_expr.index, &old_expr.index, rsx_calls)
  433. || new_expr.attrs != old_expr.attrs
  434. || new_expr.bracket_token != old_expr.bracket_token
  435. }
  436. (syn::Expr::Infer(new_expr), syn::Expr::Infer(old_expr)) => new_expr != old_expr,
  437. (syn::Expr::Let(new_expr), syn::Expr::Let(old_expr)) => {
  438. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  439. || new_expr.attrs != old_expr.attrs
  440. || new_expr.let_token != old_expr.let_token
  441. || new_expr.pat != old_expr.pat
  442. || new_expr.eq_token != old_expr.eq_token
  443. }
  444. (syn::Expr::Lit(new_expr), syn::Expr::Lit(old_expr)) => old_expr != new_expr,
  445. (syn::Expr::Loop(new_expr), syn::Expr::Loop(old_expr)) => {
  446. find_rsx_block(&new_expr.body, &old_expr.body, rsx_calls)
  447. || new_expr.attrs != old_expr.attrs
  448. || new_expr.label != old_expr.label
  449. || new_expr.loop_token != old_expr.loop_token
  450. }
  451. (syn::Expr::Macro(new_expr), syn::Expr::Macro(old_expr)) => {
  452. find_rsx_macro(&new_expr.mac, &old_expr.mac, rsx_calls)
  453. || new_expr.attrs != old_expr.attrs
  454. }
  455. (syn::Expr::Match(new_expr), syn::Expr::Match(old_expr)) => {
  456. if find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls) {
  457. return true;
  458. }
  459. for (new_arm, old_arm) in new_expr.arms.iter().zip(old_expr.arms.iter()) {
  460. match (&new_arm.guard, &old_arm.guard) {
  461. (Some((new_tok, new_expr)), Some((old_tok, old_expr))) => {
  462. if find_rsx_expr(new_expr, old_expr, rsx_calls) || new_tok != old_tok {
  463. return true;
  464. }
  465. }
  466. (None, None) => (),
  467. _ => return true,
  468. }
  469. if find_rsx_expr(&new_arm.body, &old_arm.body, rsx_calls)
  470. || new_arm.attrs != old_arm.attrs
  471. || new_arm.pat != old_arm.pat
  472. || new_arm.fat_arrow_token != old_arm.fat_arrow_token
  473. || new_arm.comma != old_arm.comma
  474. {
  475. return true;
  476. }
  477. }
  478. new_expr.attrs != old_expr.attrs
  479. || new_expr.match_token != old_expr.match_token
  480. || new_expr.brace_token != old_expr.brace_token
  481. }
  482. (syn::Expr::MethodCall(new_expr), syn::Expr::MethodCall(old_expr)) => {
  483. if find_rsx_expr(&new_expr.receiver, &old_expr.receiver, rsx_calls) {
  484. return true;
  485. }
  486. for (new_arg, old_arg) in new_expr.args.iter().zip(old_expr.args.iter()) {
  487. if find_rsx_expr(new_arg, old_arg, rsx_calls) {
  488. return true;
  489. }
  490. }
  491. new_expr.attrs != old_expr.attrs
  492. || new_expr.dot_token != old_expr.dot_token
  493. || new_expr.method != old_expr.method
  494. || new_expr.turbofish != old_expr.turbofish
  495. || new_expr.paren_token != old_expr.paren_token
  496. }
  497. (syn::Expr::Paren(new_expr), syn::Expr::Paren(old_expr)) => {
  498. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  499. || new_expr.attrs != old_expr.attrs
  500. || new_expr.paren_token != old_expr.paren_token
  501. }
  502. (syn::Expr::Path(new_expr), syn::Expr::Path(old_expr)) => old_expr != new_expr,
  503. (syn::Expr::Range(new_expr), syn::Expr::Range(old_expr)) => {
  504. match (&new_expr.start, &old_expr.start) {
  505. (Some(new_expr), Some(old_expr)) => {
  506. if find_rsx_expr(new_expr, old_expr, rsx_calls) {
  507. return true;
  508. }
  509. }
  510. (None, None) => (),
  511. _ => return true,
  512. }
  513. match (&new_expr.end, &old_expr.end) {
  514. (Some(new_inner), Some(old_inner)) => {
  515. find_rsx_expr(new_inner, old_inner, rsx_calls)
  516. || new_expr.attrs != old_expr.attrs
  517. || new_expr.limits != old_expr.limits
  518. }
  519. (None, None) => {
  520. new_expr.attrs != old_expr.attrs || new_expr.limits != old_expr.limits
  521. }
  522. _ => true,
  523. }
  524. }
  525. (syn::Expr::Reference(new_expr), syn::Expr::Reference(old_expr)) => {
  526. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  527. || new_expr.attrs != old_expr.attrs
  528. || new_expr.and_token != old_expr.and_token
  529. || new_expr.mutability != old_expr.mutability
  530. }
  531. (syn::Expr::Repeat(new_expr), syn::Expr::Repeat(old_expr)) => {
  532. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  533. || find_rsx_expr(&new_expr.len, &old_expr.len, rsx_calls)
  534. || new_expr.attrs != old_expr.attrs
  535. || new_expr.bracket_token != old_expr.bracket_token
  536. || new_expr.semi_token != old_expr.semi_token
  537. }
  538. (syn::Expr::Return(new_expr), syn::Expr::Return(old_expr)) => {
  539. match (&new_expr.expr, &old_expr.expr) {
  540. (Some(new_inner), Some(old_inner)) => {
  541. find_rsx_expr(new_inner, old_inner, rsx_calls)
  542. || new_expr.attrs != old_expr.attrs
  543. || new_expr.return_token != old_expr.return_token
  544. }
  545. (None, None) => {
  546. new_expr.attrs != old_expr.attrs
  547. || new_expr.return_token != old_expr.return_token
  548. }
  549. _ => true,
  550. }
  551. }
  552. (syn::Expr::Struct(new_expr), syn::Expr::Struct(old_expr)) => {
  553. match (&new_expr.rest, &old_expr.rest) {
  554. (Some(new_expr), Some(old_expr)) => {
  555. if find_rsx_expr(new_expr, old_expr, rsx_calls) {
  556. return true;
  557. }
  558. }
  559. (None, None) => (),
  560. _ => return true,
  561. }
  562. for (new_field, old_field) in new_expr.fields.iter().zip(old_expr.fields.iter()) {
  563. if find_rsx_expr(&new_field.expr, &old_field.expr, rsx_calls)
  564. || new_field.attrs != old_field.attrs
  565. || new_field.member != old_field.member
  566. || new_field.colon_token != old_field.colon_token
  567. {
  568. return true;
  569. }
  570. }
  571. new_expr.attrs != old_expr.attrs
  572. || new_expr.path != old_expr.path
  573. || new_expr.brace_token != old_expr.brace_token
  574. || new_expr.dot2_token != old_expr.dot2_token
  575. }
  576. (syn::Expr::Try(new_expr), syn::Expr::Try(old_expr)) => {
  577. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  578. || new_expr.attrs != old_expr.attrs
  579. || new_expr.question_token != old_expr.question_token
  580. }
  581. (syn::Expr::TryBlock(new_expr), syn::Expr::TryBlock(old_expr)) => {
  582. find_rsx_block(&new_expr.block, &old_expr.block, rsx_calls)
  583. || new_expr.attrs != old_expr.attrs
  584. || new_expr.try_token != old_expr.try_token
  585. }
  586. (syn::Expr::Tuple(new_expr), syn::Expr::Tuple(old_expr)) => {
  587. for (new_el, old_el) in new_expr.elems.iter().zip(old_expr.elems.iter()) {
  588. if find_rsx_expr(new_el, old_el, rsx_calls) {
  589. return true;
  590. }
  591. }
  592. new_expr.attrs != old_expr.attrs || new_expr.paren_token != old_expr.paren_token
  593. }
  594. (syn::Expr::Unary(new_expr), syn::Expr::Unary(old_expr)) => {
  595. find_rsx_expr(&new_expr.expr, &old_expr.expr, rsx_calls)
  596. || new_expr.attrs != old_expr.attrs
  597. || new_expr.op != old_expr.op
  598. }
  599. (syn::Expr::Unsafe(new_expr), syn::Expr::Unsafe(old_expr)) => {
  600. find_rsx_block(&new_expr.block, &old_expr.block, rsx_calls)
  601. || new_expr.attrs != old_expr.attrs
  602. || new_expr.unsafe_token != old_expr.unsafe_token
  603. }
  604. (syn::Expr::While(new_expr), syn::Expr::While(old_expr)) => {
  605. find_rsx_expr(&new_expr.cond, &old_expr.cond, rsx_calls)
  606. || find_rsx_block(&new_expr.body, &old_expr.body, rsx_calls)
  607. || new_expr.attrs != old_expr.attrs
  608. || new_expr.label != old_expr.label
  609. || new_expr.while_token != old_expr.while_token
  610. }
  611. (syn::Expr::Yield(new_expr), syn::Expr::Yield(old_expr)) => {
  612. match (&new_expr.expr, &old_expr.expr) {
  613. (Some(new_inner), Some(old_inner)) => {
  614. find_rsx_expr(new_inner, old_inner, rsx_calls)
  615. || new_expr.attrs != old_expr.attrs
  616. || new_expr.yield_token != old_expr.yield_token
  617. }
  618. (None, None) => {
  619. new_expr.attrs != old_expr.attrs || new_expr.yield_token != old_expr.yield_token
  620. }
  621. _ => true,
  622. }
  623. }
  624. (syn::Expr::Verbatim(stream), syn::Expr::Verbatim(stream2)) => {
  625. stream.to_string() != stream2.to_string()
  626. }
  627. _ => true,
  628. }
  629. }
  630. fn find_rsx_macro(
  631. new_mac: &syn::Macro,
  632. old_mac: &syn::Macro,
  633. rsx_calls: &mut Vec<ChangedRsx>,
  634. ) -> bool {
  635. if matches!(
  636. new_mac
  637. .path
  638. .get_ident()
  639. .map(|ident| ident.to_string())
  640. .as_deref(),
  641. Some("rsx" | "render")
  642. ) && matches!(
  643. old_mac
  644. .path
  645. .get_ident()
  646. .map(|ident| ident.to_string())
  647. .as_deref(),
  648. Some("rsx" | "render")
  649. ) {
  650. rsx_calls.push(ChangedRsx {
  651. old: old_mac.clone(),
  652. new: new_mac.tokens.clone(),
  653. });
  654. false
  655. } else {
  656. new_mac != old_mac
  657. }
  658. }