mirror of
https://github.com/fish-shell/fish-shell
synced 2024-12-31 23:28:45 +00:00
93fc0d06d4
"job_exit" events, despite their name, can only be created via the '--on-job-exit caller' misfeature of function. Rename it to make it clear that this event type is specifically for caller-exit.
451 lines
16 KiB
C++
451 lines
16 KiB
C++
// Implementation of the functions builtin.
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
#include "builtin_functions.h"
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <algorithm>
|
|
#include <cstddef>
|
|
#include <cwchar>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "builtin.h"
|
|
#include "common.h"
|
|
#include "complete.h"
|
|
#include "env.h"
|
|
#include "event.h"
|
|
#include "fallback.h" // IWYU pragma: keep
|
|
#include "function.h"
|
|
#include "highlight.h"
|
|
#include "io.h"
|
|
#include "parser.h"
|
|
#include "parser_keywords.h"
|
|
#include "proc.h"
|
|
#include "signal.h"
|
|
#include "wcstringutil.h"
|
|
#include "wgetopt.h"
|
|
#include "wutil.h" // IWYU pragma: keep
|
|
|
|
struct functions_cmd_opts_t {
|
|
bool print_help = false;
|
|
bool erase = false;
|
|
bool list = false;
|
|
bool show_hidden = false;
|
|
bool query = false;
|
|
bool copy = false;
|
|
bool report_metadata = false;
|
|
bool verbose = false;
|
|
bool handlers = false;
|
|
wchar_t *handlers_type = nullptr;
|
|
wchar_t *description = nullptr;
|
|
};
|
|
static const wchar_t *const short_options = L":HDacd:ehnqv";
|
|
static const struct woption long_options[] = {{L"erase", no_argument, nullptr, 'e'},
|
|
{L"description", required_argument, nullptr, 'd'},
|
|
{L"names", no_argument, nullptr, 'n'},
|
|
{L"all", no_argument, nullptr, 'a'},
|
|
{L"help", no_argument, nullptr, 'h'},
|
|
{L"query", no_argument, nullptr, 'q'},
|
|
{L"copy", no_argument, nullptr, 'c'},
|
|
{L"details", no_argument, nullptr, 'D'},
|
|
{L"verbose", no_argument, nullptr, 'v'},
|
|
{L"handlers", no_argument, nullptr, 'H'},
|
|
{L"handlers-type", required_argument, nullptr, 't'},
|
|
{nullptr, 0, nullptr, 0}};
|
|
|
|
static int parse_cmd_opts(functions_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, nullptr)) != -1) {
|
|
switch (opt) {
|
|
case 'v': {
|
|
opts.verbose = true;
|
|
break;
|
|
}
|
|
case 'e': {
|
|
opts.erase = true;
|
|
break;
|
|
}
|
|
case 'D': {
|
|
opts.report_metadata = true;
|
|
break;
|
|
}
|
|
case 'd': {
|
|
opts.description = w.woptarg;
|
|
break;
|
|
}
|
|
case 'n': {
|
|
opts.list = true;
|
|
break;
|
|
}
|
|
case 'a': {
|
|
opts.show_hidden = true;
|
|
break;
|
|
}
|
|
case 'h': {
|
|
opts.print_help = true;
|
|
break;
|
|
}
|
|
case 'q': {
|
|
opts.query = true;
|
|
break;
|
|
}
|
|
case 'c': {
|
|
opts.copy = true;
|
|
break;
|
|
}
|
|
case 'H': {
|
|
opts.handlers = true;
|
|
break;
|
|
}
|
|
case 't': {
|
|
opts.handlers_type = w.woptarg;
|
|
opts.handlers = 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;
|
|
}
|
|
|
|
/// Return a definition of the specified function. Used by the functions builtin.
|
|
static wcstring functions_def(const wcstring &name) {
|
|
assert(!name.empty() && "Empty name");
|
|
wcstring out;
|
|
wcstring desc, def;
|
|
function_get_desc(name, desc);
|
|
function_get_definition(name, def);
|
|
std::vector<std::shared_ptr<event_handler_t>> ev = event_get_function_handlers(name);
|
|
|
|
out.append(L"function ");
|
|
|
|
// Typically we prefer to specify the function name first, e.g. "function foo --description bar"
|
|
// But if the function name starts with a -, we'll need to output it after all the options.
|
|
bool defer_function_name = (name.at(0) == L'-');
|
|
if (!defer_function_name) {
|
|
out.append(escape_string(name, ESCAPE_ALL));
|
|
}
|
|
|
|
// Output wrap targets.
|
|
for (const wcstring &wrap : complete_get_wrap_targets(name)) {
|
|
out.append(L" --wraps=");
|
|
out.append(escape_string(wrap, ESCAPE_ALL));
|
|
}
|
|
|
|
if (!desc.empty()) {
|
|
wcstring esc_desc = escape_string(desc, ESCAPE_ALL);
|
|
out.append(L" --description ");
|
|
out.append(esc_desc);
|
|
}
|
|
|
|
auto props = function_get_properties(name);
|
|
assert(props && "Should have function properties");
|
|
if (!props->shadow_scope) {
|
|
out.append(L" --no-scope-shadowing");
|
|
}
|
|
|
|
for (const auto &next : ev) {
|
|
const event_description_t &d = next->desc;
|
|
switch (d.type) {
|
|
case event_type_t::signal: {
|
|
append_format(out, L" --on-signal %ls", sig2wcs(d.param1.signal));
|
|
break;
|
|
}
|
|
case event_type_t::variable: {
|
|
append_format(out, L" --on-variable %ls", d.str_param1.c_str());
|
|
break;
|
|
}
|
|
case event_type_t::exit: {
|
|
if (d.param1.pid > 0)
|
|
append_format(out, L" --on-process-exit %d", d.param1.pid);
|
|
else
|
|
append_format(out, L" --on-job-exit %d", -d.param1.pid);
|
|
break;
|
|
}
|
|
case event_type_t::caller_exit: {
|
|
append_format(out, L" --on-job-exit caller");
|
|
break;
|
|
}
|
|
case event_type_t::generic: {
|
|
append_format(out, L" --on-event %ls", d.str_param1.c_str());
|
|
break;
|
|
}
|
|
case event_type_t::any:
|
|
default: {
|
|
DIE("unexpected next->type");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
const wcstring_list_t &named = props->named_arguments;
|
|
if (!named.empty()) {
|
|
append_format(out, L" --argument");
|
|
for (const auto &name : named) {
|
|
append_format(out, L" %ls", name.c_str());
|
|
}
|
|
}
|
|
|
|
// Output the function name if we deferred it.
|
|
if (defer_function_name) {
|
|
out.append(L" -- ");
|
|
out.append(escape_string(name, ESCAPE_ALL));
|
|
}
|
|
|
|
// Output any inherited variables as `set -l` lines.
|
|
for (const auto &kv : props->inherit_vars) {
|
|
// We don't know what indentation style the function uses,
|
|
// so we do what fish_indent would.
|
|
append_format(out, L"\n set -l %ls", kv.first.c_str());
|
|
for (const auto &arg : kv.second) {
|
|
wcstring earg = escape_string(arg, ESCAPE_ALL);
|
|
out.push_back(L' ');
|
|
out.append(earg);
|
|
}
|
|
}
|
|
out.push_back('\n');
|
|
out.append(def);
|
|
|
|
// Append a newline before the 'end', unless there already is one there.
|
|
if (!string_suffixes_string(L"\n", def)) {
|
|
out.push_back(L'\n');
|
|
}
|
|
out.append(L"end\n");
|
|
return out;
|
|
}
|
|
|
|
static int report_function_metadata(const wchar_t *funcname, bool verbose, io_streams_t &streams,
|
|
parser_t &parser, bool metadata_as_comments) {
|
|
const wchar_t *path = L"n/a";
|
|
const wchar_t *autoloaded = L"n/a";
|
|
const wchar_t *shadows_scope = L"n/a";
|
|
wcstring description = L"n/a";
|
|
int line_number = 0;
|
|
|
|
if (function_exists(funcname, parser)) {
|
|
auto props = function_get_properties(funcname);
|
|
path = function_get_definition_file(funcname);
|
|
if (path) {
|
|
autoloaded = function_is_autoloaded(funcname) ? L"autoloaded" : L"not-autoloaded";
|
|
line_number = function_get_definition_lineno(funcname);
|
|
} else {
|
|
path = L"stdin";
|
|
}
|
|
shadows_scope = props->shadow_scope ? L"scope-shadowing" : L"no-scope-shadowing";
|
|
function_get_desc(funcname, description);
|
|
description = escape_string(description, ESCAPE_NO_QUOTED);
|
|
}
|
|
|
|
if (metadata_as_comments) {
|
|
if (std::wcscmp(path, L"stdin") != 0) {
|
|
wcstring comment;
|
|
append_format(comment, L"# Defined in %ls @ line %d\n", path, line_number);
|
|
if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) {
|
|
std::vector<highlight_spec_t> colors;
|
|
highlight_shell_no_io(
|
|
comment, colors, comment.size(),
|
|
operation_context_t{nullptr, env_stack_t::globals(), no_cancel});
|
|
streams.out.append(str2wcstring(colorize(comment, colors)));
|
|
} else {
|
|
streams.out.append(comment);
|
|
}
|
|
}
|
|
} else {
|
|
streams.out.append_format(L"%ls\n", path);
|
|
if (verbose) {
|
|
streams.out.append_format(L"%ls\n", autoloaded);
|
|
streams.out.append_format(L"%d\n", line_number);
|
|
streams.out.append_format(L"%ls\n", shadows_scope);
|
|
streams.out.append_format(L"%ls\n", description.c_str());
|
|
}
|
|
}
|
|
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
/// The functions builtin, used for listing and erasing functions.
|
|
int builtin_functions(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
|
const wchar_t *cmd = argv[0];
|
|
int argc = builtin_count_args(argv);
|
|
functions_cmd_opts_t opts;
|
|
|
|
int optind;
|
|
int retval = parse_cmd_opts(opts, &optind, argc, argv, parser, streams);
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
if (opts.print_help) {
|
|
builtin_print_help(parser, streams, cmd);
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
// Erase, desc, query, copy and list are mutually exclusive.
|
|
bool describe = opts.description != nullptr;
|
|
if (describe + opts.erase + opts.list + opts.query + opts.copy > 1) {
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, cmd);
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
|
|
if (opts.erase) {
|
|
for (int i = optind; i < argc; i++) function_remove(argv[i]);
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
if (opts.description) {
|
|
wchar_t *func;
|
|
|
|
if (argc - optind != 1) {
|
|
streams.err.append_format(_(L"%ls: Expected exactly one function name\n"), cmd);
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
|
|
func = argv[optind];
|
|
if (!function_exists(func, parser)) {
|
|
streams.err.append_format(_(L"%ls: Function '%ls' does not exist\n"), cmd, func);
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
function_set_desc(func, opts.description, parser);
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
if (opts.report_metadata) {
|
|
if (argc - optind != 1) {
|
|
streams.err.append_format(BUILTIN_ERR_ARG_COUNT2, cmd, argv[optind - 1], 1,
|
|
argc - optind);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
|
|
const wchar_t *funcname = argv[optind];
|
|
return report_function_metadata(funcname, opts.verbose, streams, parser, false);
|
|
}
|
|
|
|
if (opts.handlers) {
|
|
maybe_t<event_type_t> type_filter;
|
|
if (opts.handlers_type) {
|
|
type_filter = event_type_for_name(opts.handlers_type);
|
|
if (!type_filter) {
|
|
streams.err.append_format(_(L"%ls: Expected generic | variable | signal | exit | "
|
|
L"job-id for --handlers-type\n"),
|
|
cmd);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
}
|
|
event_print(streams, type_filter);
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
// If we query with no argument, just return false.
|
|
if (opts.query && argc == optind) {
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
if (opts.list || argc == optind) {
|
|
wcstring_list_t names = function_get_names(opts.show_hidden);
|
|
std::sort(names.begin(), names.end());
|
|
bool is_screen = !streams.out_is_redirected && isatty(STDOUT_FILENO);
|
|
if (is_screen) {
|
|
wcstring buff;
|
|
for (const auto &name : names) {
|
|
buff.append(name);
|
|
buff.append(L", ");
|
|
}
|
|
streams.out.append(reformat_for_screen(buff));
|
|
} else {
|
|
for (const auto &name : names) {
|
|
streams.out.append(name.c_str());
|
|
streams.out.append(L"\n");
|
|
}
|
|
}
|
|
|
|
return STATUS_CMD_OK;
|
|
}
|
|
|
|
if (opts.copy) {
|
|
wcstring current_func;
|
|
wcstring new_func;
|
|
|
|
if (argc - optind != 2) {
|
|
streams.err.append_format(_(L"%ls: Expected exactly two names (current function name, "
|
|
L"and new function name)\n"),
|
|
cmd);
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
current_func = argv[optind];
|
|
new_func = argv[optind + 1];
|
|
|
|
if (!function_exists(current_func, parser)) {
|
|
streams.err.append_format(_(L"%ls: Function '%ls' does not exist\n"), cmd,
|
|
current_func.c_str());
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
if (!valid_func_name(new_func) || parser_keywords_is_reserved(new_func)) {
|
|
streams.err.append_format(_(L"%ls: Illegal function name '%ls'\n"), cmd,
|
|
new_func.c_str());
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_INVALID_ARGS;
|
|
}
|
|
|
|
// Keep things simple: don't allow existing names to be copy targets.
|
|
if (function_exists(new_func, parser)) {
|
|
streams.err.append_format(
|
|
_(L"%ls: Function '%ls' already exists. Cannot create copy '%ls'\n"), cmd,
|
|
new_func.c_str(), current_func.c_str());
|
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
if (function_copy(current_func, new_func)) return STATUS_CMD_OK;
|
|
return STATUS_CMD_ERROR;
|
|
}
|
|
|
|
int res = STATUS_CMD_OK;
|
|
for (int i = optind; i < argc; i++) {
|
|
if (!function_exists(argv[i], parser)) {
|
|
res++;
|
|
} else {
|
|
if (!opts.query) {
|
|
if (i != optind) streams.out.append(L"\n");
|
|
const wchar_t *funcname = argv[optind];
|
|
report_function_metadata(funcname, opts.verbose, streams, parser, true);
|
|
wcstring def = functions_def(funcname);
|
|
|
|
if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) {
|
|
std::vector<highlight_spec_t> colors;
|
|
highlight_shell_no_io(def, colors, def.size(), operation_context_t::globals());
|
|
streams.out.append(str2wcstring(colorize(def, colors)));
|
|
} else {
|
|
streams.out.append(def);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|