mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-16 15:04:05 +00:00
Allow to omit indices in index range expansions
Missing range limits in, say $PATH[..] default to the first/last element, just like Python/Go/Rust slices.
This commit is contained in:
parent
4c66e69cd9
commit
be06f842a2
4 changed files with 68 additions and 9 deletions
|
@ -5,6 +5,7 @@
|
||||||
### Syntax changes and new commands
|
### Syntax changes and new commands
|
||||||
|
|
||||||
### Scripting improvements
|
### Scripting improvements
|
||||||
|
- Range limits in index range expansions like `$x[$start..$end]` may be omitted: `$start` and `$end` default to 1 and -1 (the last item) respectively.
|
||||||
|
|
||||||
### Interactive improvements
|
### Interactive improvements
|
||||||
|
|
||||||
|
|
|
@ -661,6 +661,8 @@ Sequences of elements can be written with the range operator '``..``'. A range '
|
||||||
|
|
||||||
If the end is smaller than the start, or the start is larger than the end, range expansion will go in reverse. This is unless exactly one of the given indices is negative, so the direction doesn't change if the list has fewer elements than expected.
|
If the end is smaller than the start, or the start is larger than the end, range expansion will go in reverse. This is unless exactly one of the given indices is negative, so the direction doesn't change if the list has fewer elements than expected.
|
||||||
|
|
||||||
|
A missing starting index in a range defaults to 1. This is allowed if the range is the first index expression of the sequence. Similarly, a missing ending index, defaulting to -1 is allowed for the last index range in the sequence.
|
||||||
|
|
||||||
Some examples::
|
Some examples::
|
||||||
|
|
||||||
|
|
||||||
|
@ -672,6 +674,9 @@ Some examples::
|
||||||
# Uses elements from 2 to 5
|
# Uses elements from 2 to 5
|
||||||
# Output is: 2 3 4 5
|
# Output is: 2 3 4 5
|
||||||
|
|
||||||
|
echo (seq 10)[7..]
|
||||||
|
# Prints: 7 8 9 10
|
||||||
|
|
||||||
# Use overlapping ranges:
|
# Use overlapping ranges:
|
||||||
echo (seq 10)[2..5 1..3]
|
echo (seq 10)[2..5 1..3]
|
||||||
# Takes elements from 2 to 5 and then elements from 1 to 3
|
# Takes elements from 2 to 5 and then elements from 1 to 3
|
||||||
|
|
|
@ -188,12 +188,21 @@ static size_t parse_slice(const wchar_t *in, wchar_t **end_ptr, std::vector<long
|
||||||
}
|
}
|
||||||
|
|
||||||
const wchar_t *end;
|
const wchar_t *end;
|
||||||
long tmp = fish_wcstol(&in[pos], &end);
|
long tmp;
|
||||||
// We don't test `*end` as is typically done because we expect it to not be the null char.
|
if (idx.empty() && in[pos] == L'.' && in[pos + 1] == L'.') {
|
||||||
// Ignore the case of errno==-1 because it means the end char wasn't the null char.
|
// If we are at the first index expression, a missing start index means the range starts
|
||||||
|
// at the first item.
|
||||||
|
tmp = 1; // first index
|
||||||
|
end = &in[pos];
|
||||||
|
} else {
|
||||||
|
tmp = fish_wcstol(&in[pos], &end);
|
||||||
if (errno > 0) {
|
if (errno > 0) {
|
||||||
|
// We don't test `*end` as is typically done because we expect it to not be the null
|
||||||
|
// char. Ignore the case of errno==-1 because it means the end char wasn't the null
|
||||||
|
// char.
|
||||||
return pos;
|
return pos;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
long i1 = tmp > -1 ? tmp : size + tmp + 1;
|
long i1 = tmp > -1 ? tmp : size + tmp + 1;
|
||||||
pos = end - in;
|
pos = end - in;
|
||||||
|
@ -201,12 +210,21 @@ static size_t parse_slice(const wchar_t *in, wchar_t **end_ptr, std::vector<long
|
||||||
if (in[pos] == L'.' && in[pos + 1] == L'.') {
|
if (in[pos] == L'.' && in[pos + 1] == L'.') {
|
||||||
pos += 2;
|
pos += 2;
|
||||||
while (in[pos] == INTERNAL_SEPARATOR) pos++;
|
while (in[pos] == INTERNAL_SEPARATOR) pos++;
|
||||||
|
while (iswspace(in[pos])) pos++; // Allow the space in "[.. ]".
|
||||||
|
|
||||||
long tmp1 = fish_wcstol(&in[pos], &end);
|
long tmp1;
|
||||||
|
// Check if we are at the last index range expression, a missing end index means the
|
||||||
|
// range spans until the last item.
|
||||||
|
if (in[pos] == L']') {
|
||||||
|
tmp1 = -1; // last index
|
||||||
|
end = &in[pos];
|
||||||
|
} else {
|
||||||
|
tmp1 = fish_wcstol(&in[pos], &end);
|
||||||
// Ignore the case of errno==-1 because it means the end char wasn't the null char.
|
// Ignore the case of errno==-1 because it means the end char wasn't the null char.
|
||||||
if (errno > 0) {
|
if (errno > 0) {
|
||||||
return pos;
|
return pos;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
pos = end - in;
|
pos = end - in;
|
||||||
|
|
||||||
long i2 = tmp1 > -1 ? tmp1 : size + tmp1 + 1;
|
long i2 = tmp1 > -1 ? tmp1 : size + tmp1 + 1;
|
||||||
|
|
|
@ -3,6 +3,8 @@ set n 10
|
||||||
set test (seq $n)
|
set test (seq $n)
|
||||||
echo $test[1..$n] # normal range
|
echo $test[1..$n] # normal range
|
||||||
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
||||||
|
echo $test[1 .. 2] # spaces are allowed
|
||||||
|
#CHECK: 1 2
|
||||||
echo $test[$n..1] # inverted range
|
echo $test[$n..1] # inverted range
|
||||||
#CHECK: 10 9 8 7 6 5 4 3 2 1
|
#CHECK: 10 9 8 7 6 5 4 3 2 1
|
||||||
echo $test[2..5 8..6] # several ranges
|
echo $test[2..5 8..6] # several ranges
|
||||||
|
@ -31,3 +33,36 @@ echo $test[(count $test)..1]
|
||||||
#CHECK: 10 9 8 7 6 5 4 3 2 1
|
#CHECK: 10 9 8 7 6 5 4 3 2 1
|
||||||
echo $test[1..(count $test)]
|
echo $test[1..(count $test)]
|
||||||
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
||||||
|
|
||||||
|
echo $test[ .. ]
|
||||||
|
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
||||||
|
echo $test[ ..3]
|
||||||
|
#CHECK: 1 2 3
|
||||||
|
echo $test[8.. ]
|
||||||
|
#CHECK: 8 9 10
|
||||||
|
echo $test[..2 5]
|
||||||
|
# CHECK: 1 2 5
|
||||||
|
echo $test[2 9..]
|
||||||
|
# CHECK: 2 9 10
|
||||||
|
|
||||||
|
# missing start, cannot use implied range
|
||||||
|
echo $test[1..2..]
|
||||||
|
#CHECKERR: {{.*}}: Invalid index value
|
||||||
|
#CHECKERR: echo $test[1..2..]
|
||||||
|
#CHECKERR: ^
|
||||||
|
echo $test[..1..2]
|
||||||
|
#CHECKERR: {{.*}}: Invalid index value
|
||||||
|
#CHECKERR: echo $test[..1..2]
|
||||||
|
#CHECKERR: ^
|
||||||
|
|
||||||
|
set -l empty
|
||||||
|
echo $test[ $empty..]
|
||||||
|
#CHECK:
|
||||||
|
echo $test[.."$empty"]
|
||||||
|
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
||||||
|
echo $test["$empty"..]
|
||||||
|
#CHECK: 1 2 3 4 5 6 7 8 9 10
|
||||||
|
echo $test[ (true)..3]
|
||||||
|
#CHECK:
|
||||||
|
echo $test[ (string join \n 1 2 3)..3 ]
|
||||||
|
#CHECK: 1 2 3 2 3 3
|
||||||
|
|
Loading…
Reference in a new issue