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,
    )
}