mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-02 16:18:44 +00:00
Store narrow version of uvar path
This sets the explicit path to the default one, which should be okay, since the default path never changes (not even if $XDG_CONFIG_HOME does). Then it saves a narrow version of that, which saves most of the time needed to `sync` in most cases. Fixes #5905.
This commit is contained in:
parent
bc19647be2
commit
5ca14904d0
2 changed files with 34 additions and 19 deletions
|
@ -248,7 +248,7 @@ static wcstring encode_serialized(const wcstring_list_t &vals) {
|
||||||
return join_strings(vals, UVAR_ARRAY_SEP);
|
return join_strings(vals, UVAR_ARRAY_SEP);
|
||||||
}
|
}
|
||||||
|
|
||||||
env_universal_t::env_universal_t(wcstring path) : explicit_vars_path(std::move(path)) {}
|
env_universal_t::env_universal_t(wcstring path) : narrow_vars_path(wcs2string(path)), explicit_vars_path(std::move(path)) {}
|
||||||
|
|
||||||
maybe_t<env_var_t> env_universal_t::get(const wcstring &name) const {
|
maybe_t<env_var_t> env_universal_t::get(const wcstring &name) const {
|
||||||
var_table_t::const_iterator where = vars.find(name);
|
var_table_t::const_iterator where = vars.find(name);
|
||||||
|
@ -401,7 +401,7 @@ void env_universal_t::load_from_fd(int fd, callback_data_list_t &callbacks) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool env_universal_t::load_from_path(const wcstring &path, callback_data_list_t &callbacks) {
|
bool env_universal_t::load_from_path(const std::string &path, callback_data_list_t &callbacks) {
|
||||||
ASSERT_IS_LOCKED(lock);
|
ASSERT_IS_LOCKED(lock);
|
||||||
|
|
||||||
// Check to see if the file is unchanged. We do this again in load_from_fd, but this avoids
|
// Check to see if the file is unchanged. We do this again in load_from_fd, but this avoids
|
||||||
|
@ -412,7 +412,7 @@ bool env_universal_t::load_from_path(const wcstring &path, callback_data_list_t
|
||||||
}
|
}
|
||||||
|
|
||||||
bool result = false;
|
bool result = false;
|
||||||
int fd = wopen_cloexec(path, O_RDONLY);
|
int fd = open_cloexec(path, O_RDONLY);
|
||||||
if (fd >= 0) {
|
if (fd >= 0) {
|
||||||
debug(5, L"universal log reading from file");
|
debug(5, L"universal log reading from file");
|
||||||
this->load_from_fd(fd, callbacks);
|
this->load_from_fd(fd, callbacks);
|
||||||
|
@ -422,6 +422,10 @@ bool env_universal_t::load_from_path(const wcstring &path, callback_data_list_t
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool env_universal_t::load_from_path(const wcstring &path, callback_data_list_t &callbacks) {
|
||||||
|
std::string tmp = wcs2string(path);
|
||||||
|
return load_from_path(tmp, callbacks);
|
||||||
|
}
|
||||||
/// Serialize the contents to a string.
|
/// Serialize the contents to a string.
|
||||||
std::string env_universal_t::serialize_with_vars(const var_table_t &vars) {
|
std::string env_universal_t::serialize_with_vars(const var_table_t &vars) {
|
||||||
std::string storage;
|
std::string storage;
|
||||||
|
@ -473,7 +477,7 @@ bool env_universal_t::move_new_vars_file_into_place(const wcstring &src, const w
|
||||||
bool env_universal_t::initialize(callback_data_list_t &callbacks) {
|
bool env_universal_t::initialize(callback_data_list_t &callbacks) {
|
||||||
scoped_lock locker(lock);
|
scoped_lock locker(lock);
|
||||||
if (!explicit_vars_path.empty()) {
|
if (!explicit_vars_path.empty()) {
|
||||||
return load_from_path(explicit_vars_path, callbacks);
|
return load_from_path(narrow_vars_path, callbacks);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the variables path; if there is none (e.g. HOME is bogus) it's hopeless.
|
// Get the variables path; if there is none (e.g. HOME is bogus) it's hopeless.
|
||||||
|
@ -499,6 +503,11 @@ bool env_universal_t::initialize(callback_data_list_t &callbacks) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// If we succeeded, we store the *default path*, not the legacy one.
|
||||||
|
if (success) {
|
||||||
|
explicit_vars_path = *vars_path;
|
||||||
|
narrow_vars_path = wcs2string(*vars_path);
|
||||||
|
}
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -554,7 +563,7 @@ static bool lock_uvar_file(int fd) {
|
||||||
return check_duration(start_time);
|
return check_duration(start_time);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool env_universal_t::open_and_acquire_lock(const wcstring &path, int *out_fd) {
|
bool env_universal_t::open_and_acquire_lock(const std::string &path, int *out_fd) {
|
||||||
// Attempt to open the file for reading at the given path, atomically acquiring a lock. On BSD,
|
// Attempt to open the file for reading at the given path, atomically acquiring a lock. On BSD,
|
||||||
// we can use O_EXLOCK. On Linux, we open the file, take a lock, and then compare fstat() to
|
// we can use O_EXLOCK. On Linux, we open the file, take a lock, and then compare fstat() to
|
||||||
// stat(); if they match, it means that the file was not replaced before we acquired the lock.
|
// stat(); if they match, it means that the file was not replaced before we acquired the lock.
|
||||||
|
@ -575,7 +584,7 @@ bool env_universal_t::open_and_acquire_lock(const wcstring &path, int *out_fd) {
|
||||||
int fd = -1;
|
int fd = -1;
|
||||||
while (fd == -1) {
|
while (fd == -1) {
|
||||||
double start_time = timef();
|
double start_time = timef();
|
||||||
fd = wopen_cloexec(path, flags, 0644);
|
fd = open_cloexec(path, flags, 0644);
|
||||||
if (fd == -1) {
|
if (fd == -1) {
|
||||||
if (errno == EINTR) continue; // signaled; try again
|
if (errno == EINTR) continue; // signaled; try again
|
||||||
#ifdef O_EXLOCK
|
#ifdef O_EXLOCK
|
||||||
|
@ -652,25 +661,28 @@ bool env_universal_t::sync(callback_data_list_t &callbacks) {
|
||||||
// instances of fish will not be able to obtain it. This seems to be a greater risk than that of
|
// instances of fish will not be able to obtain it. This seems to be a greater risk than that of
|
||||||
// data loss on lockless NFS. Users who put their home directory on lockless NFS are playing
|
// data loss on lockless NFS. Users who put their home directory on lockless NFS are playing
|
||||||
// with fire anyways.
|
// with fire anyways.
|
||||||
wcstring vars_path = explicit_vars_path;
|
if (explicit_vars_path.empty()) {
|
||||||
if (vars_path.empty()) {
|
// Initialize the vars path from the default one.
|
||||||
if (auto default_path = default_vars_path()) {
|
// In some cases we don't "initialize()" before, so we're doing that now.
|
||||||
vars_path = default_path.acquire();
|
// This should only happen once.
|
||||||
|
// FIXME: Why don't we initialize()?
|
||||||
|
auto def_vars_path = default_vars_path();
|
||||||
|
if (!def_vars_path) {
|
||||||
|
debug(2, L"No universal variable path available");
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
explicit_vars_path = *def_vars_path;
|
||||||
if (vars_path.empty()) {
|
narrow_vars_path = wcs2string(explicit_vars_path);
|
||||||
debug(2, L"No universal variable path available");
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we have no changes, just load.
|
// If we have no changes, just load.
|
||||||
if (modified.empty()) {
|
if (modified.empty()) {
|
||||||
this->load_from_path(vars_path, callbacks);
|
this->load_from_path(narrow_path, callbacks);
|
||||||
debug(5, L"universal log no modifications");
|
debug(5, L"universal log no modifications");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const wcstring directory = wdirname(vars_path);
|
const wcstring directory = wdirname(explicit_vars_path);
|
||||||
bool success = true;
|
bool success = true;
|
||||||
int vars_fd = -1;
|
int vars_fd = -1;
|
||||||
|
|
||||||
|
@ -678,7 +690,7 @@ bool env_universal_t::sync(callback_data_list_t &callbacks) {
|
||||||
|
|
||||||
// Open the file.
|
// Open the file.
|
||||||
if (success) {
|
if (success) {
|
||||||
success = this->open_and_acquire_lock(vars_path, &vars_fd);
|
success = this->open_and_acquire_lock(narrow_vars_path, &vars_fd);
|
||||||
if (!success) debug(5, L"universal log open_and_acquire_lock() failed");
|
if (!success) debug(5, L"universal log open_and_acquire_lock() failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -45,14 +45,17 @@ class env_universal_t {
|
||||||
// vars indicates a deleted value.
|
// vars indicates a deleted value.
|
||||||
std::unordered_set<wcstring> modified;
|
std::unordered_set<wcstring> modified;
|
||||||
|
|
||||||
|
std::string narrow_vars_path;
|
||||||
// Path that we save to. If empty, use the default.
|
// Path that we save to. If empty, use the default.
|
||||||
const wcstring explicit_vars_path;
|
wcstring explicit_vars_path;
|
||||||
|
|
||||||
|
|
||||||
// Whether it's OK to save. This may be set to false if we discover that a future version of
|
// Whether it's OK to save. This may be set to false if we discover that a future version of
|
||||||
// fish wrote the uvars contents.
|
// fish wrote the uvars contents.
|
||||||
bool ok_to_save{true};
|
bool ok_to_save{true};
|
||||||
|
|
||||||
mutable std::mutex lock;
|
mutable std::mutex lock;
|
||||||
|
bool load_from_path(const std::string &path, callback_data_list_t &callbacks);
|
||||||
bool load_from_path(const wcstring &path, callback_data_list_t &callbacks);
|
bool load_from_path(const wcstring &path, callback_data_list_t &callbacks);
|
||||||
void load_from_fd(int fd, callback_data_list_t &callbacks);
|
void load_from_fd(int fd, callback_data_list_t &callbacks);
|
||||||
|
|
||||||
|
@ -60,7 +63,7 @@ class env_universal_t {
|
||||||
bool remove_internal(const wcstring &name);
|
bool remove_internal(const wcstring &name);
|
||||||
|
|
||||||
// Functions concerned with saving.
|
// Functions concerned with saving.
|
||||||
bool open_and_acquire_lock(const wcstring &path, int *out_fd);
|
bool open_and_acquire_lock(const std::string &path, int *out_fd);
|
||||||
bool open_temporary_file(const wcstring &directory, wcstring *out_path, int *out_fd);
|
bool open_temporary_file(const wcstring &directory, wcstring *out_path, int *out_fd);
|
||||||
bool write_to_fd(int fd, const wcstring &path);
|
bool write_to_fd(int fd, const wcstring &path);
|
||||||
bool move_new_vars_file_into_place(const wcstring &src, const wcstring &dst);
|
bool move_new_vars_file_into_place(const wcstring &src, const wcstring &dst);
|
||||||
|
|
Loading…
Reference in a new issue