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
use crate::parsers::language_rust;
use ast_grep_core::language::{Language, TSLanguage};
use std::borrow::Cow;
#[derive(Clone, Copy)]
pub struct Rust;
impl Language for Rust {
fn get_ts_language(&self) -> TSLanguage {
language_rust()
}
fn expando_char(&self) -> char {
'µ'
}
fn pre_process_pattern<'q>(&self, query: &'q str) -> Cow<'q, str> {
let mut buf = [0; 4];
let expando = self.expando_char().encode_utf8(&mut buf);
let replaced = query.replace(self.meta_var_char(), expando);
Cow::Owned(replaced)
}
}
#[cfg(test)]
mod test {
use super::*;
use ast_grep_core::{Matcher, Pattern};
fn test_match(s1: &str, s2: &str) {
let pattern = Pattern::new(s1, Rust);
let cand = Rust.ast_grep(s2);
assert!(
pattern.find_node(cand.root()).is_some(),
"goal: {:?}, candidate: {}",
pattern,
cand.root().to_sexp(),
);
}
#[test]
fn test_rust_pattern() {
test_match("Some($A)", "fn test() { Some(123) }");
test_match(
"
match $A {
Some($B) => $B,
None => $C,
}",
r#"fn test() {
patterns = match config.include.clone() {
Some(patterns) => patterns,
None => Vec::from([cwd
.join("**/*.toml")
.normalize()
.to_string_lossy()
.into_owned()]),
};
}"#,
);
}
fn test_replace(src: &str, pattern: &str, replacer: &str) -> String {
let mut source = Rust.ast_grep(src);
let replacer = Pattern::new(replacer, Rust);
assert!(source.replace(pattern, replacer).unwrap());
source.generate()
}
#[test]
fn test_rust_replace() {
let ret = test_replace("fn test() { Some(123) }", "Some($A)", "Ok($A)");
assert_eq!(ret, "fn test() { Ok(123) }");
let ret = test_replace(
r#"
patterns = match config.include.clone() {
Some(patterns) => patterns,
None => 123,
}"#,
"match $A {
Some($B) => $B,
None => $C,
}",
"$A.unwrap_or($C)",
);
assert_eq!(ret, "\npatterns = config.include.clone().unwrap_or(123)")
}
}