The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
#!/usr/bin/perl -w
# $Id: 40_postback.t 1044 2012-11-28 16:35:54Z fil $

use strict;

# sub POE::Kernel::TRACE_REFCNT () { 1 }
# sub POE::Kernel::TRACE_SIGNALS () { 1 }

sub DEBUG () { 0 }

use Test::More tests => 14;

use FindBin;
use lib "$FindBin::Bin/..";

use POE;
use POE::Component::Generic;

my $N = 2;
my $alt_fork = 1;
if( $ENV{HARNESS_PERL_SWITCHES} ) {
    $N *= 5;
#    $alt_fork = 0;
}
#$alt_fork = 0 if $^O eq 'MSWin32';

my $daemon=0;
# eval "use POE::Component::Daemon; \$daemon++";


my $generic = POE::Component::Generic->spawn( 
          package   => 't::P40',
          methods   => [ qw( new something otherthing twothing holder runner ) ],
          postbacks => { something=>1, otherthing=>[0], 
                         holder => [ 1 ] 
                       },
          alt_fork  => $alt_fork,
          verbose   => 1,
          debug     => DEBUG
      );
      
my $C1 = 0;
my $C2 = 0;
my $C3 = -17;
my $PID = $$;

POE::Session->create(
    inline_states => {
        _start => sub {
            $poe_kernel->alias_set( 'worker1' );
        },

        _stop => sub {
            DEBUG and warn "_stop";
        },
        
        otherthing_back => sub {
            my( $res, $answer ) = @_[ ARG0, ARG1 ];
            ok( (not exists $res->{error}), "No errors" );
            is( $answer, 10, "Got the right answer" );
            $generic->twothing({});
        },
    }


);

my $session_id;

POE::Session->create(
    inline_states => {
        _start => sub {
            $poe_kernel->alias_set( 'worker' );
            diag( "$N seconds" );
            $poe_kernel->delay( 'something', $N );
            if( $daemon ) {
                $poe_kernel->sig( USR1=>'USR1' );
            }
        },
        USR1 => sub { Daemon->__peek( 1 ); },

        _stop => sub {
            DEBUG and warn "_stop";
        },
  
  
        ################
        something => sub {
            $generic->something( {event=>'something_back'},
                               10, 
                               'some_postback',
                               17,
                             );

            $generic->holder( {}, 'Quiznos', 'Quiznos1' );
            $generic->holder( {}, 'Subway', 'Subway' );
        },
        something_back => sub {
            my( $res, $answer ) = @_[ ARG0, ARG1 ];
            ok( (not exists $res->{error}), "No errors" );
            is( $answer, 27, "Got an answer" );

            $poe_kernel->yield( 'otherthing' );
        },
        some_postback => sub {
            my( $answer ) = $_[ ARG0 ];
           
            is( $answer, 17, "Got 17" );
            $C1++;
        },
      
        ###############
        otherthing => sub {
            $generic->otherthing( { event=>'otherthing_back', 
                                    session => 'worker1' },
                                  { event=>'other_postback'},
                                    0..9 );
        },
        other_postback => sub {
            my( $answer ) = $_[ ARG0 ];
            is( $answer, 42, "Got 42" );
            diag( "If this doesn't exit, try kill -USR1 $$" ) if $daemon;
            $C2++;

            $generic->runner( { event=>'ignore' }, 'Quiznos' ); 
        },

        Quiznos1 => sub {
            pass( "Toasty postback" );
            $C3 = keys( %{ $generic->{postback_defs} } );

            $generic->runner( { event=>'ignore' }, 'Subway' ); 
            $generic->holder( {}, 'Quiznos', { event => 'Quiznos2' } );
        },

        Subway => sub {
            pass( "Not-toasty postback" );

            $generic->runner( { event=>'ignore' }, 'Quiznos', 
                                    "Quiznos2" ); 
        },

        Quiznos2 => sub {
            pass( "Substitue postback" );
            is( keys( %{ $generic->{postback_defs} } ), $C3, 
                            "Still $C3 postbacks" );

            $generic->shutdown;
            # Test that the Generic session has _stoped
            $session_id = $generic->session_id;
            diag( "$N seconds" );
            $poe_kernel->delay( Done => $N );
        },

        Done => sub {
            ok( $session_id, "We expect to exit" );
            my $ses = eval {
                        $poe_kernel->ID_id_to_session( $session_id );
                    };
            is( $ses, undef(), "Generic session is gone" );
        }
    }     
);


$poe_kernel->run;

is( $C1, 1, "some_postback was posted" );
is( $C2, 1, "other_postback was posted" );