2016-05-03 00:22:44 +00:00
|
|
|
// Functions having to do with parser keywords, like testing if a function is a block command.
|
2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2017-04-05 04:28:57 +00:00
|
|
|
#include <string>
|
2018-11-02 11:28:09 +00:00
|
|
|
#include <unordered_set>
|
2017-04-05 04:28:57 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2016-05-27 21:41:16 +00:00
|
|
|
#include "parser_keywords.h"
|
2007-04-22 09:50:26 +00:00
|
|
|
|
2019-04-04 01:38:29 +00:00
|
|
|
typedef std::unordered_set<wcstring> string_set_t;
|
|
|
|
|
2019-05-05 10:09:25 +00:00
|
|
|
static const wcstring skip_keywords[]{
|
|
|
|
L"else",
|
|
|
|
L"begin",
|
2019-04-04 01:38:29 +00:00
|
|
|
};
|
|
|
|
|
2019-05-05 10:09:25 +00:00
|
|
|
static const wcstring subcommand_keywords[]{L"command", L"builtin", L"while", L"exec",
|
|
|
|
L"if", L"and", L"or", L"not"};
|
2019-04-04 01:38:29 +00:00
|
|
|
|
2019-05-05 10:09:25 +00:00
|
|
|
static const string_set_t block_keywords = {L"for", L"while", L"if",
|
|
|
|
L"function", L"switch", L"begin"};
|
2019-04-04 01:38:29 +00:00
|
|
|
|
2019-05-05 10:09:25 +00:00
|
|
|
static const wcstring reserved_keywords[] = {L"end", L"case", L"else", L"return",
|
|
|
|
L"continue", L"break", L"argparse", L"read",
|
|
|
|
L"set", L"status", L"test", L"["};
|
2019-04-04 01:38:29 +00:00
|
|
|
|
|
|
|
// The lists above are purposely implemented separately from the logic below, so that future
|
|
|
|
// maintainers may assume the contents of the list based off their names, and not off what the
|
|
|
|
// functions below require them to contain.
|
|
|
|
|
|
|
|
static size_t list_max_length(const string_set_t &list) {
|
|
|
|
size_t result = 0;
|
2019-05-05 10:09:25 +00:00
|
|
|
for (const auto &w : list) {
|
2019-04-04 01:38:29 +00:00
|
|
|
if (w.length() > result) {
|
|
|
|
result = w.length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
bool parser_keywords_skip_arguments(const wcstring &cmd) {
|
2019-04-04 01:38:29 +00:00
|
|
|
return cmd == skip_keywords[0] || cmd == skip_keywords[1];
|
2007-04-22 09:50:26 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
bool parser_keywords_is_subcommand(const wcstring &cmd) {
|
2019-05-05 10:09:25 +00:00
|
|
|
const static string_set_t search_list = ([]() {
|
2019-04-04 01:38:29 +00:00
|
|
|
string_set_t results;
|
|
|
|
results.insert(std::begin(subcommand_keywords), std::end(subcommand_keywords));
|
|
|
|
results.insert(std::begin(skip_keywords), std::end(skip_keywords));
|
|
|
|
return results;
|
|
|
|
})();
|
|
|
|
|
|
|
|
const static auto max_len = list_max_length(search_list);
|
|
|
|
const static auto not_found = search_list.end();
|
|
|
|
|
|
|
|
// Everything above is executed only at startup, this is the actual optimized search routine:
|
|
|
|
return cmd.length() <= max_len && search_list.find(cmd) != not_found;
|
2007-04-22 09:50:26 +00:00
|
|
|
}
|
|
|
|
|
2019-04-04 01:38:29 +00:00
|
|
|
bool parser_keywords_is_block(const wcstring &word) {
|
|
|
|
const static auto max_len = list_max_length(block_keywords);
|
|
|
|
const static auto not_found = block_keywords.end();
|
|
|
|
|
|
|
|
// Everything above is executed only at startup, this is the actual optimized search routine:
|
|
|
|
return word.length() <= max_len && block_keywords.find(word) != not_found;
|
|
|
|
}
|
2007-04-22 09:50:26 +00:00
|
|
|
|
2016-05-03 00:22:44 +00:00
|
|
|
bool parser_keywords_is_reserved(const wcstring &word) {
|
2019-05-05 10:09:25 +00:00
|
|
|
const static string_set_t search_list = ([]() {
|
2019-04-04 01:38:29 +00:00
|
|
|
string_set_t results;
|
|
|
|
results.insert(std::begin(subcommand_keywords), std::end(subcommand_keywords));
|
|
|
|
results.insert(std::begin(skip_keywords), std::end(skip_keywords));
|
|
|
|
results.insert(std::begin(block_keywords), std::end(block_keywords));
|
|
|
|
results.insert(std::begin(reserved_keywords), std::end(reserved_keywords));
|
|
|
|
return results;
|
|
|
|
})();
|
|
|
|
const static auto max_len = list_max_length(search_list);
|
|
|
|
const static auto not_found = search_list.end();
|
|
|
|
|
|
|
|
// Everything above is executed only at startup, this is the actual optimized search routine:
|
|
|
|
return word.length() <= max_len && search_list.find(word) != not_found;
|
2007-04-22 09:50:26 +00:00
|
|
|
}
|