Allow user to set and unset path property of variables

This adds flags --path and --unpath to builtin set, analogous to
--export and --unexport. These flags change whether a variable is
marked as a path variable.

Universal variables cannot yet be path variables.
This commit is contained in:
ridiculousfish 2018-10-06 20:22:35 -07:00
parent 5947aa0171
commit f4d666f56c
4 changed files with 47 additions and 6 deletions

View file

@ -37,6 +37,8 @@ struct set_cmd_opts_t {
bool erase = false; bool erase = false;
bool list = false; bool list = false;
bool unexport = false; bool unexport = false;
bool pathvar = false;
bool unpathvar = false;
bool universal = false; bool universal = false;
bool query = false; bool query = false;
bool shorten_ok = true; bool shorten_ok = true;
@ -45,6 +47,12 @@ struct set_cmd_opts_t {
bool preserve_failure_exit_status = true; bool preserve_failure_exit_status = true;
}; };
/// Values used for long-only options.
enum {
opt_path = 1,
opt_unpath = 2,
};
// Variables used for parsing the argument list. This command is atypical in using the "+" // Variables used for parsing the argument list. This command is atypical in using the "+"
// (REQUIRE_ORDER) option for flag parsing. This is not typical of most fish commands. It means // (REQUIRE_ORDER) option for flag parsing. This is not typical of most fish commands. It means
// we stop scanning for flags when the first non-flag argument is seen. // we stop scanning for flags when the first non-flag argument is seen.
@ -56,6 +64,7 @@ static const struct woption long_options[] = {
{L"universal", no_argument, NULL, 'U'}, {L"long", no_argument, NULL, 'L'}, {L"universal", no_argument, NULL, 'U'}, {L"long", no_argument, NULL, 'L'},
{L"query", no_argument, NULL, 'q'}, {L"show", no_argument, NULL, 'S'}, {L"query", no_argument, NULL, 'q'}, {L"show", no_argument, NULL, 'S'},
{L"append", no_argument, NULL, 'a'}, {L"prepend", no_argument, NULL, 'p'}, {L"append", no_argument, NULL, 'a'}, {L"prepend", no_argument, NULL, 'p'},
{L"path", no_argument, NULL, opt_path}, {L"unpath", no_argument, NULL, opt_unpath},
{L"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0}}; {L"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0}};
// Hint for invalid path operation with a colon. // Hint for invalid path operation with a colon.
@ -122,6 +131,14 @@ static int parse_cmd_opts(set_cmd_opts_t &opts, int *optind, //!OCLINT(high ncs
opts.unexport = true; opts.unexport = true;
break; break;
} }
case opt_path: {
opts.pathvar = true;
break;
}
case opt_unpath: {
opts.unpathvar = true;
break;
}
case 'U': { case 'U': {
opts.universal = true; opts.universal = true;
break; break;
@ -184,6 +201,13 @@ static int validate_cmd_opts(const wchar_t *cmd, set_cmd_opts_t &opts, //!OCLIN
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
} }
// Variables can only have one path status.
if (opts.pathvar && opts.unpathvar) {
streams.err.append_format(BUILTIN_ERR_EXPUNEXP, cmd);
builtin_print_help(parser, streams, cmd, streams.err);
return STATUS_INVALID_ARGS;
}
// Trying to erase and (un)export at the same time doesn't make sense. // Trying to erase and (un)export at the same time doesn't make sense.
if (opts.erase && (opts.exportv || opts.unexport)) { if (opts.erase && (opts.exportv || opts.unexport)) {
streams.err.append_format(BUILTIN_ERR_COMBO, cmd); streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
@ -429,13 +453,15 @@ static void erase_values(wcstring_list_t &list, const std::vector<long> &indexes
} }
} }
static int compute_scope(set_cmd_opts_t &opts) { static env_mode_flags_t compute_scope(set_cmd_opts_t &opts) {
int scope = ENV_USER; int scope = ENV_USER;
if (opts.local) scope |= ENV_LOCAL; if (opts.local) scope |= ENV_LOCAL;
if (opts.global) scope |= ENV_GLOBAL; if (opts.global) scope |= ENV_GLOBAL;
if (opts.exportv) scope |= ENV_EXPORT; if (opts.exportv) scope |= ENV_EXPORT;
if (opts.unexport) scope |= ENV_UNEXPORT; if (opts.unexport) scope |= ENV_UNEXPORT;
if (opts.universal) scope |= ENV_UNIVERSAL; if (opts.universal) scope |= ENV_UNIVERSAL;
if (opts.pathvar) scope |= ENV_PATHVAR;
if (opts.unpathvar) scope |= ENV_UNPATHVAR;
return scope; return scope;
} }

View file

@ -21,4 +21,4 @@ echo $$paren
# Test tilde expansion # Test tilde expansion
#################### ####################
# Test delimiters # Test path variables

View file

@ -127,7 +127,17 @@ unlink $tmpdir/linkhome
rmdir $tmpdir/realhome rmdir $tmpdir/realhome
rmdir $tmpdir rmdir $tmpdir
logmsg Test delimiters logmsg Test path variables
set TEST_DELIMITER one two three set TEST_DELIMITER one two three
set TEST_DELIMITER_PATH one two three set TEST_DELIMITER_PATH one two three
echo "$TEST_DELIMITER" "$TEST_DELIMITER_PATH" echo TEST_DELIMITER: $TEST_DELIMITER "$TEST_DELIMITER"
echo TEST_DELIMITER_PATH: $TEST_DELIMITER_PATH "$TEST_DELIMITER_PATH"
set testvar ONE:TWO:THREE
echo "Not a path: $testvar" (count $testvar)
set --path testvar $testvar
echo "As a path: $testvar" (count $testvar)
set testvar "$testvar:FOUR"
echo "Appended path: $testvar" (count $testvar)
set --unpath testvar $testvar
echo "Back to normal variable: $testvar" (count $testvar)

View file

@ -75,5 +75,10 @@ All digits: 0
# Test tilde expansion # Test tilde expansion
#################### ####################
# Test delimiters # Test path variables
one two three one:two:three TEST_DELIMITER: one two three one two three
TEST_DELIMITER_PATH: one two three one:two:three
Not a path: ONE:TWO:THREE 1
As a path: ONE:TWO:THREE 3
Appended path: ONE:TWO:THREE:FOUR 4
Back to normal variable: ONE TWO THREE FOUR 4