mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-11 20:48:49 +00:00
stop subclassing env_var_t from wcstring
This is the first step to implementing issue #4200 is to stop subclassing env_var_t from wcstring. Not too surprisingly doing this identified several places that were incorrectly treating env_var_t and wcstring as interchangeable types. I'm not talking about those places that passed an env_var_t instance to a function that takes a wcstring. I'm talking about doing things like assigning the former to the latter type, relying on the implicit conversion, and thus losing information. We also rename `env_get_string()` to `env_get()` for symmetry with `env_set()` and to make it clear the function does not return a string.
This commit is contained in:
parent
d87c0424d8
commit
c36ad27618
22 changed files with 205 additions and 186 deletions
|
@ -69,7 +69,7 @@ int autoload_t::load(const wcstring &cmd, bool reload) {
|
||||||
CHECK_BLOCK(0);
|
CHECK_BLOCK(0);
|
||||||
ASSERT_IS_MAIN_THREAD();
|
ASSERT_IS_MAIN_THREAD();
|
||||||
|
|
||||||
env_var_t path_var = env_get_string(env_var_name);
|
env_var_t path_var = env_get(env_var_name);
|
||||||
|
|
||||||
// Do we know where to look?
|
// Do we know where to look?
|
||||||
if (path_var.empty()) return 0;
|
if (path_var.empty()) return 0;
|
||||||
|
@ -79,7 +79,7 @@ int autoload_t::load(const wcstring &cmd, bool reload) {
|
||||||
if (path_var != this->last_path) {
|
if (path_var != this->last_path) {
|
||||||
this->last_path = path_var;
|
this->last_path = path_var;
|
||||||
this->last_path_tokenized.clear();
|
this->last_path_tokenized.clear();
|
||||||
tokenize_variable_array(this->last_path, this->last_path_tokenized);
|
this->last_path.to_list(this->last_path_tokenized);
|
||||||
|
|
||||||
scoped_lock locker(lock);
|
scoped_lock locker(lock);
|
||||||
this->evict_all_nodes();
|
this->evict_all_nodes();
|
||||||
|
@ -115,7 +115,7 @@ bool autoload_t::can_load(const wcstring &cmd, const env_vars_snapshot_t &vars)
|
||||||
if (path_var.missing_or_empty()) return false;
|
if (path_var.missing_or_empty()) return false;
|
||||||
|
|
||||||
std::vector<wcstring> path_list;
|
std::vector<wcstring> path_list;
|
||||||
tokenize_variable_array(path_var, path_list);
|
path_var.to_list(path_list);
|
||||||
return this->locate_file_and_maybe_load_it(cmd, false, false, path_list);
|
return this->locate_file_and_maybe_load_it(cmd, false, false, path_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
#include "env.h"
|
||||||
#include "lru.h"
|
#include "lru.h"
|
||||||
|
|
||||||
/// Record of an attempt to access a file.
|
/// Record of an attempt to access a file.
|
||||||
|
@ -50,7 +51,7 @@ class autoload_t : public lru_cache_t<autoload_t, autoload_function_t> {
|
||||||
/// The environment variable name.
|
/// The environment variable name.
|
||||||
const wcstring env_var_name;
|
const wcstring env_var_name;
|
||||||
/// The path from which we most recently autoloaded.
|
/// The path from which we most recently autoloaded.
|
||||||
wcstring last_path;
|
env_var_t last_path;
|
||||||
/// the most reecently autoloaded path, tokenized (split on separators).
|
/// the most reecently autoloaded path, tokenized (split on separators).
|
||||||
wcstring_list_t last_path_tokenized;
|
wcstring_list_t last_path_tokenized;
|
||||||
/// A table containing all the files that are currently being loaded.
|
/// A table containing all the files that are currently being loaded.
|
||||||
|
|
|
@ -39,7 +39,7 @@ int builtin_cd(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
||||||
if (argv[optind]) {
|
if (argv[optind]) {
|
||||||
dir_in = env_var_t(argv[optind]);
|
dir_in = env_var_t(argv[optind]);
|
||||||
} else {
|
} else {
|
||||||
dir_in = env_get_string(L"HOME");
|
dir_in = env_get(L"HOME");
|
||||||
if (dir_in.missing_or_empty()) {
|
if (dir_in.missing_or_empty()) {
|
||||||
streams.err.append_format(_(L"%ls: Could not find home directory\n"), cmd);
|
streams.err.append_format(_(L"%ls: Could not find home directory\n"), cmd);
|
||||||
return STATUS_CMD_ERROR;
|
return STATUS_CMD_ERROR;
|
||||||
|
|
|
@ -193,7 +193,7 @@ static wcstring functions_def(const wcstring &name) {
|
||||||
it != end; ++it) {
|
it != end; ++it) {
|
||||||
wcstring_list_t lst;
|
wcstring_list_t lst;
|
||||||
if (!it->second.missing()) {
|
if (!it->second.missing()) {
|
||||||
tokenize_variable_array(it->second, lst);
|
it->second.to_list(lst);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This forced tab is crummy, but we don't know what indentation style the function uses.
|
// This forced tab is crummy, but we don't know what indentation style the function uses.
|
||||||
|
|
|
@ -427,9 +427,9 @@ int builtin_read(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!opts.have_delimiter) {
|
if (!opts.have_delimiter) {
|
||||||
env_var_t ifs = env_get_string(L"IFS");
|
env_var_t ifs = env_get(L"IFS");
|
||||||
if (!ifs.missing_or_empty()) {
|
if (!ifs.missing_or_empty()) {
|
||||||
opts.delimiter = ifs;
|
opts.delimiter = ifs.as_string();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (opts.delimiter.empty()) {
|
if (opts.delimiter.empty()) {
|
||||||
|
|
|
@ -212,7 +212,7 @@ static int validate_cmd_opts(const wchar_t *cmd, set_cmd_opts_t &opts, //!OCLIN
|
||||||
static int check_global_scope_exists(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_t *dest,
|
static int check_global_scope_exists(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_t *dest,
|
||||||
io_streams_t &streams) {
|
io_streams_t &streams) {
|
||||||
if (opts.universal) {
|
if (opts.universal) {
|
||||||
env_var_t global_dest = env_get_string(dest, ENV_GLOBAL);
|
env_var_t global_dest = env_get(dest, ENV_GLOBAL);
|
||||||
if (!global_dest.missing()) {
|
if (!global_dest.missing()) {
|
||||||
streams.err.append_format(
|
streams.err.append_format(
|
||||||
_(L"%ls: Warning: universal scope selected, but a global variable '%ls' exists.\n"),
|
_(L"%ls: Warning: universal scope selected, but a global variable '%ls' exists.\n"),
|
||||||
|
@ -239,9 +239,8 @@ static int my_env_path_setup(const wchar_t *cmd, const wchar_t *key, //!OCLINT(
|
||||||
// not the (missing) local value. Also don't bother to complain about relative paths, which
|
// not the (missing) local value. Also don't bother to complain about relative paths, which
|
||||||
// don't start with /.
|
// don't start with /.
|
||||||
wcstring_list_t existing_values;
|
wcstring_list_t existing_values;
|
||||||
const env_var_t existing_variable = env_get_string(key, ENV_DEFAULT);
|
const env_var_t existing_variable = env_get(key, ENV_DEFAULT);
|
||||||
if (!existing_variable.missing_or_empty())
|
if (!existing_variable.missing_or_empty()) existing_variable.to_list(existing_values);
|
||||||
tokenize_variable_array(existing_variable, existing_values);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < list.size(); i++) {
|
for (size_t i = 0; i < list.size(); i++) {
|
||||||
const wcstring &dir = list.at(i);
|
const wcstring &dir = list.at(i);
|
||||||
|
@ -346,9 +345,9 @@ static int parse_index(std::vector<long> &indexes, wchar_t *src, int scope, io_s
|
||||||
*p = L'\0'; // split the var name from the indexes/slices
|
*p = L'\0'; // split the var name from the indexes/slices
|
||||||
p++;
|
p++;
|
||||||
|
|
||||||
env_var_t var_str = env_get_string(src, scope);
|
env_var_t var_str = env_get(src, scope);
|
||||||
wcstring_list_t var;
|
wcstring_list_t var;
|
||||||
if (!var_str.missing()) tokenize_variable_array(var_str, var);
|
if (!var_str.missing()) var_str.to_list(var);
|
||||||
|
|
||||||
int count = 0;
|
int count = 0;
|
||||||
|
|
||||||
|
@ -457,15 +456,16 @@ static int builtin_set_list(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
||||||
streams.out.append(e_key);
|
streams.out.append(e_key);
|
||||||
|
|
||||||
if (!names_only) {
|
if (!names_only) {
|
||||||
env_var_t value = env_get_string(key, compute_scope(opts));
|
env_var_t var = env_get(key, compute_scope(opts));
|
||||||
if (!value.missing()) {
|
if (!var.missing()) {
|
||||||
bool shorten = false;
|
bool shorten = false;
|
||||||
if (opts.shorten_ok && value.length() > 64) {
|
wcstring val = var.as_string();
|
||||||
|
if (opts.shorten_ok && val.length() > 64) {
|
||||||
shorten = true;
|
shorten = true;
|
||||||
value.resize(60);
|
val.resize(60);
|
||||||
}
|
}
|
||||||
|
|
||||||
wcstring e_value = expand_escape_variable(value);
|
wcstring e_value = expand_escape_variable(val);
|
||||||
streams.out.append(L" ");
|
streams.out.append(L" ");
|
||||||
streams.out.append(e_value);
|
streams.out.append(e_value);
|
||||||
|
|
||||||
|
@ -500,8 +500,8 @@ static int builtin_set_query(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
||||||
|
|
||||||
if (idx_count) {
|
if (idx_count) {
|
||||||
wcstring_list_t result;
|
wcstring_list_t result;
|
||||||
env_var_t dest_str = env_get_string(dest, scope);
|
env_var_t dest_str = env_get(dest, scope);
|
||||||
if (!dest_str.missing()) tokenize_variable_array(dest_str, result);
|
if (!dest_str.missing()) dest_str.to_list(result);
|
||||||
|
|
||||||
for (auto idx : indexes) {
|
for (auto idx : indexes) {
|
||||||
if (idx < 1 || (size_t)idx > result.size()) retval++;
|
if (idx < 1 || (size_t)idx > result.size()) retval++;
|
||||||
|
@ -538,12 +538,12 @@ static void show_scope(const wchar_t *var_name, int scope, io_streams_t &streams
|
||||||
}
|
}
|
||||||
|
|
||||||
if (env_exist(var_name, scope)) {
|
if (env_exist(var_name, scope)) {
|
||||||
const env_var_t evar = env_get_string(var_name, scope | ENV_EXPORT | ENV_USER);
|
const env_var_t evar = env_get(var_name, scope | ENV_EXPORT | ENV_USER);
|
||||||
const wchar_t *exportv = evar.missing() ? _(L"unexported") : _(L"exported");
|
const wchar_t *exportv = evar.missing() ? _(L"unexported") : _(L"exported");
|
||||||
|
|
||||||
const env_var_t var = env_get_string(var_name, scope | ENV_USER);
|
const env_var_t var = env_get(var_name, scope | ENV_USER);
|
||||||
wcstring_list_t result;
|
wcstring_list_t result;
|
||||||
if (!var.empty()) tokenize_variable_array(var, result);
|
if (!var.empty()) var.to_list(result);
|
||||||
|
|
||||||
streams.out.append_format(_(L"$%ls: set in %ls scope, %ls, with %d elements\n"), var_name,
|
streams.out.append_format(_(L"$%ls: set in %ls scope, %ls, with %d elements\n"), var_name,
|
||||||
scope_name, exportv, result.size());
|
scope_name, exportv, result.size());
|
||||||
|
@ -632,10 +632,10 @@ static int builtin_set_erase(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
||||||
if (idx_count == 0) { // unset the var
|
if (idx_count == 0) { // unset the var
|
||||||
retval = env_remove(dest, scope);
|
retval = env_remove(dest, scope);
|
||||||
} else { // remove just the specified indexes of the var
|
} else { // remove just the specified indexes of the var
|
||||||
const env_var_t dest_var = env_get_string(dest, scope);
|
const env_var_t dest_var = env_get(dest, scope);
|
||||||
if (dest_var.missing()) return STATUS_CMD_ERROR;
|
if (dest_var.missing()) return STATUS_CMD_ERROR;
|
||||||
wcstring_list_t result;
|
wcstring_list_t result;
|
||||||
tokenize_variable_array(dest_var, result);
|
dest_var.to_list(result);
|
||||||
erase_values(result, indexes);
|
erase_values(result, indexes);
|
||||||
retval = my_env_set(cmd, dest, result, scope, streams);
|
retval = my_env_set(cmd, dest, result, scope, streams);
|
||||||
}
|
}
|
||||||
|
@ -658,9 +658,9 @@ static int set_var_array(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_t
|
||||||
for (int i = 0; i < argc; i++) new_values.push_back(argv[i]);
|
for (int i = 0; i < argc; i++) new_values.push_back(argv[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
env_var_t var_str = env_get_string(varname, scope);
|
env_var_t var_str = env_get(varname, scope);
|
||||||
wcstring_list_t var_array;
|
wcstring_list_t var_array;
|
||||||
if (!var_str.missing()) tokenize_variable_array(var_str, var_array);
|
if (!var_str.missing()) var_str.to_list(var_array);
|
||||||
new_values.insert(new_values.end(), var_array.begin(), var_array.end());
|
new_values.insert(new_values.end(), var_array.begin(), var_array.end());
|
||||||
|
|
||||||
if (opts.append) {
|
if (opts.append) {
|
||||||
|
@ -691,8 +691,8 @@ static int set_var_slices(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_
|
||||||
}
|
}
|
||||||
|
|
||||||
int scope = compute_scope(opts); // calculate the variable scope based on the provided options
|
int scope = compute_scope(opts); // calculate the variable scope based on the provided options
|
||||||
const env_var_t var_str = env_get_string(varname, scope);
|
const env_var_t var_str = env_get(varname, scope);
|
||||||
if (!var_str.missing()) tokenize_variable_array(var_str, new_values);
|
if (!var_str.missing()) var_str.to_list(new_values);
|
||||||
|
|
||||||
// Slice indexes have been calculated, do the actual work.
|
// Slice indexes have been calculated, do the actual work.
|
||||||
wcstring_list_t result;
|
wcstring_list_t result;
|
||||||
|
|
|
@ -1554,8 +1554,8 @@ static void validate_new_termsize(struct winsize *new_termsize) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
// Fallback to the environment vars.
|
// Fallback to the environment vars.
|
||||||
env_var_t col_var = env_get_string(L"COLUMNS");
|
env_var_t col_var = env_get(L"COLUMNS");
|
||||||
env_var_t row_var = env_get_string(L"LINES");
|
env_var_t row_var = env_get(L"LINES");
|
||||||
if (!col_var.missing_or_empty() && !row_var.missing_or_empty()) {
|
if (!col_var.missing_or_empty() && !row_var.missing_or_empty()) {
|
||||||
// Both vars have to have valid values.
|
// Both vars have to have valid values.
|
||||||
int col = fish_wcstoi(col_var.c_str());
|
int col = fish_wcstoi(col_var.c_str());
|
||||||
|
@ -1582,11 +1582,11 @@ static void validate_new_termsize(struct winsize *new_termsize) {
|
||||||
/// Export the new terminal size as env vars and to the kernel if possible.
|
/// Export the new terminal size as env vars and to the kernel if possible.
|
||||||
static void export_new_termsize(struct winsize *new_termsize) {
|
static void export_new_termsize(struct winsize *new_termsize) {
|
||||||
wchar_t buf[64];
|
wchar_t buf[64];
|
||||||
env_var_t cols = env_get_string(L"COLUMNS", ENV_EXPORT);
|
env_var_t cols = env_get(L"COLUMNS", ENV_EXPORT);
|
||||||
swprintf(buf, 64, L"%d", (int)new_termsize->ws_col);
|
swprintf(buf, 64, L"%d", (int)new_termsize->ws_col);
|
||||||
env_set(L"COLUMNS", buf, ENV_GLOBAL | (cols.missing_or_empty() ? 0 : ENV_EXPORT));
|
env_set(L"COLUMNS", buf, ENV_GLOBAL | (cols.missing_or_empty() ? 0 : ENV_EXPORT));
|
||||||
|
|
||||||
env_var_t lines = env_get_string(L"LINES", ENV_EXPORT);
|
env_var_t lines = env_get(L"LINES", ENV_EXPORT);
|
||||||
swprintf(buf, 64, L"%d", (int)new_termsize->ws_row);
|
swprintf(buf, 64, L"%d", (int)new_termsize->ws_row);
|
||||||
env_set(L"LINES", buf, ENV_GLOBAL | (lines.missing_or_empty() ? 0 : ENV_EXPORT));
|
env_set(L"LINES", buf, ENV_GLOBAL | (lines.missing_or_empty() ? 0 : ENV_EXPORT));
|
||||||
|
|
||||||
|
|
|
@ -1106,13 +1106,14 @@ bool completer_t::complete_variable(const wcstring &str, size_t start_offset) {
|
||||||
wcstring desc;
|
wcstring desc;
|
||||||
if (this->wants_descriptions()) {
|
if (this->wants_descriptions()) {
|
||||||
// Can't use this->vars here, it could be any variable.
|
// Can't use this->vars here, it could be any variable.
|
||||||
env_var_t value_unescaped = env_get_string(env_name);
|
env_var_t var = env_get(env_name);
|
||||||
if (value_unescaped.missing()) continue;
|
if (var.missing()) continue;
|
||||||
|
|
||||||
wcstring value = expand_escape_variable(value_unescaped);
|
wcstring value = expand_escape_variable(var.as_string());
|
||||||
if (this->type() != COMPLETE_AUTOSUGGEST)
|
if (this->type() != COMPLETE_AUTOSUGGEST) {
|
||||||
desc = format_string(COMPLETE_VAR_DESC_VAL, value.c_str());
|
desc = format_string(COMPLETE_VAR_DESC_VAL, value.c_str());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
append_completion(&this->completions, comp, desc, flags, match);
|
append_completion(&this->completions, comp, desc, flags, match);
|
||||||
|
|
||||||
|
|
160
src/env.cpp
160
src/env.cpp
|
@ -331,13 +331,13 @@ static bool var_is_timezone(const wcstring &key) { return key == L"TZ"; }
|
||||||
/// Properly sets all timezone information.
|
/// Properly sets all timezone information.
|
||||||
static void handle_timezone(const wchar_t *env_var_name) {
|
static void handle_timezone(const wchar_t *env_var_name) {
|
||||||
debug(2, L"handle_timezone() called in response to '%ls' changing", env_var_name);
|
debug(2, L"handle_timezone() called in response to '%ls' changing", env_var_name);
|
||||||
const env_var_t val = env_get_string(env_var_name, ENV_EXPORT);
|
const env_var_t var = env_get(env_var_name, ENV_EXPORT);
|
||||||
const std::string &value = wcs2string(val);
|
|
||||||
const std::string &name = wcs2string(env_var_name);
|
const std::string &name = wcs2string(env_var_name);
|
||||||
debug(2, L"timezone var %s='%s'", name.c_str(), value.c_str());
|
debug(2, L"timezone var %s='%s'", name.c_str(), var.c_str());
|
||||||
if (val.empty()) {
|
if (var.missing_or_empty()) {
|
||||||
unsetenv(name.c_str());
|
unsetenv(name.c_str());
|
||||||
} else {
|
} else {
|
||||||
|
const std::string &value = wcs2string(var.as_string());
|
||||||
setenv(name.c_str(), value.c_str(), 1);
|
setenv(name.c_str(), value.c_str(), 1);
|
||||||
}
|
}
|
||||||
tzset();
|
tzset();
|
||||||
|
@ -350,13 +350,13 @@ static void fix_colon_delimited_var(const wcstring &var_name) {
|
||||||
// While we auto split/join MANPATH we do not want to replace empty elements with "." (#4158).
|
// While we auto split/join MANPATH we do not want to replace empty elements with "." (#4158).
|
||||||
if (var_name == L"MANPATH") return;
|
if (var_name == L"MANPATH") return;
|
||||||
|
|
||||||
const env_var_t paths = env_get_string(var_name);
|
const env_var_t paths = env_get(var_name);
|
||||||
if (paths.missing_or_empty()) return;
|
if (paths.missing_or_empty()) return;
|
||||||
|
|
||||||
bool modified = false;
|
bool modified = false;
|
||||||
wcstring_list_t pathsv;
|
wcstring_list_t pathsv;
|
||||||
wcstring_list_t new_pathsv;
|
wcstring_list_t new_pathsv;
|
||||||
tokenize_variable_array(paths, pathsv);
|
paths.to_list(pathsv);
|
||||||
for (auto next_path : pathsv) {
|
for (auto next_path : pathsv) {
|
||||||
if (next_path.empty()) {
|
if (next_path.empty()) {
|
||||||
next_path = L".";
|
next_path = L".";
|
||||||
|
@ -400,13 +400,14 @@ static void init_locale() {
|
||||||
char *old_msg_locale = strdup(setlocale(LC_MESSAGES, NULL));
|
char *old_msg_locale = strdup(setlocale(LC_MESSAGES, NULL));
|
||||||
|
|
||||||
for (auto var_name : locale_variables) {
|
for (auto var_name : locale_variables) {
|
||||||
const env_var_t val = env_get_string(var_name, ENV_EXPORT);
|
const env_var_t var = env_get(var_name, ENV_EXPORT);
|
||||||
const std::string &name = wcs2string(var_name);
|
const std::string &name = wcs2string(var_name);
|
||||||
const std::string &value = wcs2string(val);
|
if (var.missing_or_empty()) {
|
||||||
debug(2, L"locale var %s='%s'", name.c_str(), value.c_str());
|
debug(2, L"locale var %s missing or empty", name.c_str());
|
||||||
if (val.empty()) {
|
|
||||||
unsetenv(name.c_str());
|
unsetenv(name.c_str());
|
||||||
} else {
|
} else {
|
||||||
|
const std::string &value = wcs2string(var.as_string());
|
||||||
|
debug(2, L"locale var %s='%s'", name.c_str(), value.c_str());
|
||||||
setenv(name.c_str(), value.c_str(), 1);
|
setenv(name.c_str(), value.c_str(), 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -455,17 +456,17 @@ bool term_supports_setting_title() { return can_set_term_title; }
|
||||||
/// don't. Since we can't see the underlying terminal below screen there is no way to fix this.
|
/// don't. Since we can't see the underlying terminal below screen there is no way to fix this.
|
||||||
static const wcstring_list_t title_terms({L"xterm", L"screen", L"tmux", L"nxterm", L"rxvt"});
|
static const wcstring_list_t title_terms({L"xterm", L"screen", L"tmux", L"nxterm", L"rxvt"});
|
||||||
static bool does_term_support_setting_title() {
|
static bool does_term_support_setting_title() {
|
||||||
const env_var_t term_str = env_get_string(L"TERM");
|
const env_var_t term_var = env_get(L"TERM");
|
||||||
if (term_str.missing()) return false;
|
if (term_var.missing_or_empty()) return false;
|
||||||
|
|
||||||
const wchar_t *term = term_str.c_str();
|
const wchar_t *term = term_var.c_str();
|
||||||
bool recognized = contains(title_terms, term_str);
|
bool recognized = contains(title_terms, term_var.as_string());
|
||||||
if (!recognized) recognized = !wcsncmp(term, L"xterm-", wcslen(L"xterm-"));
|
if (!recognized) recognized = !wcsncmp(term, L"xterm-", wcslen(L"xterm-"));
|
||||||
if (!recognized) recognized = !wcsncmp(term, L"screen-", wcslen(L"screen-"));
|
if (!recognized) recognized = !wcsncmp(term, L"screen-", wcslen(L"screen-"));
|
||||||
if (!recognized) recognized = !wcsncmp(term, L"tmux-", wcslen(L"tmux-"));
|
if (!recognized) recognized = !wcsncmp(term, L"tmux-", wcslen(L"tmux-"));
|
||||||
if (!recognized) {
|
if (!recognized) {
|
||||||
if (term_str == L"linux") return false;
|
if (wcscmp(term, L"linux") == 0) return false;
|
||||||
if (term_str == L"dumb") return false;
|
if (wcscmp(term, L"dumb") == 0) return false;
|
||||||
|
|
||||||
char buf[PATH_MAX];
|
char buf[PATH_MAX];
|
||||||
int retval = ttyname_r(STDIN_FILENO, buf, PATH_MAX);
|
int retval = ttyname_r(STDIN_FILENO, buf, PATH_MAX);
|
||||||
|
@ -479,11 +480,12 @@ static bool does_term_support_setting_title() {
|
||||||
static void update_fish_color_support() {
|
static void update_fish_color_support() {
|
||||||
// Detect or infer term256 support. If fish_term256 is set, we respect it;
|
// Detect or infer term256 support. If fish_term256 is set, we respect it;
|
||||||
// otherwise infer it from the TERM variable or use terminfo.
|
// otherwise infer it from the TERM variable or use terminfo.
|
||||||
env_var_t fish_term256 = env_get_string(L"fish_term256");
|
env_var_t fish_term256 = env_get(L"fish_term256");
|
||||||
env_var_t term = env_get_string(L"TERM");
|
env_var_t term_var = env_get(L"TERM");
|
||||||
|
wcstring term = term_var.missing_or_empty() ? L"" : term_var.as_string();
|
||||||
bool support_term256 = false; // default to no support
|
bool support_term256 = false; // default to no support
|
||||||
if (!fish_term256.missing_or_empty()) {
|
if (!fish_term256.missing_or_empty()) {
|
||||||
support_term256 = from_string<bool>(fish_term256);
|
support_term256 = from_string<bool>(fish_term256.as_string());
|
||||||
debug(2, L"256 color support determined by 'fish_term256'");
|
debug(2, L"256 color support determined by 'fish_term256'");
|
||||||
} else if (term.find(L"256color") != wcstring::npos) {
|
} else if (term.find(L"256color") != wcstring::npos) {
|
||||||
// TERM=*256color*: Explicitly supported.
|
// TERM=*256color*: Explicitly supported.
|
||||||
|
@ -491,11 +493,12 @@ static void update_fish_color_support() {
|
||||||
debug(2, L"256 color support enabled for '256color' in TERM");
|
debug(2, L"256 color support enabled for '256color' in TERM");
|
||||||
} else if (term.find(L"xterm") != wcstring::npos) {
|
} else if (term.find(L"xterm") != wcstring::npos) {
|
||||||
// Assume that all xterms are 256, except for OS X SnowLeopard
|
// Assume that all xterms are 256, except for OS X SnowLeopard
|
||||||
const env_var_t prog = env_get_string(L"TERM_PROGRAM");
|
const env_var_t prog_var = env_get(L"TERM_PROGRAM");
|
||||||
const env_var_t progver = env_get_string(L"TERM_PROGRAM_VERSION");
|
const env_var_t progver_var = env_get(L"TERM_PROGRAM_VERSION");
|
||||||
if (prog == L"Apple_Terminal" && !progver.missing_or_empty()) {
|
wcstring term_program = prog_var.missing_or_empty() ? L"" : prog_var.as_string();
|
||||||
|
if (term_program == L"Apple_Terminal" && !progver_var.missing_or_empty()) {
|
||||||
// OS X Lion is version 300+, it has 256 color support
|
// OS X Lion is version 300+, it has 256 color support
|
||||||
if (strtod(wcs2str(progver), NULL) > 300) {
|
if (strtod(wcs2str(progver_var.as_string()), NULL) > 300) {
|
||||||
support_term256 = true;
|
support_term256 = true;
|
||||||
debug(2, L"256 color support enabled for TERM=xterm + modern Terminal.app");
|
debug(2, L"256 color support enabled for TERM=xterm + modern Terminal.app");
|
||||||
}
|
}
|
||||||
|
@ -511,10 +514,10 @@ static void update_fish_color_support() {
|
||||||
debug(2, L"256 color support not enabled (yet)");
|
debug(2, L"256 color support not enabled (yet)");
|
||||||
}
|
}
|
||||||
|
|
||||||
env_var_t fish_term24bit = env_get_string(L"fish_term24bit");
|
env_var_t fish_term24bit = env_get(L"fish_term24bit");
|
||||||
bool support_term24bit;
|
bool support_term24bit;
|
||||||
if (!fish_term24bit.missing_or_empty()) {
|
if (!fish_term24bit.missing_or_empty()) {
|
||||||
support_term24bit = from_string<bool>(fish_term24bit);
|
support_term24bit = from_string<bool>(fish_term24bit.as_string());
|
||||||
debug(2, L"'fish_term24bit' preference: 24-bit color %s",
|
debug(2, L"'fish_term24bit' preference: 24-bit color %s",
|
||||||
support_term24bit ? L"enabled" : L"disabled");
|
support_term24bit ? L"enabled" : L"disabled");
|
||||||
} else {
|
} else {
|
||||||
|
@ -534,12 +537,14 @@ static void update_fish_color_support() {
|
||||||
static bool initialize_curses_using_fallback(const char *term) {
|
static bool initialize_curses_using_fallback(const char *term) {
|
||||||
// If $TERM is already set to the fallback name we're about to use there isn't any point in
|
// If $TERM is already set to the fallback name we're about to use there isn't any point in
|
||||||
// seeing if the fallback name can be used.
|
// seeing if the fallback name can be used.
|
||||||
const char *term_env = wcs2str(env_get_string(L"TERM"));
|
env_var_t term_var = env_get(L"TERM");
|
||||||
|
if (term_var.missing_or_empty()) return false;
|
||||||
|
|
||||||
|
const char *term_env = wcs2str(term_var.as_string());
|
||||||
if (!strcmp(term_env, DEFAULT_TERM1) || !strcmp(term_env, DEFAULT_TERM2)) return false;
|
if (!strcmp(term_env, DEFAULT_TERM1) || !strcmp(term_env, DEFAULT_TERM2)) return false;
|
||||||
|
|
||||||
if (is_interactive_session) {
|
if (is_interactive_session) debug(1, _(L"Using fallback terminal type '%s'."), term);
|
||||||
debug(1, _(L"Using fallback terminal type '%s'."), term);
|
|
||||||
}
|
|
||||||
int err_ret;
|
int err_ret;
|
||||||
if (setupterm((char *)term, STDOUT_FILENO, &err_ret) == OK) return true;
|
if (setupterm((char *)term, STDOUT_FILENO, &err_ret) == OK) return true;
|
||||||
if (is_interactive_session) {
|
if (is_interactive_session) {
|
||||||
|
@ -559,20 +564,21 @@ static void init_path_vars() {
|
||||||
/// Initialize the curses subsystem.
|
/// Initialize the curses subsystem.
|
||||||
static void init_curses() {
|
static void init_curses() {
|
||||||
for (auto var_name : curses_variables) {
|
for (auto var_name : curses_variables) {
|
||||||
const env_var_t val = env_get_string(var_name, ENV_EXPORT);
|
const env_var_t var = env_get(var_name, ENV_EXPORT);
|
||||||
const std::string &name = wcs2string(var_name);
|
const std::string &name = wcs2string(var_name);
|
||||||
const std::string &value = wcs2string(val);
|
if (var.missing_or_empty()) {
|
||||||
debug(2, L"curses var %s='%s'", name.c_str(), value.c_str());
|
debug(2, L"curses var %s missing or empty", name.c_str());
|
||||||
if (val.empty()) {
|
|
||||||
unsetenv(name.c_str());
|
unsetenv(name.c_str());
|
||||||
} else {
|
} else {
|
||||||
|
const std::string &value = wcs2string(var.as_string());
|
||||||
|
debug(2, L"curses var %s='%s'", name.c_str(), value.c_str());
|
||||||
setenv(name.c_str(), value.c_str(), 1);
|
setenv(name.c_str(), value.c_str(), 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int err_ret;
|
int err_ret;
|
||||||
if (setupterm(NULL, STDOUT_FILENO, &err_ret) == ERR) {
|
if (setupterm(NULL, STDOUT_FILENO, &err_ret) == ERR) {
|
||||||
env_var_t term = env_get_string(L"TERM");
|
env_var_t term = env_get(L"TERM");
|
||||||
if (is_interactive_session) {
|
if (is_interactive_session) {
|
||||||
debug(1, _(L"Could not set up terminal."));
|
debug(1, _(L"Could not set up terminal."));
|
||||||
if (term.missing_or_empty()) {
|
if (term.missing_or_empty()) {
|
||||||
|
@ -666,7 +672,7 @@ static void universal_callback(fish_message_type_t type, const wchar_t *name) {
|
||||||
|
|
||||||
/// Make sure the PATH variable contains something.
|
/// Make sure the PATH variable contains something.
|
||||||
static void setup_path() {
|
static void setup_path() {
|
||||||
const env_var_t path = env_get_string(L"PATH");
|
const env_var_t path = env_get(L"PATH");
|
||||||
if (path.missing_or_empty()) {
|
if (path.missing_or_empty()) {
|
||||||
const wchar_t *value = L"/usr/bin" ARRAY_SEP_STR L"/bin";
|
const wchar_t *value = L"/usr/bin" ARRAY_SEP_STR L"/bin";
|
||||||
env_set(L"PATH", value, ENV_GLOBAL | ENV_EXPORT);
|
env_set(L"PATH", value, ENV_GLOBAL | ENV_EXPORT);
|
||||||
|
@ -677,10 +683,10 @@ static void setup_path() {
|
||||||
/// defaults. They will be updated later by the `get_current_winsize()` function if they need to be
|
/// defaults. They will be updated later by the `get_current_winsize()` function if they need to be
|
||||||
/// adjusted.
|
/// adjusted.
|
||||||
static void env_set_termsize() {
|
static void env_set_termsize() {
|
||||||
env_var_t cols = env_get_string(L"COLUMNS");
|
env_var_t cols = env_get(L"COLUMNS");
|
||||||
if (cols.missing_or_empty()) env_set(L"COLUMNS", DFLT_TERM_COL_STR, ENV_GLOBAL);
|
if (cols.missing_or_empty()) env_set(L"COLUMNS", DFLT_TERM_COL_STR, ENV_GLOBAL);
|
||||||
|
|
||||||
env_var_t rows = env_get_string(L"LINES");
|
env_var_t rows = env_get(L"LINES");
|
||||||
if (rows.missing_or_empty()) env_set(L"LINES", DFLT_TERM_ROW_STR, ENV_GLOBAL);
|
if (rows.missing_or_empty()) env_set(L"LINES", DFLT_TERM_ROW_STR, ENV_GLOBAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -698,7 +704,7 @@ bool env_set_pwd() {
|
||||||
/// Allow the user to override the limit on how much data the `read` command will process.
|
/// Allow the user to override the limit on how much data the `read` command will process.
|
||||||
/// This is primarily for testing but could be used by users in special situations.
|
/// This is primarily for testing but could be used by users in special situations.
|
||||||
void env_set_read_limit() {
|
void env_set_read_limit() {
|
||||||
env_var_t read_byte_limit_var = env_get_string(L"FISH_READ_BYTE_LIMIT");
|
env_var_t read_byte_limit_var = env_get(L"FISH_READ_BYTE_LIMIT");
|
||||||
if (!read_byte_limit_var.missing_or_empty()) {
|
if (!read_byte_limit_var.missing_or_empty()) {
|
||||||
size_t limit = fish_wcstoull(read_byte_limit_var.c_str());
|
size_t limit = fish_wcstoull(read_byte_limit_var.c_str());
|
||||||
if (errno) {
|
if (errno) {
|
||||||
|
@ -710,10 +716,11 @@ void env_set_read_limit() {
|
||||||
}
|
}
|
||||||
|
|
||||||
wcstring env_get_pwd_slash(void) {
|
wcstring env_get_pwd_slash(void) {
|
||||||
env_var_t pwd = env_get_string(L"PWD");
|
env_var_t pwd_var = env_get(L"PWD");
|
||||||
if (pwd.missing_or_empty()) {
|
if (pwd_var.missing_or_empty()) {
|
||||||
return L"";
|
return L"";
|
||||||
}
|
}
|
||||||
|
wcstring pwd = pwd_var.as_string();
|
||||||
if (!string_suffixes_string(L"/", pwd)) {
|
if (!string_suffixes_string(L"/", pwd)) {
|
||||||
pwd.push_back(L'/');
|
pwd.push_back(L'/');
|
||||||
}
|
}
|
||||||
|
@ -722,7 +729,7 @@ wcstring env_get_pwd_slash(void) {
|
||||||
|
|
||||||
/// Set up the USER variable.
|
/// Set up the USER variable.
|
||||||
static void setup_user(bool force) {
|
static void setup_user(bool force) {
|
||||||
if (env_get_string(L"USER").missing_or_empty() || force) {
|
if (env_get(L"USER").missing_or_empty() || force) {
|
||||||
struct passwd userinfo;
|
struct passwd userinfo;
|
||||||
struct passwd *result;
|
struct passwd *result;
|
||||||
char buf[8192];
|
char buf[8192];
|
||||||
|
@ -863,12 +870,12 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
|
||||||
env_set(L"FISH_VERSION", version.c_str(), ENV_GLOBAL);
|
env_set(L"FISH_VERSION", version.c_str(), ENV_GLOBAL);
|
||||||
|
|
||||||
// Set up SHLVL variable.
|
// Set up SHLVL variable.
|
||||||
const env_var_t shlvl_str = env_get_string(L"SHLVL");
|
const env_var_t shlvl_var = env_get(L"SHLVL");
|
||||||
wcstring nshlvl_str = L"1";
|
wcstring nshlvl_str = L"1";
|
||||||
if (!shlvl_str.missing()) {
|
if (!shlvl_var.missing_or_empty()) {
|
||||||
const wchar_t *end;
|
const wchar_t *end;
|
||||||
// TODO: Figure out how to handle invalid numbers better. Shouldn't we issue a diagnostic?
|
// TODO: Figure out how to handle invalid numbers better. Shouldn't we issue a diagnostic?
|
||||||
long shlvl_i = fish_wcstol(shlvl_str.c_str(), &end);
|
long shlvl_i = fish_wcstol(shlvl_var.c_str(), &end);
|
||||||
if (!errno && shlvl_i >= 0) {
|
if (!errno && shlvl_i >= 0) {
|
||||||
nshlvl_str = to_string<long>(shlvl_i + 1);
|
nshlvl_str = to_string<long>(shlvl_i + 1);
|
||||||
}
|
}
|
||||||
|
@ -881,10 +888,10 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
|
||||||
// if the target user is root, unless "--preserve-environment" is used.
|
// if the target user is root, unless "--preserve-environment" is used.
|
||||||
// Since that is an explicit choice, we should allow it to enable e.g.
|
// Since that is an explicit choice, we should allow it to enable e.g.
|
||||||
// env HOME=(mktemp -d) su --preserve-environment fish
|
// env HOME=(mktemp -d) su --preserve-environment fish
|
||||||
if (env_get_string(L"HOME").missing_or_empty()) {
|
if (env_get(L"HOME").missing_or_empty()) {
|
||||||
const env_var_t unam = env_get_string(L"USER");
|
env_var_t user_var = env_get(L"USER");
|
||||||
if (!unam.missing_or_empty()) {
|
if (!user_var.missing_or_empty()) {
|
||||||
char *unam_narrow = wcs2str(unam.c_str());
|
char *unam_narrow = wcs2str(user_var.c_str());
|
||||||
struct passwd userinfo;
|
struct passwd userinfo;
|
||||||
struct passwd *result;
|
struct passwd *result;
|
||||||
char buf[8192];
|
char buf[8192];
|
||||||
|
@ -892,10 +899,12 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
|
||||||
if (retval || !result) {
|
if (retval || !result) {
|
||||||
// Maybe USER is set but it's bogus. Reset USER from the db and try again.
|
// Maybe USER is set but it's bogus. Reset USER from the db and try again.
|
||||||
setup_user(true);
|
setup_user(true);
|
||||||
const env_var_t unam = env_get_string(L"USER");
|
user_var = env_get(L"USER");
|
||||||
unam_narrow = wcs2str(unam.c_str());
|
if (!user_var.missing_or_empty()) {
|
||||||
|
unam_narrow = wcs2str(user_var.c_str());
|
||||||
retval = getpwnam_r(unam_narrow, &userinfo, buf, sizeof(buf), &result);
|
retval = getpwnam_r(unam_narrow, &userinfo, buf, sizeof(buf), &result);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (!retval && result && userinfo.pw_dir) {
|
if (!retval && result && userinfo.pw_dir) {
|
||||||
const wcstring dir = str2wcstring(userinfo.pw_dir);
|
const wcstring dir = str2wcstring(userinfo.pw_dir);
|
||||||
env_set(L"HOME", dir.c_str(), ENV_GLOBAL | ENV_EXPORT);
|
env_set(L"HOME", dir.c_str(), ENV_GLOBAL | ENV_EXPORT);
|
||||||
|
@ -918,9 +927,8 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
|
||||||
env_set_read_limit(); // initialize the read_byte_limit
|
env_set_read_limit(); // initialize the read_byte_limit
|
||||||
|
|
||||||
// Set g_use_posix_spawn. Default to true.
|
// Set g_use_posix_spawn. Default to true.
|
||||||
env_var_t use_posix_spawn = env_get_string(L"fish_use_posix_spawn");
|
env_var_t use_posix_spawn = env_get(L"fish_use_posix_spawn");
|
||||||
g_use_posix_spawn =
|
g_use_posix_spawn = use_posix_spawn.missing_or_empty() ? true : from_string<bool>(use_posix_spawn.as_string());
|
||||||
(use_posix_spawn.missing_or_empty() ? true : from_string<bool>(use_posix_spawn));
|
|
||||||
|
|
||||||
// Set fish_bind_mode to "default".
|
// Set fish_bind_mode to "default".
|
||||||
env_set(FISH_BIND_MODE_VAR, DEFAULT_BIND_MODE, ENV_GLOBAL);
|
env_set(FISH_BIND_MODE_VAR, DEFAULT_BIND_MODE, ENV_GLOBAL);
|
||||||
|
@ -1208,12 +1216,22 @@ int env_remove(const wcstring &key, int var_mode) {
|
||||||
return !erased;
|
return !erased;
|
||||||
}
|
}
|
||||||
|
|
||||||
const wchar_t *env_var_t::c_str(void) const {
|
wcstring env_var_t::as_string(void) const {
|
||||||
assert(!is_missing); //!OCLINT(multiple unary operator)
|
assert(!is_missing);
|
||||||
return wcstring::c_str();
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
env_var_t env_get_string(const wcstring &key, env_mode_flags_t mode) {
|
const wchar_t *env_var_t::c_str(void) const {
|
||||||
|
assert(!is_missing);
|
||||||
|
return val.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
void env_var_t::to_list(wcstring_list_t &out) const {
|
||||||
|
assert(!is_missing);
|
||||||
|
tokenize_variable_array(val, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
env_var_t env_get(const wcstring &key, env_mode_flags_t mode) {
|
||||||
const bool has_scope = mode & (ENV_LOCAL | ENV_GLOBAL | ENV_UNIVERSAL);
|
const bool has_scope = mode & (ENV_LOCAL | ENV_GLOBAL | ENV_UNIVERSAL);
|
||||||
const bool search_local = !has_scope || (mode & ENV_LOCAL);
|
const bool search_local = !has_scope || (mode & ENV_LOCAL);
|
||||||
const bool search_global = !has_scope || (mode & ENV_GLOBAL);
|
const bool search_global = !has_scope || (mode & ENV_GLOBAL);
|
||||||
|
@ -1232,18 +1250,18 @@ env_var_t env_get_string(const wcstring &key, env_mode_flags_t mode) {
|
||||||
if (!is_main_thread()) {
|
if (!is_main_thread()) {
|
||||||
return env_var_t::missing_var();
|
return env_var_t::missing_var();
|
||||||
}
|
}
|
||||||
env_var_t result;
|
|
||||||
|
|
||||||
history_t *history = reader_get_history();
|
history_t *history = reader_get_history();
|
||||||
if (!history) {
|
if (!history) {
|
||||||
history = &history_t::history_with_name(history_session_id());
|
history = &history_t::history_with_name(history_session_id());
|
||||||
}
|
}
|
||||||
|
wcstring result;
|
||||||
if (history) history->get_string_representation(&result, ARRAY_SEP_STR);
|
if (history) history->get_string_representation(&result, ARRAY_SEP_STR);
|
||||||
return result;
|
return env_var_t(result);
|
||||||
} else if (key == L"status") {
|
} else if (key == L"status") {
|
||||||
return to_string(proc_get_last_status());
|
return env_var_t(to_string(proc_get_last_status()));
|
||||||
} else if (key == L"umask") {
|
} else if (key == L"umask") {
|
||||||
return format_string(L"0%0.3o", get_umask());
|
return env_var_t(format_string(L"0%0.3o", get_umask()));
|
||||||
}
|
}
|
||||||
// We should never get here unless the electric var list is out of sync with the above code.
|
// We should never get here unless the electric var list is out of sync with the above code.
|
||||||
DIE("unerecognized electric var name");
|
DIE("unerecognized electric var name");
|
||||||
|
@ -1276,7 +1294,7 @@ env_var_t env_get_string(const wcstring &key, env_mode_flags_t mode) {
|
||||||
if (!search_universal) return env_var_t::missing_var();
|
if (!search_universal) return env_var_t::missing_var();
|
||||||
|
|
||||||
// Another hack. Only do a universal barrier on the main thread (since it can change variable
|
// Another hack. Only do a universal barrier on the main thread (since it can change variable
|
||||||
// values). Make sure we do this outside the env_lock because it may itself call env_get_string.
|
// values). Make sure we do this outside the env_lock because it may itself call `env_get()`.
|
||||||
if (is_main_thread() && !get_proc_had_barrier()) {
|
if (is_main_thread() && !get_proc_had_barrier()) {
|
||||||
set_proc_had_barrier(true);
|
set_proc_had_barrier(true);
|
||||||
env_universal_barrier();
|
env_universal_barrier();
|
||||||
|
@ -1487,12 +1505,12 @@ void var_stack_t::update_export_array_if_necessary() {
|
||||||
const wcstring_list_t uni = uvars()->get_names(true, false);
|
const wcstring_list_t uni = uvars()->get_names(true, false);
|
||||||
for (size_t i = 0; i < uni.size(); i++) {
|
for (size_t i = 0; i < uni.size(); i++) {
|
||||||
const wcstring &key = uni.at(i);
|
const wcstring &key = uni.at(i);
|
||||||
const env_var_t val = uvars()->get(key);
|
const env_var_t var = uvars()->get(key);
|
||||||
|
|
||||||
if (!val.missing() && val != ENV_NULL) {
|
if (!var.missing() && var.as_string() != ENV_NULL) {
|
||||||
// Note that std::map::insert does NOT overwrite a value already in the map,
|
// Note that std::map::insert does NOT overwrite a value already in the map,
|
||||||
// which we depend on here.
|
// which we depend on here.
|
||||||
vals.insert(std::pair<wcstring, wcstring>(key, val));
|
vals.insert(std::pair<wcstring, wcstring>(key, var.as_string()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1533,16 +1551,16 @@ env_vars_snapshot_t::env_vars_snapshot_t(const wchar_t *const *keys) {
|
||||||
wcstring key;
|
wcstring key;
|
||||||
for (size_t i = 0; keys[i]; i++) {
|
for (size_t i = 0; keys[i]; i++) {
|
||||||
key.assign(keys[i]);
|
key.assign(keys[i]);
|
||||||
const env_var_t val = env_get_string(key);
|
const env_var_t var = env_get(key);
|
||||||
if (!val.missing()) {
|
if (!var.missing()) {
|
||||||
vars[key] = val;
|
vars[key] = var.as_string();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
env_vars_snapshot_t::env_vars_snapshot_t() {}
|
env_vars_snapshot_t::env_vars_snapshot_t() {}
|
||||||
|
|
||||||
// The "current" variables are not a snapshot at all, but instead trampoline to env_get_string, etc.
|
// The "current" variables are not a snapshot at all, but instead trampoline to env_get, etc.
|
||||||
// We identify the current snapshot based on pointer values.
|
// We identify the current snapshot based on pointer values.
|
||||||
static const env_vars_snapshot_t sCurrentSnapshot;
|
static const env_vars_snapshot_t sCurrentSnapshot;
|
||||||
const env_vars_snapshot_t &env_vars_snapshot_t::current() { return sCurrentSnapshot; }
|
const env_vars_snapshot_t &env_vars_snapshot_t::current() { return sCurrentSnapshot; }
|
||||||
|
@ -1550,9 +1568,9 @@ const env_vars_snapshot_t &env_vars_snapshot_t::current() { return sCurrentSnaps
|
||||||
bool env_vars_snapshot_t::is_current() const { return this == &sCurrentSnapshot; }
|
bool env_vars_snapshot_t::is_current() const { return this == &sCurrentSnapshot; }
|
||||||
|
|
||||||
env_var_t env_vars_snapshot_t::get(const wcstring &key) const {
|
env_var_t env_vars_snapshot_t::get(const wcstring &key) const {
|
||||||
// If we represent the current state, bounce to env_get_string.
|
// If we represent the current state, bounce to env_get.
|
||||||
if (this->is_current()) {
|
if (this->is_current()) {
|
||||||
return env_get_string(key);
|
return env_get(key);
|
||||||
}
|
}
|
||||||
std::map<wcstring, wcstring>::const_iterator iter = vars.find(key);
|
std::map<wcstring, wcstring>::const_iterator iter = vars.find(key);
|
||||||
return iter == vars.end() ? env_var_t::missing_var() : env_var_t(iter->second);
|
return iter == vars.end() ? env_var_t::missing_var() : env_var_t(iter->second);
|
||||||
|
|
60
src/env.h
60
src/env.h
|
@ -24,7 +24,7 @@ extern bool curses_initialized;
|
||||||
/// Value denoting a null string.
|
/// Value denoting a null string.
|
||||||
#define ENV_NULL L"\x1d"
|
#define ENV_NULL L"\x1d"
|
||||||
|
|
||||||
// Flags that may be passed as the 'mode' in env_set / env_get_string.
|
// Flags that may be passed as the 'mode' in env_set / env_get.
|
||||||
enum {
|
enum {
|
||||||
/// Default mode.
|
/// Default mode.
|
||||||
ENV_DEFAULT = 0,
|
ENV_DEFAULT = 0,
|
||||||
|
@ -69,11 +69,16 @@ void env_init(const struct config_paths_t *paths = NULL);
|
||||||
/// routines.
|
/// routines.
|
||||||
void misc_init();
|
void misc_init();
|
||||||
|
|
||||||
int env_set(const wcstring &key, const wchar_t *val, env_mode_flags_t mode);
|
/// Tokenize the specified string into the specified wcstring_list_t.
|
||||||
|
///
|
||||||
|
/// \param val the input string. The contents of this string is not changed.
|
||||||
|
/// \param out the list in which to place the elements.
|
||||||
|
void tokenize_variable_array(const wcstring &val, wcstring_list_t &out);
|
||||||
|
|
||||||
class env_var_t : public wcstring {
|
class env_var_t {
|
||||||
private:
|
private:
|
||||||
bool is_missing;
|
bool is_missing;
|
||||||
|
wcstring val;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static env_var_t missing_var() {
|
static env_var_t missing_var() {
|
||||||
|
@ -82,41 +87,36 @@ class env_var_t : public wcstring {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
env_var_t(const env_var_t &x) : wcstring(x), is_missing(x.is_missing) {}
|
env_var_t(const env_var_t &x) : val(x.val), is_missing(x.is_missing) {}
|
||||||
env_var_t(const wcstring &x) : wcstring(x), is_missing(false) {}
|
env_var_t(const wcstring &x) : val(x), is_missing(false) {}
|
||||||
env_var_t(const wchar_t *x) : wcstring(x), is_missing(false) {}
|
env_var_t(const wchar_t *x) : val(x), is_missing(false) {}
|
||||||
env_var_t() : wcstring(L""), is_missing(false) {}
|
env_var_t() : val(L""), is_missing(false) {}
|
||||||
|
|
||||||
|
bool empty(void) const { return val.empty(); };
|
||||||
bool missing(void) const { return is_missing; }
|
bool missing(void) const { return is_missing; }
|
||||||
|
bool missing_or_empty(void) const { return missing() || val.empty(); }
|
||||||
bool missing_or_empty(void) const { return missing() || empty(); }
|
|
||||||
|
|
||||||
const wchar_t *c_str(void) const;
|
const wchar_t *c_str(void) const;
|
||||||
|
void to_list(wcstring_list_t &out) const;
|
||||||
|
wcstring as_string() const;
|
||||||
|
|
||||||
env_var_t &operator=(const env_var_t &s) {
|
env_var_t &operator=(const env_var_t &v) {
|
||||||
is_missing = s.is_missing;
|
is_missing = v.is_missing;
|
||||||
wcstring::operator=(s);
|
val = v.val;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator==(const env_var_t &s) const {
|
bool operator==(const env_var_t &s) const { return is_missing == s.is_missing && val == s.val; }
|
||||||
return is_missing == s.is_missing &&
|
|
||||||
static_cast<const wcstring &>(*this) == static_cast<const wcstring &>(s);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator==(const wcstring &s) const {
|
bool operator==(const wcstring &s) const { return !is_missing && val == s; }
|
||||||
return !is_missing && static_cast<const wcstring &>(*this) == s;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(const env_var_t &s) const { return !(*this == s); }
|
bool operator!=(const env_var_t &v) const { return val != v.val; }
|
||||||
|
|
||||||
bool operator!=(const wcstring &s) const { return !(*this == s); }
|
bool operator!=(const wcstring &s) const { return val != s; }
|
||||||
|
|
||||||
bool operator==(const wchar_t *s) const {
|
bool operator!=(const wchar_t *s) const { return val != s; }
|
||||||
return !is_missing && static_cast<const wcstring &>(*this) == s;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(const wchar_t *s) const { return !(*this == s); }
|
bool operator==(const wchar_t *s) const { return !is_missing && val == s; }
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Gets the variable with the specified name, or env_var_t::missing_var if it does not exist or is
|
/// Gets the variable with the specified name, or env_var_t::missing_var if it does not exist or is
|
||||||
|
@ -124,7 +124,9 @@ class env_var_t : public wcstring {
|
||||||
///
|
///
|
||||||
/// \param key The name of the variable to get
|
/// \param key The name of the variable to get
|
||||||
/// \param mode An optional scope to search in. All scopes are searched if unset
|
/// \param mode An optional scope to search in. All scopes are searched if unset
|
||||||
env_var_t env_get_string(const wcstring &key, env_mode_flags_t mode = ENV_DEFAULT);
|
env_var_t env_get(const wcstring &key, env_mode_flags_t mode = ENV_DEFAULT);
|
||||||
|
|
||||||
|
int env_set(const wcstring &key, const wchar_t *val, env_mode_flags_t mode);
|
||||||
|
|
||||||
/// Returns true if the specified key exists. This can't be reliably done using env_get, since
|
/// Returns true if the specified key exists. This can't be reliably done using env_get, since
|
||||||
/// env_get returns null for 0-element arrays.
|
/// env_get returns null for 0-element arrays.
|
||||||
|
@ -214,10 +216,4 @@ bool term_supports_setting_title();
|
||||||
/// Returns the fish internal representation for an array of strings.
|
/// Returns the fish internal representation for an array of strings.
|
||||||
std::unique_ptr<wcstring> list_to_array_val(const wcstring_list_t &list);
|
std::unique_ptr<wcstring> list_to_array_val(const wcstring_list_t &list);
|
||||||
std::unique_ptr<wcstring> list_to_array_val(const wchar_t **list);
|
std::unique_ptr<wcstring> list_to_array_val(const wchar_t **list);
|
||||||
|
|
||||||
/// Tokenize the specified string into the specified wcstring_list_t.
|
|
||||||
///
|
|
||||||
/// \param val the input string. The contents of this string is not changed.
|
|
||||||
/// \param out the list in which to place the elements.
|
|
||||||
void tokenize_variable_array(const wcstring &val, wcstring_list_t &out);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -405,7 +405,7 @@ void exec_job(parser_t &parser, job_t *j) {
|
||||||
// really make sense, so I'm not trying to fix it here.
|
// really make sense, so I'm not trying to fix it here.
|
||||||
if (!setup_child_process(j, 0, all_ios)) {
|
if (!setup_child_process(j, 0, all_ios)) {
|
||||||
// Decrement SHLVL as we're removing ourselves from the shell "stack".
|
// Decrement SHLVL as we're removing ourselves from the shell "stack".
|
||||||
const env_var_t shlvl_str = env_get_string(L"SHLVL", ENV_GLOBAL | ENV_EXPORT);
|
const env_var_t shlvl_str = env_get(L"SHLVL", ENV_GLOBAL | ENV_EXPORT);
|
||||||
wcstring nshlvl_str = L"0";
|
wcstring nshlvl_str = L"0";
|
||||||
if (!shlvl_str.missing()) {
|
if (!shlvl_str.missing()) {
|
||||||
long shlvl_i = fish_wcstol(shlvl_str.c_str());
|
long shlvl_i = fish_wcstol(shlvl_str.c_str());
|
||||||
|
@ -1111,7 +1111,7 @@ static int exec_subshell_internal(const wcstring &cmd, wcstring_list_t *lst, boo
|
||||||
const int prev_status = proc_get_last_status();
|
const int prev_status = proc_get_last_status();
|
||||||
bool split_output = false;
|
bool split_output = false;
|
||||||
|
|
||||||
const env_var_t ifs = env_get_string(L"IFS");
|
const env_var_t ifs = env_get(L"IFS");
|
||||||
if (!ifs.missing_or_empty()) {
|
if (!ifs.missing_or_empty()) {
|
||||||
split_output = true;
|
split_output = true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -144,7 +144,7 @@ static void append_cmdsub_error(parse_error_list_t *errors, size_t source_start,
|
||||||
/// Return the environment variable value for the string starting at \c in.
|
/// Return the environment variable value for the string starting at \c in.
|
||||||
static env_var_t expand_var(const wchar_t *in) {
|
static env_var_t expand_var(const wchar_t *in) {
|
||||||
if (!in) return env_var_t::missing_var();
|
if (!in) return env_var_t::missing_var();
|
||||||
return env_get_string(in);
|
return env_get(in);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Test if the specified string does not contain character which can not be used inside a quoted
|
/// Test if the specified string does not contain character which can not be used inside a quoted
|
||||||
|
@ -777,19 +777,19 @@ static int expand_variables(const wcstring &instr, std::vector<completion_t> *ou
|
||||||
}
|
}
|
||||||
|
|
||||||
var_tmp.append(instr, start_pos, var_len);
|
var_tmp.append(instr, start_pos, var_len);
|
||||||
env_var_t var_val;
|
env_var_t var;
|
||||||
if (var_len == 1 && var_tmp[0] == VARIABLE_EXPAND_EMPTY) {
|
if (var_len == 1 && var_tmp[0] == VARIABLE_EXPAND_EMPTY) {
|
||||||
var_val = env_var_t::missing_var();
|
var = env_var_t::missing_var();
|
||||||
} else {
|
} else {
|
||||||
var_val = expand_var(var_tmp.c_str());
|
var = expand_var(var_tmp.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!var_val.missing()) {
|
if (!var.missing()) {
|
||||||
int all_vars = 1;
|
int all_vars = 1;
|
||||||
wcstring_list_t var_item_list;
|
wcstring_list_t var_item_list;
|
||||||
|
|
||||||
if (is_ok) {
|
if (is_ok) {
|
||||||
tokenize_variable_array(var_val, var_item_list);
|
var.to_list(var_item_list);
|
||||||
|
|
||||||
const size_t slice_start = stop_pos;
|
const size_t slice_start = stop_pos;
|
||||||
if (slice_start < insize && instr.at(slice_start) == L'[') {
|
if (slice_start < insize && instr.at(slice_start) == L'[') {
|
||||||
|
@ -1173,7 +1173,7 @@ static void expand_home_directory(wcstring &input) {
|
||||||
env_var_t home;
|
env_var_t home;
|
||||||
if (username.empty()) {
|
if (username.empty()) {
|
||||||
// Current users home directory.
|
// Current users home directory.
|
||||||
home = env_get_string(L"HOME");
|
home = env_get(L"HOME");
|
||||||
// If home is either missing or empty,
|
// If home is either missing or empty,
|
||||||
// treat it like an empty list.
|
// treat it like an empty list.
|
||||||
// $HOME is defined to be a _path_,
|
// $HOME is defined to be a _path_,
|
||||||
|
@ -1201,7 +1201,7 @@ static void expand_home_directory(wcstring &input) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wchar_t *realhome = wrealpath(home, NULL);
|
wchar_t *realhome = wrealpath(home.as_string(), NULL);
|
||||||
|
|
||||||
if (!tilde_error && realhome) {
|
if (!tilde_error && realhome) {
|
||||||
input.replace(input.begin(), input.begin() + tail_idx, realhome);
|
input.replace(input.begin(), input.begin() + tail_idx, realhome);
|
||||||
|
@ -1433,12 +1433,12 @@ static expand_error_t expand_stage_wildcards(const wcstring &input, std::vector<
|
||||||
} else {
|
} else {
|
||||||
// Get the PATH/CDPATH and cwd. Perhaps these should be passed in. An empty CDPATH
|
// Get the PATH/CDPATH and cwd. Perhaps these should be passed in. An empty CDPATH
|
||||||
// implies just the current directory, while an empty PATH is left empty.
|
// implies just the current directory, while an empty PATH is left empty.
|
||||||
env_var_t paths = env_get_string(for_cd ? L"CDPATH" : L"PATH");
|
env_var_t paths = env_get(for_cd ? L"CDPATH" : L"PATH");
|
||||||
if (paths.missing_or_empty()) paths = for_cd ? L"." : L"";
|
if (paths.missing_or_empty()) paths = for_cd ? L"." : L"";
|
||||||
|
|
||||||
// Tokenize it into path names.
|
// Tokenize it into path names.
|
||||||
std::vector<wcstring> pathsv;
|
std::vector<wcstring> pathsv;
|
||||||
tokenize_variable_array(paths, pathsv);
|
paths.to_list(pathsv);
|
||||||
for (auto next_path : pathsv) {
|
for (auto next_path : pathsv) {
|
||||||
effective_working_dirs.push_back(
|
effective_working_dirs.push_back(
|
||||||
path_apply_working_directory(next_path, working_dir));
|
path_apply_working_directory(next_path, working_dir));
|
||||||
|
@ -1592,9 +1592,9 @@ void update_abbr_cache(const wchar_t *op, const wcstring varname) {
|
||||||
}
|
}
|
||||||
abbreviations.erase(abbr);
|
abbreviations.erase(abbr);
|
||||||
if (wcscmp(op, L"ERASE") != 0) {
|
if (wcscmp(op, L"ERASE") != 0) {
|
||||||
const env_var_t expansion = env_get_string(varname);
|
const env_var_t expansion = env_get(varname);
|
||||||
if (!expansion.missing_or_empty()) {
|
if (!expansion.missing_or_empty()) {
|
||||||
abbreviations.emplace(std::make_pair(abbr, expansion));
|
abbreviations.emplace(std::make_pair(abbr, expansion.as_string()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,7 +78,7 @@ static int load(const wcstring &name) {
|
||||||
static void autoload_names(std::set<wcstring> &names, int get_hidden) {
|
static void autoload_names(std::set<wcstring> &names, int get_hidden) {
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
const env_var_t path_var_wstr = env_get_string(L"fish_function_path");
|
const env_var_t path_var_wstr = env_get(L"fish_function_path");
|
||||||
if (path_var_wstr.missing()) return;
|
if (path_var_wstr.missing()) return;
|
||||||
const wchar_t *path_var = path_var_wstr.c_str();
|
const wchar_t *path_var = path_var_wstr.c_str();
|
||||||
|
|
||||||
|
@ -121,7 +121,7 @@ void function_init() {
|
||||||
static std::map<wcstring, env_var_t> snapshot_vars(const wcstring_list_t &vars) {
|
static std::map<wcstring, env_var_t> snapshot_vars(const wcstring_list_t &vars) {
|
||||||
std::map<wcstring, env_var_t> result;
|
std::map<wcstring, env_var_t> result;
|
||||||
for (wcstring_list_t::const_iterator it = vars.begin(), end = vars.end(); it != end; ++it) {
|
for (wcstring_list_t::const_iterator it = vars.begin(), end = vars.end(); it != end; ++it) {
|
||||||
result.insert(std::make_pair(*it, env_get_string(*it)));
|
result.insert(std::make_pair(*it, env_get(*it)));
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
@ -221,12 +221,12 @@ static bool is_potential_cd_path(const wcstring &path, const wcstring &working_d
|
||||||
directories.push_back(working_directory);
|
directories.push_back(working_directory);
|
||||||
} else {
|
} else {
|
||||||
// Get the CDPATH.
|
// Get the CDPATH.
|
||||||
env_var_t cdpath = env_get_string(L"CDPATH");
|
env_var_t cdpath = env_get(L"CDPATH");
|
||||||
if (cdpath.missing_or_empty()) cdpath = L".";
|
if (cdpath.missing_or_empty()) cdpath = L".";
|
||||||
|
|
||||||
// Tokenize it into directories.
|
// Tokenize it into directories.
|
||||||
std::vector<wcstring> pathsv;
|
std::vector<wcstring> pathsv;
|
||||||
tokenize_variable_array(cdpath, pathsv);
|
cdpath.to_list(pathsv);
|
||||||
for (auto next_path : pathsv) {
|
for (auto next_path : pathsv) {
|
||||||
if (next_path.empty()) next_path = L".";
|
if (next_path.empty()) next_path = L".";
|
||||||
// Ensure that we use the working directory for relative cdpaths like ".".
|
// Ensure that we use the working directory for relative cdpaths like ".".
|
||||||
|
@ -268,17 +268,17 @@ rgb_color_t highlight_get_color(highlight_spec_t highlight, bool is_background)
|
||||||
return rgb_color_t::normal();
|
return rgb_color_t::normal();
|
||||||
}
|
}
|
||||||
|
|
||||||
env_var_t val_wstr = env_get_string(highlight_var[idx]);
|
env_var_t var = env_get(highlight_var[idx]);
|
||||||
|
|
||||||
// debug( 1, L"%d -> %d -> %ls", highlight, idx, val );
|
// debug( 1, L"%d -> %d -> %ls", highlight, idx, val );
|
||||||
|
|
||||||
if (val_wstr.missing()) val_wstr = env_get_string(highlight_var[0]);
|
if (var.missing()) var = env_get(highlight_var[0]);
|
||||||
|
|
||||||
if (!val_wstr.missing()) result = parse_color(val_wstr, treat_as_background);
|
if (!var.missing()) result = parse_color(var.as_string(), treat_as_background);
|
||||||
|
|
||||||
// Handle modifiers.
|
// Handle modifiers.
|
||||||
if (highlight & highlight_modifier_valid_path) {
|
if (highlight & highlight_modifier_valid_path) {
|
||||||
env_var_t val2_wstr = env_get_string(L"fish_color_valid_path");
|
env_var_t val2_wstr = env_get(L"fish_color_valid_path");
|
||||||
const wcstring val2 = val2_wstr.missing() ? L"" : val2_wstr.c_str();
|
const wcstring val2 = val2_wstr.missing() ? L"" : val2_wstr.c_str();
|
||||||
|
|
||||||
rgb_color_t result2 = parse_color(val2, is_background);
|
rgb_color_t result2 = parse_color(val2, is_background);
|
||||||
|
|
|
@ -1796,8 +1796,9 @@ void history_sanity_check() {
|
||||||
wcstring history_session_id() {
|
wcstring history_session_id() {
|
||||||
wcstring result = DFLT_FISH_HISTORY_SESSION_ID;
|
wcstring result = DFLT_FISH_HISTORY_SESSION_ID;
|
||||||
|
|
||||||
const env_var_t session_id = env_get_string(L"FISH_HISTORY");
|
const env_var_t var = env_get(L"FISH_HISTORY");
|
||||||
if (!session_id.missing()) {
|
if (!var.missing()) {
|
||||||
|
wcstring session_id = var.as_string();
|
||||||
if (session_id.empty()) {
|
if (session_id.empty()) {
|
||||||
result = L"";
|
result = L"";
|
||||||
} else if (session_id == L"default") {
|
} else if (session_id == L"default") {
|
||||||
|
|
|
@ -201,8 +201,8 @@ static int input_function_args_index = 0;
|
||||||
|
|
||||||
/// Return the current bind mode.
|
/// Return the current bind mode.
|
||||||
wcstring input_get_bind_mode() {
|
wcstring input_get_bind_mode() {
|
||||||
env_var_t mode = env_get_string(FISH_BIND_MODE_VAR);
|
env_var_t mode = env_get(FISH_BIND_MODE_VAR);
|
||||||
return mode.missing() ? DEFAULT_BIND_MODE : mode;
|
return mode.missing() ? DEFAULT_BIND_MODE : mode.as_string();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set the current bind mode.
|
/// Set the current bind mode.
|
||||||
|
|
|
@ -160,7 +160,7 @@ static wint_t readb() {
|
||||||
// Update the wait_on_escape_ms value in response to the fish_escape_delay_ms user variable being
|
// Update the wait_on_escape_ms value in response to the fish_escape_delay_ms user variable being
|
||||||
// set.
|
// set.
|
||||||
void update_wait_on_escape_ms() {
|
void update_wait_on_escape_ms() {
|
||||||
env_var_t escape_time_ms = env_get_string(L"fish_escape_delay_ms");
|
env_var_t escape_time_ms = env_get(L"fish_escape_delay_ms");
|
||||||
if (escape_time_ms.missing_or_empty()) {
|
if (escape_time_ms.missing_or_empty()) {
|
||||||
wait_on_escape_ms = WAIT_ON_ESCAPE_DEFAULT;
|
wait_on_escape_ms = WAIT_ON_ESCAPE_DEFAULT;
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -554,7 +554,7 @@ void writembs_check(char *mbs, const char *mbs_name, const char *file, long line
|
||||||
if (mbs != NULL) {
|
if (mbs != NULL) {
|
||||||
tputs(mbs, 1, &writeb);
|
tputs(mbs, 1, &writeb);
|
||||||
} else {
|
} else {
|
||||||
env_var_t term = env_get_string(L"TERM");
|
env_var_t term = env_get(L"TERM");
|
||||||
const wchar_t *fmt =
|
const wchar_t *fmt =
|
||||||
_(L"Tried to use terminfo string %s on line %ld of %s, which is "
|
_(L"Tried to use terminfo string %s on line %ld of %s, which is "
|
||||||
L"undefined in terminal of type \"%ls\". Please report this error to %s");
|
L"undefined in terminal of type \"%ls\". Please report this error to %s");
|
||||||
|
|
24
src/path.cpp
24
src/path.cpp
|
@ -50,7 +50,7 @@ static bool path_get_path_core(const wcstring &cmd, wcstring *out_path,
|
||||||
int err = ENOENT;
|
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.as_string();
|
||||||
} else {
|
} else {
|
||||||
// Note that PREFIX is defined in the `Makefile` and is thus defined when this module is
|
// Note that PREFIX is defined in the `Makefile` and is thus defined when this module is
|
||||||
// compiled. This ensures we always default to "/bin", "/usr/bin" and the bin dir defined
|
// compiled. This ensures we always default to "/bin", "/usr/bin" and the bin dir defined
|
||||||
|
@ -104,7 +104,7 @@ bool path_get_path(const wcstring &cmd, wcstring *out_path, const env_vars_snaps
|
||||||
}
|
}
|
||||||
|
|
||||||
bool path_get_path(const wcstring &cmd, wcstring *out_path) {
|
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(L"PATH"));
|
||||||
}
|
}
|
||||||
|
|
||||||
wcstring_list_t path_get_paths(const wcstring &cmd) {
|
wcstring_list_t path_get_paths(const wcstring &cmd) {
|
||||||
|
@ -122,9 +122,9 @@ wcstring_list_t path_get_paths(const wcstring &cmd) {
|
||||||
return paths;
|
return paths;
|
||||||
}
|
}
|
||||||
|
|
||||||
wcstring env_path = env_get_string(L"PATH");
|
env_var_t path_var = env_get(L"PATH");
|
||||||
std::vector<wcstring> pathsv;
|
std::vector<wcstring> pathsv;
|
||||||
tokenize_variable_array(env_path, pathsv);
|
path_var.to_list(pathsv);
|
||||||
for (auto path : pathsv) {
|
for (auto path : pathsv) {
|
||||||
if (path.empty()) continue;
|
if (path.empty()) continue;
|
||||||
append_path_component(path, cmd);
|
append_path_component(path, cmd);
|
||||||
|
@ -141,10 +141,11 @@ wcstring_list_t path_get_paths(const wcstring &cmd) {
|
||||||
return paths;
|
return paths;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool path_get_cdpath(const wcstring &dir, wcstring *out, const wchar_t *wd,
|
bool path_get_cdpath(const env_var_t &dir_var, 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;
|
||||||
if (dir.empty()) return false;
|
if (dir_var.missing_or_empty()) return false;
|
||||||
|
wcstring dir = dir_var.as_string();
|
||||||
|
|
||||||
if (wd) {
|
if (wd) {
|
||||||
size_t len = wcslen(wd);
|
size_t len = wcslen(wd);
|
||||||
|
@ -168,7 +169,7 @@ bool path_get_cdpath(const wcstring &dir, wcstring *out, const wchar_t *wd,
|
||||||
if (cdpaths.missing_or_empty()) cdpaths = L".";
|
if (cdpaths.missing_or_empty()) cdpaths = L".";
|
||||||
|
|
||||||
std::vector<wcstring> cdpathsv;
|
std::vector<wcstring> cdpathsv;
|
||||||
tokenize_variable_array(cdpaths, cdpathsv);
|
cdpaths.to_list(cdpathsv);
|
||||||
for (auto next_path : cdpathsv) {
|
for (auto next_path : cdpathsv) {
|
||||||
if (next_path.empty()) next_path = L".";
|
if (next_path.empty()) next_path = L".";
|
||||||
if (next_path == L"." && wd != NULL) {
|
if (next_path == L"." && wd != NULL) {
|
||||||
|
@ -287,19 +288,20 @@ static void path_create(wcstring &path, const wcstring &xdg_var, const wcstring
|
||||||
// The vars we fetch must be exported. Allowing them to be universal doesn't make sense and
|
// The vars we fetch must be exported. Allowing them to be universal doesn't make sense and
|
||||||
// allowing that creates a lock inversion that deadlocks the shell since we're called before
|
// allowing that creates a lock inversion that deadlocks the shell since we're called before
|
||||||
// uvars are available.
|
// uvars are available.
|
||||||
const env_var_t xdg_dir = env_get_string(xdg_var, ENV_GLOBAL | ENV_EXPORT);
|
const env_var_t xdg_dir = env_get(xdg_var, ENV_GLOBAL | ENV_EXPORT);
|
||||||
if (!xdg_dir.missing_or_empty()) {
|
if (!xdg_dir.missing_or_empty()) {
|
||||||
using_xdg = true;
|
using_xdg = true;
|
||||||
path = xdg_dir + L"/fish";
|
path = xdg_dir.as_string() + L"/fish";
|
||||||
if (create_directory(path) != -1) {
|
if (create_directory(path) != -1) {
|
||||||
path_done = true;
|
path_done = true;
|
||||||
} else {
|
} else {
|
||||||
saved_errno = errno;
|
saved_errno = errno;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const env_var_t home = env_get_string(L"HOME", ENV_GLOBAL | ENV_EXPORT);
|
const env_var_t home = env_get(L"HOME", ENV_GLOBAL | ENV_EXPORT);
|
||||||
if (!home.missing_or_empty()) {
|
if (!home.missing_or_empty()) {
|
||||||
path = home + (which_dir == L"config" ? L"/.config/fish" : L"/.local/share/fish");
|
path = home.as_string() +
|
||||||
|
(which_dir == L"config" ? L"/.config/fish" : L"/.local/share/fish");
|
||||||
if (create_directory(path) != -1) {
|
if (create_directory(path) != -1) {
|
||||||
path_done = true;
|
path_done = true;
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -61,7 +61,7 @@ wcstring_list_t path_get_paths(const wcstring &cmd);
|
||||||
/// \param vars The environment variable snapshot to use (for the CDPATH variable)
|
/// \param vars The environment variable snapshot to use (for the CDPATH variable)
|
||||||
/// \return 0 if the command can not be found, the path of the command otherwise. The path should be
|
/// \return 0 if the command can not be found, the path of the command otherwise. The path should be
|
||||||
/// free'd with free().
|
/// free'd with free().
|
||||||
bool path_get_cdpath(const wcstring &dir, wcstring *out_or_NULL, const wchar_t *wd = NULL,
|
bool path_get_cdpath(const env_var_t &dir, wcstring *out_or_NULL, const wchar_t *wd = NULL,
|
||||||
const env_vars_snapshot_t &vars = env_vars_snapshot_t::current());
|
const env_vars_snapshot_t &vars = env_vars_snapshot_t::current());
|
||||||
|
|
||||||
/// Returns whether the path can be used for an implicit cd command; if so, also returns the path by
|
/// Returns whether the path can be used for an implicit cd command; if so, also returns the path by
|
||||||
|
|
|
@ -2063,8 +2063,8 @@ void reader_import_history_if_necessary(void) {
|
||||||
// Try opening a bash file. We make an effort to respect $HISTFILE; this isn't very complete
|
// Try opening a bash file. We make an effort to respect $HISTFILE; this isn't very complete
|
||||||
// (AFAIK it doesn't have to be exported), and to really get this right we ought to ask bash
|
// (AFAIK it doesn't have to be exported), and to really get this right we ought to ask bash
|
||||||
// itself. But this is better than nothing.
|
// itself. But this is better than nothing.
|
||||||
const env_var_t var = env_get_string(L"HISTFILE");
|
const env_var_t var = env_get(L"HISTFILE");
|
||||||
wcstring path = (var.missing() ? L"~/.bash_history" : var);
|
wcstring path = (var.missing() ? L"~/.bash_history" : var.as_string());
|
||||||
expand_tilde(path);
|
expand_tilde(path);
|
||||||
FILE *f = wfopen(path, "r");
|
FILE *f = wfopen(path, "r");
|
||||||
if (f) {
|
if (f) {
|
||||||
|
|
|
@ -120,8 +120,8 @@ static bool is_screen_name_escape_seq(const wchar_t *code, size_t *resulting_len
|
||||||
#if 0
|
#if 0
|
||||||
// TODO: Decide if this should be removed or modified to also test for TERM values that begin
|
// TODO: Decide if this should be removed or modified to also test for TERM values that begin
|
||||||
// with "tmux". See issue #3512.
|
// with "tmux". See issue #3512.
|
||||||
const env_var_t term_name = env_get_string(L"TERM");
|
const env_var_t term_name = env_get(L"TERM");
|
||||||
if (term_name.missing() || !string_prefixes_string(L"screen", term_name)) {
|
if (term_name.missing_or_empty() || !string_prefixes_string(L"screen", term_name)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in a new issue