Iterator-Flex

 view release on metacpan or  search on metacpan

lib/Iterator/Flex/Cat.pm  view on Meta::CPAN

package Iterator::Flex::Cat;

# ABSTRACT: An iterator which concatenates a set of iterators

use v5.28;
use strict;
use warnings;
use experimental qw( signatures declared_refs refaliasing );

our $VERSION = '0.34';

use Iterator::Flex::Utils
  qw( RETURN STATE EXHAUSTION :IterAttrs :IterStates can_meth throw_failure );
use Iterator::Flex::Factory 'to_iterator';
use parent 'Iterator::Flex::Base';
use List::Util 'all';

use namespace::clean;





































sub new ( $class, @args ) {
    my $pars = Ref::Util::is_hashref( $args[-1] ) ? pop @args : {};

    @args
      or throw_failure( parameter => 'not enough parameters' );

    $class->SUPER::new( {
            depends                => \@args,
            current_iterator_index => undef,
        },
        $pars,
    );
}

sub construct ( $class, $state ) {
    throw_failure( parameter => q{state must be a HASH reference} )
      unless Ref::Util::is_hashref( $state );

    $state->{value} //= [];

    my ( \@depends, $current_iterator, $prev, $current, $next, undef )
      = @{$state}{ 'depends', 'current_iterator', 'prev', 'current', 'next', 'thaw' };

    # transform into iterators if required.
    my @iterators
      = map { to_iterator( $_, { ( +EXHAUSTION ) => RETURN } ) } @depends;

    # -- not sure if this is just cargo-culted
    # my $value;
    # $value = $current
    #   if $thaw;

    my $self;
    my $iterator_state;
    my $iter;
    my $is_exhausted;

    my sub init {
        $current_iterator //= 0;
        if ( $current_iterator >= @iterators ) {
            $iterator_state = IterState_EXHAUSTED;
        }
        else {
            $iter         = $iterators[$current_iterator];
            $is_exhausted = $iter->can( 'is_exhausted' );
        }
    }

    init();

    my %params = (

        ( +_SELF ) => \$self,

        ( +STATE ) => \$iterator_state,

        ( +NEXT ) => sub {
            return $self->signal_exhaustion if $iterator_state == IterState_EXHAUSTED;

            while ( 1 ) {
                my $value = $iter->();
                if ( defined $value || !$iter->$is_exhausted ) {
                    $prev    = $current;
                    $current = $value;
                    return $current;
                }

                last if ++$current_iterator >= @iterators;
                $iter         = $iterators[$current_iterator];
                $is_exhausted = $iter->can( 'is_exhausted' );
            }

            # if we haven't returned, we've exhausted things
            $current_iterator = undef;
            $prev             = $current;
            return $current = $self->signal_exhaustion;
        },



( run in 1.882 second using v1.01-cache-2.11-cpan-99c4e6809bf )