2016-04-20 02:49:15 +00:00
|
|
|
// Functions used for implementing the set builtin.
|
2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
#include <errno.h>
|
2017-02-13 04:24:22 +00:00
|
|
|
#include <stddef.h>
|
2005-09-20 13:26:39 +00:00
|
|
|
#include <stdlib.h>
|
2016-11-15 05:31:51 +00:00
|
|
|
#include <string.h>
|
2016-04-20 02:49:15 +00:00
|
|
|
#include <sys/stat.h>
|
2016-11-15 05:31:51 +00:00
|
|
|
#include <unistd.h>
|
2005-09-20 13:26:39 +00:00
|
|
|
#include <wchar.h>
|
2017-02-13 04:24:22 +00:00
|
|
|
|
2012-01-09 19:49:37 +00:00
|
|
|
#include <algorithm>
|
2016-04-20 02:49:15 +00:00
|
|
|
#include <iterator>
|
2016-04-21 06:00:54 +00:00
|
|
|
#include <memory>
|
|
|
|
#include <set>
|
2016-04-20 02:49:15 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2006-02-28 13:17:16 +00:00
|
|
|
|
2005-09-20 13:26:39 +00:00
|
|
|
#include "builtin.h"
|
2016-04-20 02:49:15 +00:00
|
|
|
#include "common.h"
|
2005-09-20 13:26:39 +00:00
|
|
|
#include "env.h"
|
|
|
|
#include "expand.h"
|
2016-04-20 02:49:15 +00:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2016-04-21 06:00:54 +00:00
|
|
|
#include "io.h"
|
2016-04-20 02:49:15 +00:00
|
|
|
#include "proc.h"
|
|
|
|
#include "wgetopt.h"
|
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
2016-04-21 06:00:54 +00:00
|
|
|
|
|
|
|
class parser_t;
|
2006-07-19 22:55:49 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
struct set_cmd_opts_t {
|
|
|
|
bool print_help = false;
|
2017-08-02 03:58:40 +00:00
|
|
|
bool show = false;
|
2017-07-23 04:15:20 +00:00
|
|
|
bool local = false;
|
|
|
|
bool global = false;
|
|
|
|
bool exportv = false;
|
|
|
|
bool erase = false;
|
|
|
|
bool list = false;
|
|
|
|
bool unexport = false;
|
|
|
|
bool universal = false;
|
|
|
|
bool query = false;
|
|
|
|
bool shorten_ok = true;
|
|
|
|
bool preserve_failure_exit_status = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Variables used for parsing the argument list. This command is atypical in using the "+"
|
|
|
|
// (REQUIRE_ORDER) option for flag parsing. This is not typical of most fish commands. It means
|
|
|
|
// we stop scanning for flags when the first non-flag argument is seen.
|
2017-08-02 03:58:40 +00:00
|
|
|
static const wchar_t *short_options = L"+:LSUeghlnqux";
|
|
|
|
static const struct woption long_options[] = {
|
|
|
|
{L"export", no_argument, NULL, 'x'}, {L"global", no_argument, NULL, 'g'},
|
|
|
|
{L"local", no_argument, NULL, 'l'}, {L"erase", no_argument, NULL, 'e'},
|
|
|
|
{L"names", no_argument, NULL, 'n'}, {L"unexport", no_argument, NULL, 'u'},
|
|
|
|
{L"universal", no_argument, NULL, 'U'}, {L"long", no_argument, NULL, 'L'},
|
|
|
|
{L"query", no_argument, NULL, 'q'}, {L"show", no_argument, NULL, 'S'},
|
|
|
|
{L"help", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0}};
|
2017-07-23 04:15:20 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
// Error message for invalid path operations.
|
2017-07-24 18:42:54 +00:00
|
|
|
#define BUILTIN_SET_PATH_ERROR _(L"%ls: Warning: $%ls entry \"%ls\" is not valid (%s)\n")
|
2016-04-20 02:49:15 +00:00
|
|
|
// Hint for invalid path operation with a colon.
|
2017-07-24 18:42:54 +00:00
|
|
|
#define BUILTIN_SET_PATH_HINT _(L"%ls: Did you mean 'set %ls $%ls %ls'?\n")
|
|
|
|
#define BUILTIN_SET_MISMATCHED_ARGS _(L"%ls: You provided %d indexes but %d values\n")
|
|
|
|
#define BUILTIN_SET_ERASE_NO_VAR _(L"%ls: Erase needs a variable name\n")
|
|
|
|
#define BUILTIN_SET_ARRAY_BOUNDS_ERR _(L"%ls: Array index out of bounds\n")
|
2016-04-20 02:49:15 +00:00
|
|
|
|
|
|
|
// Test if the specified variable should be subject to path validation.
|
2017-04-05 04:28:57 +00:00
|
|
|
static const wcstring_list_t path_variables({L"PATH", L"CDPATH"});
|
|
|
|
static int is_path_variable(const wchar_t *env) { return contains(path_variables, env); }
|
2006-04-20 18:29:44 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
static int parse_cmd_opts(set_cmd_opts_t &opts, int *optind, //!OCLINT(high ncss method)
|
|
|
|
int argc, wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
|
|
|
wchar_t *cmd = argv[0];
|
|
|
|
|
|
|
|
int opt;
|
|
|
|
wgetopter_t w;
|
|
|
|
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, NULL)) != -1) {
|
|
|
|
switch (opt) {
|
|
|
|
case 'e': {
|
|
|
|
opts.erase = true;
|
|
|
|
opts.preserve_failure_exit_status = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'n': {
|
|
|
|
opts.list = true;
|
|
|
|
opts.preserve_failure_exit_status = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'x': {
|
|
|
|
opts.exportv = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'l': {
|
|
|
|
opts.local = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'g': {
|
|
|
|
opts.global = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'u': {
|
|
|
|
opts.unexport = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'U': {
|
|
|
|
opts.universal = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'L': {
|
|
|
|
opts.shorten_ok = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'q': {
|
|
|
|
opts.query = true;
|
|
|
|
opts.preserve_failure_exit_status = false;
|
|
|
|
break;
|
|
|
|
}
|
2017-08-02 03:58:40 +00:00
|
|
|
case 'S': {
|
|
|
|
opts.show = true;
|
|
|
|
opts.preserve_failure_exit_status = false;
|
|
|
|
break;
|
|
|
|
}
|
2017-07-23 04:15:20 +00:00
|
|
|
case 'h': {
|
|
|
|
opts.print_help = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ':': {
|
|
|
|
builtin_missing_argument(parser, streams, cmd, argv[w.woptind - 1]);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
case '?': {
|
|
|
|
builtin_unknown_option(parser, streams, cmd, argv[w.woptind - 1]);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
DIE("unexpected retval from wgetopt_long");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*optind = w.woptind;
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
2017-08-02 03:58:40 +00:00
|
|
|
static int validate_cmd_opts(const wchar_t *cmd, set_cmd_opts_t &opts, //!OCLINT(npath complexity)
|
|
|
|
int argc, parser_t &parser, io_streams_t &streams) {
|
2017-07-23 04:15:20 +00:00
|
|
|
// Can't query and erase or list.
|
|
|
|
if (opts.query && (opts.erase || opts.list)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can't both list and erase variables.
|
|
|
|
if (opts.erase && opts.list) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Variables can only have one scope.
|
|
|
|
if (opts.local + opts.global + opts.universal > 1) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_GLOCAL, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Variables can only have one export status.
|
|
|
|
if (opts.exportv && opts.unexport) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_EXPUNEXP, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
2017-07-28 21:11:14 +00:00
|
|
|
// Trying to erase and (un)export at the same time doesn't make sense.
|
|
|
|
if (opts.erase && (opts.exportv || opts.unexport)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
2017-08-02 03:58:40 +00:00
|
|
|
// The --show flag cannot be combined with any other flag.
|
|
|
|
if (opts.show &&
|
|
|
|
(opts.local || opts.global || opts.erase || opts.list || opts.exportv || opts.universal)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (argc == 0 && opts.erase) {
|
|
|
|
streams.err.append_format(BUILTIN_SET_ERASE_NO_VAR, cmd);
|
2017-07-23 04:15:20 +00:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
// Check if we are setting a uvar and a global of the same name exists. See
|
|
|
|
// https://github.com/fish-shell/fish-shell/issues/806
|
|
|
|
static int check_global_scope_exists(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_t *dest,
|
|
|
|
io_streams_t &streams) {
|
|
|
|
if (opts.universal) {
|
|
|
|
env_var_t global_dest = env_get_string(dest, ENV_GLOBAL);
|
|
|
|
if (!global_dest.missing()) {
|
|
|
|
streams.err.append_format(
|
|
|
|
_(L"%ls: Warning: universal scope selected, but a global variable '%ls' exists.\n"),
|
|
|
|
cmd, dest);
|
|
|
|
}
|
|
|
|
}
|
2013-02-20 01:48:51 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
// Fix for https://github.com/fish-shell/fish-shell/issues/199 . Return success if any path setting
|
|
|
|
// succeeds.
|
|
|
|
static int my_env_path_setup(const wchar_t *cmd, const wchar_t *key, //!OCLINT(npath complexity)
|
|
|
|
const wcstring_list_t &list, io_streams_t &streams) {
|
|
|
|
bool any_success = false;
|
|
|
|
|
|
|
|
// Don't bother validating (or complaining about) values that are already present. When
|
|
|
|
// determining already-present values, use ENV_DEFAULT instead of the passed-in scope because
|
|
|
|
// in:
|
|
|
|
//
|
|
|
|
// set -l PATH stuff $PATH
|
|
|
|
//
|
|
|
|
// where we are temporarily shadowing a variable, we want to compare against the shadowed value,
|
|
|
|
// not the (missing) local value. Also don't bother to complain about relative paths, which
|
|
|
|
// don't start with /.
|
|
|
|
wcstring_list_t existing_values;
|
|
|
|
const env_var_t existing_variable = env_get_string(key, ENV_DEFAULT);
|
|
|
|
if (!existing_variable.missing_or_empty())
|
|
|
|
tokenize_variable_array(existing_variable, existing_values);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < list.size(); i++) {
|
|
|
|
const wcstring &dir = list.at(i);
|
|
|
|
if (!string_prefixes_string(L"/", dir) || contains(existing_values, dir)) {
|
|
|
|
any_success = true;
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
bool show_hint = false;
|
|
|
|
bool error = false;
|
|
|
|
struct stat buff;
|
|
|
|
if (wstat(dir, &buff) == -1) {
|
|
|
|
error = true;
|
|
|
|
} else if (!S_ISDIR(buff.st_mode)) {
|
|
|
|
error = true;
|
|
|
|
errno = ENOTDIR;
|
|
|
|
} else if (waccess(dir, X_OK) == -1) {
|
|
|
|
error = true;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (!error) {
|
|
|
|
any_success = true;
|
|
|
|
} else {
|
|
|
|
streams.err.append_format(BUILTIN_SET_PATH_ERROR, cmd, key, dir.c_str(),
|
|
|
|
strerror(errno));
|
|
|
|
const wchar_t *colon = wcschr(dir.c_str(), L':');
|
|
|
|
if (colon && *(colon + 1)) show_hint = true;
|
2012-07-08 06:04:02 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (show_hint) {
|
|
|
|
streams.err.append_format(BUILTIN_SET_PATH_HINT, cmd, key, key,
|
|
|
|
wcschr(dir.c_str(), L':') + 1);
|
2012-07-08 06:04:02 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
// Fail at setting the path if we tried to set it to something non-empty, but it wound up
|
|
|
|
// empty.
|
|
|
|
if (!list.empty() && !any_success) return STATUS_CMD_ERROR;
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Call env_set. If this is a path variable, e.g. PATH, validate the elements. On error, print a
|
|
|
|
/// description of the problem to stderr.
|
|
|
|
static int my_env_set(const wchar_t *cmd, const wchar_t *key, const wcstring_list_t &list,
|
|
|
|
int scope, io_streams_t &streams) {
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
if (is_path_variable(key)) {
|
|
|
|
retval = my_env_path_setup(cmd, key, list, streams);
|
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
}
|
|
|
|
|
2017-07-08 20:14:30 +00:00
|
|
|
// We don't check `val->empty()` because an array var with a single empty string will be
|
|
|
|
// "empty". A truly empty array var is set to the special value `ENV_NULL`.
|
|
|
|
auto val = list_to_array_val(list);
|
2017-07-24 18:42:54 +00:00
|
|
|
retval = env_set(key, *val == ENV_NULL ? NULL : val->c_str(), scope | ENV_USER);
|
|
|
|
switch (retval) {
|
2016-10-30 00:25:48 +00:00
|
|
|
case ENV_OK: {
|
2017-07-24 18:42:54 +00:00
|
|
|
retval = STATUS_CMD_OK;
|
2016-10-30 00:25:48 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case ENV_PERM: {
|
|
|
|
streams.err.append_format(_(L"%ls: Tried to change the read-only variable '%ls'\n"),
|
2017-07-24 18:42:54 +00:00
|
|
|
cmd, key);
|
|
|
|
retval = STATUS_CMD_ERROR;
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case ENV_SCOPE: {
|
|
|
|
streams.err.append_format(
|
2017-07-24 18:42:54 +00:00
|
|
|
_(L"%ls: Tried to set the special variable '%ls' with the wrong scope\n"), cmd,
|
2016-04-20 02:49:15 +00:00
|
|
|
key);
|
2017-07-24 18:42:54 +00:00
|
|
|
retval = STATUS_CMD_ERROR;
|
2014-07-12 21:05:42 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
case ENV_INVALID: {
|
|
|
|
streams.err.append_format(
|
2017-07-24 18:42:54 +00:00
|
|
|
_(L"%ls: Tried to set the special variable '%ls' to an invalid value\n"), cmd, key);
|
|
|
|
retval = STATUS_CMD_ERROR;
|
2012-11-19 08:31:03 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-10-30 00:25:48 +00:00
|
|
|
default: {
|
|
|
|
DIE("unexpected env_set() ret val");
|
|
|
|
break;
|
|
|
|
}
|
2012-07-08 06:04:02 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
return retval;
|
2012-01-09 19:49:37 +00:00
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Extract indexes from an argument of the form `var_name[index1 index2...]`.
|
2016-04-20 02:49:15 +00:00
|
|
|
///
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Inputs:
|
|
|
|
/// indexes: the list to insert the new indexes into
|
|
|
|
/// src: The source string to parse. This must be a var spec of the form "var[...]"
|
2016-04-20 02:49:15 +00:00
|
|
|
///
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Returns:
|
|
|
|
/// The total number of indexes parsed, or -1 on error. If any indexes were found the `src` string
|
|
|
|
/// is modified to omit the index expression leaving just the var name.
|
|
|
|
static int parse_index(std::vector<long> &indexes, wchar_t *src, int scope, io_streams_t &streams) {
|
|
|
|
wchar_t *p = wcschr(src, L'[');
|
|
|
|
if (!p) return 0; // no slices so nothing for us to do
|
|
|
|
*p = L'\0'; // split the var name from the indexes/slices
|
|
|
|
p++;
|
|
|
|
|
|
|
|
env_var_t var_str = env_get_string(src, scope);
|
|
|
|
wcstring_list_t var;
|
|
|
|
if (!var_str.missing()) tokenize_variable_array(var_str, var);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
int count = 0;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
while (*p != L']') {
|
2016-11-23 04:24:03 +00:00
|
|
|
const wchar_t *end;
|
2017-07-24 18:42:54 +00:00
|
|
|
long l_ind = fish_wcstol(p, &end);
|
2016-11-23 04:24:03 +00:00
|
|
|
if (errno > 0) { // ignore errno == -1 meaning the int did not end with a '\0'
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.err.append_format(_(L"%ls: Invalid index starting at '%ls'\n"), L"set", src);
|
2017-07-24 18:42:54 +00:00
|
|
|
return -1;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2017-07-24 18:42:54 +00:00
|
|
|
p = (wchar_t *)end;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
// Convert negative index to a positive index.
|
|
|
|
if (l_ind < 0) l_ind = var.size() + l_ind + 1;
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (*p == L'.' && *(p + 1) == L'.') {
|
|
|
|
p += 2;
|
|
|
|
long l_ind2 = fish_wcstol(p, &end);
|
2016-11-23 04:24:03 +00:00
|
|
|
if (errno > 0) { // ignore errno == -1 meaning the int did not end with a '\0'
|
2017-07-24 18:42:54 +00:00
|
|
|
return -1;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2017-07-24 18:42:54 +00:00
|
|
|
p = (wchar_t *)end;
|
|
|
|
|
|
|
|
// Convert negative index to a positive index.
|
|
|
|
if (l_ind2 < 0) l_ind2 = var.size() + l_ind2 + 1;
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
int direction = l_ind2 < l_ind ? -1 : 1;
|
|
|
|
for (long jjj = l_ind; jjj * direction <= l_ind2 * direction; jjj += direction) {
|
2012-11-19 00:30:30 +00:00
|
|
|
indexes.push_back(jjj);
|
|
|
|
count++;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
} else {
|
2012-11-19 00:30:30 +00:00
|
|
|
indexes.push_back(l_ind);
|
|
|
|
count++;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2012-11-19 00:30:30 +00:00
|
|
|
return count;
|
2012-01-09 19:49:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
static int update_values(wcstring_list_t &list, std::vector<long> &indexes,
|
|
|
|
wcstring_list_t &values) {
|
|
|
|
// Replace values where needed.
|
2017-07-24 18:42:54 +00:00
|
|
|
for (size_t i = 0; i < indexes.size(); i++) {
|
2016-04-20 02:49:15 +00:00
|
|
|
// The '- 1' below is because the indices in fish are one-based, but the vector uses
|
|
|
|
// zero-based indices.
|
2012-11-19 00:30:30 +00:00
|
|
|
long ind = indexes[i] - 1;
|
2016-04-20 02:49:15 +00:00
|
|
|
const wcstring newv = values[i];
|
|
|
|
if (ind < 0) {
|
2012-11-19 00:30:30 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
if ((size_t)ind >= list.size()) {
|
|
|
|
list.resize(ind + 1);
|
2012-06-21 17:24:49 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
list[ind] = newv;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2012-11-19 00:30:30 +00:00
|
|
|
return 0;
|
2012-01-09 19:49:37 +00:00
|
|
|
}
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
/// Erase from a list of wcstring values at specified indexes.
|
|
|
|
static void erase_values(wcstring_list_t &list, const std::vector<long> &indexes) {
|
2012-05-10 08:04:18 +00:00
|
|
|
// Make a set of indexes.
|
|
|
|
// This both sorts them into ascending order and removes duplicates.
|
|
|
|
const std::set<long> indexes_set(indexes.begin(), indexes.end());
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
// Now walk the set backwards, so we encounter larger indexes first, and remove elements at the
|
|
|
|
// given (1-based) indexes.
|
2012-05-10 08:04:18 +00:00
|
|
|
std::set<long>::const_reverse_iterator iter;
|
2016-04-20 02:49:15 +00:00
|
|
|
for (iter = indexes_set.rbegin(); iter != indexes_set.rend(); ++iter) {
|
2012-05-10 08:04:18 +00:00
|
|
|
long val = *iter;
|
2016-04-20 02:49:15 +00:00
|
|
|
if (val > 0 && (size_t)val <= list.size()) {
|
2012-05-10 08:04:18 +00:00
|
|
|
// One-based indexing!
|
|
|
|
list.erase(list.begin() + val - 1);
|
|
|
|
}
|
|
|
|
}
|
2012-01-09 19:49:37 +00:00
|
|
|
}
|
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
static int compute_scope(set_cmd_opts_t &opts) {
|
|
|
|
int scope = ENV_USER;
|
|
|
|
if (opts.local) scope |= ENV_LOCAL;
|
|
|
|
if (opts.global) scope |= ENV_GLOBAL;
|
|
|
|
if (opts.exportv) scope |= ENV_EXPORT;
|
|
|
|
if (opts.unexport) scope |= ENV_UNEXPORT;
|
|
|
|
if (opts.universal) scope |= ENV_UNIVERSAL;
|
|
|
|
return scope;
|
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Print the names of all environment variables in the scope. It will include the values unless the
|
|
|
|
/// `set --list` flag was used.
|
|
|
|
static int builtin_set_list(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
2017-07-23 04:15:20 +00:00
|
|
|
wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
2017-07-24 18:42:54 +00:00
|
|
|
UNUSED(cmd);
|
|
|
|
UNUSED(argc);
|
|
|
|
UNUSED(argv);
|
|
|
|
UNUSED(parser);
|
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
bool names_only = opts.list;
|
|
|
|
wcstring_list_t names = env_get_names(compute_scope(opts));
|
2012-02-04 02:39:41 +00:00
|
|
|
sort(names.begin(), names.end());
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-20 02:49:15 +00:00
|
|
|
for (size_t i = 0; i < names.size(); i++) {
|
2012-11-19 00:30:30 +00:00
|
|
|
const wcstring key = names.at(i);
|
2012-02-04 02:39:41 +00:00
|
|
|
const wcstring e_key = escape_string(key, 0);
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(e_key);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
if (!names_only) {
|
|
|
|
env_var_t value = env_get_string(key, compute_scope(opts));
|
2016-04-20 02:49:15 +00:00
|
|
|
if (!value.missing()) {
|
2017-07-23 04:15:20 +00:00
|
|
|
bool shorten = false;
|
|
|
|
if (opts.shorten_ok && value.length() > 64) {
|
|
|
|
shorten = true;
|
2012-11-19 00:30:30 +00:00
|
|
|
value.resize(60);
|
|
|
|
}
|
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
wcstring e_value = expand_escape_variable(value);
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(L" ");
|
|
|
|
streams.out.append(e_value);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
if (shorten) streams.out.push_back(ellipsis_char);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2015-09-21 18:24:49 +00:00
|
|
|
streams.out.append(L"\n");
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
// Query mode. Return the number of variables that do not exist out of the specified variables.
|
2017-07-24 18:42:54 +00:00
|
|
|
static int builtin_set_query(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
2017-07-23 04:15:20 +00:00
|
|
|
wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
|
|
|
int retval = 0;
|
2017-07-24 18:42:54 +00:00
|
|
|
int scope = compute_scope(opts);
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
for (int i = 0; i < argc; i++) {
|
2017-07-23 04:15:20 +00:00
|
|
|
wchar_t *arg = argv[i];
|
|
|
|
wchar_t *dest = wcsdup(arg);
|
|
|
|
assert(dest);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
std::vector<long> indexes;
|
|
|
|
int idx_count = parse_index(indexes, dest, scope, streams);
|
|
|
|
if (idx_count == -1) {
|
|
|
|
free(dest);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_CMD_ERROR;
|
2017-07-23 04:15:20 +00:00
|
|
|
}
|
2016-04-20 02:49:15 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (idx_count) {
|
2017-07-23 04:15:20 +00:00
|
|
|
wcstring_list_t result;
|
2017-07-24 18:42:54 +00:00
|
|
|
env_var_t dest_str = env_get_string(dest, scope);
|
2017-07-23 04:15:20 +00:00
|
|
|
if (!dest_str.missing()) tokenize_variable_array(dest_str, result);
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
for (auto idx : indexes) {
|
|
|
|
if (idx < 1 || (size_t)idx > result.size()) retval++;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2017-07-24 18:42:54 +00:00
|
|
|
} else if (!env_exist(arg, scope)) {
|
2017-07-23 04:15:20 +00:00
|
|
|
retval++;
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
2017-07-23 04:15:20 +00:00
|
|
|
|
|
|
|
free(dest);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2017-08-02 03:58:40 +00:00
|
|
|
static void show_scope(const wchar_t *var_name, int scope, io_streams_t &streams) {
|
|
|
|
const wchar_t *scope_name;
|
|
|
|
switch (scope) {
|
|
|
|
case ENV_LOCAL: {
|
|
|
|
scope_name = L"local";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ENV_GLOBAL: {
|
|
|
|
scope_name = L"global";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ENV_UNIVERSAL: {
|
|
|
|
scope_name = L"universal";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
DIE("invalid scope");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (env_exist(var_name, scope)) {
|
|
|
|
const env_var_t evar = env_get_string(var_name, scope | ENV_EXPORT | ENV_USER);
|
|
|
|
const wchar_t *exportv = evar.missing() ? _(L"unexported") : _(L"exported");
|
|
|
|
|
|
|
|
const env_var_t var = env_get_string(var_name, scope | ENV_USER);
|
|
|
|
wcstring_list_t result;
|
|
|
|
if (!var.empty()) tokenize_variable_array(var, result);
|
|
|
|
|
|
|
|
streams.out.append_format(_(L"$%ls: set in %ls scope, %ls, with %d elements\n"), var_name,
|
|
|
|
scope_name, exportv, result.size());
|
|
|
|
for (size_t i = 0; i < result.size(); i++) {
|
|
|
|
if (result.size() > 100) {
|
|
|
|
if (i == 50) streams.out.append(L"...\n");
|
|
|
|
if (i >= 50 && i < result.size() - 50) continue;
|
|
|
|
}
|
|
|
|
const wcstring value = result[i];
|
|
|
|
const wcstring escaped_val =
|
|
|
|
escape_string(value.c_str(), ESCAPE_NO_QUOTED, STRING_STYLE_SCRIPT);
|
|
|
|
streams.out.append_format(_(L"$%ls[%d]: length=%d value=|%ls|\n"), var_name, i,
|
|
|
|
value.size(), escaped_val.c_str());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
streams.out.append_format(_(L"$%ls: not set in %ls scope\n"), var_name, scope_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Show mode. Show information about the named variable(s).
|
|
|
|
static int builtin_set_show(const wchar_t *cmd, set_cmd_opts_t &opts, int argc, wchar_t **argv,
|
|
|
|
parser_t &parser, io_streams_t &streams) {
|
|
|
|
UNUSED(opts);
|
|
|
|
|
|
|
|
if (argc == 0) { // show all vars
|
|
|
|
wcstring_list_t names = env_get_names(ENV_USER);
|
|
|
|
sort(names.begin(), names.end());
|
|
|
|
for (auto it : names) {
|
|
|
|
show_scope(it.c_str(), ENV_LOCAL, streams);
|
|
|
|
show_scope(it.c_str(), ENV_GLOBAL, streams);
|
|
|
|
show_scope(it.c_str(), ENV_UNIVERSAL, streams);
|
|
|
|
streams.out.push_back(L'\n');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (int i = 0; i < argc; i++) {
|
|
|
|
wchar_t *arg = argv[i];
|
|
|
|
|
|
|
|
if (!valid_var_name(arg)) {
|
|
|
|
streams.err.append_format(_(L"$%ls: invalid var name\n"), arg);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wcschr(arg, L'[')) {
|
|
|
|
streams.err.append_format(
|
|
|
|
_(L"%ls: `set --show` does not allow slices with the var names\n"), cmd);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_CMD_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
show_scope(arg, ENV_LOCAL, streams);
|
|
|
|
show_scope(arg, ENV_GLOBAL, streams);
|
|
|
|
show_scope(arg, ENV_UNIVERSAL, streams);
|
|
|
|
streams.out.push_back(L'\n');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Erase a variable.
|
|
|
|
static int builtin_set_erase(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
|
|
|
wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
|
|
|
if (argc != 1) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_ARG_COUNT2, cmd, L"--erase", 1, argc);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_CMD_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
int scope = compute_scope(opts); // calculate the variable scope based on the provided options
|
|
|
|
wchar_t *dest = argv[0];
|
|
|
|
|
|
|
|
std::vector<long> indexes;
|
|
|
|
int idx_count = parse_index(indexes, dest, scope, streams);
|
|
|
|
if (idx_count == -1) {
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_CMD_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
int retval;
|
|
|
|
if (!valid_var_name(dest)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_VARNAME, cmd, dest);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idx_count == 0) { // unset the var
|
|
|
|
retval = env_remove(dest, scope);
|
|
|
|
} else { // remove just the specified indexes of the var
|
|
|
|
const env_var_t dest_var = env_get_string(dest, scope);
|
|
|
|
if (dest_var.missing()) return STATUS_CMD_ERROR;
|
|
|
|
wcstring_list_t result;
|
|
|
|
tokenize_variable_array(dest_var, result);
|
|
|
|
erase_values(result, indexes);
|
|
|
|
retval = my_env_set(cmd, dest, result, scope, streams);
|
|
|
|
}
|
2017-07-23 04:15:20 +00:00
|
|
|
|
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
2017-07-24 18:42:54 +00:00
|
|
|
return check_global_scope_exists(cmd, opts, dest, streams);
|
|
|
|
}
|
2017-07-23 04:15:20 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// This handles the more difficult case of setting individual slices of a var.
|
|
|
|
static int set_slices(const wchar_t *cmd, set_cmd_opts_t &opts, const wchar_t *dest,
|
|
|
|
std::vector<long> &indexes, int argc, wchar_t **argv, parser_t &parser,
|
|
|
|
io_streams_t &streams) {
|
|
|
|
UNUSED(parser);
|
|
|
|
|
|
|
|
if (indexes.size() != static_cast<size_t>(argc)) {
|
|
|
|
streams.err.append_format(BUILTIN_SET_MISMATCHED_ARGS, cmd, indexes.size(), argc);
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
int scope = compute_scope(opts); // calculate the variable scope based on the provided options
|
|
|
|
wcstring_list_t result;
|
|
|
|
const env_var_t dest_str = env_get_string(dest, scope);
|
|
|
|
if (!dest_str.missing()) tokenize_variable_array(dest_str, result);
|
|
|
|
|
|
|
|
// Slice indexes have been calculated, do the actual work.
|
|
|
|
wcstring_list_t new_values;
|
|
|
|
for (int i = 0; i < argc; i++) new_values.push_back(argv[i]);
|
|
|
|
|
|
|
|
int retval = update_values(result, indexes, new_values);
|
|
|
|
if (retval != STATUS_CMD_OK) {
|
|
|
|
streams.err.append_format(BUILTIN_SET_ARRAY_BOUNDS_ERR, cmd);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
retval = my_env_set(cmd, dest, result, scope, streams);
|
2017-07-23 04:15:20 +00:00
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
2017-07-24 18:42:54 +00:00
|
|
|
return check_global_scope_exists(cmd, opts, dest, streams);
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// Set a variable.
|
|
|
|
static int builtin_set_set(const wchar_t *cmd, set_cmd_opts_t &opts, int argc,
|
|
|
|
wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
|
|
|
if (argc == 0) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_MIN_ARG_COUNT1, cmd, 1);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
int retval;
|
2017-07-23 04:15:20 +00:00
|
|
|
int scope = compute_scope(opts); // calculate the variable scope based on the provided options
|
2017-07-24 18:42:54 +00:00
|
|
|
wchar_t *dest = argv[0];
|
|
|
|
argv++;
|
|
|
|
argc--;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
std::vector<long> indexes;
|
|
|
|
int idx_count = parse_index(indexes, dest, scope, streams);
|
|
|
|
if (idx_count == -1) {
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-19 00:30:30 +00:00
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-04-20 06:43:02 +00:00
|
|
|
if (!valid_var_name(dest)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_VARNAME, cmd, dest);
|
2017-07-23 04:15:20 +00:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 04:35:41 +00:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-18 10:23:22 +00:00
|
|
|
}
|
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (idx_count != 0) {
|
|
|
|
// Handle the uncommon case of setting specific slices of a var.
|
|
|
|
return set_slices(cmd, opts, dest, indexes, argc, argv, parser, streams);
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
// This is the simple, common, case. Set the var to the specified values.
|
|
|
|
wcstring_list_t values;
|
|
|
|
for (int i = 0; i < argc; i++) values.push_back(argv[i]);
|
|
|
|
retval = my_env_set(cmd, dest, values, scope, streams);
|
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
return check_global_scope_exists(cmd, opts, dest, streams);
|
|
|
|
}
|
2014-07-12 22:43:32 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
/// The set builtin creates, updates, and erases (removes, deletes) variables.
|
|
|
|
int builtin_set(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
|
|
|
const int incoming_exit_status = proc_get_last_status();
|
|
|
|
wchar_t *cmd = argv[0];
|
|
|
|
int argc = builtin_count_args(argv);
|
|
|
|
set_cmd_opts_t opts;
|
2012-11-19 00:30:30 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
int optind;
|
|
|
|
int retval = parse_cmd_opts(opts, &optind, argc, argv, parser, streams);
|
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
argv += optind;
|
|
|
|
argc -= optind;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (opts.print_help) {
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.out);
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
retval = validate_cmd_opts(cmd, opts, argc, parser, streams);
|
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2017-07-24 18:42:54 +00:00
|
|
|
if (opts.query) {
|
|
|
|
retval = builtin_set_query(cmd, opts, argc, argv, parser, streams);
|
|
|
|
} else if (opts.erase) {
|
|
|
|
retval = builtin_set_erase(cmd, opts, argc, argv, parser, streams);
|
|
|
|
} else if (opts.list) { // explicit list the vars we know about
|
|
|
|
retval = builtin_set_list(cmd, opts, argc, argv, parser, streams);
|
2017-08-02 03:58:40 +00:00
|
|
|
} else if (opts.show) {
|
|
|
|
retval = builtin_set_show(cmd, opts, argc, argv, parser, streams);
|
2017-07-24 18:42:54 +00:00
|
|
|
} else if (argc == 0) { // implicit list the vars we know about
|
|
|
|
retval = builtin_set_list(cmd, opts, argc, argv, parser, streams);
|
2016-04-20 02:49:15 +00:00
|
|
|
} else {
|
2017-07-24 18:42:54 +00:00
|
|
|
retval = builtin_set_set(cmd, opts, argc, argv, parser, streams);
|
2014-10-06 07:10:56 +00:00
|
|
|
}
|
|
|
|
|
2017-07-23 04:15:20 +00:00
|
|
|
if (retval == STATUS_CMD_OK && opts.preserve_failure_exit_status) retval = incoming_exit_status;
|
|
|
|
return retval;
|
2005-09-20 13:26:39 +00:00
|
|
|
}
|