env_var_t to forget its name

Store properties associated with the name via flags instead
This commit is contained in:
ridiculousfish 2018-01-30 12:36:50 -08:00
parent 5c2e6734c1
commit f025269195
7 changed files with 61 additions and 42 deletions

View file

@ -37,7 +37,7 @@ int builtin_cd(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
wcstring dir; wcstring dir;
if (argv[optind]) { if (argv[optind]) {
dir_in = env_var_t(L"", argv[optind]); // unamed var dir_in = env_var_t(wcstring(argv[optind]), 0); // unamed var
} else { } else {
auto maybe_dir_in = env_get(L"HOME"); auto maybe_dir_in = env_get(L"HOME");
if (maybe_dir_in.missing_or_empty()) { if (maybe_dir_in.missing_or_empty()) {

View file

@ -532,8 +532,7 @@ static void show_scope(const wchar_t *var_name, int scope, io_streams_t &streams
return; return;
} }
const wchar_t *exportv = var->exports() ? _(L"exported") : _(L"unexported");
const wchar_t *exportv = var->exportv ? _(L"exported") : _(L"unexported");
wcstring_list_t vals = var->as_list(); wcstring_list_t vals = var->as_list();
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, vals.size()); scope_name, exportv, vals.size());

View file

@ -218,7 +218,7 @@ void var_stack_t::push(bool new_scope) {
// "--no-scope-shadowing". // "--no-scope-shadowing".
if (new_scope && top_node != this->global_env) { if (new_scope && top_node != this->global_env) {
for (const auto &var : top_node->env) { for (const auto &var : top_node->env) {
if (var.second.exportv) node->env.insert(var); if (var.second.exports()) node->env.insert(var);
} }
} }
@ -265,7 +265,7 @@ void var_stack_t::pop() {
for (const auto &entry_pair : old_top->env) { for (const auto &entry_pair : old_top->env) {
const env_var_t &var = entry_pair.second; const env_var_t &var = entry_pair.second;
if (var.exportv) { if (var.exports()) {
this->mark_changed_exported(); this->mark_changed_exported();
break; break;
} }
@ -323,8 +323,6 @@ static bool is_read_only(const wcstring &key) {
return env_read_only.find(key) != env_read_only.end(); return env_read_only.find(key) != env_read_only.end();
} }
bool env_var_t::read_only() const { return is_read_only(name); }
/// Table of variables whose value is dynamically calculated, such as umask, status, etc. /// Table of variables whose value is dynamically calculated, such as umask, status, etc.
static const_string_set_t env_electric; static const_string_set_t env_electric;
@ -1090,7 +1088,7 @@ static int env_set_internal(const wcstring &key, env_mode_flags_t var_mode, wcst
var_table_t::const_iterator result = preexisting_node->env.find(key); var_table_t::const_iterator result = preexisting_node->env.find(key);
assert(result != preexisting_node->env.end()); assert(result != preexisting_node->env.end());
const env_var_t &var = result->second; const env_var_t &var = result->second;
if (var.exportv) { if (var.exports()) {
preexisting_entry_exportv = true; preexisting_entry_exportv = true;
has_changed_new = true; has_changed_new = true;
} }
@ -1137,7 +1135,7 @@ static int env_set_internal(const wcstring &key, env_mode_flags_t var_mode, wcst
// Set the entry in the node. Note that operator[] accesses the existing entry, or // Set the entry in the node. Note that operator[] accesses the existing entry, or
// creates a new one. // creates a new one.
env_var_t &var = node->env[key]; env_var_t &var = node->env[key];
if (var.exportv) { if (var.exports()) {
// This variable already existed, and was exported. // This variable already existed, and was exported.
has_changed_new = true; has_changed_new = true;
} }
@ -1146,11 +1144,11 @@ static int env_set_internal(const wcstring &key, env_mode_flags_t var_mode, wcst
if (var_mode & ENV_EXPORT) { if (var_mode & ENV_EXPORT) {
// The new variable is exported. // The new variable is exported.
var.exportv = true; var.set_exports(true);
node->exportv = true; node->exportv = true;
has_changed_new = true; has_changed_new = true;
} else { } else {
var.exportv = false; var.set_exports(false);
// Set the node's exported when it changes something about exports // Set the node's exported when it changes something about exports
// (also when it redefines a variable to not be exported). // (also when it redefines a variable to not be exported).
node->exportv = has_changed_old != has_changed_new; node->exportv = has_changed_old != has_changed_new;
@ -1201,7 +1199,7 @@ static bool try_remove(env_node_t *n, const wchar_t *key, int var_mode) {
var_table_t::iterator result = n->env.find(key); var_table_t::iterator result = n->env.find(key);
if (result != n->env.end()) { if (result != n->env.end()) {
if (result->second.exportv) { if (result->second.exports()) {
vars_stack().mark_changed_exported(); vars_stack().mark_changed_exported();
} }
n->env.erase(result); n->env.erase(result);
@ -1272,7 +1270,7 @@ const wcstring_list_t &env_var_t::as_list() const { return vals; }
wcstring env_var_t::as_string(void) const { wcstring env_var_t::as_string(void) const {
if (this->vals.empty()) return wcstring(ENV_NULL); if (this->vals.empty()) return wcstring(ENV_NULL);
wchar_t sep = variable_is_colon_delimited_var(this->name) ? L':' : ARRAY_SEP; wchar_t sep = (flags & flag_colon_delimit) ? L':' : ARRAY_SEP;
auto it = this->vals.cbegin(); auto it = this->vals.cbegin();
wcstring result(*it); wcstring result(*it);
while (++it != vals.end()) { while (++it != vals.end()) {
@ -1286,6 +1284,14 @@ void env_var_t::to_list(wcstring_list_t &out) const {
out = vals; out = vals;
} }
env_var_t::env_var_flags_t env_var_t::flags_for(const wchar_t *name) {
env_var_flags_t result = 0;
wcstring tmp(name); // todo: eliminate
if (is_read_only(tmp)) result |= flag_read_only;
if (variable_is_colon_delimited_var(tmp)) result |= flag_colon_delimit;
return result;
}
maybe_t<env_var_t> env_get(const wcstring &key, env_mode_flags_t mode) { maybe_t<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);
@ -1334,7 +1340,7 @@ maybe_t<env_var_t> env_get(const wcstring &key, env_mode_flags_t mode) {
var_table_t::const_iterator result = env->env.find(key); var_table_t::const_iterator result = env->env.find(key);
if (result != env->env.end()) { if (result != env->env.end()) {
const env_var_t &var = result->second; const env_var_t &var = result->second;
if (var.exportv ? search_exported : search_unexported) { if (var.exports() ? search_exported : search_unexported) {
return var; return var;
} }
} }
@ -1387,7 +1393,7 @@ static void add_key_to_string_set(const var_table_t &envs, std::set<wcstring> *s
for (iter = envs.begin(); iter != envs.end(); ++iter) { for (iter = envs.begin(); iter != envs.end(); ++iter) {
const env_var_t &var = iter->second; const env_var_t &var = iter->second;
if ((var.exportv && show_exported) || (!var.exportv && show_unexported)) { if ((var.exports() && show_exported) || (!var.exports() && show_unexported)) {
// Insert this key. // Insert this key.
str_set->insert(iter->first); str_set->insert(iter->first);
} }
@ -1454,7 +1460,7 @@ static void get_exported(const env_node_t *n, var_table_t &h) {
const wcstring &key = iter->first; const wcstring &key = iter->first;
const env_var_t var = iter->second; const env_var_t var = iter->second;
if (var.exportv) { if (var.exports()) {
// Export the variable. Don't use std::map::insert here, since we need to overwrite // Export the variable. Don't use std::map::insert here, since we need to overwrite
// existing values from previous scopes. // existing values from previous scopes.
h[key] = var; h[key] = var;
@ -1570,7 +1576,7 @@ maybe_t<env_var_t> env_vars_snapshot_t::get(const wcstring &key) const {
} }
auto iter = vars.find(key); auto iter = vars.find(key);
if (iter == vars.end()) return none(); if (iter == vars.end()) return none();
return env_var_t(iter->second); return iter->second;
} }
const wchar_t *const env_vars_snapshot_t::highlighting_keys[] = {L"PATH", L"CDPATH", const wchar_t *const env_vars_snapshot_t::highlighting_keys[] = {L"PATH", L"CDPATH",

View file

@ -70,28 +70,34 @@ void misc_init();
class env_var_t { class env_var_t {
private: private:
wcstring name; // name of the var using env_var_flags_t = uint8_t;
wcstring_list_t vals; // list of values assigned to the var wcstring_list_t vals; // list of values assigned to the var
env_var_flags_t flags;
public: public:
bool exportv = false; // whether the variable should be exported enum {
flag_export = 1 << 0, // whether the variable is exported
flag_colon_delimit = 1 << 1, // whether the variable is colon delimited
flag_read_only = 1 << 2 // whether the variable is read only
};
// Constructors. // Constructors.
env_var_t(const env_var_t &) = default; env_var_t(const env_var_t &) = default;
env_var_t(env_var_t &&) = default; env_var_t(env_var_t &&) = default;
env_var_t(wcstring name, wcstring_list_t vals) : name(std::move(name)), vals(std::move(vals)) {} env_var_t(wcstring_list_t vals, env_var_flags_t flags) : vals(std::move(vals)), flags(flags) {}
env_var_t(wcstring val, env_var_flags_t flags)
: env_var_t(wcstring_list_t{std::move(val)}, flags) {}
env_var_t(wcstring name, wcstring val) // Constructors that infer the flags from a name.
: name(std::move(name)), env_var_t(const wchar_t *name, wcstring_list_t vals)
vals({ : env_var_t(std::move(vals), flags_for(name)) {}
std::move(val), env_var_t(const wchar_t *name, wcstring val) : env_var_t(std::move(val), flags_for(name)) {}
}),
exportv(false) {}
env_var_t() = default; env_var_t() = default;
bool empty() const { return vals.empty() || (vals.size() == 1 && vals[0].empty()); }; bool empty() const { return vals.empty() || (vals.size() == 1 && vals[0].empty()); };
bool read_only() const; bool read_only() const { return flags & flag_read_only; }
bool exports() const { return flags & flag_export; }
wcstring as_string() const; wcstring as_string() const;
void to_list(wcstring_list_t &out) const; void to_list(wcstring_list_t &out) const;
@ -99,6 +105,16 @@ class env_var_t {
void set_vals(wcstring_list_t v) { vals = std::move(v); } void set_vals(wcstring_list_t v) { vals = std::move(v); }
void set_exports(bool exportv) {
if (exportv) {
flags |= flag_export;
} else {
flags &= ~flag_export;
}
}
static env_var_flags_t flags_for(const wchar_t *name);
env_var_t &operator=(const env_var_t &var) = default; env_var_t &operator=(const env_var_t &var) = default;
env_var_t &operator=(env_var_t &&) = default; env_var_t &operator=(env_var_t &&) = default;

View file

@ -268,7 +268,7 @@ bool env_universal_t::get_export(const wcstring &name) const {
bool result = false; bool result = false;
var_table_t::const_iterator where = vars.find(name); var_table_t::const_iterator where = vars.find(name);
if (where != vars.end()) { if (where != vars.end()) {
result = where->second.exportv; result = where->second.exports();
} }
return result; return result;
} }
@ -282,9 +282,9 @@ void env_universal_t::set_internal(const wcstring &key, wcstring_list_t vals, bo
} }
env_var_t &entry = vars[key]; env_var_t &entry = vars[key];
if (entry.exportv != exportv || entry.as_list() != vals) { if (entry.exports() != exportv || entry.as_list() != vals) {
entry.set_vals(std::move(vals)); entry.set_vals(std::move(vals));
entry.exportv = exportv; entry.set_exports(exportv);
// If we are overwriting, then this is now modified. // If we are overwriting, then this is now modified.
if (overwrite) { if (overwrite) {
@ -319,7 +319,7 @@ wcstring_list_t env_universal_t::get_names(bool show_exported, bool show_unexpor
for (iter = vars.begin(); iter != vars.end(); ++iter) { for (iter = vars.begin(); iter != vars.end(); ++iter) {
const wcstring &key = iter->first; const wcstring &key = iter->first;
const env_var_t &var = iter->second; const env_var_t &var = iter->second;
if ((var.exportv && show_exported) || (!var.exportv && show_unexported)) { if ((var.exports() && show_exported) || (!var.exports() && show_unexported)) {
result.push_back(key); result.push_back(key);
} }
} }
@ -357,11 +357,11 @@ void env_universal_t::generate_callbacks(const var_table_t &new_vars,
// See if the value has changed. // See if the value has changed.
const env_var_t &new_entry = iter->second; const env_var_t &new_entry = iter->second;
var_table_t::const_iterator existing = this->vars.find(key); var_table_t::const_iterator existing = this->vars.find(key);
if (existing == this->vars.end() || existing->second.exportv != new_entry.exportv || if (existing == this->vars.end() || existing->second.exports() != new_entry.exports() ||
existing->second != new_entry) { existing->second != new_entry) {
// Value has changed. // Value has changed.
callbacks.push_back( callbacks.push_back(callback_data_t(new_entry.exports() ? SET_EXPORT : SET, key,
callback_data_t(new_entry.exportv ? SET_EXPORT : SET, key, new_entry.as_string())); new_entry.as_string()));
} }
} }
} }
@ -448,7 +448,7 @@ bool env_universal_t::write_to_fd(int fd, const wcstring &path) {
// variable; soldier on. // variable; soldier on.
const wcstring &key = iter->first; const wcstring &key = iter->first;
const env_var_t &var = iter->second; const env_var_t &var = iter->second;
append_file_entry(var.exportv ? SET_EXPORT : SET, key, var.as_string(), &contents, append_file_entry(var.exports() ? SET_EXPORT : SET, key, var.as_string(), &contents,
&storage); &storage);
// Go to next. // Go to next.
@ -830,7 +830,7 @@ void env_universal_t::parse_message_internal(const wcstring &msgstr, var_table_t
wcstring val; wcstring val;
if (unescape_string(tmp + 1, &val, 0)) { if (unescape_string(tmp + 1, &val, 0)) {
env_var_t &entry = (*vars)[key]; env_var_t &entry = (*vars)[key];
entry.exportv = exportv; entry.set_exports(exportv);
entry.set_vals(decode_serialized(val)); entry.set_vals(decode_serialized(val));
} }
} else { } else {

View file

@ -2634,7 +2634,7 @@ static void test_universal() {
if (j == 0) { if (j == 0) {
expected_val = none(); expected_val = none();
} else { } else {
expected_val = env_var_t(L"", format_string(L"val_%d_%d", i, j)); expected_val = env_var_t(format_string(L"val_%d_%d", i, j), 0);
} }
const maybe_t<env_var_t> var = uvars.get(key); const maybe_t<env_var_t> var = uvars.get(key);
if (j == 0) assert(!expected_val); if (j == 0) assert(!expected_val);

View file

@ -224,11 +224,9 @@ static bool is_potential_cd_path(const wcstring &path, const wcstring &working_d
} else { } else {
// Get the CDPATH. // Get the CDPATH.
auto cdpath = env_get(L"CDPATH"); auto cdpath = env_get(L"CDPATH");
if (cdpath.missing_or_empty()) cdpath = env_var_t(L"CDPATH", L"."); std::vector<wcstring> pathsv =
cdpath.missing_or_empty() ? wcstring_list_t{L"."} : cdpath->as_list();
// Tokenize it into directories.
std::vector<wcstring> 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 ".".
@ -355,7 +353,7 @@ bool autosuggest_validate_from_history(const history_item_t &item,
string_prefixes_string(cd_dir, L"--help") || string_prefixes_string(cd_dir, L"-h"); string_prefixes_string(cd_dir, L"--help") || string_prefixes_string(cd_dir, L"-h");
if (!is_help) { if (!is_help) {
wcstring path; wcstring path;
env_var_t dir_var(L"n/a", cd_dir); env_var_t dir_var(cd_dir, 0);
bool can_cd = path_get_cdpath(dir_var, &path, working_directory.c_str(), vars); bool can_cd = path_get_cdpath(dir_var, &path, working_directory.c_str(), vars);
if (can_cd && !paths_are_same_file(working_directory, path)) { if (can_cd && !paths_are_same_file(working_directory, path)) {
suggestionOK = true; suggestionOK = true;