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
|
//! this mod achieves the transformation of a string containing
//! one or several commands into a vec of parsed commands
use {
super::{Command, CommandParts},
crate::{
app::AppContext,
errors::ProgramError,
verb::*,
},
};
/// an unparsed sequence with its separator (which may be
/// different from the one provided by local_separator())
#[derive(Debug, Clone)]
pub struct Sequence {
pub raw: String,
pub separator: String,
}
impl Sequence {
/// return the separator to use to parse sequences.
pub fn local_separator() -> String {
match std::env::var("BROOT_CMD_SEPARATOR") {
Ok(sep) if !sep.is_empty() => sep,
_ => String::from(";"),
}
}
pub fn new<S: Into<String>>(raw: S, separator: Option<S>) -> Self {
Self {
raw: raw.into(),
separator: separator.map_or_else(Sequence::local_separator, |s| s.into()),
}
}
pub fn new_single(cmd: String) -> Self {
Self {
separator: "".to_string(),
raw: cmd,
}
}
pub fn new_local(raw: String) -> Self {
Self {
separator: Self::local_separator(),
raw,
}
}
pub fn parse(
&self,
con: &AppContext,
) -> Result<Vec<(String, Command)>, ProgramError> {
debug!("Splitting cmd sequence with {:?}", &self.separator);
let mut commands = Vec::new();
if self.separator.is_empty() {
add_commands(&self.raw, &mut commands, con)?;
} else {
for input in self.raw.split(&self.separator) {
add_commands(input, &mut commands, con)?;
}
}
Ok(commands)
}
pub fn has_selection_group(&self) -> bool {
str_has_selection_group(&self.raw)
}
pub fn has_other_panel_group(&self) -> bool {
str_has_other_panel_group(&self.raw)
}
}
/// an input may be made of two parts:
/// - a search pattern
/// - a verb followed by its arguments
/// we need to build a command for each part so
/// that the search is effectively done before
/// the verb invocation
fn add_commands(
input: &str,
commands: &mut Vec<(String, Command)>,
con: &AppContext,
) -> Result<(), ProgramError> {
let raw_parts = CommandParts::from(input.to_string());
let (pattern, verb_invocation) = raw_parts.split();
if let Some(pattern) = pattern {
commands.push((input.to_string(), Command::from_parts(pattern, false)));
}
if let Some(verb_invocation) = verb_invocation {
let command = Command::from_parts(verb_invocation, true);
if let Command::VerbInvocate(invocation) = &command {
// we check that the verb exists to avoid running a sequence
// of actions with some missing
match con.verb_store.search_prefix(&invocation.name) {
PrefixSearchResult::NoMatch => {
return Err(ProgramError::UnknownVerb {
name: invocation.name.to_string(),
});
}
PrefixSearchResult::Matches(_) => {
return Err(ProgramError::AmbiguousVerbName {
name: invocation.name.to_string(),
});
}
_ => {}
}
commands.push((input.to_string(), command));
}
}
Ok(())
}
|