iterator: add functionality to move indices
Extract the index acrobatics from the message store and move it to iterator package for re-use and to add unit tests. Signed-off-by: Koni Marti <koni.marti@gmail.com> Acked-by: Robin Jarry <robin@jarry.cc>
This commit is contained in:
parent
c5face0b6f
commit
206665a2d9
2 changed files with 184 additions and 0 deletions
51
lib/iterator/index.go
Normal file
51
lib/iterator/index.go
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
package iterator
|
||||||
|
|
||||||
|
// IndexProvider implements a subset of the Interator interface
|
||||||
|
type IndexProvider interface {
|
||||||
|
StartIndex() int
|
||||||
|
EndIndex() int
|
||||||
|
}
|
||||||
|
|
||||||
|
// FixBounds will force the index i to either its lower- or upper-bound value
|
||||||
|
// if out-of-bound
|
||||||
|
func FixBounds(i, lower, upper int) int {
|
||||||
|
switch {
|
||||||
|
case i > upper:
|
||||||
|
i = upper
|
||||||
|
case i < lower:
|
||||||
|
i = lower
|
||||||
|
}
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
// WrapBounds will wrap the index i around its upper- or lower-bound if
|
||||||
|
// out-of-bound
|
||||||
|
func WrapBounds(i, lower, upper int) int {
|
||||||
|
switch {
|
||||||
|
case i > upper:
|
||||||
|
i = lower + (i-upper-1)%upper
|
||||||
|
case i < lower:
|
||||||
|
i = upper - (lower-i-1)%upper
|
||||||
|
}
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoundsCheckFunc func(int, int, int) int
|
||||||
|
|
||||||
|
// MoveIndex moves the index variable idx forward by delta steps and ensures
|
||||||
|
// that the boundary policy as defined by the CheckBoundsFunc is enforced.
|
||||||
|
//
|
||||||
|
// If CheckBoundsFunc is nil, fix boundary checks are performed.
|
||||||
|
func MoveIndex(idx, delta int, indexer IndexProvider, cb BoundsCheckFunc) int {
|
||||||
|
lower, upper := indexer.StartIndex(), indexer.EndIndex()
|
||||||
|
sign := 1
|
||||||
|
if upper < lower {
|
||||||
|
lower, upper = upper, lower
|
||||||
|
sign = -1
|
||||||
|
}
|
||||||
|
result := idx + sign*delta
|
||||||
|
if cb == nil {
|
||||||
|
return FixBounds(result, lower, upper)
|
||||||
|
}
|
||||||
|
return cb(result, lower, upper)
|
||||||
|
}
|
133
lib/iterator/index_test.go
Normal file
133
lib/iterator/index_test.go
Normal file
|
@ -0,0 +1,133 @@
|
||||||
|
package iterator_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"git.sr.ht/~rjarry/aerc/lib/iterator"
|
||||||
|
)
|
||||||
|
|
||||||
|
type indexer struct {
|
||||||
|
start int
|
||||||
|
end int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ip *indexer) StartIndex() int {
|
||||||
|
return ip.start
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ip *indexer) EndIndex() int {
|
||||||
|
return ip.end
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMoveIndex(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
idx int
|
||||||
|
delta int
|
||||||
|
start int
|
||||||
|
end int
|
||||||
|
cb iterator.BoundsCheckFunc
|
||||||
|
expected int
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 1,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.FixBounds,
|
||||||
|
expected: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 5,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.FixBounds,
|
||||||
|
expected: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: -1,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.FixBounds,
|
||||||
|
expected: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 2,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 3,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: -1,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 2,
|
||||||
|
delta: 2,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: -2,
|
||||||
|
start: 0,
|
||||||
|
end: 2,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 1,
|
||||||
|
delta: 1,
|
||||||
|
start: 2,
|
||||||
|
end: 0,
|
||||||
|
cb: iterator.FixBounds,
|
||||||
|
expected: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 1,
|
||||||
|
start: 2,
|
||||||
|
end: 0,
|
||||||
|
cb: iterator.FixBounds,
|
||||||
|
expected: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
idx: 0,
|
||||||
|
delta: 1,
|
||||||
|
start: 2,
|
||||||
|
end: 0,
|
||||||
|
cb: iterator.WrapBounds,
|
||||||
|
expected: 2,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, test := range tests {
|
||||||
|
idx := iterator.MoveIndex(
|
||||||
|
test.idx,
|
||||||
|
test.delta,
|
||||||
|
&indexer{test.start, test.end},
|
||||||
|
test.cb,
|
||||||
|
)
|
||||||
|
if idx != test.expected {
|
||||||
|
t.Errorf("test %d [%#v] failed: got %d but expected %d",
|
||||||
|
i, test, idx, test.expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue