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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
pub mod types;
use std::path::PathBuf;
use anyhow::{Context, Result};
use code_blocks::{Block, BlockTree};
use tree_sitter::{Language, Parser, Query};
use tree_sitter_installer::parser_installer::{self, InstallationStatus};
pub use types::*;
#[derive(Debug)]
pub struct InstallLanguageArgs<F: FnMut(InstallationStatus)> {
pub download_cmd: String,
pub library_name: String,
pub install_dir: PathBuf,
pub report_progress: Option<F>,
}
pub type InstallLanguageResponse = PathBuf;
pub fn install_language<F: FnMut(InstallationStatus)>(
args: InstallLanguageArgs<F>,
) -> Result<InstallLanguageResponse> {
if !parser_installer::is_installed_at(&args.library_name, &args.install_dir) {
parser_installer::install_parser(
&args.download_cmd,
&args.library_name,
&args.install_dir,
args.report_progress,
)
} else {
Ok(parser_installer::get_compiled_lib_path(
&args.library_name,
&args.install_dir,
))
}
}
#[derive(Debug)]
pub struct GetSubtreesArgs {
pub queries: Vec<String>,
pub text: String,
pub language: Language,
}
pub type GetSubtreesResponse = Vec<BlockLocationTree>;
pub fn get_subtrees(args: GetSubtreesArgs) -> Result<GetSubtreesResponse> {
let mut parser = Parser::new();
parser.set_language(args.language)?;
let text = args.text;
let tree = parser.parse(&text, None).context("Failed to parse text")?;
let mut queries = vec![];
for query in args.queries {
queries.push(Query::new(args.language, &query)?);
}
let items = code_blocks::get_query_subtrees(&queries, &tree, &text);
Ok(items.iter().map(Into::into).collect())
}
#[derive(Debug)]
pub struct MoveBlockArgs<C: Fn(&Block, &Block) -> Result<()>> {
pub queries: Vec<String>,
pub text: String,
pub language: Language,
pub src_block: BlockLocation,
pub dst_block: BlockLocation,
pub assert_move_legal_fn: Option<C>,
pub force: bool,
}
pub type MoveBlockResponse = String;
pub fn move_block<C: Fn(&Block, &Block) -> Result<()>>(
args: MoveBlockArgs<C>,
) -> Result<MoveBlockResponse> {
fn copy_item_at<'tree>(
location: &BlockLocation,
trees: &[BlockTree<'tree>],
) -> Option<Block<'tree>> {
for tree in trees {
if &BlockLocation::from(&tree.block) == location {
return Some(tree.block.clone());
}
if let Some(node) = copy_item_at(location, &tree.children) {
return Some(node);
}
}
None
}
let mut parser = Parser::new();
parser.set_language(args.language)?;
let tree = parser
.parse(&args.text, None)
.context("Failed to parse text")?;
let mut queries = vec![];
for query in args.queries {
queries.push(Query::new(args.language, &query)?);
}
let subtrees = code_blocks::get_query_subtrees(&queries, &tree, &args.text);
let src_block = copy_item_at(&args.src_block, &subtrees).context("Failed to find src item")?;
let dst_item = copy_item_at(&args.dst_block, &subtrees).context("Failed to find dst item")?;
code_blocks::move_block(
src_block,
dst_item,
&args.text,
args.assert_move_legal_fn,
args.force,
)
}