mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-14 14:03:58 +00:00
Fix builtin read crash with negative nchars
Also make it simpler by just passing it along as a usize
This commit is contained in:
parent
8667ed5c17
commit
9a2729d298
3 changed files with 26 additions and 25 deletions
|
@ -49,7 +49,7 @@ struct Options {
|
||||||
silent: bool,
|
silent: bool,
|
||||||
split_null: bool,
|
split_null: bool,
|
||||||
to_stdout: bool,
|
to_stdout: bool,
|
||||||
nchars: i32,
|
nchars: usize,
|
||||||
one_line: bool,
|
one_line: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -136,18 +136,17 @@ fn parse_cmd_opts(
|
||||||
}
|
}
|
||||||
'n' => {
|
'n' => {
|
||||||
opts.nchars = match fish_wcstoi(w.woptarg.unwrap()) {
|
opts.nchars = match fish_wcstoi(w.woptarg.unwrap()) {
|
||||||
Ok(n) => n,
|
Ok(n) if n >= 0 => n.try_into().unwrap(),
|
||||||
Err(err) => {
|
Err(wutil::Error::Overflow) => {
|
||||||
if err == wutil::Error::Overflow {
|
streams.err.append(wgettext_fmt!(
|
||||||
streams.err.append(wgettext_fmt!(
|
"%ls: Argument '%ls' is out of range\n",
|
||||||
"%ls: Argument '%ls' is out of range\n",
|
cmd,
|
||||||
cmd,
|
w.woptarg.unwrap()
|
||||||
w.woptarg.unwrap()
|
));
|
||||||
));
|
builtin_print_error_trailer(parser, streams.err, cmd);
|
||||||
builtin_print_error_trailer(parser, streams.err, cmd);
|
return Err(STATUS_INVALID_ARGS);
|
||||||
return Err(STATUS_INVALID_ARGS);
|
}
|
||||||
}
|
_ => {
|
||||||
|
|
||||||
streams.err.append(wgettext_fmt!(
|
streams.err.append(wgettext_fmt!(
|
||||||
BUILTIN_ERR_NOT_NUMBER,
|
BUILTIN_ERR_NOT_NUMBER,
|
||||||
cmd,
|
cmd,
|
||||||
|
@ -210,7 +209,7 @@ fn parse_cmd_opts(
|
||||||
fn read_interactive(
|
fn read_interactive(
|
||||||
parser: &Parser,
|
parser: &Parser,
|
||||||
buff: &mut WString,
|
buff: &mut WString,
|
||||||
nchars: i32,
|
nchars: usize,
|
||||||
shell: bool,
|
shell: bool,
|
||||||
silent: bool,
|
silent: bool,
|
||||||
prompt: &wstr,
|
prompt: &wstr,
|
||||||
|
@ -253,12 +252,12 @@ fn read_interactive(
|
||||||
};
|
};
|
||||||
if let Some(line) = mline {
|
if let Some(line) = mline {
|
||||||
*buff = line;
|
*buff = line;
|
||||||
if nchars > 0 && usize::try_from(nchars).unwrap() < buff.len() {
|
if nchars > 0 && nchars < buff.len() {
|
||||||
// Line may be longer than nchars if a keybinding used `commandline -i`
|
// Line may be longer than nchars if a keybinding used `commandline -i`
|
||||||
// note: we're deliberately throwing away the tail of the commandline.
|
// note: we're deliberately throwing away the tail of the commandline.
|
||||||
// It shouldn't be unread because it was produced with `commandline -i`,
|
// It shouldn't be unread because it was produced with `commandline -i`,
|
||||||
// not typed.
|
// not typed.
|
||||||
buff.truncate(usize::try_from(nchars).unwrap());
|
buff.truncate(nchars);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
exit_res = STATUS_CMD_ERROR;
|
exit_res = STATUS_CMD_ERROR;
|
||||||
|
@ -339,7 +338,7 @@ fn read_in_chunks(fd: RawFd, buff: &mut WString, split_null: bool, do_seek: bool
|
||||||
fn read_one_char_at_a_time(
|
fn read_one_char_at_a_time(
|
||||||
fd: RawFd,
|
fd: RawFd,
|
||||||
buff: &mut WString,
|
buff: &mut WString,
|
||||||
nchars: i32,
|
nchars: usize,
|
||||||
split_null: bool,
|
split_null: bool,
|
||||||
) -> Option<c_int> {
|
) -> Option<c_int> {
|
||||||
let mut exit_res = STATUS_CMD_OK;
|
let mut exit_res = STATUS_CMD_OK;
|
||||||
|
@ -398,7 +397,7 @@ fn read_one_char_at_a_time(
|
||||||
}
|
}
|
||||||
|
|
||||||
buff.push(res);
|
buff.push(res);
|
||||||
if nchars > 0 && usize::try_from(nchars).unwrap() <= buff.len() {
|
if nchars > 0 && nchars <= buff.len() {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -890,13 +890,8 @@ pub fn reader_reading_interrupted() -> i32 {
|
||||||
/// characters even if a full line has not yet been read. Note: the returned value may be longer
|
/// characters even if a full line has not yet been read. Note: the returned value may be longer
|
||||||
/// than nchars if a single keypress resulted in multiple characters being inserted into the
|
/// than nchars if a single keypress resulted in multiple characters being inserted into the
|
||||||
/// commandline.
|
/// commandline.
|
||||||
pub fn reader_readline(nchars: i32) -> Option<WString> {
|
pub fn reader_readline(nchars: usize) -> Option<WString> {
|
||||||
let nchars = usize::try_from(nchars).unwrap();
|
let nchars = NonZeroUsize::try_from(nchars).ok();
|
||||||
let nchars = if nchars == 0 {
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
Some(NonZeroUsize::try_from(nchars).unwrap())
|
|
||||||
};
|
|
||||||
let data = current_data().unwrap();
|
let data = current_data().unwrap();
|
||||||
// Apply any outstanding commandline changes (#8633).
|
// Apply any outstanding commandline changes (#8633).
|
||||||
data.apply_commandline_state_changes();
|
data.apply_commandline_state_changes();
|
||||||
|
|
|
@ -393,3 +393,10 @@ echo read $status
|
||||||
echo ' foo' | read -n 1 -la var
|
echo ' foo' | read -n 1 -la var
|
||||||
set -S var
|
set -S var
|
||||||
#CHECK: $var: set in local scope, unexported, with 0 elements
|
#CHECK: $var: set in local scope, unexported, with 0 elements
|
||||||
|
|
||||||
|
echo foo | read -n -1
|
||||||
|
# CHECKERR: read: -1: invalid integer
|
||||||
|
# CHECKERR: {{.*}}read.fish (line {{\d+}}):
|
||||||
|
# CHECKERR: echo foo | read -n -1
|
||||||
|
# CHECKERR: ^
|
||||||
|
# CHECKERR: (Type 'help read' for related documentation)
|
||||||
|
|
Loading…
Reference in a new issue