2018-04-24 19:09:13 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
|
|
|
#include "future_feature_flags.h"
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include <cwchar>
|
|
|
|
|
2020-01-15 21:16:43 +00:00
|
|
|
#include "wcstringutil.h"
|
|
|
|
|
2021-04-07 14:41:25 +00:00
|
|
|
features_t::features_t() {
|
|
|
|
for (const metadata_t &md : metadata) {
|
|
|
|
this->set(md.flag, md.default_value);
|
|
|
|
}
|
|
|
|
}
|
2018-04-24 19:09:13 +00:00
|
|
|
|
2020-09-13 00:35:21 +00:00
|
|
|
/// The set of features applying to this instance.
|
|
|
|
features_t features_t::global_features;
|
2018-04-24 19:09:13 +00:00
|
|
|
|
|
|
|
const features_t::metadata_t features_t::metadata[features_t::flag_count] = {
|
2022-05-30 23:14:32 +00:00
|
|
|
{stderr_nocaret, L"stderr-nocaret", L"3.0",
|
|
|
|
L"^ no longer redirects stderr (historical, can no longer be changed)", true,
|
|
|
|
true /* read-only */},
|
2022-04-08 15:03:53 +00:00
|
|
|
{qmark_noglob, L"qmark-noglob", L"3.0", L"? no longer globs", false, false},
|
2021-04-07 14:41:25 +00:00
|
|
|
{string_replace_backslash, L"regex-easyesc", L"3.1", L"string replace -r needs fewer \\'s",
|
2022-04-08 15:03:53 +00:00
|
|
|
true, false},
|
2021-06-29 17:30:27 +00:00
|
|
|
{ampersand_nobg_in_token, L"ampersand-nobg-in-token", L"3.4",
|
2022-04-08 15:45:41 +00:00
|
|
|
L"& only backgrounds if followed by a separator", true, false},
|
2018-04-24 19:09:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const struct features_t::metadata_t *features_t::metadata_for(const wchar_t *name) {
|
|
|
|
assert(name && "null flag name");
|
|
|
|
for (const auto &md : metadata) {
|
2019-03-12 21:06:01 +00:00
|
|
|
if (!std::wcscmp(name, md.name)) return &md;
|
2018-04-24 19:09:13 +00:00
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
2018-04-24 21:32:06 +00:00
|
|
|
|
|
|
|
void features_t::set_from_string(const wcstring &str) {
|
|
|
|
wcstring_list_t entries = split_string(str, L',');
|
|
|
|
const wchar_t *whitespace = L"\t\n\v\f\r ";
|
|
|
|
for (wcstring entry : entries) {
|
|
|
|
if (entry.empty()) continue;
|
|
|
|
|
|
|
|
// Trim leading and trailing whitespace
|
|
|
|
entry.erase(0, entry.find_first_not_of(whitespace));
|
|
|
|
entry.erase(entry.find_last_not_of(whitespace) + 1);
|
|
|
|
|
|
|
|
const wchar_t *name = entry.c_str();
|
|
|
|
bool value = true;
|
|
|
|
// A "no-" prefix inverts the sense.
|
|
|
|
if (string_prefixes_string(L"no-", name)) {
|
|
|
|
value = false;
|
2021-02-08 21:09:10 +00:00
|
|
|
name += const_strlen("no-");
|
2018-04-24 21:32:06 +00:00
|
|
|
}
|
|
|
|
// Look for a feature with this name. If we don't find it, assume it's a group name and set
|
|
|
|
// all features whose group contain it. Do nothing even if the string is unrecognized; this
|
|
|
|
// is to allow uniform invocations of fish (e.g. disable a feature that is only present in
|
|
|
|
// future versions).
|
|
|
|
// The special name 'all' may be used for those who like to live on the edge.
|
|
|
|
if (const metadata_t *md = metadata_for(name)) {
|
2022-04-08 15:03:53 +00:00
|
|
|
// Only change it if it's not read-only.
|
|
|
|
// Don't complain if it is, this is typically set from a variable.
|
|
|
|
if (!md->read_only) {
|
|
|
|
this->set(md->flag, value);
|
|
|
|
}
|
2018-04-24 21:32:06 +00:00
|
|
|
} else {
|
|
|
|
for (const metadata_t &md : metadata) {
|
2019-03-12 21:06:01 +00:00
|
|
|
if (std::wcsstr(md.groups, name) || !std::wcscmp(name, L"all")) {
|
2022-04-08 15:03:53 +00:00
|
|
|
if (!md.read_only) {
|
|
|
|
this->set(md.flag, value);
|
|
|
|
}
|
2018-04-24 21:32:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|