#!/usr/bin/perl -w

use strict;

use Test::More tests => 7;

use Time::HiRes qw( time );

use IO::Socket::UNIX;

use IO::Async::Set::Select;

my $set = IO::Async::Set::Select->new();

my ( $rvec, $wvec, $evec ) = ('') x 3;
my $timeout;

$set->pre_select( \$rvec, \$wvec, \$evec, \$timeout );
is( $timeout, undef, '$timeout idling pre_select' );

$timeout = 5;

$set->pre_select( \$rvec, \$wvec, \$evec, \$timeout );
is( $timeout, 5, '$timeout idling pre_select with timeout' );

my $done = 0;
$set->enqueue_timer( delay => 2, code => sub { $done = 1; } );

my $id = $set->enqueue_timer( delay => 5, code => sub { die "This timer should have been cancelled" } );
$set->cancel_timer( $id );

undef $id;

$set->pre_select( \$rvec, \$wvec, \$evec, \$timeout );
cmp_ok( $timeout, '>', 1.9, '$timeout while timer waiting pre_select at least 1.9' );
cmp_ok( $timeout, '<', 2.5, '$timeout while timer waiting pre_select at least 2.5' );

my ( $now, $took );

$now = time;
select( $rvec, $wvec, $evec, $timeout );
$took = time - $now;

cmp_ok( $took, '>', 1.9, 'loop_once(5) while waiting for timer takes at least 1.9 seconds' );
cmp_ok( $took, '<', 2.5, 'loop_once(5) while waiting for timer no more than 2.5 seconds' );

$set->post_select( $rvec, $evec, $wvec );

# select() might have returned just a little early, such that the TimerQueue
# doesn't think anything is ready yet. We need to handle that case.
while( !$done ) {
   $timeout = 0.1;

   $set->pre_select( \$rvec, \$wvec, \$evec, \$timeout );
   select( $rvec, $wvec, $evec, $timeout );
   $set->post_select( $rvec, $evec, $wvec );

   die "It should have been ready by now" if( time - $now > 5 );
}

is( $done, 1, '$done after post_select while waiting for timer' );


syntax highlighted by Code2HTML, v. 0.9.1