mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-19 08:24:00 +00:00
1023d322e5
When fish expands a string that starts with a tilde, like `~/stuff/*`, it first must resolve the tilde (e.g. to the user's home directory) before passing it to wildcard expansion. The wildcard expansion will produce full paths like `/home/user/stuff/file`. fish then "unexpands" the home directory back to a tilde. Previously this was only used during completions, but in the next commit we plan to use it for string expansions as well. Rationalize this behavior by adding an explicit flag to request it and explain some subtleties about completions.
219 lines
9.2 KiB
C++
219 lines
9.2 KiB
C++
// Prototypes for string expansion functions. These functions perform several kinds of parameter
|
|
// expansion. There are a lot of issues with regards to memory allocation. Overall, these functions
|
|
// would benefit from using a more clever memory allocation scheme, perhaps an evil combination of
|
|
// talloc, string buffers and reference counting.
|
|
#ifndef FISH_EXPAND_H
|
|
#define FISH_EXPAND_H
|
|
|
|
#include "config.h"
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <map>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "common.h"
|
|
#include "enum_set.h"
|
|
#include "maybe.h"
|
|
#include "parse_constants.h"
|
|
|
|
class environment_t;
|
|
class env_var_t;
|
|
class environment_t;
|
|
class operation_context_t;
|
|
|
|
/// Set of flags controlling expansions.
|
|
enum class expand_flag {
|
|
/// Skip command substitutions.
|
|
skip_cmdsubst,
|
|
/// Skip variable expansion.
|
|
skip_variables,
|
|
/// Skip wildcard expansion.
|
|
skip_wildcards,
|
|
/// The expansion is being done for tab or auto completions. Returned completions may have the
|
|
/// wildcard as a prefix instead of a match.
|
|
for_completions,
|
|
/// Only match files that are executable by the current user.
|
|
executables_only,
|
|
/// Only match directories.
|
|
directories_only,
|
|
/// Generate descriptions, stored in the description field of completions.
|
|
gen_descriptions,
|
|
/// Un-expand home directories to tildes after.
|
|
preserve_home_tildes,
|
|
/// Allow fuzzy matching.
|
|
fuzzy_match,
|
|
/// Disallow directory abbreviations like /u/l/b for /usr/local/bin. Only applicable if
|
|
/// fuzzy_match is set.
|
|
no_fuzzy_directories,
|
|
/// Do expansions specifically to support cd. This means using CDPATH as a list of potential
|
|
/// working directories, and to use logical instead of physical paths.
|
|
special_for_cd,
|
|
/// Do expansions specifically for cd autosuggestion. This is to differentiate between cd
|
|
/// completions and cd autosuggestions.
|
|
special_for_cd_autosuggestion,
|
|
/// Do expansions specifically to support external command completions. This means using PATH as
|
|
/// a list of potential working directories.
|
|
special_for_command,
|
|
|
|
COUNT,
|
|
};
|
|
|
|
template <>
|
|
struct enum_info_t<expand_flag> {
|
|
static constexpr auto count = expand_flag::COUNT;
|
|
};
|
|
|
|
using expand_flags_t = enum_set_t<expand_flag>;
|
|
|
|
class completion_t;
|
|
using completion_list_t = std::vector<completion_t>;
|
|
class completion_receiver_t;
|
|
|
|
enum : wchar_t {
|
|
/// Character representing a home directory.
|
|
HOME_DIRECTORY = EXPAND_RESERVED_BASE,
|
|
/// Character representing process expansion for %self.
|
|
PROCESS_EXPAND_SELF,
|
|
/// Character representing variable expansion.
|
|
VARIABLE_EXPAND,
|
|
/// Character representing variable expansion into a single element.
|
|
VARIABLE_EXPAND_SINGLE,
|
|
/// Character representing the start of a bracket expansion.
|
|
BRACE_BEGIN,
|
|
/// Character representing the end of a bracket expansion.
|
|
BRACE_END,
|
|
/// Character representing separation between two bracket elements.
|
|
BRACE_SEP,
|
|
/// Character that takes the place of any whitespace within non-quoted text in braces
|
|
BRACE_SPACE,
|
|
/// Separate subtokens in a token with this character.
|
|
INTERNAL_SEPARATOR,
|
|
/// Character representing an empty variable expansion. Only used transitively while expanding
|
|
/// variables.
|
|
VARIABLE_EXPAND_EMPTY,
|
|
/// This is a special pseudo-char that is not used other than to mark the end of the the special
|
|
/// characters so we can sanity check the enum range.
|
|
EXPAND_SENTINEL
|
|
};
|
|
|
|
/// These are the possible return values for expand_string.
|
|
struct expand_result_t {
|
|
enum result_t {
|
|
/// There was an error, for example, unmatched braces.
|
|
error,
|
|
/// Expansion succeeded.
|
|
ok,
|
|
/// Expansion was cancelled (e.g. control-C).
|
|
cancel,
|
|
/// Expansion succeeded, but a wildcard in the string matched no files,
|
|
/// so the output is empty.
|
|
wildcard_no_match,
|
|
};
|
|
|
|
/// The result of expansion.
|
|
result_t result;
|
|
|
|
/// If expansion resulted in an error, this is an appropriate value with which to populate
|
|
/// $status.
|
|
int status{0};
|
|
|
|
/* implicit */ expand_result_t(result_t result) : result(result) {}
|
|
|
|
/// operator== allows for comparison against result_t values.
|
|
bool operator==(result_t rhs) const { return result == rhs; }
|
|
bool operator!=(result_t rhs) const { return !(*this == rhs); }
|
|
|
|
/// Make an error value with the given status.
|
|
static expand_result_t make_error(int status) {
|
|
assert(status != 0 && "status cannot be 0 for an error result");
|
|
expand_result_t result(error);
|
|
result.status = status;
|
|
return result;
|
|
}
|
|
};
|
|
|
|
/// The string represented by PROCESS_EXPAND_SELF
|
|
#define PROCESS_EXPAND_SELF_STR L"%self"
|
|
#define PROCESS_EXPAND_SELF_STR_LEN 5
|
|
|
|
/// Perform various forms of expansion on in, such as tilde expansion (\~USER becomes the users home
|
|
/// directory), variable expansion (\$VAR_NAME becomes the value of the environment variable
|
|
/// VAR_NAME), cmdsubst expansion and wildcard expansion. The results are inserted into the list
|
|
/// out.
|
|
///
|
|
/// If the parameter does not need expansion, it is copied into the list out.
|
|
///
|
|
/// \param input The parameter to expand
|
|
/// \param output The list to which the result will be appended.
|
|
/// \param flags Specifies if any expansion pass should be skipped. Legal values are any combination
|
|
/// of skip_cmdsubst skip_variables and skip_wildcards
|
|
/// \param ctx The parser, variables, and cancellation checker for this operation. The parser may
|
|
/// be null. \param errors Resulting errors, or nullptr to ignore
|
|
///
|
|
/// \return An expand_result_t.
|
|
/// wildcard_no_match and wildcard_match are normal exit conditions used only on
|
|
/// strings containing wildcards to tell if the wildcard produced any matches.
|
|
__warn_unused expand_result_t expand_string(wcstring input, completion_list_t *output,
|
|
expand_flags_t flags, const operation_context_t &ctx,
|
|
parse_error_list_t *errors = nullptr);
|
|
|
|
/// Variant of string that inserts its results into a completion_receiver_t.
|
|
__warn_unused expand_result_t expand_string(wcstring input, completion_receiver_t *output,
|
|
expand_flags_t flags, const operation_context_t &ctx,
|
|
parse_error_list_t *errors = nullptr);
|
|
|
|
/// expand_one is identical to expand_string, except it will fail if in expands to more than one
|
|
/// string. This is used for expanding command names.
|
|
///
|
|
/// \param inout_str The parameter to expand in-place
|
|
/// \param flags Specifies if any expansion pass should be skipped. Legal values are any combination
|
|
/// of skip_cmdsubst skip_variables and skip_wildcards
|
|
/// \param ctx The parser, variables, and cancellation checker for this operation. The parser may be
|
|
/// null. \param errors Resulting errors, or nullptr to ignore
|
|
///
|
|
/// \return Whether expansion succeeded.
|
|
bool expand_one(wcstring &string, expand_flags_t flags, const operation_context_t &ctx,
|
|
parse_error_list_t *errors = nullptr);
|
|
|
|
/// Expand a command string like $HOME/bin/cmd into a command and list of arguments.
|
|
/// Return the command and arguments by reference.
|
|
/// If the expansion resulted in no or an empty command, the command will be an empty string. Note
|
|
/// that API does not distinguish between expansion resulting in an empty command (''), and
|
|
/// expansion resulting in no command (e.g. unset variable).
|
|
/// If \p skip_wildcards is true, then do not do wildcard expansion
|
|
/// \return an expand error.
|
|
expand_result_t expand_to_command_and_args(const wcstring &instr, const operation_context_t &ctx,
|
|
wcstring *out_cmd, wcstring_list_t *out_args,
|
|
parse_error_list_t *errors = nullptr,
|
|
bool skip_wildcards = false);
|
|
|
|
/// Convert the variable value to a human readable form, i.e. escape things, handle arrays, etc.
|
|
/// Suitable for pretty-printing.
|
|
wcstring expand_escape_variable(const env_var_t &var);
|
|
|
|
/// Convert a string value to a human readable form, i.e. escape things, handle arrays, etc.
|
|
/// Suitable for pretty-printing.
|
|
wcstring expand_escape_string(const wcstring &el);
|
|
|
|
/// Perform tilde expansion and nothing else on the specified string, which is modified in place.
|
|
///
|
|
/// \param input the string to tilde expand
|
|
void expand_tilde(wcstring &input, const environment_t &vars);
|
|
|
|
/// Perform the opposite of tilde expansion on the string, which is modified in place.
|
|
wcstring replace_home_directory_with_tilde(const wcstring &str, const environment_t &vars);
|
|
|
|
/// Abbreviation support. Expand src as an abbreviation, returning the expanded form if found,
|
|
/// none() if not.
|
|
maybe_t<wcstring> expand_abbreviation(const wcstring &src, const environment_t &vars);
|
|
|
|
/// \return a snapshot of all abbreviations as a map abbreviation->expansion.
|
|
/// The abbreviations are unescaped, i.e. they may not be valid variable identifiers (#6166).
|
|
std::map<wcstring, wcstring> get_abbreviations(const environment_t &vars);
|
|
|
|
// Terrible hacks
|
|
bool fish_xdm_login_hack_hack_hack_hack(std::vector<std::string> *cmds, int argc,
|
|
const char *const *argv);
|
|
#endif
|