Skip to content

Commit 1859850

Browse files
committed
Allow multiple '-o' options
The values of the multiple '-o' options are joined together with a comma. When parsing the joined option definition, allow multiple consecutive commas as separator. This requires that a comma as option-char must be escaped. Update man page and tutorial regarding characters, that need to be escaped.
1 parent 78bd725 commit 1859850

File tree

8 files changed

+47
-9
lines changed

8 files changed

+47
-9
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@ target/
44
*.add.spl
55
*.profraw
66
*.log
7+
.vscode/

CHANGELOG.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,9 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.
1414

1515
### CHANGED
1616

17-
* Generated code to test whether shell function exists.
17+
* Allow multiple '-o' / '--options' and join their values before parsing.
18+
19+
* Simplify Generated code to test whether shell function exists.
1820

1921
## [0.1.0] - 2023-09-09
2022

doc/parseargs.1.adoc

+3-2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ Typical usage in a script:
2525
== OPTIONS
2626

2727
*-o, --options* OPTION-DEFINITION::
28-
Define the options supported by the shell script.
28+
Define the options supported by the shell script. Can be given multiple times.
2929
See <<OD,*OPTION DEFINITION*>> below.
3030

3131
*-n, --name* NAME::
@@ -82,12 +82,13 @@ option-names::
8282
The option-names consist of one or more option chars or strings separated by colons.
8383
A single char defines an option that is given with one leading dash (e.g. `l` for `-l`). +
8484
Allowed option chars are all ASCII characters, except `-`, whitspace and control-characters.
85-
The following characters has to be escaped with a backslash in the definition: `#%+:=\`.
85+
The following characters has to be escaped with a backslash: `,#%+:=\`.
8686
Some of this characters need a backslash escape, when used on the command line.
8787
+
8888
A string defines a long option that is given with two leading dashes (e.g. `long` for `--long`).
8989
String may consist of any ASCII character, except `=`, whitespace and control characters.
9090
The minus (`-`) is allowed, as long it isn't the first character.
91+
The following characters has to be escaped with a backslash: `,#%+:\`.
9192

9293
option-type::
9394
Following option types are supported:

doc/tutorial.adoc

+11-2
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,12 @@ Always use exactly this notation.
7878
A simple `$@` (without the quotes) or `$*` might fail.
7979
See your shell documentation.
8080

81-
IMPORTANT: The option character may be any ASCII character, except for `-` (minus), whitespace or control character.
81+
[IMPORTANT]
82+
====
83+
The option character may be any ASCII character, except for `-` (minus), whitespace or control characters.
84+
85+
The characters `#`, `%`, `+`, `:`, `=`, `\` and `,` have to be escaped with a backslash.
86+
====
8287

8388
The trailing `|| exit 1` is just a safety net, in case something goes wrong during eval.
8489

@@ -201,8 +206,12 @@ Now we have two colon-separated options before the type marker (`#`, `=`).
201206
If a option is a single character, it defines a short option (`l` -> `-l`).
202207
With multiple characters it is a long option, that has two leading dashes (`long` -> `--long`).
203208

209+
[IMPORTANT]
210+
====
211+
Long options may contain any ASCII character, except for `=`, whitespace or control characters. The `-` (minus) is not allowed as the first character.
204212
205-
IMPORTANT: Long options may contain any ASCII character, except for `=`, whitespace or control character. The `-` (minus) is not allowed as the first character.
213+
The characters `#`, `%`, `+`, `:`, `\` and `,` have to be escaped with a backslash.
214+
====
206215

207216
Now our example script enables long output by either using `-l` or `--long` and the output file can be set with `-o out.file` or `--out-file out.file` or even `--out-file=out.file`.
208217

script-test/test-all-chars.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,6 @@ test_pa 'test "$opt" = true' -o '8#opt' -- -8
7777
test_pa 'test "$opt" = true' -o '9#opt' -- -9
7878
test_pa 'test "$opt" = true' -o '!#opt' -- -!
7979
test_pa 'test "$opt" = true' -o '$#opt' -- -$
80-
test_pa 'test "$opt" = true' -o ',#opt' -- -,
8180
test_pa 'test "$opt" = true' -o '.#opt' -- -.
8281
if [ -z "$IS_MSYS" ]; then
8382
test_pa 'test "$opt" = true' -o '/#opt' -- -/
@@ -113,5 +112,6 @@ test_pa 'test "$opt" = true' -o '\+#opt' -- -+
113112
test_pa 'test "$opt" = true' -o '\:#opt' -- -:
114113
test_pa 'test "$opt" = true' -o '\=#opt' -- -=
115114
test_pa 'test "$opt" = true' -o '\\#opt' -- -\\
115+
test_pa 'test "$opt" = true' -o '\,#opt' -- -,
116116

117117
end_test

src/main.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ const PARSEARGS_SHELL_VAR: &str = "PARSEARGS_SHELL";
4545
struct CmdLineArgs {
4646
/// Definition of supported shell options
4747
#[arg(short = 'o', long = "options", value_name = "OPT-DEFs")]
48-
options: Option<String>,
48+
options_list: Option<Vec<String>>,
4949

5050
/// Name of calling shell script. Used as prefix for error messages.
5151
#[arg(short = 'n', long = "name")]
@@ -513,8 +513,8 @@ fn parseargs(cmd_line_args: CmdLineArgs) -> ! {
513513
};
514514

515515
// parse the option definition string
516-
let result = if cmd_line_args.options.is_some() {
517-
opt_def::parse(&cmd_line_args.options.clone().unwrap())
516+
let result = if cmd_line_args.options_list.is_some() {
517+
opt_def::parse(&cmd_line_args.options_list.clone().unwrap().join(","))
518518
} else {
519519
Ok(Vec::new())
520520
};

src/opt_def.rs

+3
Original file line numberDiff line numberDiff line change
@@ -608,6 +608,9 @@ pub fn parse(opt_def_str: &String) -> Result<Vec<OptConfig>, String> {
608608
fn parse_opt_def_list(ps: &mut ParserSource) -> Result<Vec<OptConfig>, ParsingError> {
609609
let mut opt_def_list: Vec<OptConfig> = Vec::new();
610610
loop {
611+
// eat up consecutive commas
612+
while ps.next_if(|c| c == ',').is_some() {}
613+
611614
match parse_opt_def(ps) {
612615
Ok(od) => {
613616
opt_def_list.push(od);

tests/code_gen.rs

+22
Original file line numberDiff line numberDiff line change
@@ -262,6 +262,7 @@ fn test_mode_switch_option() {
262262
"parseargs: Options are mutual exclusive: -c/--copy, -m/--move",
263263
);
264264
}
265+
265266
#[test]
266267
fn test_combined_options() {
267268
let expected = &[
@@ -453,3 +454,24 @@ fn test_posix() {
453454
&["set -- 'test' '-d' '--' 'test2'"],
454455
);
455456
}
457+
458+
#[test]
459+
fn test_multiple_o_options() {
460+
exec::test_code_gen(
461+
&["-o", "d#debug", "-o", "l#long"],
462+
&["-d", "-l"],
463+
&["debug='true';", "long='true';", "set --"],
464+
);
465+
466+
exec::test_code_gen(
467+
&["-o", "d#debug,,", "-o", ",,l#long"],
468+
&["-d", "-l"],
469+
&["debug='true';", "long='true';", "set --"],
470+
);
471+
472+
exec::test_code_gen(
473+
&["-o", "d#debug", "-o", "l#long", "-o", "\\,#comma"],
474+
&["-d", "-l", "-,"],
475+
&["debug='true';", "long='true';", "comma='true';", "set --"],
476+
);
477+
}

0 commit comments

Comments
 (0)