2017-08-05 22:08:39 +00:00
|
|
|
// Prototypes for functions for manipulating fish script variables.
|
2005-10-04 15:11:39 +00:00
|
|
|
#ifndef FISH_ENV_H
|
|
|
|
#define FISH_ENV_H
|
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
#include <stddef.h>
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <stdint.h>
|
2017-02-14 04:37:27 +00:00
|
|
|
|
2011-12-27 03:18:46 +00:00
|
|
|
#include <map>
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <memory>
|
2016-04-29 01:26:46 +00:00
|
|
|
#include <string>
|
2017-08-05 22:08:39 +00:00
|
|
|
#include <vector>
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2011-12-27 03:18:46 +00:00
|
|
|
#include "common.h"
|
2017-08-28 07:25:41 +00:00
|
|
|
#include "maybe.h"
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2017-02-08 01:21:35 +00:00
|
|
|
extern size_t read_byte_limit;
|
2017-02-16 04:09:26 +00:00
|
|
|
extern bool curses_initialized;
|
2017-02-08 01:21:35 +00:00
|
|
|
|
2017-08-06 01:22:49 +00:00
|
|
|
// Flags that may be passed as the 'mode' in env_set / env_get.
|
2016-04-29 01:26:46 +00:00
|
|
|
enum {
|
2017-08-05 22:08:39 +00:00
|
|
|
/// Default mode. Used with `env_get()` to indicate the caller doesn't care what scope the var
|
|
|
|
/// is in or whether it is exported or unexported.
|
2014-07-14 00:30:48 +00:00
|
|
|
ENV_DEFAULT = 0,
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Flag for local (to the current block) variable.
|
2017-08-05 22:08:39 +00:00
|
|
|
ENV_LOCAL = 1 << 0,
|
|
|
|
/// Flag for global variable.
|
|
|
|
ENV_GLOBAL = 1 << 1,
|
|
|
|
/// Flag for universal variable.
|
|
|
|
ENV_UNIVERSAL = 1 << 2,
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Flag for exported (to commands) variable.
|
2017-08-05 22:08:39 +00:00
|
|
|
ENV_EXPORT = 1 << 3,
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Flag for unexported variable.
|
2017-08-05 22:08:39 +00:00
|
|
|
ENV_UNEXPORT = 1 << 4,
|
2018-09-10 01:32:15 +00:00
|
|
|
/// Flag to mark a variable as a path variable.
|
|
|
|
ENV_PATHVAR = 1 << 5,
|
|
|
|
/// Flag to unmark a variable as a path variable.
|
|
|
|
ENV_UNPATHVAR = 1 << 6,
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Flag for variable update request from the user. All variable changes that are made directly
|
2017-08-05 22:08:39 +00:00
|
|
|
/// by the user, such as those from the `read` and `set` builtin must have this flag set. It
|
|
|
|
/// serves one purpose: to indicate that an error should be returned if the user is attempting
|
|
|
|
/// to modify a var that should not be modified by direct user action; e.g., a read-only var.
|
2018-09-10 01:32:15 +00:00
|
|
|
ENV_USER = 1 << 7,
|
2014-07-14 00:30:48 +00:00
|
|
|
};
|
|
|
|
typedef uint32_t env_mode_flags_t;
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-10-30 00:25:48 +00:00
|
|
|
/// Return values for `env_set()`.
|
2018-04-01 03:12:52 +00:00
|
|
|
enum { ENV_OK, ENV_PERM, ENV_SCOPE, ENV_INVALID, ENV_NOT_FOUND };
|
2016-04-29 01:26:46 +00:00
|
|
|
|
|
|
|
/// A struct of configuration directories, determined in main() that fish will optionally pass to
|
|
|
|
/// env_init.
|
|
|
|
struct config_paths_t {
|
2017-08-05 22:08:39 +00:00
|
|
|
wcstring data; // e.g., /usr/local/share
|
|
|
|
wcstring sysconf; // e.g., /usr/local/etc
|
|
|
|
wcstring doc; // e.g., /usr/local/share/doc/fish
|
|
|
|
wcstring bin; // e.g., /usr/local/bin
|
2012-07-18 17:50:38 +00:00
|
|
|
};
|
2006-04-10 15:36:26 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Initialize environment variable data.
|
2012-07-18 17:50:38 +00:00
|
|
|
void env_init(const struct config_paths_t *paths = NULL);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2017-02-16 04:09:26 +00:00
|
|
|
/// Various things we need to initialize at run-time that don't really fit any of the other init
|
|
|
|
/// routines.
|
|
|
|
void misc_init();
|
|
|
|
|
2017-08-06 01:22:49 +00:00
|
|
|
class env_var_t {
|
2018-09-10 01:32:15 +00:00
|
|
|
public:
|
2018-01-30 20:36:50 +00:00
|
|
|
using env_var_flags_t = uint8_t;
|
2018-09-10 01:32:15 +00:00
|
|
|
|
|
|
|
private:
|
2017-08-05 22:08:39 +00:00
|
|
|
wcstring_list_t vals; // list of values assigned to the var
|
2018-01-30 20:36:50 +00:00
|
|
|
env_var_flags_t flags;
|
2016-04-29 01:26:46 +00:00
|
|
|
|
|
|
|
public:
|
2018-01-30 20:36:50 +00:00
|
|
|
enum {
|
2018-09-10 01:32:15 +00:00
|
|
|
flag_export = 1 << 0, // whether the variable is exported
|
|
|
|
flag_read_only = 1 << 1, // whether the variable is read only
|
|
|
|
flag_pathvar = 1 << 2, // whether the variable is a path variable
|
2018-01-30 20:36:50 +00:00
|
|
|
};
|
2017-08-09 18:11:58 +00:00
|
|
|
|
2017-08-05 22:08:39 +00:00
|
|
|
// Constructors.
|
2017-12-23 00:54:15 +00:00
|
|
|
env_var_t(const env_var_t &) = default;
|
|
|
|
env_var_t(env_var_t &&) = default;
|
2018-01-30 20:36:50 +00:00
|
|
|
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) {}
|
2017-12-23 00:54:15 +00:00
|
|
|
|
2018-01-30 20:36:50 +00:00
|
|
|
// Constructors that infer the flags from a name.
|
|
|
|
env_var_t(const wchar_t *name, wcstring_list_t vals)
|
|
|
|
: env_var_t(std::move(vals), flags_for(name)) {}
|
|
|
|
env_var_t(const wchar_t *name, wcstring val) : env_var_t(std::move(val), flags_for(name)) {}
|
2017-08-05 22:08:39 +00:00
|
|
|
|
2017-12-23 00:54:15 +00:00
|
|
|
env_var_t() = default;
|
|
|
|
|
|
|
|
bool empty() const { return vals.empty() || (vals.size() == 1 && vals[0].empty()); };
|
2018-01-30 20:36:50 +00:00
|
|
|
bool read_only() const { return flags & flag_read_only; }
|
|
|
|
bool exports() const { return flags & flag_export; }
|
2018-09-10 01:32:15 +00:00
|
|
|
bool is_pathvar() const { return flags & flag_pathvar; }
|
|
|
|
env_var_flags_t get_flags() const { return flags; }
|
2013-02-12 07:16:50 +00:00
|
|
|
|
2017-08-06 01:22:49 +00:00
|
|
|
wcstring as_string() const;
|
2017-08-05 22:08:39 +00:00
|
|
|
void to_list(wcstring_list_t &out) const;
|
2017-08-28 09:51:34 +00:00
|
|
|
const wcstring_list_t &as_list() const;
|
2017-08-05 22:08:39 +00:00
|
|
|
|
2018-09-30 23:34:01 +00:00
|
|
|
/// \return the character used when delimiting quoted expansion.
|
|
|
|
wchar_t get_delimiter() const;
|
|
|
|
|
2017-08-28 09:51:34 +00:00
|
|
|
void set_vals(wcstring_list_t v) { vals = std::move(v); }
|
2013-02-12 07:16:50 +00:00
|
|
|
|
2018-01-30 20:36:50 +00:00
|
|
|
void set_exports(bool exportv) {
|
|
|
|
if (exportv) {
|
|
|
|
flags |= flag_export;
|
|
|
|
} else {
|
|
|
|
flags &= ~flag_export;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-10 01:32:15 +00:00
|
|
|
void set_pathvar(bool pathvar) {
|
|
|
|
if (pathvar) {
|
|
|
|
flags |= flag_pathvar;
|
|
|
|
} else {
|
|
|
|
flags &= ~flag_pathvar;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-30 20:36:50 +00:00
|
|
|
static env_var_flags_t flags_for(const wchar_t *name);
|
|
|
|
|
2017-12-23 00:54:15 +00:00
|
|
|
env_var_t &operator=(const env_var_t &var) = default;
|
|
|
|
env_var_t &operator=(env_var_t &&) = default;
|
2013-02-12 07:16:50 +00:00
|
|
|
|
2018-10-20 22:55:46 +00:00
|
|
|
bool operator==(const env_var_t &rhs) const { return vals == rhs.vals && flags == rhs.flags; }
|
|
|
|
bool operator!=(const env_var_t &rhs) const { return ! (*this == rhs); }
|
2012-01-14 09:06:47 +00:00
|
|
|
};
|
2013-02-20 01:48:51 +00:00
|
|
|
|
2018-09-09 09:25:51 +00:00
|
|
|
/// An environment is read-only access to variable values.
|
|
|
|
class environment_t {
|
|
|
|
protected:
|
|
|
|
environment_t() = default;
|
|
|
|
|
|
|
|
public:
|
|
|
|
virtual maybe_t<env_var_t> get(const wcstring &key,
|
|
|
|
env_mode_flags_t mode = ENV_DEFAULT) const = 0;
|
|
|
|
virtual ~environment_t();
|
|
|
|
};
|
|
|
|
|
2017-08-28 07:25:41 +00:00
|
|
|
/// Gets the variable with the specified name, or none() if it does not exist.
|
|
|
|
maybe_t<env_var_t> env_get(const wcstring &key, env_mode_flags_t mode = ENV_DEFAULT);
|
2017-08-06 01:22:49 +00:00
|
|
|
|
2017-08-28 09:51:34 +00:00
|
|
|
/// Sets the variable with the specified name to the given values.
|
|
|
|
int env_set(const wcstring &key, env_mode_flags_t mode, wcstring_list_t vals);
|
2011-12-27 03:18:46 +00:00
|
|
|
|
2017-08-05 22:08:39 +00:00
|
|
|
/// Sets the variable with the specified name to a single value.
|
2017-08-28 09:51:34 +00:00
|
|
|
int env_set_one(const wcstring &key, env_mode_flags_t mode, wcstring val);
|
2017-08-05 22:08:39 +00:00
|
|
|
|
|
|
|
/// Sets the variable with the specified name to no values.
|
|
|
|
int env_set_empty(const wcstring &key, env_mode_flags_t mode);
|
2005-09-26 14:47:03 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Remove environment variable.
|
|
|
|
///
|
|
|
|
/// \param key The name of the variable to remove
|
|
|
|
/// \param mode should be ENV_USER if this is a remove request from the user, 0 otherwise. If this
|
|
|
|
/// is a user request, read-only variables can not be removed. The mode may also specify the scope
|
|
|
|
/// of the variable that should be erased.
|
|
|
|
///
|
|
|
|
/// \return zero if the variable existed, and non-zero if the variable did not exist
|
2012-11-19 00:30:30 +00:00
|
|
|
int env_remove(const wcstring &key, int mode);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Push the variable stack. Used for implementing local variables for functions and for-loops.
|
2013-01-19 21:21:55 +00:00
|
|
|
void env_push(bool new_scope);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Pop the variable stack. Used for implementing local variables for functions and for-loops.
|
2005-09-20 13:26:39 +00:00
|
|
|
void env_pop();
|
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Synchronizes all universal variable changes: writes everything out, reads stuff in.
|
2014-06-16 00:30:50 +00:00
|
|
|
void env_universal_barrier();
|
|
|
|
|
2017-01-26 20:03:14 +00:00
|
|
|
/// Returns an array containing all exported variables in a format suitable for execv
|
|
|
|
const char *const *env_export_arr();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Sets up argv as the given null terminated array of strings.
|
|
|
|
void env_set_argv(const wchar_t *const *argv);
|
2015-08-15 20:37:17 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Returns all variable names.
|
2012-11-19 00:30:30 +00:00
|
|
|
wcstring_list_t env_get_names(int flags);
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
/// Returns the PWD with a terminating slash.
|
2013-04-27 07:45:38 +00:00
|
|
|
wcstring env_get_pwd_slash();
|
|
|
|
|
2017-02-08 01:21:35 +00:00
|
|
|
/// Update the read_byte_limit variable.
|
|
|
|
void env_set_read_limit();
|
|
|
|
|
2018-09-09 19:17:31 +00:00
|
|
|
/// A environment stack of scopes. This is the main class that tracks fish variables.
|
|
|
|
struct var_stack_t;
|
|
|
|
class env_node_t;
|
|
|
|
class env_stack_t : public environment_t {
|
|
|
|
std::unique_ptr<var_stack_t> vars_;
|
|
|
|
|
|
|
|
int set_internal(const wcstring &key, env_mode_flags_t var_mode, wcstring_list_t val);
|
|
|
|
|
|
|
|
bool try_remove(env_node_t *n, const wchar_t *key, int var_mode);
|
|
|
|
env_node_t *get_node(const wcstring &key);
|
|
|
|
|
|
|
|
var_stack_t &vars_stack();
|
|
|
|
const var_stack_t &vars_stack() const;
|
|
|
|
|
|
|
|
env_stack_t();
|
|
|
|
~env_stack_t() override;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// Gets the variable with the specified name, or none() if it does not exist.
|
|
|
|
maybe_t<env_var_t> get(const wcstring &key, env_mode_flags_t mode = ENV_DEFAULT) const override;
|
|
|
|
|
|
|
|
/// Sets the variable with the specified name to the given values.
|
|
|
|
int set(const wcstring &key, env_mode_flags_t mode, wcstring_list_t vals);
|
|
|
|
|
|
|
|
/// Sets the variable with the specified name to a single value.
|
|
|
|
int set_one(const wcstring &key, env_mode_flags_t mode, wcstring val);
|
|
|
|
|
|
|
|
/// Sets the variable with the specified name to no values.
|
|
|
|
int set_empty(const wcstring &key, env_mode_flags_t mode);
|
|
|
|
|
|
|
|
/// Update the PWD variable based on the result of getcwd.
|
|
|
|
void set_pwd_from_getcwd();
|
|
|
|
|
|
|
|
/// Remove environment variable.
|
|
|
|
///
|
|
|
|
/// \param key The name of the variable to remove
|
|
|
|
/// \param mode should be ENV_USER if this is a remove request from the user, 0 otherwise. If
|
|
|
|
/// this is a user request, read-only variables can not be removed. The mode may also specify
|
|
|
|
/// the scope of the variable that should be erased.
|
|
|
|
///
|
|
|
|
/// \return zero if the variable existed, and non-zero if the variable did not exist
|
|
|
|
int remove(const wcstring &key, int mode);
|
|
|
|
|
|
|
|
/// Push the variable stack. Used for implementing local variables for functions and for-loops.
|
|
|
|
void push(bool new_scope);
|
|
|
|
|
|
|
|
/// Pop the variable stack. Used for implementing local variables for functions and for-loops.
|
|
|
|
void pop();
|
|
|
|
|
|
|
|
/// Synchronizes all universal variable changes: writes everything out, reads stuff in.
|
|
|
|
void universal_barrier();
|
|
|
|
|
|
|
|
/// Returns an array containing all exported variables in a format suitable for execv
|
|
|
|
const char *const *export_arr();
|
|
|
|
|
|
|
|
/// Returns all variable names.
|
|
|
|
wcstring_list_t get_names(int flags);
|
|
|
|
|
|
|
|
/// Sets up argv as the given null terminated array of strings.
|
|
|
|
void set_argv(const wchar_t *const *argv);
|
|
|
|
|
|
|
|
/// Returns the PWD with a terminating slash.
|
|
|
|
wcstring get_pwd_slash();
|
|
|
|
|
|
|
|
/// Update the read_byte_limit variable.
|
|
|
|
void set_read_limit();
|
|
|
|
|
|
|
|
/// Mark that exported variables have changed.
|
|
|
|
void mark_changed_exported();
|
|
|
|
|
|
|
|
// Compatibility hack; access the "environment stack" from back when there was just one.
|
|
|
|
static env_stack_t &principal();
|
|
|
|
};
|
|
|
|
|
2018-09-09 09:25:51 +00:00
|
|
|
class env_vars_snapshot_t : public environment_t {
|
2017-08-05 22:08:39 +00:00
|
|
|
std::map<wcstring, env_var_t> vars;
|
2012-07-21 05:11:05 +00:00
|
|
|
bool is_current() const;
|
2016-04-29 01:26:46 +00:00
|
|
|
|
2017-01-27 04:00:43 +00:00
|
|
|
public:
|
2017-01-24 16:23:24 +00:00
|
|
|
env_vars_snapshot_t(const env_vars_snapshot_t &) = default;
|
|
|
|
env_vars_snapshot_t &operator=(const env_vars_snapshot_t &) = default;
|
2016-04-29 01:26:46 +00:00
|
|
|
|
|
|
|
env_vars_snapshot_t(const wchar_t *const *keys);
|
2016-02-08 09:29:23 +00:00
|
|
|
env_vars_snapshot_t();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2018-09-09 09:25:51 +00:00
|
|
|
~env_vars_snapshot_t();
|
|
|
|
|
|
|
|
maybe_t<env_var_t> get(const wcstring &key, env_mode_flags_t mode = ENV_DEFAULT) const override;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
// Returns the fake snapshot representing the live variables array.
|
2012-07-21 05:11:05 +00:00
|
|
|
static const env_vars_snapshot_t ¤t();
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-29 01:26:46 +00:00
|
|
|
// Vars necessary for highlighting.
|
|
|
|
static const wchar_t *const highlighting_keys[];
|
|
|
|
|
|
|
|
// Vars necessary for completion.
|
|
|
|
static const wchar_t *const completing_keys[];
|
2011-12-27 03:18:46 +00:00
|
|
|
};
|
|
|
|
|
2012-04-22 03:08:08 +00:00
|
|
|
extern int g_fork_count;
|
2012-08-15 07:57:56 +00:00
|
|
|
extern bool g_use_posix_spawn;
|
|
|
|
|
2017-08-09 18:11:58 +00:00
|
|
|
typedef std::map<wcstring, env_var_t> var_table_t;
|
2016-12-23 21:08:45 +00:00
|
|
|
|
|
|
|
extern bool term_has_xn; // does the terminal have the "eat_newline_glitch"
|
2017-01-10 02:35:37 +00:00
|
|
|
|
|
|
|
/// Returns true if we think the terminal supports setting its title.
|
|
|
|
bool term_supports_setting_title();
|
2018-09-28 15:14:27 +00:00
|
|
|
|
|
|
|
/// Gets a path appropriate for runtime storage
|
|
|
|
wcstring env_get_runtime_path();
|
2005-10-04 15:11:39 +00:00
|
|
|
#endif
|