The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
use strict;
use warnings;
no  warnings 'once';

use Test::More;

use RPC::ExtDirect::Test::Util;

if ( $ENV{REGRESSION_TESTS} ) {
    plan tests => 16;
}
else {
    plan skip_all => 'Regression tests are not enabled.';
}

# We will test deprecated API and don't want the warnings
# cluttering STDERR
$SIG{__WARN__} = sub {};

use lib 't/lib2';
use RPC::ExtDirect::Test::Hooks;
use RPC::ExtDirect::Test::PollProvider;

use RPC::ExtDirect::Router;
use RPC::ExtDirect::EventProvider;
use RPC::ExtDirect::Event;

use RPC::ExtDirect::API before => \&before_hook, after => \&after_hook;

{
    package RPC::ExtDirect::Event;

    use Data::Dumper;

    use overload '""' => \&stringify,
                 'eq' => \=

    sub stringify {
        my ($self) = @_;

        local $Data::Dumper::Indent = 0;
        local $Data::Dumper::Terse  = 1;

        my $str = $self->name . ':' . Dumper( $self->data );
        $str =~ s/^'//;
        $str =~ s/'$//;

        return $str;
    }

    sub equals {
        my ($self, $comparison) = @_;

        return $self->stringify eq "$comparison";
    }

    # This cheating is to avoid rewriting test modules

    package RPC::ExtDirect::Test::Hooks;

    RPC::ExtDirect->import( Action => 'Hooks',
                            before => \&main::before_hook,
                            after  => \&main::after_hook );
}

# These variables get set when hooks are called
my ($before, $after, $modify, $throw_up, $cancel);

sub before_hook {
    my ($class, %params) = @_;

    $before = [ $class, { %params } ];

    $params{arg}->[0] = 'bar' if $modify;

    die "Exception\n" if $throw_up;

    return "Method canceled" if $cancel;

    return 1;
}

sub after_hook {
    $after = [ shift @_, { @_ } ];
}

my $tests = eval do { local $/; <DATA> };
die "Can't read DATA: $@\n" if $@;

for my $test ( @$tests ) {
    my $name       = $test->{name};
    my $input      = $test->{input};
    my $type       = $test->{type};
    my $env        = $test->{env};
    my $exp_before = $test->{expected_before};
    my $exp_after  = $test->{expected_after};
    
    $before = $after = $modify = $throw_up = $cancel = undef;
    
    $modify   = $test->{modify};
    $throw_up = $test->{throw_up};
    $cancel   = $test->{cancel};
    
    if ( $type eq 'router' ) {
        RPC::ExtDirect::Router->route($input, $env);
    }
    else {
        RPC::ExtDirect::EventProvider->poll($env);
    };

    # Orig is a closure in RPC::ExtDirect::Hook, impossible to take ref of
    {
        local $@;
        eval { delete $before->[1]->{orig}; delete $after->[1]->{orig}; };
    }

    # (before|instead|after|method)_ref hook args were introduced in 3.0
    # 2.x tests will blow up if we don't clean these up; however being
    # additions they can't harm the existing code that is not aware of them
    {
        local $@;
        eval {
            delete $before->[1]->{$_}, delete $after->[1]->{$_}
                for map { $_.'_ref' } qw/ before instead after method /;
        };
    }

    is_deep $before, $exp_before, "$name: before data";
    is_deep $after,  $exp_after,  "$name: after data";
};

__DATA__
[
    # Cancel Method call by throwing error
    {
        name => 'Router throw error',
        input => q|{"type":"rpc","tid":1,"action":"Hooks",|.
                 q|"method":"foo_hook","data":["foo"]}|,
        env => 'env',
        throw_up => 1,
        type => 'router',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::Hooks',
                method      => 'foo_hook',
                code        => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg         => [ 'foo' ],
                env         => 'env',
                param_names => undef,
                param_no    => 1,
                pollHandler => 0,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::Hooks',
                method        => 'foo_hook',
                code          => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg           => [ 'foo' ],
                env           => 'env',
                result        => undef,
                exception     => "Exception\n",
                param_names   => undef,
                param_no      => 1,
                pollHandler   => 0,
                formHandler   => 0,
                method_called => undef,
            },
        ],
    },

    # Cancel Method call by returning non-1 from before hook
    {
        name => 'Router cancel Method',
        input => q|{"type":"rpc","tid":1,"action":"Hooks",|.
                 q|"method":"foo_hook","data":["foo"]}|,
        env => 'env',
        cancel => 1,
        type => 'router',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::Hooks',
                method      => 'foo_hook',
                code        => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg         => [ 'foo' ],
                env         => 'env',
                param_names => undef,
                param_no    => 1,
                pollHandler => 0,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::Hooks',
                method        => 'foo_hook',
                code          => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg           => [ 'foo' ],
                env           => 'env',
                result        => 'Method canceled',
                exception     => undef,
                param_names   => undef,
                param_no      => 1,
                pollHandler   => 0,
                formHandler   => 0,
                method_called => undef,
            },
        ],
    },

    # Simple Router request
    {
        name => 'Router method call',
        input => q|{"type":"rpc","tid":1,"action":"Hooks",|.
                 q|"method":"foo_hook","data":["foo"]}|,
        env => 'env',
        type => 'router',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::Hooks',
                method      => 'foo_hook',
                code        => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg         => [ 'foo' ],
                env         => 'env',
                param_names => undef,
                param_no    => 1,
                pollHandler => 0,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::Hooks',
                method        => 'foo_hook',
                code          => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg           => [ 'foo' ],
                env           => 'env',
                result        => [ 'RPC::ExtDirect::Test::Hooks', 'foo' ],
                exception     => '',
                param_names   => undef,
                param_no      => 1,
                pollHandler   => 0,
                formHandler   => 0,
                method_called => \&RPC::ExtDirect::Test::Hooks::foo_hook,
            },
        ],
    },

    # Argument modification in "before" hook
    {
        name => 'Router arg modification',
        input => q|{"type":"rpc","tid":1,"action":"Hooks",|.
                 q|"method":"foo_hook","data":["foo"]}|,
        env => 'env',
        modify => 1,
        type => 'router',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::Hooks',
                method      => 'foo_hook',
                code        => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg         => [ 'bar' ],
                env         => 'env',
                param_names => undef,
                param_no    => 1,
                pollHandler => 0,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::Hooks',
                method        => 'foo_hook',
                code          => \&RPC::ExtDirect::Test::Hooks::foo_hook,
                arg           => [ 'bar' ],
                env           => 'env',
                result        => [ 'RPC::ExtDirect::Test::Hooks', 'bar' ],
                exception     => '',
                param_names   => undef,
                param_no      => 1,
                pollHandler   => 0,
                formHandler   => 0,
                method_called => \&RPC::ExtDirect::Test::Hooks::foo_hook,
            },
        ],
    },

    # Cancel EventProvider call by throwing error
    {
        name => 'Poll throw error',
        env => 'env',
        throw_up => 1,
        type => 'poll',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::PollProvider',
                method      => 'foo',
                code        => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg         => [ ],
                env         => 'env',
                param_names => undef,
                param_no    => undef,
                pollHandler => 1,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::PollProvider',
                method        => 'foo',
                code          => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg           => [ ],
                env           => 'env',
                result        => undef,
                exception     => "Exception\n",
                param_names   => undef,
                param_no      => undef,
                pollHandler   => 1,
                formHandler   => 0,
                method_called => undef,
            },
        ],
    },

    # Cancel Method call by returning non-1 from before hook
    {
        name => 'Poll cancel Method',
        env => 'env',
        cancel => 1,
        type => 'poll',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::PollProvider',
                method      => 'foo',
                code        => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg         => [ ],
                env         => 'env',
                param_names => undef,
                param_no    => undef,
                pollHandler => 1,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::PollProvider',
                method        => 'foo',
                code          => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg           => [ ],
                env           => 'env',
                result        => 'Method canceled',
                exception     => undef,
                param_names   => undef,
                param_no      => undef,
                pollHandler   => 1,
                formHandler   => 0,
                method_called => undef,
            },
        ],
    },

    # Argument modification in "before" hook
    {
        name => 'Poll arg modification',
        env => 'env',
        modify => 1,
        type => 'poll',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::PollProvider',
                method      => 'foo',
                code        => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg         => [ 'bar' ],
                env         => 'env',
                param_names => undef,
                param_no    => undef,
                pollHandler => 1,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::PollProvider',
                method        => 'foo',
                code          => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg           => [ 'bar' ],
                env           => 'env',
                result        => [q|foo_event:['foo']|, q|bar_event:{'foo' => 'bar'}|],
                exception     => '',
                param_names   => undef,
                param_no      => undef,
                pollHandler   => 1,
                formHandler   => 0,
                method_called => \&RPC::ExtDirect::Test::PollProvider::foo,
            },
        ],
    },

    # Event polling
    {
        name => 'Poll method call',
        env => 'env',
        type => 'poll',
        expected_before => [
            'main',
            {
                before      => \&before_hook,
                instead     => undef,
                after       => \&after_hook,
                package     => 'RPC::ExtDirect::Test::PollProvider',
                method      => 'foo',
                code        => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg         => [ ],
                env         => 'env',
                param_names => undef,
                param_no    => undef,
                pollHandler => 1,
                formHandler => 0,
            },
        ],
        expected_after => [
            'main',
            {
                before        => \&before_hook,
                instead       => undef,
                after         => \&after_hook,
                package       => 'RPC::ExtDirect::Test::PollProvider',
                method        => 'foo',
                code          => \&RPC::ExtDirect::Test::PollProvider::foo,
                arg           => [ ],
                env           => 'env',
                result        => [q|foo_event:['foo']|, q|bar_event:{'foo' => 'bar'}|],
                exception     => '',
                param_names   => undef,
                param_no      => undef,
                pollHandler   => 1,
                formHandler   => 0,
                method_called => \&RPC::ExtDirect::Test::PollProvider::foo,
            },
        ],
    },
]