2016-05-01 03:47:05 +00:00
|
|
|
// Functions for syntax highlighting.
|
2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2016-04-21 06:00:54 +00:00
|
|
|
// IWYU pragma: no_include <cstddef>
|
2016-05-01 03:47:05 +00:00
|
|
|
#include <dirent.h>
|
|
|
|
#include <errno.h>
|
2005-09-20 13:26:39 +00:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <wchar.h>
|
2017-02-11 02:47:02 +00:00
|
|
|
|
2013-10-13 23:58:40 +00:00
|
|
|
#include <algorithm>
|
2016-05-01 03:47:05 +00:00
|
|
|
#include <memory>
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <string>
|
2017-02-11 02:47:02 +00:00
|
|
|
#include <type_traits>
|
2017-08-19 16:55:06 +00:00
|
|
|
#include <unordered_map>
|
2017-08-19 20:29:52 +00:00
|
|
|
#include <unordered_set>
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <utility>
|
2005-09-20 13:26:39 +00:00
|
|
|
|
|
|
|
#include "builtin.h"
|
2016-05-01 03:47:05 +00:00
|
|
|
#include "color.h"
|
|
|
|
#include "common.h"
|
2005-09-20 13:26:39 +00:00
|
|
|
#include "env.h"
|
|
|
|
#include "expand.h"
|
2016-05-01 03:47:05 +00:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
|
|
|
#include "function.h"
|
2018-05-06 02:11:57 +00:00
|
|
|
#include "future_feature_flags.h"
|
2016-05-01 03:47:05 +00:00
|
|
|
#include "highlight.h"
|
2012-06-28 18:54:37 +00:00
|
|
|
#include "history.h"
|
2016-05-01 03:47:05 +00:00
|
|
|
#include "output.h"
|
|
|
|
#include "parse_constants.h"
|
|
|
|
#include "parse_util.h"
|
|
|
|
#include "path.h"
|
2018-01-20 19:58:57 +00:00
|
|
|
#include "tnode.h"
|
2016-05-01 03:47:05 +00:00
|
|
|
#include "tokenizer.h"
|
|
|
|
#include "wildcard.h"
|
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2018-01-14 00:26:27 +00:00
|
|
|
namespace g = grammar;
|
|
|
|
|
2013-10-09 01:41:35 +00:00
|
|
|
#define CURSOR_POSITION_INVALID ((size_t)(-1))
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Number of elements in the highlight_var array.
|
|
|
|
#define VAR_COUNT (sizeof(highlight_var) / sizeof(wchar_t *))
|
|
|
|
|
|
|
|
/// The environment variables used to specify the color of different tokens. This matches the order
|
|
|
|
/// in highlight_spec_t.
|
2017-05-02 04:44:30 +00:00
|
|
|
static const wchar_t *const highlight_var[] = {L"fish_color_normal",
|
|
|
|
L"fish_color_error",
|
|
|
|
L"fish_color_command",
|
|
|
|
L"fish_color_end",
|
|
|
|
L"fish_color_param",
|
|
|
|
L"fish_color_comment",
|
|
|
|
L"fish_color_match",
|
|
|
|
L"fish_color_search_match",
|
|
|
|
L"fish_color_operator",
|
|
|
|
L"fish_color_escape",
|
|
|
|
L"fish_color_quote",
|
|
|
|
L"fish_color_redirection",
|
|
|
|
L"fish_color_autosuggestion",
|
|
|
|
L"fish_color_selection",
|
|
|
|
L"fish_pager_color_prefix",
|
|
|
|
L"fish_pager_color_completion",
|
|
|
|
L"fish_pager_color_description",
|
|
|
|
L"fish_pager_color_progress",
|
|
|
|
L"fish_pager_color_secondary"};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-11-05 01:40:22 +00:00
|
|
|
/// Determine if the filesystem containing the given fd is case insensitive for lookups regardless
|
|
|
|
/// of whether it preserves the case when saving a pathname.
|
|
|
|
///
|
|
|
|
/// Returns:
|
|
|
|
/// false: the filesystem is not case insensitive
|
|
|
|
/// true: the file system is case insensitive
|
2017-08-19 23:27:24 +00:00
|
|
|
typedef std::unordered_map<wcstring, bool> case_sensitivity_cache_t;
|
2016-05-01 03:47:05 +00:00
|
|
|
bool fs_is_case_insensitive(const wcstring &path, int fd,
|
|
|
|
case_sensitivity_cache_t &case_sensitivity_cache) {
|
2012-06-16 21:08:58 +00:00
|
|
|
bool result = false;
|
|
|
|
#ifdef _PC_CASE_SENSITIVE
|
2016-05-01 03:47:05 +00:00
|
|
|
// Try the cache first.
|
2012-06-16 21:08:58 +00:00
|
|
|
case_sensitivity_cache_t::iterator cache = case_sensitivity_cache.find(path);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (cache != case_sensitivity_cache.end()) {
|
2012-06-16 21:08:58 +00:00
|
|
|
/* Use the cached value */
|
|
|
|
result = cache->second;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// Ask the system. A -1 value means error (so assume case sensitive), a 1 value means case
|
|
|
|
// sensitive, and a 0 value means case insensitive.
|
2012-06-16 21:08:58 +00:00
|
|
|
long ret = fpathconf(fd, _PC_CASE_SENSITIVE);
|
|
|
|
result = (ret == 0);
|
|
|
|
case_sensitivity_cache[path] = result;
|
|
|
|
}
|
2016-11-05 01:40:22 +00:00
|
|
|
#else
|
|
|
|
// Silence lint tools about the unused parameters.
|
|
|
|
UNUSED(path);
|
|
|
|
UNUSED(fd);
|
|
|
|
UNUSED(case_sensitivity_cache);
|
2012-06-16 21:08:58 +00:00
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Tests whether the specified string cpath is the prefix of anything we could cd to. directories
|
|
|
|
/// is a list of possible parent directories (typically either the working directory, or the
|
|
|
|
/// cdpath). This does I/O!
|
|
|
|
///
|
|
|
|
/// Hack: if out_suggested_cdpath is not NULL, it returns the autosuggestion for cd. This descends
|
|
|
|
/// the deepest unique directory hierarchy.
|
|
|
|
///
|
|
|
|
/// We expect the path to already be unescaped.
|
|
|
|
bool is_potential_path(const wcstring &potential_path_fragment, const wcstring_list_t &directories,
|
2018-09-19 04:03:01 +00:00
|
|
|
const environment_t &vars, path_flags_t flags) {
|
2011-12-27 03:18:46 +00:00
|
|
|
ASSERT_IS_BACKGROUND_THREAD();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-10-21 04:14:40 +00:00
|
|
|
const bool require_dir = static_cast<bool>(flags & PATH_REQUIRE_DIR);
|
2015-11-07 21:58:13 +00:00
|
|
|
wcstring clean_potential_path_fragment;
|
2012-11-19 00:30:30 +00:00
|
|
|
int has_magic = 0;
|
|
|
|
bool result = false;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2015-11-07 21:58:13 +00:00
|
|
|
wcstring path_with_magic(potential_path_fragment);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (flags & PATH_EXPAND_TILDE) expand_tilde(path_with_magic);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// debug( 1, L"%ls -> %ls ->%ls", path, tilde, unescaped );
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
for (size_t i = 0; i < path_with_magic.size(); i++) {
|
2015-11-07 21:58:13 +00:00
|
|
|
wchar_t c = path_with_magic.at(i);
|
2016-05-01 03:47:05 +00:00
|
|
|
switch (c) {
|
2018-10-10 21:26:29 +00:00
|
|
|
case PROCESS_EXPAND_SELF:
|
2012-11-19 08:31:03 +00:00
|
|
|
case VARIABLE_EXPAND:
|
|
|
|
case VARIABLE_EXPAND_SINGLE:
|
2018-03-10 19:16:07 +00:00
|
|
|
case BRACE_BEGIN:
|
|
|
|
case BRACE_END:
|
|
|
|
case BRACE_SEP:
|
2018-05-06 02:11:57 +00:00
|
|
|
case ANY_CHAR:
|
2012-11-19 08:31:03 +00:00
|
|
|
case ANY_STRING:
|
2016-05-01 03:47:05 +00:00
|
|
|
case ANY_STRING_RECURSIVE: {
|
2012-11-19 08:31:03 +00:00
|
|
|
has_magic = 1;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case INTERNAL_SEPARATOR: {
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
default: {
|
2015-11-07 21:58:13 +00:00
|
|
|
clean_potential_path_fragment.push_back(c);
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-12-27 03:18:46 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
if (has_magic || clean_potential_path_fragment.empty()) {
|
|
|
|
return result;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
// Don't test the same path multiple times, which can happen if the path is absolute and the
|
|
|
|
// CDPATH contains multiple entries.
|
2017-08-19 23:27:24 +00:00
|
|
|
std::unordered_set<wcstring> checked_paths;
|
2016-10-30 22:05:41 +00:00
|
|
|
|
|
|
|
// Keep a cache of which paths / filesystems are case sensitive.
|
|
|
|
case_sensitivity_cache_t case_sensitivity_cache;
|
|
|
|
|
|
|
|
for (size_t wd_idx = 0; wd_idx < directories.size() && !result; wd_idx++) {
|
|
|
|
const wcstring &wd = directories.at(wd_idx);
|
|
|
|
|
|
|
|
const wcstring abs_path = path_apply_working_directory(clean_potential_path_fragment, wd);
|
|
|
|
|
|
|
|
// Skip this if it's empty or we've already checked it.
|
|
|
|
if (abs_path.empty() || checked_paths.count(abs_path)) continue;
|
|
|
|
checked_paths.insert(abs_path);
|
|
|
|
|
|
|
|
// If we end with a slash, then it must be a directory.
|
|
|
|
bool must_be_full_dir = abs_path.at(abs_path.size() - 1) == L'/';
|
|
|
|
if (must_be_full_dir) {
|
|
|
|
struct stat buf;
|
|
|
|
if (0 == wstat(abs_path, &buf) && S_ISDIR(buf.st_mode)) {
|
|
|
|
result = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We do not end with a slash; it does not have to be a directory.
|
|
|
|
DIR *dir = NULL;
|
|
|
|
const wcstring dir_name = wdirname(abs_path);
|
|
|
|
const wcstring filename_fragment = wbasename(abs_path);
|
|
|
|
if (dir_name == L"/" && filename_fragment == L"/") {
|
|
|
|
// cd ///.... No autosuggestion.
|
|
|
|
result = true;
|
|
|
|
} else if ((dir = wopendir(dir_name))) {
|
|
|
|
// Check if we're case insensitive.
|
|
|
|
const bool do_case_insensitive =
|
|
|
|
fs_is_case_insensitive(dir_name, dirfd(dir), case_sensitivity_cache);
|
|
|
|
|
|
|
|
wcstring matched_file;
|
|
|
|
|
|
|
|
// We opened the dir_name; look for a string where the base name prefixes it Don't
|
|
|
|
// ask for the is_dir value unless we care, because it can cause extra filesystem
|
|
|
|
// access.
|
|
|
|
wcstring ent;
|
|
|
|
bool is_dir = false;
|
|
|
|
while (wreaddir_resolving(dir, dir_name, ent, require_dir ? &is_dir : NULL)) {
|
|
|
|
// Maybe skip directories.
|
|
|
|
if (require_dir && !is_dir) {
|
|
|
|
continue;
|
2011-12-27 03:18:46 +00:00
|
|
|
}
|
2016-02-18 00:25:19 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
if (string_prefixes_string(filename_fragment, ent) ||
|
|
|
|
(do_case_insensitive &&
|
|
|
|
string_prefixes_string_case_insensitive(filename_fragment, ent))) {
|
|
|
|
matched_file = ent; // we matched
|
|
|
|
break;
|
|
|
|
}
|
2011-12-27 03:18:46 +00:00
|
|
|
}
|
2016-10-30 22:05:41 +00:00
|
|
|
closedir(dir);
|
|
|
|
|
|
|
|
result = !matched_file.empty(); // we succeeded if we found a match
|
2011-12-27 03:18:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-10-30 22:05:41 +00:00
|
|
|
|
2012-05-08 00:31:24 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Given a string, return whether it prefixes a path that we could cd into. Return that path in
|
|
|
|
// out_path. Expects path to be unescaped.
|
|
|
|
static bool is_potential_cd_path(const wcstring &path, const wcstring &working_directory,
|
2018-09-19 04:03:01 +00:00
|
|
|
const environment_t &vars, path_flags_t flags) {
|
2012-05-08 00:31:24 +00:00
|
|
|
wcstring_list_t directories;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (string_prefixes_string(L"./", path)) {
|
|
|
|
// Ignore the CDPATH in this case; just use the working directory.
|
2012-05-14 03:19:02 +00:00
|
|
|
directories.push_back(working_directory);
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// Get the CDPATH.
|
2018-09-19 04:03:01 +00:00
|
|
|
auto cdpath = vars.get(L"CDPATH");
|
2018-01-30 20:36:50 +00:00
|
|
|
std::vector<wcstring> pathsv =
|
|
|
|
cdpath.missing_or_empty() ? wcstring_list_t{L"."} : cdpath->as_list();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-03-17 03:59:04 +00:00
|
|
|
for (auto next_path : pathsv) {
|
|
|
|
if (next_path.empty()) next_path = L".";
|
2016-05-01 03:47:05 +00:00
|
|
|
// Ensure that we use the working directory for relative cdpaths like ".".
|
2016-02-06 22:39:47 +00:00
|
|
|
directories.push_back(path_apply_working_directory(next_path, working_directory));
|
2012-05-14 03:19:02 +00:00
|
|
|
}
|
2012-05-08 00:31:24 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Call is_potential_path with all of these directories.
|
2018-09-19 04:03:01 +00:00
|
|
|
return is_potential_path(path, directories, vars, flags | PATH_REQUIRE_DIR);
|
2006-06-14 13:22:40 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Given a plain statement node in a parse tree, get the command and return it, expanded
|
|
|
|
// appropriately for commands. If we succeed, return true.
|
2018-01-13 23:36:14 +00:00
|
|
|
static bool plain_statement_get_expanded_command(const wcstring &src,
|
2018-01-14 00:26:27 +00:00
|
|
|
tnode_t<g::plain_statement> stmt,
|
2018-09-11 05:29:52 +00:00
|
|
|
const environment_t &vars, wcstring *out_cmd) {
|
2016-10-22 18:21:13 +00:00
|
|
|
// Get the command. Try expanding it. If we cannot, it's an error.
|
2018-01-13 23:36:14 +00:00
|
|
|
maybe_t<wcstring> cmd = command_for_plain_statement(stmt, src);
|
2018-09-01 18:45:15 +00:00
|
|
|
if (!cmd) return false;
|
2018-09-11 05:29:52 +00:00
|
|
|
expand_error_t err = expand_to_command_and_args(*cmd, vars, out_cmd, nullptr);
|
2018-09-01 18:45:15 +00:00
|
|
|
return err == EXPAND_OK || err == EXPAND_WILDCARD_MATCH;
|
2013-10-08 22:05:30 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
rgb_color_t highlight_get_color(highlight_spec_t highlight, bool is_background) {
|
2018-09-19 04:03:01 +00:00
|
|
|
// TODO: rationalize this principal_vars.
|
|
|
|
const auto &vars = env_stack_t::principal();
|
2014-01-15 09:01:25 +00:00
|
|
|
rgb_color_t result = rgb_color_t::normal();
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// If sloppy_background is set, then we look at the foreground color even if is_background is
|
|
|
|
// set.
|
2014-03-31 17:01:39 +00:00
|
|
|
bool treat_as_background = is_background && !(highlight & highlight_modifier_sloppy_background);
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Get the primary variable.
|
2014-01-15 09:01:25 +00:00
|
|
|
size_t idx = highlight_get_primary(highlight);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (idx >= VAR_COUNT) {
|
2014-01-16 07:33:34 +00:00
|
|
|
return rgb_color_t::normal();
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2018-09-19 04:03:01 +00:00
|
|
|
auto var = vars.get(highlight_var[idx]);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// debug( 1, L"%d -> %d -> %ls", highlight, idx, val );
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2018-09-19 04:03:01 +00:00
|
|
|
if (!var) var = vars.get(highlight_var[0]);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-08-28 07:25:41 +00:00
|
|
|
if (var) result = parse_color(*var, treat_as_background);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Handle modifiers.
|
|
|
|
if (highlight & highlight_modifier_valid_path) {
|
2018-09-19 04:03:01 +00:00
|
|
|
auto var2 = vars.get(L"fish_color_valid_path");
|
2017-08-28 07:25:41 +00:00
|
|
|
if (var2) {
|
|
|
|
rgb_color_t result2 = parse_color(*var2, is_background);
|
|
|
|
if (result.is_normal())
|
|
|
|
result = result2;
|
|
|
|
else {
|
|
|
|
if (result2.is_bold()) result.set_bold(true);
|
|
|
|
if (result2.is_underline()) result.set_underline(true);
|
|
|
|
if (result2.is_italics()) result.set_italics(true);
|
|
|
|
if (result2.is_dim()) result.set_dim(true);
|
|
|
|
if (result2.is_reverse()) result.set_reverse(true);
|
|
|
|
}
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (highlight & highlight_modifier_force_underline) {
|
2014-01-26 08:41:30 +00:00
|
|
|
result.set_underline(true);
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2012-11-19 00:30:30 +00:00
|
|
|
return result;
|
2006-05-26 16:46:38 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
static bool has_expand_reserved(const wcstring &str) {
|
2013-10-06 23:23:45 +00:00
|
|
|
bool result = false;
|
2016-05-01 03:47:05 +00:00
|
|
|
for (size_t i = 0; i < str.size(); i++) {
|
2013-10-06 23:23:45 +00:00
|
|
|
wchar_t wc = str.at(i);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (wc >= EXPAND_RESERVED_BASE && wc <= EXPAND_RESERVED_END) {
|
2013-10-06 23:23:45 +00:00
|
|
|
result = true;
|
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2013-10-06 23:23:45 +00:00
|
|
|
return result;
|
2008-02-04 23:09:05 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Parse a command line. Return by reference the last command, and the last argument to that command
|
2018-01-13 22:25:39 +00:00
|
|
|
// (as a string), if any. This is used by autosuggestions.
|
2018-09-11 05:29:52 +00:00
|
|
|
static bool autosuggest_parse_command(const wcstring &buff, const environment_t &vars,
|
|
|
|
wcstring *out_expanded_command, wcstring *out_last_arg) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Parse the buffer.
|
2013-10-08 22:05:30 +00:00
|
|
|
parse_node_tree_t parse_tree;
|
2016-05-01 03:47:05 +00:00
|
|
|
parse_tree_from_string(buff,
|
|
|
|
parse_flag_continue_after_error | parse_flag_accept_incomplete_tokens,
|
|
|
|
&parse_tree, NULL);
|
|
|
|
|
|
|
|
// Find the last statement.
|
2018-01-14 00:26:27 +00:00
|
|
|
auto last_statement = parse_tree.find_last_node<g::plain_statement>();
|
2018-01-13 23:36:14 +00:00
|
|
|
if (last_statement &&
|
2018-09-11 05:29:52 +00:00
|
|
|
plain_statement_get_expanded_command(buff, last_statement, vars, out_expanded_command)) {
|
2016-10-22 18:21:13 +00:00
|
|
|
// Find the last argument. If we don't get one, return an invalid node.
|
2018-01-14 00:26:27 +00:00
|
|
|
if (auto last_arg = parse_tree.find_last_node<g::argument>(last_statement)) {
|
2018-01-13 22:25:39 +00:00
|
|
|
*out_last_arg = last_arg.get_source(buff);
|
2012-05-07 19:55:13 +00:00
|
|
|
}
|
2016-10-22 18:21:13 +00:00
|
|
|
return true;
|
2012-05-08 00:43:05 +00:00
|
|
|
}
|
2016-10-22 18:21:13 +00:00
|
|
|
return false;
|
2012-05-08 00:43:05 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
bool autosuggest_validate_from_history(const history_item_t &item,
|
|
|
|
const wcstring &working_directory,
|
2018-09-09 09:25:51 +00:00
|
|
|
const environment_t &vars) {
|
2012-02-19 02:54:36 +00:00
|
|
|
ASSERT_IS_BACKGROUND_THREAD();
|
2012-06-28 18:54:37 +00:00
|
|
|
|
2012-05-07 19:55:13 +00:00
|
|
|
bool handled = false, suggestionOK = false;
|
2012-05-08 00:43:05 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Parse the string.
|
2012-05-08 00:43:05 +00:00
|
|
|
wcstring parsed_command;
|
2018-01-13 22:25:39 +00:00
|
|
|
wcstring cd_dir;
|
2018-09-11 05:29:52 +00:00
|
|
|
if (!autosuggest_parse_command(item.str(), vars, &parsed_command, &cd_dir)) return false;
|
2012-05-08 00:43:05 +00:00
|
|
|
|
2018-01-13 22:25:39 +00:00
|
|
|
if (parsed_command == L"cd" && !cd_dir.empty()) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// We can possibly handle this specially.
|
2018-09-11 05:29:52 +00:00
|
|
|
if (expand_one(cd_dir, EXPAND_SKIP_CMDSUBST, vars)) {
|
2012-05-07 19:55:13 +00:00
|
|
|
handled = true;
|
2016-05-01 03:47:05 +00:00
|
|
|
bool is_help =
|
2018-01-13 22:25:39 +00:00
|
|
|
string_prefixes_string(cd_dir, L"--help") || string_prefixes_string(cd_dir, L"-h");
|
2016-11-03 23:32:27 +00:00
|
|
|
if (!is_help) {
|
2018-09-16 11:05:17 +00:00
|
|
|
auto path = path_get_cdpath(cd_dir, working_directory, vars);
|
|
|
|
if (path && !paths_are_same_file(working_directory, *path)) {
|
2012-05-07 19:55:13 +00:00
|
|
|
suggestionOK = true;
|
|
|
|
}
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
if (handled) {
|
|
|
|
return suggestionOK;
|
|
|
|
}
|
2012-06-28 18:54:37 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
// Not handled specially so handle it here.
|
|
|
|
bool cmd_ok = false;
|
2018-09-16 19:48:50 +00:00
|
|
|
if (path_get_path(parsed_command, NULL, vars)) {
|
2016-10-30 22:05:41 +00:00
|
|
|
cmd_ok = true;
|
|
|
|
} else if (builtin_exists(parsed_command) ||
|
|
|
|
function_exists_no_autoload(parsed_command, vars)) {
|
|
|
|
cmd_ok = true;
|
|
|
|
}
|
2012-06-28 18:54:37 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
if (cmd_ok) {
|
|
|
|
const path_list_t &paths = item.get_required_paths();
|
2017-01-25 01:52:28 +00:00
|
|
|
suggestionOK = all_paths_are_valid(paths, working_directory);
|
2012-02-19 02:54:36 +00:00
|
|
|
}
|
2012-06-28 18:54:37 +00:00
|
|
|
|
|
|
|
return suggestionOK;
|
2012-02-19 02:54:36 +00:00
|
|
|
}
|
2011-12-27 03:18:46 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlights the variable starting with 'in', setting colors within the 'colors' array. Returns the
|
|
|
|
// number of characters consumed.
|
|
|
|
static size_t color_variable(const wchar_t *in, size_t in_len,
|
|
|
|
std::vector<highlight_spec_t>::iterator colors) {
|
2014-02-03 22:13:42 +00:00
|
|
|
assert(in_len > 0);
|
|
|
|
assert(in[0] == L'$');
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2014-02-03 22:13:42 +00:00
|
|
|
// Handle an initial run of $s.
|
|
|
|
size_t idx = 0;
|
2014-09-30 18:14:57 +00:00
|
|
|
size_t dollar_count = 0;
|
2016-05-01 03:47:05 +00:00
|
|
|
while (in[idx] == '$') {
|
|
|
|
// Our color depends on the next char.
|
2014-02-03 22:13:42 +00:00
|
|
|
wchar_t next = in[idx + 1];
|
2017-04-20 06:43:02 +00:00
|
|
|
if (next == L'$' || valid_var_name_char(next)) {
|
2014-02-03 22:13:42 +00:00
|
|
|
colors[idx] = highlight_spec_operator;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
2014-02-03 22:13:42 +00:00
|
|
|
colors[idx] = highlight_spec_error;
|
2012-02-21 19:45:13 +00:00
|
|
|
}
|
2014-02-03 22:13:42 +00:00
|
|
|
idx++;
|
2014-09-30 18:14:57 +00:00
|
|
|
dollar_count++;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Handle a sequence of variable characters.
|
2017-04-20 06:43:02 +00:00
|
|
|
while (valid_var_name_char(in[idx])) {
|
2014-02-03 22:13:42 +00:00
|
|
|
colors[idx++] = highlight_spec_operator;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Handle a slice, up to dollar_count of them. Note that we currently don't do any validation of
|
|
|
|
// the slice's contents, e.g. $foo[blah] will not show an error even though it's invalid.
|
|
|
|
for (size_t slice_count = 0; slice_count < dollar_count && in[idx] == L'['; slice_count++) {
|
2014-02-03 22:13:42 +00:00
|
|
|
wchar_t *slice_begin = NULL, *slice_end = NULL;
|
2014-09-30 18:14:57 +00:00
|
|
|
int located = parse_util_locate_slice(in + idx, &slice_begin, &slice_end, false);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (located == 1) {
|
2014-09-30 18:14:57 +00:00
|
|
|
size_t slice_begin_idx = slice_begin - in, slice_end_idx = slice_end - in;
|
|
|
|
assert(slice_end_idx > slice_begin_idx);
|
|
|
|
colors[slice_begin_idx] = highlight_spec_operator;
|
|
|
|
colors[slice_end_idx] = highlight_spec_operator;
|
|
|
|
idx = slice_end_idx + 1;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (located == 0) {
|
2014-09-30 18:14:57 +00:00
|
|
|
// not a slice
|
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
2014-09-30 18:14:57 +00:00
|
|
|
assert(located < 0);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Syntax error. Normally the entire token is colored red for us, but inside a
|
|
|
|
// double-quoted string that doesn't happen. As such, color the variable + the slice
|
|
|
|
// start red. Coloring any more than that looks bad, unless we're willing to try and
|
|
|
|
// detect where the double-quoted string ends, and I'd rather not do that.
|
2014-09-30 18:14:57 +00:00
|
|
|
std::fill(colors, colors + idx + 1, (highlight_spec_t)highlight_spec_error);
|
|
|
|
break;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2014-02-03 22:13:42 +00:00
|
|
|
return idx;
|
2005-09-20 13:26:39 +00:00
|
|
|
}
|
|
|
|
|
2018-09-01 18:45:15 +00:00
|
|
|
/// This function is a disaster badly in need of refactoring. It colors an argument or command,
|
|
|
|
/// without regard to command substitutions.
|
|
|
|
static void color_string_internal(const wcstring &buffstr, highlight_spec_t base_color,
|
|
|
|
std::vector<highlight_spec_t>::iterator colors) {
|
|
|
|
// Clarify what we expect.
|
|
|
|
assert((base_color == highlight_spec_param || base_color == highlight_spec_command) &&
|
|
|
|
"Unexpected base color");
|
2013-08-08 22:06:46 +00:00
|
|
|
const size_t buff_len = buffstr.size();
|
2018-09-01 18:45:15 +00:00
|
|
|
std::fill(colors, colors + buff_len, base_color);
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2018-10-10 23:25:21 +00:00
|
|
|
// Hacky support for %self which must be an unquoted literal argument.
|
|
|
|
if (buffstr == PROCESS_EXPAND_SELF_STR) {
|
|
|
|
std::fill_n(colors, wcslen(PROCESS_EXPAND_SELF_STR), highlight_spec_operator);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
enum { e_unquoted, e_single_quoted, e_double_quoted } mode = e_unquoted;
|
|
|
|
int bracket_count = 0;
|
|
|
|
for (size_t in_pos = 0; in_pos < buff_len; in_pos++) {
|
2013-08-08 22:06:46 +00:00
|
|
|
const wchar_t c = buffstr.at(in_pos);
|
2016-05-01 03:47:05 +00:00
|
|
|
switch (mode) {
|
|
|
|
case e_unquoted: {
|
|
|
|
if (c == L'\\') {
|
|
|
|
int fill_color = highlight_spec_escape; // may be set to highlight_error
|
2013-08-08 22:06:46 +00:00
|
|
|
const size_t backslash_pos = in_pos;
|
|
|
|
size_t fill_end = backslash_pos;
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Move to the escaped character.
|
2013-08-08 22:06:46 +00:00
|
|
|
in_pos++;
|
|
|
|
const wchar_t escaped_char = (in_pos < buff_len ? buffstr.at(in_pos) : L'\0');
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (escaped_char == L'\0') {
|
2013-08-08 22:06:46 +00:00
|
|
|
fill_end = in_pos;
|
2014-01-15 09:01:25 +00:00
|
|
|
fill_color = highlight_spec_error;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (wcschr(L"~%", escaped_char)) {
|
|
|
|
if (in_pos == 1) {
|
2013-08-08 22:06:46 +00:00
|
|
|
fill_end = in_pos + 1;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (escaped_char == L',') {
|
|
|
|
if (bracket_count) {
|
2013-08-08 22:06:46 +00:00
|
|
|
fill_end = in_pos + 1;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (wcschr(L"abefnrtv*?$(){}[]'\"<>^ \\#;|&", escaped_char)) {
|
2013-08-08 22:06:46 +00:00
|
|
|
fill_end = in_pos + 1;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (wcschr(L"c", escaped_char)) {
|
|
|
|
// Like \ci. So highlight three characters.
|
2013-08-08 22:06:46 +00:00
|
|
|
fill_end = in_pos + 1;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (wcschr(L"uUxX01234567", escaped_char)) {
|
|
|
|
long long res = 0;
|
|
|
|
int chars = 2;
|
|
|
|
int base = 16;
|
2013-08-08 22:06:46 +00:00
|
|
|
wchar_t max_val = ASCII_MAX;
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
switch (escaped_char) {
|
|
|
|
case L'u': {
|
|
|
|
chars = 4;
|
2013-08-08 22:06:46 +00:00
|
|
|
max_val = UCS2_MAX;
|
|
|
|
in_pos++;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'U': {
|
|
|
|
chars = 8;
|
2013-08-08 22:06:46 +00:00
|
|
|
max_val = WCHAR_MAX;
|
|
|
|
in_pos++;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'x': {
|
2013-08-08 22:06:46 +00:00
|
|
|
in_pos++;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'X': {
|
2013-08-08 22:06:46 +00:00
|
|
|
max_val = BYTE_MAX;
|
|
|
|
in_pos++;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
default: {
|
2013-08-08 22:06:46 +00:00
|
|
|
// a digit like \12
|
2016-05-01 03:47:05 +00:00
|
|
|
base = 8;
|
|
|
|
chars = 3;
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2013-08-08 22:06:46 +00:00
|
|
|
// Consume
|
2016-05-01 03:47:05 +00:00
|
|
|
for (int i = 0; i < chars && in_pos < buff_len; i++) {
|
2013-08-08 22:06:46 +00:00
|
|
|
long d = convert_digit(buffstr.at(in_pos), base);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (d < 0) break;
|
2013-08-08 22:06:46 +00:00
|
|
|
res = (res * base) + d;
|
|
|
|
in_pos++;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
// in_pos is now at the first character that could not be converted (or
|
|
|
|
// buff_len).
|
2013-08-08 22:06:46 +00:00
|
|
|
assert(in_pos >= backslash_pos && in_pos <= buff_len);
|
|
|
|
fill_end = in_pos;
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// It's an error if we exceeded the max value.
|
|
|
|
if (res > max_val) fill_color = highlight_spec_error;
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Subtract one from in_pos, so that the increment in the loop will move to
|
|
|
|
// the next character.
|
2013-08-08 22:06:46 +00:00
|
|
|
in_pos--;
|
|
|
|
}
|
|
|
|
assert(fill_end >= backslash_pos);
|
|
|
|
std::fill(colors + backslash_pos, colors + fill_end, fill_color);
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// Not a backslash.
|
|
|
|
switch (c) {
|
2018-03-09 09:36:10 +00:00
|
|
|
case L'~': {
|
2016-05-01 03:47:05 +00:00
|
|
|
if (in_pos == 0) {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'$': {
|
2013-08-08 22:06:46 +00:00
|
|
|
assert(in_pos < buff_len);
|
2016-05-01 03:47:05 +00:00
|
|
|
in_pos += color_variable(buffstr.c_str() + in_pos, buff_len - in_pos,
|
|
|
|
colors + in_pos);
|
|
|
|
// Subtract one to account for the upcoming loop increment.
|
2014-02-09 23:33:34 +00:00
|
|
|
in_pos -= 1;
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
|
|
|
}
|
2018-05-06 02:11:57 +00:00
|
|
|
case L'?': {
|
2018-05-06 02:44:57 +00:00
|
|
|
if (!feature_test(features_t::qmark_noglob)) {
|
2018-05-06 02:11:57 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
case L'*':
|
|
|
|
case L'(':
|
2016-05-01 03:47:05 +00:00
|
|
|
case L')': {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'{': {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
2013-08-08 22:06:46 +00:00
|
|
|
bracket_count++;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'}': {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
2013-08-08 22:06:46 +00:00
|
|
|
bracket_count--;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L',': {
|
|
|
|
if (bracket_count > 0) {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_operator;
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'\'': {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_quote;
|
2013-08-08 22:06:46 +00:00
|
|
|
mode = e_single_quoted;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'\"': {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_quote;
|
2013-08-08 22:06:46 +00:00
|
|
|
mode = e_double_quoted;
|
|
|
|
break;
|
|
|
|
}
|
2016-11-03 01:29:14 +00:00
|
|
|
default: {
|
|
|
|
break; // we ignore all other characters
|
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
// Mode 1 means single quoted string, i.e 'foo'.
|
|
|
|
case e_single_quoted: {
|
2014-01-15 09:01:25 +00:00
|
|
|
colors[in_pos] = highlight_spec_quote;
|
2016-05-01 03:47:05 +00:00
|
|
|
if (c == L'\\') {
|
2013-08-08 22:06:46 +00:00
|
|
|
// backslash
|
2016-05-01 03:47:05 +00:00
|
|
|
if (in_pos + 1 < buff_len) {
|
2013-08-08 22:06:46 +00:00
|
|
|
const wchar_t escaped_char = buffstr.at(in_pos + 1);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (escaped_char == L'\\' || escaped_char == L'\'') {
|
|
|
|
colors[in_pos] = highlight_spec_escape; // backslash
|
|
|
|
colors[in_pos + 1] = highlight_spec_escape; // escaped char
|
|
|
|
in_pos += 1; // skip over backslash
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (c == L'\'') {
|
2013-08-08 22:06:46 +00:00
|
|
|
mode = e_unquoted;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
// Mode 2 means double quoted string, i.e. "foo".
|
|
|
|
case e_double_quoted: {
|
|
|
|
// Slices are colored in advance, past `in_pos`, and we don't want to overwrite
|
|
|
|
// that.
|
2018-09-01 18:45:15 +00:00
|
|
|
if (colors[in_pos] == base_color) {
|
2014-08-21 05:28:42 +00:00
|
|
|
colors[in_pos] = highlight_spec_quote;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
switch (c) {
|
|
|
|
case L'"': {
|
2013-08-08 22:06:46 +00:00
|
|
|
mode = e_unquoted;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'\\': {
|
|
|
|
// Backslash
|
|
|
|
if (in_pos + 1 < buff_len) {
|
2013-08-08 22:06:46 +00:00
|
|
|
const wchar_t escaped_char = buffstr.at(in_pos + 1);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (wcschr(L"\\\"\n$", escaped_char)) {
|
|
|
|
colors[in_pos] = highlight_spec_escape; // backslash
|
|
|
|
colors[in_pos + 1] = highlight_spec_escape; // escaped char
|
|
|
|
in_pos += 1; // skip over backslash
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'$': {
|
|
|
|
in_pos += color_variable(buffstr.c_str() + in_pos, buff_len - in_pos,
|
|
|
|
colors + in_pos);
|
|
|
|
// Subtract one to account for the upcoming increment in the loop.
|
2014-02-09 23:33:34 +00:00
|
|
|
in_pos -= 1;
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-11-03 01:29:14 +00:00
|
|
|
default: {
|
|
|
|
break; // we ignore all other characters
|
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Syntax highlighter helper.
|
|
|
|
class highlighter_t {
|
|
|
|
// The string we're highlighting. Note this is a reference memmber variable (to avoid copying)!
|
|
|
|
// We must not outlive this!
|
2013-10-09 01:41:35 +00:00
|
|
|
const wcstring &buff;
|
2016-05-01 03:47:05 +00:00
|
|
|
// Cursor position.
|
2013-10-09 01:41:35 +00:00
|
|
|
const size_t cursor_pos;
|
2016-05-01 03:47:05 +00:00
|
|
|
// Environment variables. Again, a reference member variable!
|
2018-09-09 09:25:51 +00:00
|
|
|
const environment_t &vars;
|
2016-05-01 03:47:05 +00:00
|
|
|
// Whether it's OK to do I/O.
|
2014-03-27 01:49:09 +00:00
|
|
|
const bool io_ok;
|
2016-05-01 03:47:05 +00:00
|
|
|
// Working directory.
|
2013-10-09 01:41:35 +00:00
|
|
|
const wcstring working_directory;
|
2016-05-01 03:47:05 +00:00
|
|
|
// The resulting colors.
|
2014-01-15 09:01:25 +00:00
|
|
|
typedef std::vector<highlight_spec_t> color_array_t;
|
2013-10-09 01:41:35 +00:00
|
|
|
color_array_t color_array;
|
2016-05-01 03:47:05 +00:00
|
|
|
// The parse tree of the buff.
|
2013-10-09 01:41:35 +00:00
|
|
|
parse_node_tree_t parse_tree;
|
2018-09-01 18:45:15 +00:00
|
|
|
// Color a command.
|
|
|
|
void color_command(tnode_t<g::tok_string> node);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color an argument.
|
2018-01-14 00:26:27 +00:00
|
|
|
void color_argument(tnode_t<g::tok_string> node);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color a redirection.
|
2018-01-14 00:26:27 +00:00
|
|
|
void color_redirection(tnode_t<g::redirection> node);
|
2018-01-14 00:11:58 +00:00
|
|
|
// Color a list of arguments. If cmd_is_cd is true, then the arguments are for 'cd'; detect
|
|
|
|
// invalid directories.
|
2018-01-14 02:00:24 +00:00
|
|
|
void color_arguments(const std::vector<tnode_t<g::argument>> &args, bool cmd_is_cd = false);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color the redirections of the given node.
|
2018-01-14 00:26:27 +00:00
|
|
|
void color_redirections(tnode_t<g::arguments_or_redirections_list> list);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color all the children of the command with the given type.
|
|
|
|
void color_children(const parse_node_t &parent, parse_token_type_t type,
|
|
|
|
highlight_spec_t color);
|
|
|
|
// Colors the source range of a node with a given color.
|
2014-02-22 02:01:40 +00:00
|
|
|
void color_node(const parse_node_t &node, highlight_spec_t color);
|
2018-01-14 00:11:58 +00:00
|
|
|
// return whether a plain statement is 'cd'.
|
2018-01-14 00:26:27 +00:00
|
|
|
bool is_cd(tnode_t<g::plain_statement> stmt) const;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
public:
|
|
|
|
// Constructor
|
2018-09-09 09:25:51 +00:00
|
|
|
highlighter_t(const wcstring &str, size_t pos, const environment_t &ev, wcstring wd,
|
2018-10-10 23:25:21 +00:00
|
|
|
bool can_do_io)
|
2016-05-01 03:47:05 +00:00
|
|
|
: buff(str),
|
|
|
|
cursor_pos(pos),
|
|
|
|
vars(ev),
|
|
|
|
io_ok(can_do_io),
|
2018-02-19 02:39:03 +00:00
|
|
|
working_directory(std::move(wd)),
|
2016-05-01 03:47:05 +00:00
|
|
|
color_array(str.size()) {
|
|
|
|
// Parse the tree.
|
|
|
|
parse_tree_from_string(buff, parse_flag_continue_after_error | parse_flag_include_comments,
|
|
|
|
&this->parse_tree, NULL);
|
2013-10-09 01:41:35 +00:00
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Perform highlighting, returning an array of colors.
|
2013-10-09 01:41:35 +00:00
|
|
|
const color_array_t &highlight();
|
|
|
|
};
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
void highlighter_t::color_node(const parse_node_t &node, highlight_spec_t color) {
|
|
|
|
// Can only color nodes with valid source ranges.
|
|
|
|
if (!node.has_source() || node.source_length == 0) return;
|
2013-10-09 01:41:35 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Fill the color array with our color in the corresponding range.
|
2013-10-09 01:41:35 +00:00
|
|
|
size_t source_end = node.source_start + node.source_length;
|
|
|
|
assert(source_end >= node.source_start);
|
|
|
|
assert(source_end <= color_array.size());
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
std::fill(this->color_array.begin() + node.source_start, this->color_array.begin() + source_end,
|
|
|
|
color);
|
2013-10-09 01:41:35 +00:00
|
|
|
}
|
|
|
|
|
2018-09-01 18:45:15 +00:00
|
|
|
void highlighter_t::color_command(tnode_t<g::tok_string> node) {
|
|
|
|
auto source_range = node.source_range();
|
|
|
|
if (!source_range) return;
|
|
|
|
|
|
|
|
const wcstring cmd_str = node.get_source(this->buff);
|
|
|
|
|
|
|
|
// Get an iterator to the colors associated with the argument.
|
|
|
|
const size_t arg_start = source_range->start;
|
|
|
|
const color_array_t::iterator colors = color_array.begin() + arg_start;
|
|
|
|
color_string_internal(cmd_str, highlight_spec_command, colors);
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// node does not necessarily have type symbol_argument here.
|
2018-01-14 00:26:27 +00:00
|
|
|
void highlighter_t::color_argument(tnode_t<g::tok_string> node) {
|
2018-01-13 22:51:37 +00:00
|
|
|
auto source_range = node.source_range();
|
|
|
|
if (!source_range) return;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2013-10-09 01:41:35 +00:00
|
|
|
const wcstring arg_str = node.get_source(this->buff);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Get an iterator to the colors associated with the argument.
|
2018-01-13 22:51:37 +00:00
|
|
|
const size_t arg_start = source_range->start;
|
2013-10-09 01:41:35 +00:00
|
|
|
const color_array_t::iterator arg_colors = color_array.begin() + arg_start;
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color this argument without concern for command substitutions.
|
2018-09-01 18:45:15 +00:00
|
|
|
color_string_internal(arg_str, highlight_spec_param, arg_colors);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Now do command substitutions.
|
2013-10-09 01:41:35 +00:00
|
|
|
size_t cmdsub_cursor = 0, cmdsub_start = 0, cmdsub_end = 0;
|
|
|
|
wcstring cmdsub_contents;
|
2016-05-01 03:47:05 +00:00
|
|
|
while (parse_util_locate_cmdsubst_range(arg_str, &cmdsub_cursor, &cmdsub_contents,
|
|
|
|
&cmdsub_start, &cmdsub_end,
|
|
|
|
true /* accept incomplete */) > 0) {
|
|
|
|
// The cmdsub_start is the open paren. cmdsub_end is either the close paren or the end of
|
|
|
|
// the string. cmdsub_contents extends from one past cmdsub_start to cmdsub_end.
|
2013-10-09 01:41:35 +00:00
|
|
|
assert(cmdsub_end > cmdsub_start);
|
|
|
|
assert(cmdsub_end - cmdsub_start - 1 == cmdsub_contents.size());
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Found a command substitution. Compute the position of the start and end of the cmdsub
|
|
|
|
// contents, within our overall src.
|
|
|
|
const size_t arg_subcmd_start = arg_start + cmdsub_start,
|
|
|
|
arg_subcmd_end = arg_start + cmdsub_end;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlight the parens. The open paren must exist; the closed paren may not if it was
|
|
|
|
// incomplete.
|
2013-10-09 01:41:35 +00:00
|
|
|
assert(cmdsub_start < arg_str.size());
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_array.at(arg_subcmd_start) = highlight_spec_operator;
|
2013-10-09 01:41:35 +00:00
|
|
|
if (arg_subcmd_end < this->buff.size())
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_array.at(arg_subcmd_end) = highlight_spec_operator;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Compute the cursor's position within the cmdsub. We must be past the open paren (hence >)
|
|
|
|
// but can be at the end of the string or closed paren (hence <=).
|
2013-10-09 01:41:35 +00:00
|
|
|
size_t cursor_subpos = CURSOR_POSITION_INVALID;
|
2016-05-01 03:47:05 +00:00
|
|
|
if (cursor_pos != CURSOR_POSITION_INVALID && cursor_pos > arg_subcmd_start &&
|
|
|
|
cursor_pos <= arg_subcmd_end) {
|
|
|
|
// The -1 because the cmdsub_contents does not include the open paren.
|
2013-10-09 01:41:35 +00:00
|
|
|
cursor_subpos = cursor_pos - arg_subcmd_start - 1;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlight it recursively.
|
|
|
|
highlighter_t cmdsub_highlighter(cmdsub_contents, cursor_subpos, this->vars,
|
|
|
|
this->working_directory, this->io_ok);
|
2013-10-09 01:41:35 +00:00
|
|
|
const color_array_t &subcolors = cmdsub_highlighter.highlight();
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Copy out the subcolors back into our array.
|
2013-10-09 01:41:35 +00:00
|
|
|
assert(subcolors.size() == cmdsub_contents.size());
|
2016-05-01 03:47:05 +00:00
|
|
|
std::copy(subcolors.begin(), subcolors.end(),
|
|
|
|
this->color_array.begin() + arg_subcmd_start + 1);
|
2013-10-09 01:41:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Indicates whether the source range of the given node forms a valid path in the given
|
|
|
|
/// working_directory.
|
|
|
|
static bool node_is_potential_path(const wcstring &src, const parse_node_t &node,
|
2018-09-19 04:03:01 +00:00
|
|
|
const environment_t &vars, const wcstring &working_directory) {
|
2016-05-01 03:47:05 +00:00
|
|
|
if (!node.has_source()) return false;
|
2013-10-06 23:23:45 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Get the node source, unescape it, and then pass it to is_potential_path along with the
|
|
|
|
// working directory (as a one element list).
|
2013-10-06 23:23:45 +00:00
|
|
|
bool result = false;
|
|
|
|
wcstring token(src, node.source_start, node.source_length);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (unescape_string_in_place(&token, UNESCAPE_SPECIAL)) {
|
|
|
|
// Big hack: is_potential_path expects a tilde, but unescape_string gives us HOME_DIRECTORY.
|
|
|
|
// Put it back.
|
|
|
|
if (!token.empty() && token.at(0) == HOME_DIRECTORY) token.at(0) = L'~';
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2013-10-06 23:23:45 +00:00
|
|
|
const wcstring_list_t working_directory_list(1, working_directory);
|
2018-09-19 04:03:01 +00:00
|
|
|
result = is_potential_path(token, working_directory_list, vars, PATH_EXPAND_TILDE);
|
2013-10-06 23:23:45 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-01-14 00:26:27 +00:00
|
|
|
bool highlighter_t::is_cd(tnode_t<g::plain_statement> stmt) const {
|
2013-10-07 10:56:09 +00:00
|
|
|
bool cmd_is_cd = false;
|
2018-01-14 00:11:58 +00:00
|
|
|
if (this->io_ok && stmt.has_source()) {
|
|
|
|
wcstring cmd_str;
|
2018-09-11 05:29:52 +00:00
|
|
|
if (plain_statement_get_expanded_command(this->buff, stmt, vars, &cmd_str)) {
|
2018-01-14 00:11:58 +00:00
|
|
|
cmd_is_cd = (cmd_str == L"cd");
|
2013-10-07 10:56:09 +00:00
|
|
|
}
|
|
|
|
}
|
2018-01-14 00:11:58 +00:00
|
|
|
return cmd_is_cd;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2018-01-14 00:11:58 +00:00
|
|
|
// Color all of the arguments of the given node list, which should be argument_list or
|
|
|
|
// argument_or_redirection_list.
|
2018-01-14 00:26:27 +00:00
|
|
|
void highlighter_t::color_arguments(const std::vector<tnode_t<g::argument>> &args, bool cmd_is_cd) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Find all the arguments of this list.
|
2018-01-14 00:26:27 +00:00
|
|
|
for (tnode_t<g::argument> arg : args) {
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_argument(arg.child<0>());
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (cmd_is_cd) {
|
|
|
|
// Mark this as an error if it's not 'help' and not a valid cd path.
|
2018-01-13 22:51:37 +00:00
|
|
|
wcstring param = arg.get_source(this->buff);
|
2018-09-11 05:29:52 +00:00
|
|
|
if (expand_one(param, EXPAND_SKIP_CMDSUBST, vars)) {
|
2016-05-01 03:47:05 +00:00
|
|
|
bool is_help = string_prefixes_string(param, L"--help") ||
|
|
|
|
string_prefixes_string(param, L"-h");
|
|
|
|
if (!is_help && this->io_ok &&
|
2018-09-19 04:03:01 +00:00
|
|
|
!is_potential_cd_path(param, working_directory, vars, PATH_EXPAND_TILDE)) {
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_node(arg, highlight_spec_error);
|
2013-10-07 10:56:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-14 00:26:27 +00:00
|
|
|
void highlighter_t::color_redirection(tnode_t<g::redirection> redirection_node) {
|
2016-05-01 03:47:05 +00:00
|
|
|
if (!redirection_node.has_source()) return;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2018-01-14 00:26:27 +00:00
|
|
|
tnode_t<g::tok_redirection> redir_prim = redirection_node.child<0>(); // like 2>
|
|
|
|
tnode_t<g::tok_string> redir_target = redirection_node.child<1>(); // like &1 or file path
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2018-01-13 22:51:37 +00:00
|
|
|
if (redir_prim) {
|
2013-10-13 23:58:40 +00:00
|
|
|
wcstring target;
|
2018-02-23 23:19:58 +00:00
|
|
|
const maybe_t<redirection_type_t> redirect_type =
|
2018-01-15 23:37:13 +00:00
|
|
|
redirection_type(redirection_node, this->buff, nullptr, &target);
|
2016-05-01 03:47:05 +00:00
|
|
|
|
2018-02-23 23:19:58 +00:00
|
|
|
// We may get a missing redirection type if the redirection is invalid.
|
|
|
|
auto hl = redirect_type ? highlight_spec_redirection : highlight_spec_error;
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_node(redir_prim, hl);
|
2016-05-01 03:47:05 +00:00
|
|
|
|
|
|
|
// Check if the argument contains a command substitution. If so, highlight it as a param
|
|
|
|
// even though it's a command redirection, and don't try to do any other validation.
|
|
|
|
if (parse_util_locate_cmdsubst(target.c_str(), NULL, NULL, true) != 0) {
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_argument(redir_target);
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// No command substitution, so we can highlight the target file or fd. For example,
|
|
|
|
// disallow redirections into a non-existent directory.
|
2013-10-13 23:58:40 +00:00
|
|
|
bool target_is_valid = true;
|
|
|
|
|
2018-02-23 23:19:58 +00:00
|
|
|
if (!redirect_type) {
|
|
|
|
// not a valid redirection
|
|
|
|
target_is_valid = false;
|
|
|
|
} else if (!this->io_ok) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// I/O is disallowed, so we don't have much hope of catching anything but gross
|
|
|
|
// errors. Assume it's valid.
|
2014-03-27 01:49:09 +00:00
|
|
|
target_is_valid = true;
|
2018-09-11 05:29:52 +00:00
|
|
|
} else if (!expand_one(target, EXPAND_SKIP_CMDSUBST, vars)) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Could not be expanded.
|
2013-10-13 23:58:40 +00:00
|
|
|
target_is_valid = false;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// Ok, we successfully expanded our target. Now verify that it works with this
|
|
|
|
// redirection. We will probably need it as a path (but not in the case of fd
|
|
|
|
// redirections). Note that the target is now unescaped.
|
|
|
|
const wcstring target_path =
|
|
|
|
path_apply_working_directory(target, this->working_directory);
|
2018-02-23 23:19:58 +00:00
|
|
|
switch (*redirect_type) {
|
|
|
|
case redirection_type_t::fd: {
|
2016-11-23 04:24:03 +00:00
|
|
|
int fd = fish_wcstoi(target.c_str());
|
|
|
|
target_is_valid = !errno && fd >= 0;
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2013-10-13 23:58:40 +00:00
|
|
|
}
|
2018-02-23 23:19:58 +00:00
|
|
|
case redirection_type_t::input: {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Input redirections must have a readable non-directory.
|
2013-10-13 23:58:40 +00:00
|
|
|
struct stat buf = {};
|
2016-05-01 03:47:05 +00:00
|
|
|
target_is_valid = !waccess(target_path, R_OK) &&
|
|
|
|
!wstat(target_path, &buf) && !S_ISDIR(buf.st_mode);
|
|
|
|
break;
|
2013-10-13 23:58:40 +00:00
|
|
|
}
|
2018-02-23 23:19:58 +00:00
|
|
|
case redirection_type_t::overwrite:
|
|
|
|
case redirection_type_t::append:
|
|
|
|
case redirection_type_t::noclob: {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Test whether the file exists, and whether it's writable (possibly after
|
|
|
|
// creating it). access() returns failure if the file does not exist.
|
2013-10-13 23:58:40 +00:00
|
|
|
bool file_exists = false, file_is_writable = false;
|
|
|
|
int err = 0;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2013-10-13 23:58:40 +00:00
|
|
|
struct stat buf = {};
|
2016-05-01 03:47:05 +00:00
|
|
|
if (wstat(target_path, &buf) < 0) {
|
2013-10-13 23:58:40 +00:00
|
|
|
err = errno;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (string_suffixes_string(L"/", target)) {
|
|
|
|
// Redirections to things that are directories is definitely not
|
|
|
|
// allowed.
|
2013-10-13 23:58:40 +00:00
|
|
|
file_exists = false;
|
|
|
|
file_is_writable = false;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (err == 0) {
|
|
|
|
// No err. We can write to it if it's not a directory and we have
|
|
|
|
// permission.
|
2013-10-13 23:58:40 +00:00
|
|
|
file_exists = true;
|
2016-05-01 03:47:05 +00:00
|
|
|
file_is_writable = !S_ISDIR(buf.st_mode) && !waccess(target_path, W_OK);
|
|
|
|
} else if (err == ENOENT) {
|
|
|
|
// File does not exist. Check if its parent directory is writable.
|
2013-10-13 23:58:40 +00:00
|
|
|
wcstring parent = wdirname(target_path);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Ensure that the parent ends with the path separator. This will ensure
|
|
|
|
// that we get an error if the parent directory is not really a
|
|
|
|
// directory.
|
|
|
|
if (!string_suffixes_string(L"/", parent)) parent.push_back(L'/');
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Now the file is considered writable if the parent directory is
|
|
|
|
// writable.
|
2013-10-13 23:58:40 +00:00
|
|
|
file_exists = false;
|
|
|
|
file_is_writable = (0 == waccess(parent, W_OK));
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
// Other errors we treat as not writable. This includes things like
|
|
|
|
// ENOTDIR.
|
2013-10-13 23:58:40 +00:00
|
|
|
file_exists = false;
|
|
|
|
file_is_writable = false;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// NOCLOB means that we must not overwrite files that exist.
|
2018-02-23 23:19:58 +00:00
|
|
|
target_is_valid =
|
|
|
|
file_is_writable &&
|
|
|
|
!(file_exists && redirect_type == redirection_type_t::noclob);
|
2013-10-13 23:58:40 +00:00
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
2013-10-13 23:58:40 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2018-01-13 22:51:37 +00:00
|
|
|
if (redir_target) {
|
2017-05-02 04:44:30 +00:00
|
|
|
auto hl = target_is_valid ? highlight_spec_redirection : highlight_spec_error;
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_node(redir_target, hl);
|
2013-10-13 23:58:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Color all of the redirections of the given command.
|
2018-01-14 00:26:27 +00:00
|
|
|
void highlighter_t::color_redirections(tnode_t<g::arguments_or_redirections_list> list) {
|
|
|
|
for (const auto &node : list.descendants<g::redirection>()) {
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_redirection(node);
|
2013-10-13 23:58:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Color all the children of the command with the given type.
|
|
|
|
void highlighter_t::color_children(const parse_node_t &parent, parse_token_type_t type,
|
|
|
|
highlight_spec_t color) {
|
|
|
|
for (node_offset_t idx = 0; idx < parent.child_count; idx++) {
|
2013-10-09 01:41:35 +00:00
|
|
|
const parse_node_t *child = this->parse_tree.get_child(parent, idx);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (child != NULL && child->type == type) {
|
2013-10-09 01:41:35 +00:00
|
|
|
this->color_node(*child, color);
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Determine if a command is valid.
|
|
|
|
static bool command_is_valid(const wcstring &cmd, enum parse_statement_decoration_t decoration,
|
2018-09-09 09:25:51 +00:00
|
|
|
const wcstring &working_directory, const environment_t &vars) {
|
2016-05-01 03:47:05 +00:00
|
|
|
// Determine which types we check, based on the decoration.
|
|
|
|
bool builtin_ok = true, function_ok = true, abbreviation_ok = true, command_ok = true,
|
|
|
|
implicit_cd_ok = true;
|
|
|
|
if (decoration == parse_statement_decoration_command ||
|
|
|
|
decoration == parse_statement_decoration_exec) {
|
2013-10-06 23:23:45 +00:00
|
|
|
builtin_ok = false;
|
|
|
|
function_ok = false;
|
|
|
|
abbreviation_ok = false;
|
|
|
|
command_ok = true;
|
|
|
|
implicit_cd_ok = false;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else if (decoration == parse_statement_decoration_builtin) {
|
2013-10-06 23:23:45 +00:00
|
|
|
builtin_ok = true;
|
|
|
|
function_ok = false;
|
|
|
|
abbreviation_ok = false;
|
|
|
|
command_ok = false;
|
|
|
|
implicit_cd_ok = false;
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Check them.
|
2013-10-06 23:23:45 +00:00
|
|
|
bool is_valid = false;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Builtins
|
|
|
|
if (!is_valid && builtin_ok) is_valid = builtin_exists(cmd);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Functions
|
|
|
|
if (!is_valid && function_ok) is_valid = function_exists_no_autoload(cmd, vars);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Abbreviations
|
|
|
|
if (!is_valid && abbreviation_ok) is_valid = expand_abbreviation(cmd, NULL);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Regular commands
|
|
|
|
if (!is_valid && command_ok) is_valid = path_get_path(cmd, NULL, vars);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Implicit cd
|
2017-08-09 18:11:58 +00:00
|
|
|
if (!is_valid && implicit_cd_ok) {
|
2018-09-16 11:05:17 +00:00
|
|
|
is_valid = path_as_implicit_cd(cmd, working_directory, vars).has_value();
|
2017-08-09 18:11:58 +00:00
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Return what we got.
|
2013-10-07 08:04:37 +00:00
|
|
|
return is_valid;
|
2013-10-06 23:23:45 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
const highlighter_t::color_array_t &highlighter_t::highlight() {
|
|
|
|
// If we are doing I/O, we must be in a background thread.
|
|
|
|
if (io_ok) {
|
2014-03-27 01:49:09 +00:00
|
|
|
ASSERT_IS_BACKGROUND_THREAD();
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2013-08-08 22:06:46 +00:00
|
|
|
const size_t length = buff.size();
|
2013-10-09 01:41:35 +00:00
|
|
|
assert(this->buff.size() == this->color_array.size());
|
2016-05-01 03:47:05 +00:00
|
|
|
if (length == 0) return color_array;
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Start out at zero.
|
2013-10-09 01:41:35 +00:00
|
|
|
std::fill(this->color_array.begin(), this->color_array.end(), 0);
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Walk the node tree.
|
2018-01-13 23:46:07 +00:00
|
|
|
for (const parse_node_t &node : parse_tree) {
|
2016-05-01 03:47:05 +00:00
|
|
|
switch (node.type) {
|
|
|
|
// Color direct string descendants, e.g. 'for' and 'in'.
|
2013-08-08 22:06:46 +00:00
|
|
|
case symbol_while_header:
|
|
|
|
case symbol_begin_header:
|
|
|
|
case symbol_function_header:
|
|
|
|
case symbol_if_clause:
|
|
|
|
case symbol_else_clause:
|
|
|
|
case symbol_case_item:
|
|
|
|
case symbol_decorated_statement:
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_if_statement: {
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_children(node, parse_token_type_string, highlight_spec_command);
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2014-02-22 02:01:40 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_switch_statement: {
|
2018-01-14 00:26:27 +00:00
|
|
|
tnode_t<g::switch_statement> switchn(&parse_tree, &node);
|
2018-01-13 23:46:07 +00:00
|
|
|
auto literal_switch = switchn.child<0>();
|
|
|
|
auto switch_arg = switchn.child<1>();
|
|
|
|
this->color_node(literal_switch, highlight_spec_command);
|
|
|
|
this->color_node(switch_arg, highlight_spec_param);
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2014-03-29 00:09:08 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_for_header: {
|
2018-01-14 00:26:27 +00:00
|
|
|
tnode_t<g::for_header> fhead(&parse_tree, &node);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color the 'for' and 'in' as commands.
|
2018-01-13 22:51:37 +00:00
|
|
|
auto literal_for = fhead.child<0>();
|
|
|
|
auto literal_in = fhead.child<2>();
|
|
|
|
this->color_node(literal_for, highlight_spec_command);
|
|
|
|
this->color_node(literal_in, highlight_spec_command);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Color the variable name as a parameter.
|
2018-01-13 22:51:37 +00:00
|
|
|
this->color_argument(fhead.child<1>());
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2013-08-11 07:35:00 +00:00
|
|
|
}
|
2018-03-04 23:03:56 +00:00
|
|
|
|
|
|
|
case parse_token_type_andand:
|
|
|
|
case parse_token_type_oror:
|
|
|
|
this->color_node(node, highlight_spec_operator);
|
|
|
|
break;
|
|
|
|
|
2018-03-05 00:06:32 +00:00
|
|
|
case symbol_not_statement:
|
|
|
|
this->color_children(node, parse_token_type_string, highlight_spec_operator);
|
|
|
|
break;
|
|
|
|
|
2018-03-04 23:03:56 +00:00
|
|
|
case symbol_job_decorator:
|
|
|
|
this->color_node(node, highlight_spec_operator);
|
|
|
|
break;
|
|
|
|
|
2014-08-21 01:40:14 +00:00
|
|
|
case parse_token_type_pipe:
|
2013-08-08 22:06:46 +00:00
|
|
|
case parse_token_type_background:
|
|
|
|
case parse_token_type_end:
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_optional_background: {
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_node(node, highlight_spec_statement_terminator);
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2013-10-07 08:04:37 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_plain_statement: {
|
2018-01-14 00:43:12 +00:00
|
|
|
tnode_t<g::plain_statement> stmt(&parse_tree, &node);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Get the decoration from the parent.
|
2018-01-14 00:43:12 +00:00
|
|
|
enum parse_statement_decoration_t decoration = get_decoration(stmt);
|
2016-05-01 03:47:05 +00:00
|
|
|
|
|
|
|
// Color the command.
|
2018-01-14 00:43:12 +00:00
|
|
|
tnode_t<g::tok_string> cmd_node = stmt.child<0>();
|
|
|
|
maybe_t<wcstring> cmd = cmd_node.get_source(buff);
|
|
|
|
if (!cmd) {
|
2016-10-30 22:05:41 +00:00
|
|
|
break; // not much as we can do without a node that has source text
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_valid_cmd = false;
|
|
|
|
if (!this->io_ok) {
|
|
|
|
// We cannot check if the command is invalid, so just assume it's valid.
|
|
|
|
is_valid_cmd = true;
|
|
|
|
} else {
|
2018-09-01 18:45:15 +00:00
|
|
|
wcstring expanded_cmd;
|
2016-10-30 22:05:41 +00:00
|
|
|
// Check to see if the command is valid.
|
|
|
|
// Try expanding it. If we cannot, it's an error.
|
2018-09-11 05:29:52 +00:00
|
|
|
bool expanded =
|
|
|
|
plain_statement_get_expanded_command(buff, stmt, vars, &expanded_cmd);
|
2018-09-01 18:45:15 +00:00
|
|
|
if (expanded && !has_expand_reserved(expanded_cmd)) {
|
|
|
|
is_valid_cmd =
|
|
|
|
command_is_valid(expanded_cmd, decoration, working_directory, vars);
|
2013-10-07 08:04:37 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-01 18:45:15 +00:00
|
|
|
if (!is_valid_cmd) {
|
|
|
|
this->color_node(*cmd_node, highlight_spec_error);
|
|
|
|
} else {
|
|
|
|
this->color_command(cmd_node);
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
2018-01-14 00:11:58 +00:00
|
|
|
// Only work on root lists, so that we don't re-color child lists.
|
|
|
|
case symbol_arguments_or_redirections_list: {
|
2018-01-16 02:57:28 +00:00
|
|
|
tnode_t<g::arguments_or_redirections_list> list(&parse_tree, &node);
|
|
|
|
if (argument_list_is_root(list)) {
|
2018-01-14 00:26:27 +00:00
|
|
|
bool cmd_is_cd = is_cd(list.try_get_parent<g::plain_statement>());
|
|
|
|
this->color_arguments(list.descendants<g::argument>(), cmd_is_cd);
|
2018-01-14 00:11:58 +00:00
|
|
|
this->color_redirections(list);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_argument_list: {
|
2018-01-16 02:57:28 +00:00
|
|
|
tnode_t<g::argument_list> list(&parse_tree, &node);
|
|
|
|
if (argument_list_is_root(list)) {
|
2018-01-14 02:00:24 +00:00
|
|
|
this->color_arguments(list.descendants<g::argument>());
|
2013-10-07 10:56:09 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
break;
|
2013-10-07 10:56:09 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
case symbol_end_command: {
|
2014-02-22 02:01:40 +00:00
|
|
|
this->color_node(node, highlight_spec_command);
|
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
case parse_special_type_parse_error:
|
2016-05-01 03:47:05 +00:00
|
|
|
case parse_special_type_tokenizer_error: {
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_node(node, highlight_spec_error);
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
|
|
|
case parse_special_type_comment: {
|
2014-01-15 09:01:25 +00:00
|
|
|
this->color_node(node, highlight_spec_comment);
|
2013-08-08 22:06:46 +00:00
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
|
|
|
default: { break; }
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-10-30 22:05:41 +00:00
|
|
|
if (!this->io_ok || this->cursor_pos > this->buff.size()) {
|
|
|
|
return color_array;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the cursor is over an argument, and that argument is a valid path, underline it.
|
|
|
|
for (parse_node_tree_t::const_iterator iter = parse_tree.begin(); iter != parse_tree.end();
|
|
|
|
++iter) {
|
|
|
|
const parse_node_t &node = *iter;
|
|
|
|
|
|
|
|
// Must be an argument with source.
|
|
|
|
if (node.type != symbol_argument || !node.has_source()) continue;
|
|
|
|
|
|
|
|
// See if this node contains the cursor. We check <= source_length so that, when backspacing
|
|
|
|
// (and the cursor is just beyond the last token), we may still underline it.
|
|
|
|
if (this->cursor_pos >= node.source_start &&
|
|
|
|
this->cursor_pos - node.source_start <= node.source_length &&
|
2018-09-19 04:03:01 +00:00
|
|
|
node_is_potential_path(buff, node, vars, working_directory)) {
|
2016-10-30 22:05:41 +00:00
|
|
|
// It is, underline it.
|
|
|
|
for (size_t i = node.source_start; i < node.source_start + node.source_length; i++) {
|
|
|
|
// Don't color highlight_spec_error because it looks dorky. For example,
|
|
|
|
// trying to cd into a non-directory would show an underline and also red.
|
|
|
|
if (highlight_get_primary(this->color_array.at(i)) != highlight_spec_error) {
|
|
|
|
this->color_array.at(i) |= highlight_modifier_valid_path;
|
2013-10-06 23:23:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2013-10-09 01:41:35 +00:00
|
|
|
return color_array;
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
void highlight_shell(const wcstring &buff, std::vector<highlight_spec_t> &color, size_t pos,
|
2018-09-09 09:25:51 +00:00
|
|
|
wcstring_list_t *error, const environment_t &vars) {
|
2016-10-09 21:38:26 +00:00
|
|
|
UNUSED(error);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Do something sucky and get the current working directory on this background thread. This
|
|
|
|
// should really be passed in.
|
2018-09-11 05:29:52 +00:00
|
|
|
const wcstring working_directory = vars.get_pwd_slash();
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlight it!
|
2014-03-27 01:49:09 +00:00
|
|
|
highlighter_t highlighter(buff, pos, vars, working_directory, true /* can do IO */);
|
|
|
|
color = highlighter.highlight();
|
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
void highlight_shell_no_io(const wcstring &buff, std::vector<highlight_spec_t> &color, size_t pos,
|
2018-09-09 09:25:51 +00:00
|
|
|
wcstring_list_t *error, const environment_t &vars) {
|
2016-10-09 21:38:26 +00:00
|
|
|
UNUSED(error);
|
2016-05-01 03:47:05 +00:00
|
|
|
// Do something sucky and get the current working directory on this background thread. This
|
|
|
|
// should really be passed in.
|
2018-09-11 05:29:52 +00:00
|
|
|
const wcstring working_directory = vars.get_pwd_slash();
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlight it!
|
2014-03-27 01:49:09 +00:00
|
|
|
highlighter_t highlighter(buff, pos, vars, working_directory, false /* no IO allowed */);
|
2013-10-09 01:41:35 +00:00
|
|
|
color = highlighter.highlight();
|
2013-08-08 22:06:46 +00:00
|
|
|
}
|
2011-12-27 03:18:46 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
/// Perform quote and parenthesis highlighting on the specified string.
|
|
|
|
static void highlight_universal_internal(const wcstring &buffstr,
|
|
|
|
std::vector<highlight_spec_t> &color, size_t pos) {
|
2012-02-22 01:55:56 +00:00
|
|
|
assert(buffstr.size() == color.size());
|
2016-05-01 03:47:05 +00:00
|
|
|
if (pos < buffstr.size()) {
|
|
|
|
// Highlight matching quotes.
|
|
|
|
if ((buffstr.at(pos) == L'\'') || (buffstr.at(pos) == L'\"')) {
|
2012-12-21 01:37:09 +00:00
|
|
|
std::vector<size_t> lst;
|
2016-05-01 03:47:05 +00:00
|
|
|
int level = 0;
|
|
|
|
wchar_t prev_q = 0;
|
|
|
|
const wchar_t *const buff = buffstr.c_str();
|
2012-11-19 00:30:30 +00:00
|
|
|
const wchar_t *str = buff;
|
2018-12-31 06:43:26 +00:00
|
|
|
bool match_found = false;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
while (*str) {
|
|
|
|
switch (*str) {
|
|
|
|
case L'\\': {
|
2012-11-19 08:31:03 +00:00
|
|
|
str++;
|
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
2012-11-19 08:31:03 +00:00
|
|
|
case L'\"':
|
2016-05-01 03:47:05 +00:00
|
|
|
case L'\'': {
|
|
|
|
if (level == 0) {
|
2012-11-19 00:30:30 +00:00
|
|
|
level++;
|
2016-05-01 03:47:05 +00:00
|
|
|
lst.push_back(str - buff);
|
2012-11-19 00:30:30 +00:00
|
|
|
prev_q = *str;
|
2016-05-01 03:47:05 +00:00
|
|
|
} else {
|
|
|
|
if (prev_q == *str) {
|
2012-12-21 01:37:09 +00:00
|
|
|
size_t pos1, pos2;
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2012-11-19 08:31:03 +00:00
|
|
|
level--;
|
|
|
|
pos1 = lst.back();
|
2016-05-01 03:47:05 +00:00
|
|
|
pos2 = str - buff;
|
|
|
|
if (pos1 == pos || pos2 == pos) {
|
|
|
|
color.at(pos1) |=
|
|
|
|
highlight_make_background(highlight_spec_match);
|
|
|
|
color.at(pos2) |=
|
|
|
|
highlight_make_background(highlight_spec_match);
|
2018-12-31 06:43:26 +00:00
|
|
|
match_found = true;
|
2012-11-19 08:31:03 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
prev_q = *str == L'\"' ? L'\'' : L'\"';
|
|
|
|
} else {
|
2012-11-19 08:31:03 +00:00
|
|
|
level++;
|
2016-05-01 03:47:05 +00:00
|
|
|
lst.push_back(str - buff);
|
2012-11-19 08:31:03 +00:00
|
|
|
prev_q = *str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2016-05-01 03:47:05 +00:00
|
|
|
}
|
2016-11-03 01:29:14 +00:00
|
|
|
default: {
|
|
|
|
break; // we ignore all other characters
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2016-05-01 03:47:05 +00:00
|
|
|
if ((*str == L'\0')) break;
|
2012-11-19 00:30:30 +00:00
|
|
|
str++;
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (!match_found) color.at(pos) = highlight_make_background(highlight_spec_error);
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
// Highlight matching parenthesis.
|
2012-02-22 01:55:56 +00:00
|
|
|
const wchar_t c = buffstr.at(pos);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (wcschr(L"()[]{}", c)) {
|
|
|
|
int step = wcschr(L"({[", c) ? 1 : -1;
|
2012-11-19 00:30:30 +00:00
|
|
|
wchar_t dec_char = *(wcschr(L"()[]{}", c) + step);
|
|
|
|
wchar_t inc_char = c;
|
|
|
|
int level = 0;
|
2018-12-31 06:43:26 +00:00
|
|
|
bool match_found = false;
|
2016-05-01 03:47:05 +00:00
|
|
|
for (long i = pos; i >= 0 && (size_t)i < buffstr.size(); i += step) {
|
2012-11-19 00:30:30 +00:00
|
|
|
const wchar_t test_char = buffstr.at(i);
|
2016-05-01 03:47:05 +00:00
|
|
|
if (test_char == inc_char) level++;
|
|
|
|
if (test_char == dec_char) level--;
|
|
|
|
if (level == 0) {
|
2012-11-19 00:30:30 +00:00
|
|
|
long pos2 = i;
|
2016-05-01 03:47:05 +00:00
|
|
|
color.at(pos) |= highlight_spec_match << 16;
|
|
|
|
color.at(pos2) |= highlight_spec_match << 16;
|
2018-12-31 06:43:26 +00:00
|
|
|
match_found = true;
|
2012-11-19 00:30:30 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
if (!match_found) color[pos] = highlight_make_background(highlight_spec_error);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2005-09-20 13:26:39 +00:00
|
|
|
}
|
|
|
|
|
2016-05-01 03:47:05 +00:00
|
|
|
void highlight_universal(const wcstring &buff, std::vector<highlight_spec_t> &color, size_t pos,
|
2018-09-09 09:25:51 +00:00
|
|
|
wcstring_list_t *error, const environment_t &vars) {
|
2016-10-09 21:38:26 +00:00
|
|
|
UNUSED(error);
|
|
|
|
UNUSED(vars);
|
2012-02-22 01:55:56 +00:00
|
|
|
assert(buff.size() == color.size());
|
2012-11-18 10:23:22 +00:00
|
|
|
std::fill(color.begin(), color.end(), 0);
|
2012-11-19 00:30:30 +00:00
|
|
|
highlight_universal_internal(buff, color, pos);
|
2005-09-20 13:26:39 +00:00
|
|
|
}
|