mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-11 20:48:49 +00:00
parent
2d42baac35
commit
c31b9f430f
4 changed files with 74 additions and 15 deletions
|
@ -11,7 +11,9 @@ command [OPTIONS] COMMANDNAME [ARGS...]
|
||||||
|
|
||||||
The following options are available:
|
The following options are available:
|
||||||
|
|
||||||
- `-s` or `--search` returns the name of the disk file that would be executed, or nothing if no file with the specified name could be found in the `$PATH`.
|
- `-a` or `--all` returns all the external commands that are found in `$PATH` in the order they are found.
|
||||||
|
|
||||||
|
- `-s` or `--search` returns the name of the external command that would be executed, or nothing if no file with the specified name could be found in the `$PATH`.
|
||||||
|
|
||||||
With the `-s` option, `command` treats every argument as a separate command to look up and sets the exit status to 0 if any of the specified commands were found, or 1 if no commands could be found. Additionally passing a `-q` or `--quiet` option prevents any paths from being printed, like the `type -q`, for testing only the exit status.
|
With the `-s` option, `command` treats every argument as a separate command to look up and sets the exit status to 0 if any of the specified commands were found, or 1 if no commands could be found. Additionally passing a `-q` or `--quiet` option prevents any paths from being printed, like the `type -q`, for testing only the exit status.
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,8 @@
|
||||||
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
#include "builtin.h"
|
#include "builtin.h"
|
||||||
#include "builtin_command.h"
|
#include "builtin_command.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
@ -16,9 +18,11 @@ struct command_cmd_opts_t {
|
||||||
bool print_help = false;
|
bool print_help = false;
|
||||||
bool find_path = false;
|
bool find_path = false;
|
||||||
bool quiet = false;
|
bool quiet = false;
|
||||||
|
bool all_paths = false;
|
||||||
};
|
};
|
||||||
static const wchar_t *short_options = L"hqsv";
|
static const wchar_t *short_options = L":ahqsv";
|
||||||
static const struct woption long_options[] = {{L"help", no_argument, NULL, 'h'},
|
static const struct woption long_options[] = {{L"help", no_argument, NULL, 'h'},
|
||||||
|
{L"all", no_argument, NULL, 'a'},
|
||||||
{L"quiet", no_argument, NULL, 'q'},
|
{L"quiet", no_argument, NULL, 'q'},
|
||||||
{L"search", no_argument, NULL, 's'},
|
{L"search", no_argument, NULL, 's'},
|
||||||
{NULL, 0, NULL, 0}};
|
{NULL, 0, NULL, 0}};
|
||||||
|
@ -30,6 +34,10 @@ static int parse_cmd_opts(command_cmd_opts_t &opts, int *optind, int argc, wchar
|
||||||
wgetopter_t w;
|
wgetopter_t w;
|
||||||
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, NULL)) != -1) {
|
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, NULL)) != -1) {
|
||||||
switch (opt) {
|
switch (opt) {
|
||||||
|
case 'a': {
|
||||||
|
opts.all_paths = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
case 'h': {
|
case 'h': {
|
||||||
opts.print_help = true;
|
opts.print_help = true;
|
||||||
break;
|
break;
|
||||||
|
@ -74,7 +82,7 @@ int builtin_command(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
||||||
return STATUS_CMD_OK;
|
return STATUS_CMD_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!opts.find_path) {
|
if (!opts.find_path && !opts.all_paths) {
|
||||||
builtin_print_help(parser, streams, cmd, streams.out);
|
builtin_print_help(parser, streams, cmd, streams.out);
|
||||||
return STATUS_INVALID_ARGS;
|
return STATUS_INVALID_ARGS;
|
||||||
}
|
}
|
||||||
|
@ -82,10 +90,18 @@ int builtin_command(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
||||||
int found = 0;
|
int found = 0;
|
||||||
for (int idx = optind; argv[idx]; ++idx) {
|
for (int idx = optind; argv[idx]; ++idx) {
|
||||||
const wchar_t *command_name = argv[idx];
|
const wchar_t *command_name = argv[idx];
|
||||||
wcstring path;
|
if (opts.all_paths) {
|
||||||
if (path_get_path(command_name, &path)) {
|
wcstring_list_t paths = path_get_paths(command_name);
|
||||||
if (!opts.quiet) streams.out.append_format(L"%ls\n", path.c_str());
|
for (auto path : paths) {
|
||||||
++found;
|
if (!opts.quiet) streams.out.append_format(L"%ls\n", path.c_str());
|
||||||
|
++found;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
wcstring path;
|
||||||
|
if (path_get_path(command_name, &path)) {
|
||||||
|
if (!opts.quiet) streams.out.append_format(L"%ls\n", path.c_str());
|
||||||
|
++found;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
39
src/path.cpp
39
src/path.cpp
|
@ -25,10 +25,10 @@
|
||||||
|
|
||||||
static bool path_get_path_core(const wcstring &cmd, wcstring *out_path,
|
static bool path_get_path_core(const wcstring &cmd, wcstring *out_path,
|
||||||
const env_var_t &bin_path_var) {
|
const env_var_t &bin_path_var) {
|
||||||
int err = ENOENT;
|
|
||||||
debug(3, L"path_get_path( '%ls' )", cmd.c_str());
|
debug(3, L"path_get_path( '%ls' )", cmd.c_str());
|
||||||
|
|
||||||
// If the command has a slash, it must be a full path.
|
// If the command has a slash, it must be an absolute or relative path and thus we don't bother
|
||||||
|
// looking for a matching command.
|
||||||
if (cmd.find(L'/') != wcstring::npos) {
|
if (cmd.find(L'/') != wcstring::npos) {
|
||||||
if (waccess(cmd, X_OK) != 0) {
|
if (waccess(cmd, X_OK) != 0) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -46,6 +46,7 @@ static bool path_get_path_core(const wcstring &cmd, wcstring *out_path,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int err = ENOENT;
|
||||||
wcstring bin_path;
|
wcstring bin_path;
|
||||||
if (!bin_path_var.missing()) {
|
if (!bin_path_var.missing()) {
|
||||||
bin_path = bin_path_var;
|
bin_path = bin_path_var;
|
||||||
|
@ -105,6 +106,40 @@ bool path_get_path(const wcstring &cmd, wcstring *out_path) {
|
||||||
return path_get_path_core(cmd, out_path, env_get_string(L"PATH"));
|
return path_get_path_core(cmd, out_path, env_get_string(L"PATH"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wcstring_list_t path_get_paths(const wcstring &cmd) {
|
||||||
|
debug(3, L"path_get_paths('%ls')", cmd.c_str());
|
||||||
|
wcstring_list_t paths;
|
||||||
|
|
||||||
|
// If the command has a slash, it must be an absolute or relative path and thus we don't bother
|
||||||
|
// looking for matching commands in the PATH var.
|
||||||
|
if (cmd.find(L'/') != wcstring::npos) {
|
||||||
|
struct stat buff;
|
||||||
|
if (wstat(cmd, &buff)) return paths;
|
||||||
|
if (!S_ISREG(buff.st_mode)) return paths;
|
||||||
|
if (waccess(cmd, X_OK)) return paths;
|
||||||
|
paths.push_back(cmd);
|
||||||
|
return paths;
|
||||||
|
}
|
||||||
|
|
||||||
|
wcstring env_path = env_get_string(L"PATH");
|
||||||
|
std::vector<wcstring> pathsv;
|
||||||
|
tokenize_variable_array(env_path, pathsv);
|
||||||
|
for (auto path : pathsv) {
|
||||||
|
if (path.empty()) continue;
|
||||||
|
append_path_component(path, cmd);
|
||||||
|
if (waccess(path, X_OK) == 0) {
|
||||||
|
struct stat buff;
|
||||||
|
if (wstat(path, &buff) == -1) {
|
||||||
|
if (errno != EACCES) wperror(L"stat");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (S_ISREG(buff.st_mode)) paths.push_back(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return paths;
|
||||||
|
}
|
||||||
|
|
||||||
bool path_get_cdpath(const wcstring &dir, wcstring *out, const wchar_t *wd,
|
bool path_get_cdpath(const wcstring &dir, wcstring *out, const wchar_t *wd,
|
||||||
const env_vars_snapshot_t &env_vars) {
|
const env_vars_snapshot_t &env_vars) {
|
||||||
int err = ENOENT;
|
int err = ENOENT;
|
||||||
|
|
18
src/path.h
18
src/path.h
|
@ -29,16 +29,22 @@ bool path_get_config(wcstring &path);
|
||||||
/// \return whether the directory was returned successfully
|
/// \return whether the directory was returned successfully
|
||||||
bool path_get_data(wcstring &path);
|
bool path_get_data(wcstring &path);
|
||||||
|
|
||||||
/// Finds the full path of an executable. Returns YES if successful.
|
/// Finds the full path of an executable.
|
||||||
///
|
///
|
||||||
/// \param cmd The name of the executable.
|
/// Args:
|
||||||
/// \param output_or_NULL If non-NULL, store the full path.
|
/// cmd - The name of the executable.
|
||||||
/// \param vars The environment variables snapshot to use
|
/// output_or_NULL - If non-NULL, store the full path.
|
||||||
/// \return 0 if the command can not be found, the path of the command otherwise. The result should
|
/// vars - The environment variables snapshot to use
|
||||||
/// be freed with free().
|
///
|
||||||
|
/// Returns:
|
||||||
|
/// false if the command can not be found else true. The result
|
||||||
|
/// should be freed with free().
|
||||||
bool path_get_path(const wcstring &cmd, wcstring *output_or_NULL,
|
bool path_get_path(const wcstring &cmd, wcstring *output_or_NULL,
|
||||||
const env_vars_snapshot_t &vars = env_vars_snapshot_t::current());
|
const env_vars_snapshot_t &vars = env_vars_snapshot_t::current());
|
||||||
|
|
||||||
|
/// Return all the paths that match the given command.
|
||||||
|
wcstring_list_t path_get_paths(const wcstring &cmd);
|
||||||
|
|
||||||
/// Returns the full path of the specified directory, using the CDPATH variable as a list of base
|
/// Returns the full path of the specified directory, using the CDPATH variable as a list of base
|
||||||
/// directories for relative paths. The returned string is allocated using halloc and the specified
|
/// directories for relative paths. The returned string is allocated using halloc and the specified
|
||||||
/// context.
|
/// context.
|
||||||
|
|
Loading…
Reference in a new issue