Merge pull request #12 from elferherrera/similar-name

Similar name check for signature
This commit is contained in:
JT 2021-09-04 19:49:36 +12:00 committed by GitHub
commit f259992b4b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 155 additions and 23 deletions

View file

@ -0,0 +1,119 @@
use nu_parser::{Flag, PositionalArg, Signature, SyntaxShape};
#[test]
fn test_signature() {
let signature = Signature::new("new_signature");
let from_build = Signature::build("new_signature");
// asserting partial eq implementation
assert_eq!(signature, from_build);
// constructing signature with description
let signature = Signature::new("signature").desc("example usage");
assert_eq!(signature.usage, "example usage".to_string())
}
#[test]
fn test_signature_chained() {
let signature = Signature::new("new_signature")
.desc("description")
.required("required", SyntaxShape::String, "required description")
.optional("optional", SyntaxShape::String, "optional description")
.required_named(
"req_named",
SyntaxShape::String,
"required named description",
Some('r'),
)
.named("named", SyntaxShape::String, "named description", Some('n'))
.switch("switch", "switch description", None)
.rest(SyntaxShape::String, "rest description");
assert_eq!(signature.required_positional.len(), 1);
assert_eq!(signature.optional_positional.len(), 1);
assert_eq!(signature.named.len(), 3);
assert!(signature.rest_positional.is_some());
assert_eq!(signature.get_shorts(), vec!['r', 'n']);
assert_eq!(signature.get_names(), vec!["req_named", "named", "switch"]);
assert_eq!(signature.num_positionals(), 2);
assert_eq!(
signature.get_positional(0),
Some(PositionalArg {
name: "required".to_string(),
desc: "required description".to_string(),
shape: SyntaxShape::String,
var_id: None
})
);
assert_eq!(
signature.get_positional(1),
Some(PositionalArg {
name: "optional".to_string(),
desc: "optional description".to_string(),
shape: SyntaxShape::String,
var_id: None
})
);
assert_eq!(
signature.get_positional(2),
Some(PositionalArg {
name: "rest".to_string(),
desc: "rest description".to_string(),
shape: SyntaxShape::String,
var_id: None
})
);
assert_eq!(
signature.get_long_flag("req_named"),
Some(Flag {
long: "req_named".to_string(),
short: Some('r'),
arg: Some(SyntaxShape::String),
required: true,
desc: "required named description".to_string(),
var_id: None
})
);
assert_eq!(
signature.get_short_flag('r'),
Some(Flag {
long: "req_named".to_string(),
short: Some('r'),
arg: Some(SyntaxShape::String),
required: true,
desc: "required named description".to_string(),
var_id: None
})
);
}
#[test]
#[should_panic(expected = "There may be duplicate short flags, such as -h")]
fn test_signature_same_short() {
// Creating signature with same short name should panic
Signature::new("new_signature")
.required_named(
"required_named",
SyntaxShape::String,
"required named description",
Some('n'),
)
.named("named", SyntaxShape::String, "named description", Some('n'));
}
#[test]
#[should_panic(expected = "There may be duplicate name flags, such as --help")]
fn test_signature_same_name() {
// Creating signature with same short name should panic
Signature::new("new_signature")
.required_named(
"name",
SyntaxShape::String,
"required named description",
Some('r'),
)
.named("name", SyntaxShape::String, "named description", Some('n'));
}

View file

@ -6,7 +6,7 @@ use crate::SyntaxShape;
use crate::Value; use crate::Value;
use crate::VarId; use crate::VarId;
#[derive(Debug, Clone)] #[derive(Debug, Clone, PartialEq, Eq)]
pub struct Flag { pub struct Flag {
pub long: String, pub long: String,
pub short: Option<char>, pub short: Option<char>,
@ -127,10 +127,8 @@ impl Signature {
desc: impl Into<String>, desc: impl Into<String>,
short: Option<char>, short: Option<char>,
) -> Signature { ) -> Signature {
let s = short.map(|c| { let (name, s) = self.check_names(name, short);
debug_assert!(!self.get_shorts().contains(&c));
c
});
self.named.push(Flag { self.named.push(Flag {
long: name.into(), long: name.into(),
short: s, short: s,
@ -151,10 +149,8 @@ impl Signature {
desc: impl Into<String>, desc: impl Into<String>,
short: Option<char>, short: Option<char>,
) -> Signature { ) -> Signature {
let s = short.map(|c| { let (name, s) = self.check_names(name, short);
debug_assert!(!self.get_shorts().contains(&c));
c
});
self.named.push(Flag { self.named.push(Flag {
long: name.into(), long: name.into(),
short: s, short: s,
@ -174,13 +170,7 @@ impl Signature {
desc: impl Into<String>, desc: impl Into<String>,
short: Option<char>, short: Option<char>,
) -> Signature { ) -> Signature {
let s = short.map(|c| { let (name, s) = self.check_names(name, short);
debug_assert!(
!self.get_shorts().contains(&c),
"There may be duplicate short flags, such as -h"
);
c
});
self.named.push(Flag { self.named.push(Flag {
long: name.into(), long: name.into(),
@ -190,18 +180,41 @@ impl Signature {
desc: desc.into(), desc: desc.into(),
var_id: None, var_id: None,
}); });
self self
} }
/// Get list of the short-hand flags /// Get list of the short-hand flags
pub fn get_shorts(&self) -> Vec<char> { pub fn get_shorts(&self) -> Vec<char> {
let mut shorts = Vec::new(); self.named.iter().filter_map(|f| f.short).collect()
for Flag { short, .. } in &self.named {
if let Some(c) = short {
shorts.push(*c);
} }
/// Get list of the long-hand flags
pub fn get_names(&self) -> Vec<String> {
self.named.iter().map(|f| f.long.clone()).collect()
} }
shorts
/// Checks if short or long are already present
/// Panics if one of them is found
fn check_names(&self, name: impl Into<String>, short: Option<char>) -> (String, Option<char>) {
let s = short.map(|c| {
debug_assert!(
!self.get_shorts().contains(&c),
"There may be duplicate short flags, such as -h"
);
c
});
let name = {
let name = name.into();
debug_assert!(
!self.get_names().contains(&name),
"There may be duplicate name flags, such as --help"
);
name
};
(name, s)
} }
pub fn get_positional(&self, position: usize) -> Option<PositionalArg> { pub fn get_positional(&self, position: usize) -> Option<PositionalArg> {