Revert accidental merge of #7340

This reverts back to commit d8e2cac83e.
I accidentally did a 'git push' during code review.
This commit is contained in:
ridiculousfish 2020-09-19 19:31:30 -07:00
parent 32f8b0c531
commit 5c3571d626
4 changed files with 10 additions and 163 deletions

View file

@ -1,52 +0,0 @@
string-pad - pad characters before and after string
===================================================
Synopsis
--------
.. BEGIN SYNOPSIS
::
string pad [(-r | --right)] [(-c | --char) CHAR] [(-w | --width) INTEGER] [STRING...]
.. END SYNOPSIS
Description
-----------
.. BEGIN DESCRIPTION
``string pad`` pads specified character before and after the string for each STRING.
The default behaviour is left padding with spaces and default width is the length of string (hence, no padding).
If ``-r`` or ``--right`` is given, only pad after string.
The ``-c`` or ``--char`` switch causes the characters in CHAR to be padded instead of default whitespace character.
If ``-w`` or ``--width`` is given, pad the string to given length. Width less than the string length will result in an unchanged string.
Exit status: 0 if string was padded or haven't changed, or 1 otherwise.
.. END DESCRIPTION
Examples
--------
.. BEGIN EXAMPLES
::
>_ string pad -w 10 -c ' ' 'abc'
abc
>_ string pad --right --width 12 --char=z foo barbaz
foozzzzzzzzz
barbazzzzzzz
>_ string pad -w 6 --char=- foo | string pad --right -w 9 --char=-
---foo---
.. END EXAMPLES

View file

@ -15,7 +15,6 @@ Synopsis
string length [(-q | --quiet)] [STRING...] string length [(-q | --quiet)] [STRING...]
string lower [(-q | --quiet)] [STRING...] string lower [(-q | --quiet)] [STRING...]
string match [(-a | --all)] [(-e | --entire)] [(-i | --ignore-case)] [(-r | --regex)] [(-n | --index)] [(-q | --quiet)] [(-v | --invert)] PATTERN [STRING...] string match [(-a | --all)] [(-e | --entire)] [(-i | --ignore-case)] [(-r | --regex)] [(-n | --index)] [(-q | --quiet)] [(-v | --invert)] PATTERN [STRING...]
string pad [(-r | --right)] [(-c | --char) CHAR] [(-w | --width) INTEGER] [STRING...]
string repeat [(-n | --count) COUNT] [(-m | --max) MAX] [(-N | --no-newline)] [(-q | --quiet)] [STRING...] string repeat [(-n | --count) COUNT] [(-m | --max) MAX] [(-N | --no-newline)] [(-q | --quiet)] [STRING...]
string replace [(-a | --all)] [(-f | --filter)] [(-i | --ignore-case)] [(-r | --regex)] [(-q | --quiet)] PATTERN REPLACEMENT [STRING...] string replace [(-a | --all)] [(-f | --filter)] [(-i | --ignore-case)] [(-r | --regex)] [(-q | --quiet)] PATTERN REPLACEMENT [STRING...]
string split [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...] string split [(-m | --max) MAX] [(-n | --no-empty)] [(-q | --quiet)] [(-r | --right)] SEP [STRING...]
@ -142,21 +141,6 @@ Examples
:start-after: BEGIN EXAMPLES :start-after: BEGIN EXAMPLES
:end-before: END EXAMPLES :end-before: END EXAMPLES
"pad" subcommand
------------------
.. include:: string-pad.rst
:start-after: BEGIN SYNOPSIS
:end-before: END SYNOPSIS
.. include:: string-pad.rst
:start-after: BEGIN DESCRIPTION
:end-before: END DESCRIPTION
.. include:: string-pad.rst
:start-after: BEGIN EXAMPLES
:end-before: END EXAMPLES
"repeat" subcommand "repeat" subcommand
------------------- -------------------

View file

@ -135,8 +135,7 @@ class arg_iterator_t {
// valid and get the result of parsing the command for flags. // valid and get the result of parsing the command for flags.
using options_t = struct options_t { //!OCLINT(too many fields) using options_t = struct options_t { //!OCLINT(too many fields)
bool all_valid = false; bool all_valid = false;
bool char_to_pad_valid = false; bool chars_valid = false;
bool chars_to_trim_valid = false;
bool count_valid = false; bool count_valid = false;
bool entire_valid = false; bool entire_valid = false;
bool filter_valid = false; bool filter_valid = false;
@ -158,7 +157,6 @@ using options_t = struct options_t { //!OCLINT(too many fields)
bool no_trim_newlines_valid = false; bool no_trim_newlines_valid = false;
bool fields_valid = false; bool fields_valid = false;
bool allow_empty_valid = false; bool allow_empty_valid = false;
bool width_valid = false;
bool all = false; bool all = false;
bool entire = false; bool entire = false;
@ -181,9 +179,6 @@ using options_t = struct options_t { //!OCLINT(too many fields)
long max = 0; long max = 0;
long start = 0; long start = 0;
long end = 0; long end = 0;
size_t width = 0;
wchar_t char_to_pad = ' ';
std::vector<int> fields; std::vector<int> fields;
@ -247,16 +242,9 @@ static int handle_flag_a(wchar_t **argv, parser_t &parser, io_streams_t &streams
static int handle_flag_c(wchar_t **argv, parser_t &parser, io_streams_t &streams, static int handle_flag_c(wchar_t **argv, parser_t &parser, io_streams_t &streams,
const wgetopter_t &w, options_t *opts) { const wgetopter_t &w, options_t *opts) {
if (opts->chars_to_trim_valid) { if (opts->chars_valid) {
opts->chars_to_trim = w.woptarg; opts->chars_to_trim = w.woptarg;
return STATUS_CMD_OK; return STATUS_CMD_OK;
} else if (opts->char_to_pad_valid) {
if (wcslen(w.woptarg) != 1) {
string_error(streams, _(L"%ls: Padding should be a character '%ls'\n"), argv[0], w.woptarg);
return STATUS_INVALID_ARGS;
}
opts->char_to_pad = w.woptarg[0];
return STATUS_CMD_OK;
} }
string_unknown_option(parser, streams, argv[0], argv[w.woptind - 1]); string_unknown_option(parser, streams, argv[0], argv[w.woptind - 1]);
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
@ -463,33 +451,13 @@ static int handle_flag_v(wchar_t **argv, parser_t &parser, io_streams_t &streams
return STATUS_INVALID_ARGS; return STATUS_INVALID_ARGS;
} }
static int handle_flag_w(wchar_t **argv, parser_t &parser, io_streams_t &streams,
const wgetopter_t &w, options_t *opts) {
long width = 0;
if (opts->width_valid) {
width = fish_wcstol(w.woptarg);
if (width < 0) {
string_error(streams, _(L"%ls: Invalid width value '%ls'\n"), argv[0], w.woptarg);
return STATUS_INVALID_ARGS;
} else if (errno) {
string_error(streams, BUILTIN_ERR_NOT_NUMBER, argv[0], w.woptarg);
return STATUS_INVALID_ARGS;
}
opts->width = static_cast<size_t>(width);
return STATUS_CMD_OK;
}
string_unknown_option(parser, streams, argv[0], argv[w.woptind - 1]);
return STATUS_INVALID_ARGS;
}
/// This constructs the wgetopt() short options string based on which arguments are valid for the /// This constructs the wgetopt() short options string based on which arguments are valid for the
/// subcommand. We have to do this because many short flags have multiple meanings and may or may /// subcommand. We have to do this because many short flags have multiple meanings and may or may
/// not require an argument depending on the meaning. /// not require an argument depending on the meaning.
static wcstring construct_short_opts(options_t *opts) { //!OCLINT(high npath complexity) static wcstring construct_short_opts(options_t *opts) { //!OCLINT(high npath complexity)
wcstring short_opts(L":"); wcstring short_opts(L":");
if (opts->all_valid) short_opts.append(L"a"); if (opts->all_valid) short_opts.append(L"a");
if (opts->char_to_pad_valid) short_opts.append(L"c:"); if (opts->chars_valid) short_opts.append(L"c:");
if (opts->chars_to_trim_valid) short_opts.append(L"c:");
if (opts->count_valid) short_opts.append(L"n:"); if (opts->count_valid) short_opts.append(L"n:");
if (opts->entire_valid) short_opts.append(L"e"); if (opts->entire_valid) short_opts.append(L"e");
if (opts->filter_valid) short_opts.append(L"f"); if (opts->filter_valid) short_opts.append(L"f");
@ -510,7 +478,6 @@ static wcstring construct_short_opts(options_t *opts) { //!OCLINT(high npath co
if (opts->no_trim_newlines_valid) short_opts.append(L"N"); if (opts->no_trim_newlines_valid) short_opts.append(L"N");
if (opts->fields_valid) short_opts.append(L"f:"); if (opts->fields_valid) short_opts.append(L"f:");
if (opts->allow_empty_valid) short_opts.append(L"a"); if (opts->allow_empty_valid) short_opts.append(L"a");
if (opts->width_valid) short_opts.append(L"w:");
return short_opts; return short_opts;
} }
@ -540,14 +507,13 @@ static const struct woption long_options[] = {{L"all", no_argument, nullptr, 'a'
{L"no-trim-newlines", no_argument, nullptr, 'N'}, {L"no-trim-newlines", no_argument, nullptr, 'N'},
{L"fields", required_argument, nullptr, 'f'}, {L"fields", required_argument, nullptr, 'f'},
{L"allow-empty", no_argument, nullptr, 'a'}, {L"allow-empty", no_argument, nullptr, 'a'},
{L"width", required_argument, nullptr, 'w'},
{nullptr, 0, nullptr, 0}}; {nullptr, 0, nullptr, 0}};
static const std::unordered_map<char, decltype(*handle_flag_N)> flag_to_function = { static const std::unordered_map<char, decltype(*handle_flag_N)> flag_to_function = {
{'N', handle_flag_N}, {'a', handle_flag_a}, {'c', handle_flag_c}, {'e', handle_flag_e}, {'N', handle_flag_N}, {'a', handle_flag_a}, {'c', handle_flag_c}, {'e', handle_flag_e},
{'f', handle_flag_f}, {'i', handle_flag_i}, {'l', handle_flag_l}, {'m', handle_flag_m}, {'f', handle_flag_f}, {'i', handle_flag_i}, {'l', handle_flag_l}, {'m', handle_flag_m},
{'n', handle_flag_n}, {'q', handle_flag_q}, {'r', handle_flag_r}, {'s', handle_flag_s}, {'n', handle_flag_n}, {'q', handle_flag_q}, {'r', handle_flag_r}, {'s', handle_flag_s},
{'v', handle_flag_v}, {'w', handle_flag_w}, {1, handle_flag_1}}; {'v', handle_flag_v}, {1, handle_flag_1}};
/// Parse the arguments for flags recognized by a specific string subcommand. /// Parse the arguments for flags recognized by a specific string subcommand.
static int parse_opts(options_t *opts, int *optind, int n_req_args, int argc, wchar_t **argv, static int parse_opts(options_t *opts, int *optind, int n_req_args, int argc, wchar_t **argv,
@ -971,41 +937,6 @@ static int string_match(parser_t &parser, io_streams_t &streams, int argc, wchar
return matcher->match_count() > 0 ? STATUS_CMD_OK : STATUS_CMD_ERROR; return matcher->match_count() > 0 ? STATUS_CMD_OK : STATUS_CMD_ERROR;
} }
static int string_pad(parser_t &parser, io_streams_t &streams, int argc, wchar_t **argv) {
options_t opts;
opts.char_to_pad_valid = true;
opts.right_valid = true;
opts.width_valid = true;
int optind;
int retval = parse_opts(&opts, &optind, 0, argc, argv, parser, streams);
if (retval != STATUS_CMD_OK) return retval;
// Pad left by default
if (!opts.right) {
opts.left = true;
}
arg_iterator_t aiter(argv, optind, streams);
while (const wcstring *arg = aiter.nextstr()) {
wcstring padded = *arg;
if (opts.width >= padded.size()) {
size_t pad = opts.width - padded.size();
if (opts.left) {
padded.insert(0, pad, opts.char_to_pad);
}
if (opts.right) {
padded.append(pad, opts.char_to_pad);
}
}
if (!opts.quiet) {
streams.out.append(padded);
streams.out.append(L'\n');
}
}
return STATUS_CMD_OK;
}
class string_replacer_t { class string_replacer_t {
protected: protected:
const wchar_t *argv0; const wchar_t *argv0;
@ -1432,7 +1363,7 @@ static int string_sub(parser_t &parser, io_streams_t &streams, int argc, wchar_t
static int string_trim(parser_t &parser, io_streams_t &streams, int argc, wchar_t **argv) { static int string_trim(parser_t &parser, io_streams_t &streams, int argc, wchar_t **argv) {
options_t opts; options_t opts;
opts.chars_to_trim_valid = true; opts.chars_valid = true;
opts.left_valid = true; opts.left_valid = true;
opts.right_valid = true; opts.right_valid = true;
opts.quiet_valid = true; opts.quiet_valid = true;
@ -1517,7 +1448,6 @@ string_subcommands[] = {
{L"split", &string_split}, {L"split0", &string_split0}, {L"sub", &string_sub}, {L"split", &string_split}, {L"split0", &string_split0}, {L"sub", &string_sub},
{L"trim", &string_trim}, {L"lower", &string_lower}, {L"upper", &string_upper}, {L"trim", &string_trim}, {L"lower", &string_lower}, {L"upper", &string_upper},
{L"repeat", &string_repeat}, {L"unescape", &string_unescape}, {L"collect", &string_collect}, {L"repeat", &string_repeat}, {L"unescape", &string_unescape}, {L"collect", &string_collect},
{L"pad", &string_pad},
{nullptr, nullptr}}; {nullptr, nullptr}};
/// The string builtin, for manipulating strings. /// The string builtin, for manipulating strings.

View file

@ -42,21 +42,6 @@ string length "hello, world"
string length -q ""; and echo not zero length; or echo zero length string length -q ""; and echo not zero length; or echo zero length
# CHECK: zero length # CHECK: zero length
string pad foo
# CHECK: foo
string pad -r -w 4 foo
# CHECK: foo
string pad -r -w 7 -c '-' foo
# CHECK: foo----
string pad --width 7 -c '=' foo
# CHECK: ====foo
string pad --width 10 --right foo
# CHECK: foo
string sub --length 2 abcde string sub --length 2 abcde
# CHECK: ab # CHECK: ab
@ -207,7 +192,7 @@ string unescape --style=url (string escape --style=url 'a b#c"\'d')
# CHECK: a b#c"'d # CHECK: a b#c"'d
string unescape --style=url (string escape --style=url \na\nb%c~d\n) string unescape --style=url (string escape --style=url \na\nb%c~d\n)
# CHECK: # CHECK:
# CHECK: a # CHECK: a
# CHECK: b%c~d # CHECK: b%c~d
@ -275,7 +260,7 @@ string replace -a " " _ "spaces to underscores"
# CHECK: spaces_to_underscores # CHECK: spaces_to_underscores
string replace -r -a "[^\d.]+" " " "0 one two 3.14 four 5x" string replace -r -a "[^\d.]+" " " "0 one two 3.14 four 5x"
# CHECK: 0 3.14 5 # CHECK: 0 3.14 5
string replace -r "(\w+)\s+(\w+)" "\$2 \$1 \$\$" "left right" string replace -r "(\w+)\s+(\w+)" "\$2 \$1 \$\$" "left right"
# CHECK: right left $ # CHECK: right left $
@ -310,7 +295,7 @@ and echo Unexpected exit status at line (status --current-line-number)
# 'string match -r with empty capture groups' # 'string match -r with empty capture groups'
string match -r '^([ugoa]*)([=+-]?)([rwx]*)$' '=r' string match -r '^([ugoa]*)([=+-]?)([rwx]*)$' '=r'
#CHECK: =r #CHECK: =r
#CHECK: #CHECK:
#CHECK: = #CHECK: =
#CHECK: r #CHECK: r
@ -590,13 +575,13 @@ printf '[%s]\n' (string collect one\n\n two\n)
# CHECK: [two] # CHECK: [two]
printf '[%s]\n' (string collect -N one\n\n two\n) printf '[%s]\n' (string collect -N one\n\n two\n)
# CHECK: [one # CHECK: [one
# CHECK: # CHECK:
# CHECK: ] # CHECK: ]
# CHECK: [two # CHECK: [two
# CHECK: ] # CHECK: ]
printf '[%s]\n' (string collect --no-trim-newlines one\n\n two\n) printf '[%s]\n' (string collect --no-trim-newlines one\n\n two\n)
# CHECK: [one # CHECK: [one
# CHECK: # CHECK:
# CHECK: ] # CHECK: ]
# CHECK: [two # CHECK: [two
# CHECK: ] # CHECK: ]