mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-27 20:25:12 +00:00
Clean up env_dispatch_table
This commit is contained in:
parent
b7fceddfc8
commit
b67174b4a3
3 changed files with 74 additions and 42 deletions
|
@ -789,7 +789,7 @@ void env_init(const struct config_paths_t *paths /* or NULL */) {
|
||||||
vars.set_one(FISH_BIND_MODE_VAR, ENV_GLOBAL, DEFAULT_BIND_MODE);
|
vars.set_one(FISH_BIND_MODE_VAR, ENV_GLOBAL, DEFAULT_BIND_MODE);
|
||||||
|
|
||||||
// Allow changes to variables to produce events.
|
// Allow changes to variables to produce events.
|
||||||
env_dispatch_mark_initialization_finished();
|
env_dispatch_init();
|
||||||
|
|
||||||
// Set up universal variables. The empty string means to use the default path.
|
// Set up universal variables. The empty string means to use the default path.
|
||||||
assert(s_universal_variables == NULL);
|
assert(s_universal_variables == NULL);
|
||||||
|
|
|
@ -82,19 +82,53 @@ extern const wcstring_list_t locale_variables({L"LANG", L"LANGUAGE", L"LC_ALL",
|
||||||
/// subsystem.
|
/// subsystem.
|
||||||
extern const wcstring_list_t curses_variables({L"TERM", L"TERMINFO", L"TERMINFO_DIRS"});
|
extern const wcstring_list_t curses_variables({L"TERM", L"TERMINFO", L"TERMINFO_DIRS"});
|
||||||
|
|
||||||
typedef std::unordered_map<wcstring, void (*)(const wcstring &, const wcstring &, env_stack_t &)>
|
class var_dispatch_table_t {
|
||||||
var_dispatch_table_t;
|
using named_callback_t = std::function<void(const wcstring &, const wcstring &, env_stack_t &)>;
|
||||||
static var_dispatch_table_t create_var_dispatch_table();
|
std::unordered_map<wcstring, named_callback_t> named_table_;
|
||||||
static const var_dispatch_table_t s_var_dispatch_table = create_var_dispatch_table();
|
|
||||||
|
|
||||||
/// This is used to ensure that we don't perform any callbacks from `env_dispatch_var_change()`
|
using anon_callback_t = std::function<void(env_stack_t &)>;
|
||||||
/// when we're importing environment variables in `env_init()`. That's because we don't have any
|
std::unordered_map<wcstring, anon_callback_t> anon_table_;
|
||||||
/// control over the order in which the vars are imported and some of them work in combination.
|
|
||||||
/// For example, `TERMINFO_DIRS` and `TERM`. If the user has set `TERM` to a custom value that is
|
|
||||||
/// found in `TERMINFO_DIRS` we don't to call `handle_curses()` before we've imported the latter.
|
|
||||||
static bool env_initialized = false;
|
|
||||||
|
|
||||||
void env_dispatch_mark_initialization_finished() { env_initialized = true; }
|
bool observes_var(const wcstring &name) {
|
||||||
|
return named_table_.count(name) || anon_table_.count(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
/// Add a callback for the given variable, which expects the name.
|
||||||
|
/// We must not already be observing this variable.
|
||||||
|
void add(wcstring name, named_callback_t cb) {
|
||||||
|
assert(!observes_var(name) && "Already observing that variable");
|
||||||
|
named_table_.emplace(std::move(name), std::move(cb));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Add a callback for the given variable, which ignores the name.
|
||||||
|
/// We must not already be observing this variable.
|
||||||
|
void add(wcstring name, anon_callback_t cb) {
|
||||||
|
assert(!observes_var(name) && "Already observing that variable");
|
||||||
|
anon_table_.emplace(std::move(name), std::move(cb));
|
||||||
|
}
|
||||||
|
|
||||||
|
void dispatch(const wchar_t *op, const wcstring &key, env_stack_t &vars) const {
|
||||||
|
auto named = named_table_.find(key);
|
||||||
|
if (named != named_table_.end()) {
|
||||||
|
named->second(op, key, vars);
|
||||||
|
}
|
||||||
|
auto anon = anon_table_.find(key);
|
||||||
|
if (anon != anon_table_.end()) {
|
||||||
|
anon->second(vars);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// return a new-ly allocated dispatch table, running those dispatch functions which should be
|
||||||
|
// initialized.
|
||||||
|
static std::unique_ptr<const var_dispatch_table_t> create_dispatch_table();
|
||||||
|
|
||||||
|
// A pointer to the variable dispatch table. This is allocated with new() and deliberately leaked to
|
||||||
|
// avoid shutdown destructors. This is set during startup and should not be modified after.
|
||||||
|
static const var_dispatch_table_t *s_var_dispatch_table;
|
||||||
|
|
||||||
|
void env_dispatch_init() { s_var_dispatch_table = create_dispatch_table().release(); }
|
||||||
|
|
||||||
/// Properly sets all timezone information.
|
/// Properly sets all timezone information.
|
||||||
static void handle_timezone(const wchar_t *env_var_name, const environment_t &vars) {
|
static void handle_timezone(const wchar_t *env_var_name, const environment_t &vars) {
|
||||||
|
@ -145,17 +179,14 @@ void guess_emoji_width() {
|
||||||
|
|
||||||
/// React to modifying the given variable.
|
/// React to modifying the given variable.
|
||||||
void env_dispatch_var_change(const wchar_t *op, const wcstring &key, env_stack_t &vars) {
|
void env_dispatch_var_change(const wchar_t *op, const wcstring &key, env_stack_t &vars) {
|
||||||
// Don't do any of this until `env_init()` has run. We only want to do this in response to
|
ASSERT_IS_MAIN_THREAD();
|
||||||
// variables set by the user; e.g., in a script like *config.fish* or interactively or as part
|
// Do nothing if not yet fully initialized.
|
||||||
// of loading the universal variables for the first time. Variables we import from the
|
if (!s_var_dispatch_table) return;
|
||||||
// environment or that are otherwise set by fish before this gets called have to explicitly
|
|
||||||
// call the appropriate functions to put the value of the var into effect.
|
|
||||||
if (!env_initialized) return;
|
|
||||||
|
|
||||||
auto dispatch = s_var_dispatch_table.find(key);
|
s_var_dispatch_table->dispatch(op, key, vars);
|
||||||
if (dispatch != s_var_dispatch_table.end()) {
|
|
||||||
(*dispatch->second)(op, key, vars);
|
// Eww.
|
||||||
} else if (string_prefixes_string(L"fish_color_", key)) {
|
if (string_prefixes_string(L"fish_color_", key)) {
|
||||||
reader_react_to_color_change();
|
reader_react_to_color_change();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -280,29 +311,30 @@ static void handle_curses_change(const wcstring &op, const wcstring &var_name, e
|
||||||
|
|
||||||
/// Populate the dispatch table used by `env_dispatch_var_change()` to efficiently call the
|
/// Populate the dispatch table used by `env_dispatch_var_change()` to efficiently call the
|
||||||
/// appropriate function to handle a change to a variable.
|
/// appropriate function to handle a change to a variable.
|
||||||
static var_dispatch_table_t create_var_dispatch_table() {
|
/// Note this returns a new-allocated value that we expect to leak.
|
||||||
var_dispatch_table_t var_dispatch_table;
|
static std::unique_ptr<const var_dispatch_table_t> create_dispatch_table() {
|
||||||
|
auto var_dispatch_table = make_unique<var_dispatch_table_t>();
|
||||||
for (const auto &var_name : locale_variables) {
|
for (const auto &var_name : locale_variables) {
|
||||||
var_dispatch_table.emplace(var_name, handle_locale_change);
|
var_dispatch_table->add(var_name, handle_locale_change);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto &var_name : curses_variables) {
|
for (const auto &var_name : curses_variables) {
|
||||||
var_dispatch_table.emplace(var_name, handle_curses_change);
|
var_dispatch_table->add(var_name, handle_curses_change);
|
||||||
}
|
}
|
||||||
|
|
||||||
var_dispatch_table.emplace(L"PATH", handle_magic_colon_var_change);
|
var_dispatch_table->add(L"PATH", handle_magic_colon_var_change);
|
||||||
var_dispatch_table.emplace(L"CDPATH", handle_magic_colon_var_change);
|
var_dispatch_table->add(L"CDPATH", handle_magic_colon_var_change);
|
||||||
var_dispatch_table.emplace(L"fish_term256", handle_fish_term_change);
|
var_dispatch_table->add(L"fish_term256", handle_fish_term_change);
|
||||||
var_dispatch_table.emplace(L"fish_term24bit", handle_fish_term_change);
|
var_dispatch_table->add(L"fish_term24bit", handle_fish_term_change);
|
||||||
var_dispatch_table.emplace(L"fish_escape_delay_ms", handle_escape_delay_change);
|
var_dispatch_table->add(L"fish_escape_delay_ms", handle_escape_delay_change);
|
||||||
var_dispatch_table.emplace(L"fish_emoji_width", handle_change_emoji_width);
|
var_dispatch_table->add(L"fish_emoji_width", handle_change_emoji_width);
|
||||||
var_dispatch_table.emplace(L"fish_ambiguous_width", handle_change_ambiguous_width);
|
var_dispatch_table->add(L"fish_ambiguous_width", handle_change_ambiguous_width);
|
||||||
var_dispatch_table.emplace(L"LINES", handle_term_size_change);
|
var_dispatch_table->add(L"LINES", handle_term_size_change);
|
||||||
var_dispatch_table.emplace(L"COLUMNS", handle_term_size_change);
|
var_dispatch_table->add(L"COLUMNS", handle_term_size_change);
|
||||||
var_dispatch_table.emplace(L"fish_complete_path", handle_complete_path_change);
|
var_dispatch_table->add(L"fish_complete_path", handle_complete_path_change);
|
||||||
var_dispatch_table.emplace(L"fish_function_path", handle_function_path_change);
|
var_dispatch_table->add(L"fish_function_path", handle_function_path_change);
|
||||||
var_dispatch_table.emplace(L"fish_read_limit", handle_read_limit_change);
|
var_dispatch_table->add(L"fish_read_limit", handle_read_limit_change);
|
||||||
var_dispatch_table.emplace(L"fish_history", handle_fish_history_change);
|
var_dispatch_table->add(L"fish_history", handle_fish_history_change);
|
||||||
var_dispatch_table.emplace(L"TZ", handle_tz_change);
|
var_dispatch_table->add(L"TZ", handle_tz_change);
|
||||||
return var_dispatch_table;
|
return var_dispatch_table;
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,8 +9,8 @@
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
/// Mark initialization as finished.
|
/// Initialize variable dispatch.
|
||||||
void env_dispatch_mark_initialization_finished();
|
void env_dispatch_init();
|
||||||
|
|
||||||
class env_stack_t;
|
class env_stack_t;
|
||||||
void env_dispatch_var_change(const wchar_t *op, const wcstring &key, env_stack_t &vars);
|
void env_dispatch_var_change(const wchar_t *op, const wcstring &key, env_stack_t &vars);
|
||||||
|
|
Loading…
Reference in a new issue