highlighter: underline valid "cd" arguments also if they come from CDPATH

When visiting the "cd" node, we mark invalid paths as error, but don't
underline valid paths.  This works fine most of the time because we later
underline paths (for any command, not just "cd").
However the latter check fails to honor CDPATH.  Let's correct that, which
also allows to simplify the logic.
This commit is contained in:
Johannes Altmanninger 2022-10-26 15:46:56 +02:00
parent dfb0c00d72
commit 861ac00a61
2 changed files with 24 additions and 23 deletions

View file

@ -5348,6 +5348,7 @@ static void test_highlighting() {
if (!pushd("test/fish_highlight_test/")) return; if (!pushd("test/fish_highlight_test/")) return;
cleanup_t pop{[] { popd(); }}; cleanup_t pop{[] { popd(); }};
if (system("mkdir -p dir")) err(L"mkdir failed"); if (system("mkdir -p dir")) err(L"mkdir failed");
if (system("mkdir -p cdpath-entry/dir-in-cdpath")) err(L"mkdir failed");
if (system("touch foo")) err(L"touch failed"); if (system("touch foo")) err(L"touch failed");
if (system("touch bar")) err(L"touch failed"); if (system("touch bar")) err(L"touch failed");
@ -5418,6 +5419,11 @@ static void test_highlighting() {
{L"definitely_not_a_directory", highlight_role_t::error}, {L"definitely_not_a_directory", highlight_role_t::error},
}); });
highlight_tests.push_back({
{L"cd", highlight_role_t::command},
{L"dir-in-cdpath", param_valid_path},
});
// Command substitutions. // Command substitutions.
highlight_tests.push_back({ highlight_tests.push_back({
{L"echo", highlight_role_t::command}, {L"echo", highlight_role_t::command},
@ -5691,6 +5697,7 @@ static void test_highlighting() {
// Verify variables and wildcards in commands using /bin/cat. // Verify variables and wildcards in commands using /bin/cat.
vars.set(L"VARIABLE_IN_COMMAND", ENV_LOCAL, {L"a"}); vars.set(L"VARIABLE_IN_COMMAND", ENV_LOCAL, {L"a"});
vars.set(L"VARIABLE_IN_COMMAND2", ENV_LOCAL, {L"at"}); vars.set(L"VARIABLE_IN_COMMAND2", ENV_LOCAL, {L"at"});
vars.set(L"CDPATH", ENV_LOCAL, {L"./cdpath-entry"});
highlight_tests.push_back( highlight_tests.push_back(
{{L"/bin/ca", highlight_role_t::command, ns}, {L"*", highlight_role_t::operat, ns}}); {{L"/bin/ca", highlight_role_t::command, ns}, {L"*", highlight_role_t::operat, ns}});

View file

@ -1008,18 +1008,31 @@ void highlighter_t::visit(const ast::semi_nl_t &semi_nl) {
void highlighter_t::visit(const ast::argument_t &arg, bool cmd_is_cd, bool options_allowed) { void highlighter_t::visit(const ast::argument_t &arg, bool cmd_is_cd, bool options_allowed) {
color_as_argument(arg, options_allowed); color_as_argument(arg, options_allowed);
if (cmd_is_cd && io_still_ok()) { if (!io_still_ok()) {
return;
}
// Underline every valid path.
bool is_valid_path = false;
if (cmd_is_cd) {
// Mark this as an error if it's not 'help' and not a valid cd path. // Mark this as an error if it's not 'help' and not a valid cd path.
wcstring param = arg.source(this->buff); wcstring param = arg.source(this->buff);
if (expand_one(param, expand_flag::skip_cmdsubst, ctx)) { if (expand_one(param, expand_flag::skip_cmdsubst, ctx)) {
bool is_help = bool is_help =
string_prefixes_string(param, L"--help") || string_prefixes_string(param, L"-h"); string_prefixes_string(param, L"--help") || string_prefixes_string(param, L"-h");
if (!is_help && if (!is_help) {
!is_potential_cd_path(param, working_directory, ctx, PATH_EXPAND_TILDE)) { is_valid_path =
this->color_node(arg, highlight_role_t::error); is_potential_cd_path(param, working_directory, ctx, PATH_EXPAND_TILDE);
if (!is_valid_path) {
this->color_node(arg, highlight_role_t::error);
}
} }
} }
} else if (range_is_potential_path(buff, arg.range, ctx, working_directory)) {
is_valid_path = true;
} }
if (is_valid_path)
for (size_t i = arg.range.start, end = arg.range.start + arg.range.length; i < end; i++)
this->color_array.at(i).valid_path = true;
} }
void highlighter_t::visit(const ast::variable_assignment_t &varas) { void highlighter_t::visit(const ast::variable_assignment_t &varas) {
@ -1269,25 +1282,6 @@ highlighter_t::color_array_t highlighter_t::highlight() {
this->color_range(r, highlight_role_t::error); this->color_range(r, highlight_role_t::error);
} }
// Underline every valid path.
if (io_still_ok()) {
for (const ast::node_t &node : ast) {
const auto arg = node.try_as<ast::argument_t>();
if (!arg || arg->unsourced) continue;
if (ctx.check_cancel()) break;
if (range_is_potential_path(buff, arg->range, ctx, working_directory)) {
// Don't color highlight_role_t::error because it looks dorky. For example,
// trying to cd into a non-directory would show an underline and also red.
for (size_t i = arg->range.start, end = arg->range.start + arg->range.length;
i < end; i++) {
if (this->color_array.at(i).foreground != highlight_role_t::error) {
this->color_array.at(i).valid_path = true;
}
}
}
}
}
return std::move(color_array); return std::move(color_array);
} }
} // namespace } // namespace