Skip to content

Commit

Permalink
add new peek iterator for single item look ahead
Browse files Browse the repository at this point in the history
  • Loading branch information
0x5a17ed committed May 23, 2024
1 parent 526dd11 commit ac82baf
Show file tree
Hide file tree
Showing 2 changed files with 146 additions and 0 deletions.
77 changes: 77 additions & 0 deletions itlib/peek.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
// Copyright (c) 2024 individual contributors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// <https://www.apache.org/licenses/LICENSE-2.0>
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package itlib

import (
"github.com/0x5a17ed/itkit"
)

// PeekIterator represents an iterator allowing to peek the next item
// before advancing to it.
type PeekIterator[T any] struct {
src itkit.Iterator[T]

cur T
cached T
has bool
}

// Ensure PeekIterator implements the iterator interface.
var _ itkit.Iterator[struct{}] = &PeekIterator[struct{}]{}

// Next implements the [itkit.Iterator.Next] interface.
func (it *PeekIterator[T]) Next() (ok bool) {
if ok = it.has; ok {
// Try to consume the cached item first.
it.cur, it.has = it.cached, false
} else if ok = it.src.Next(); ok {
// Try to get an item from the source iterator.
it.cur = it.src.Value()
}
return
}

// Value implements the [itkit.Iterator.Value] interface.
func (it *PeekIterator[T]) Value() T {
return it.cur
}

// Peek returns the next item without advancing the iterator.
//
// Advances the source iterator to the next item only if necessary.
func (it *PeekIterator[T]) Peek() (v T, ok bool) {
if it.has {
return it.cached, true
}

if it.has = it.src.Next(); it.has {
it.cached = it.src.Value()
}
return it.cached, it.has
}

// Iter returns the [TeeIterator] as an [itkit.Iterator] value.
func (it *PeekIterator[T]) Iter() itkit.Iterator[T] {
return it
}

func newPeekIterator[T any](src itkit.Iterator[T]) *PeekIterator[T] {
return &PeekIterator[T]{src: src}
}

// Peek returns an Iterator that is always exhausted.
func Peek[T any](src itkit.Iterator[T]) *PeekIterator[T] {
return newPeekIterator(src)
}
69 changes: 69 additions & 0 deletions itlib/peek_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
// Copyright (c) 2024 individual contributors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// <https://www.apache.org/licenses/LICENSE-2.0>
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package itlib_test

import (
"testing"

"github.com/0x5a17ed/itkit/iters/rangeit"
"github.com/0x5a17ed/itkit/iters/sliceit"
"github.com/0x5a17ed/itkit/itlib"
"github.com/stretchr/testify/assert"
)

func TestPeek(t *testing.T) {
t.Run("empty", func(t *testing.T) {
it := itlib.Peek(itlib.Empty[int]())

_, ok := it.Peek()
assert.False(t, ok)

assert.False(t, it.Next())
})

t.Run("one", func(t *testing.T) {
asserter := assert.New(t)

it := itlib.Peek(rangeit.RangeFrom(1, 3))

// Peeking into the iterator yields the first item.
v, ok := it.Peek()
asserter.True(ok)
asserter.Equal(1, v)

// Peeking again yields the same value as before.
v, ok = it.Peek()
asserter.True(ok)
asserter.Equal(1, v)

// Advancing the iterator yields the same value again.
asserter.True(it.Next())
asserter.Equal(1, v)

// Peeking again yields the next value in the source iterator.
v, ok = it.Peek()
asserter.True(ok)
asserter.Equal(2, v)
})

t.Run("mirror", func(t *testing.T) {
asserter := assert.New(t)

it := itlib.Peek(rangeit.RangeFrom(1, 5))

// Iterating the Peek iterator yields all items normally.
asserter.Equal([]int{1, 2, 3, 4}, sliceit.To(it.Iter()))
})
}

0 comments on commit ac82baf

Please sign in to comment.