The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
#!perl -T
## no critic (ValuesAndExpressions::ProhibitMagicNumbers)

use 5.006;
use strict;
use warnings FATAL => 'all';
use Test::More;
use Test::Exception;
use Log::Any::Test;    # should appear before 'use Log::Any'!
use Log::Any qw($log);

use lib 't';
use lib 'integ_t';
require 'iron_io_integ_tests_common.pl'; ## no critic (Modules::RequireBarewordIncludes)

plan tests => 4; # Setup, Do, Verify, Cleanup

require IO::Iron::IronMQ::Client;
require IO::Iron::IronMQ::Queue;
require IO::Iron::IronMQ::Message;

#use Log::Any::Adapter ('Stderr'); # Activate to get all log messages.
use Data::Dumper; $Data::Dumper::Maxdepth = 2;

diag("Testing IO::Iron::IronMQ::Client, Perl $], $^X");

## Test case
diag('Testing IO::Iron::IronMQ::Queue method post_messages()');

my $iron_mq_client;
my $project_id;
my $queue_name;
my $queue;
my @send_messages;
my @send_messages_ids;
my %msg_body_hash_02;

subtest 'Setup for testing' => sub {
    plan tests => 2;
    # Create an IronMQ client.
    $iron_mq_client = IO::Iron::IronMQ::Client->new( 'config' => 'iron_mq.json' );
    $project_id = $iron_mq_client->{'connection'}->{'project_id'};
    # Create a new queue name.
    $queue_name = create_unique_queue_name();
    # Create a new queue.
    $queue = $iron_mq_client->create_and_get_queue( 'name' => $queue_name );
    isa_ok($queue, 'IO::Iron::IronMQ::Queue', 'create_and_get_queue returns a IO::Iron::IronMQ::Queue.');
    is($queue->name(), $queue_name, 'Created queue has the given name.');
    diag('Created message queue ' . $queue_name . q{'.});

    use utf8;
    # Let's create some messages
    my $iron_mq_msg_send_01 = IO::Iron::IronMQ::Message->new(
            'body' => 'My message #01',
            );
    diag('Test with YAML in JSON.');
    use YAML::Tiny; # For serializing/deserializing a hash.
    %msg_body_hash_02 = (msg_body_text => 'My message #02', msg_body_item => {sub_item => 'Sub text'});
    my $yaml = YAML::Tiny->new(); $yaml->[0] = \%msg_body_hash_02;
    my $msg_body = $yaml->write_string();
    my $iron_mq_msg_send_02 = IO::Iron::IronMQ::Message->new(
            'body' => $msg_body,
            'delay' => 0,     # The item will not be available on the queue until this many seconds have passed.
            );
    my $iron_mq_msg_send_03 = IO::Iron::IronMQ::Message->new( 'body' => 'My message #03 in Swedish: Räksmörgås' );
    my $iron_mq_msg_send_04 = IO::Iron::IronMQ::Message->new( 'body' => 'My message #04 in Russian: бутерброд' );
    my $iron_mq_msg_send_05 = IO::Iron::IronMQ::Message->new( 'body' => 'My message #05 in Chinese: 三明治' );
    my $iron_mq_msg_send_06 = IO::Iron::IronMQ::Message->new( 'body' => 'My message #06 in Yiddish: סענדוויטש' );
    diag('Test with JSON in JSON.');
    require JSON::MaybeXS;
    my $json = JSON::MaybeXS->new(utf8 => 1, pretty => 1);
    $msg_body = $json->encode(\%msg_body_hash_02);
    my $iron_mq_msg_send_07 = IO::Iron::IronMQ::Message->new( 'body' => $msg_body );
    diag('Test with perl Storable serializer module.');
    require Storable;
    $msg_body = Storable::freeze(\%msg_body_hash_02);
    my $iron_mq_msg_send_08 = IO::Iron::IronMQ::Message->new( 'body' => $msg_body );
    no utf8;
    diag('Created 8 messages for sending.');
    push @send_messages, $iron_mq_msg_send_01, $iron_mq_msg_send_02, $iron_mq_msg_send_03, $iron_mq_msg_send_04, $iron_mq_msg_send_05, $iron_mq_msg_send_06, $iron_mq_msg_send_07, $iron_mq_msg_send_08;
};

my @sent_msg_ids;
subtest 'Pushing' => sub {
    plan tests => 7;
    #Queue is empty
    my @msg_pulls_00 = $queue->reserve_messages( 'n' => 2, 'timeout' => 120 );
    is(scalar @msg_pulls_00, 0, 'No messages pulled from queue, size 0.');
    is($queue->size(), 0, 'Queue size is 0.');
    diag('Empty queue at the start.');

    # Let's push the messages.
    $log->clear();
    my $msg_send_id_01 = $queue->post_messages( 'messages' => [ $send_messages[0] ] );
    #my $log_test = 0;
    #map { $log_test = 1 if ($_->{level} eq 'info' 
    #        && $_->{category} eq 'IO::Iron::IronMQ::Queue' 
    #        && $_->{message} =~ /^Pushed IronMQ Message\(s\) \(queue name=$queue_name; message id\(s\)=$msg_send_id_01\)\.$/gs
    #    ) } @{$log->msgs};
    #is($log_test, 1, 'Push() logged correctly.');
    push @send_messages_ids, $msg_send_id_01;
    is($queue->size(), 1, 'One message pushed, queue size is 1.');
    push @sent_msg_ids, $msg_send_id_01;

    $log->clear();
    my @msg_send_ids_02 = $queue->post_messages( 'messages' => [ @send_messages[1,2] ] );
    #$log_test = 0;
    #my $send_ids_text = join ',', @msg_send_ids_02;
    ##diag(Dumper($log->msgs));
    #map { $log_test = 1 if ($_->{level} eq 'info' 
    #        && $_->{category} eq 'IO::Iron::IronMQ::Queue' 
    #        && $_->{message} =~ m/^Pushed IronMQ Message\(s\) \(queue name=$queue_name; message id\(s\)=$send_ids_text\)\.$/gs
    #    ) } @{$log->msgs};
    #is($log_test, 1, 'Push() logged correctly.');
    is(scalar @msg_send_ids_02, 2, 'Two messages pushed');
    push @send_messages_ids, @msg_send_ids_02;
    is($queue->size(), 3, 'Two messages pushed, queue size is 3.');
    push @sent_msg_ids, @msg_send_ids_02;

    my $number_of_msgs_sent = $queue->post_messages( 'messages' => [ @send_messages[3,4,5,6,7] ] );
    is($number_of_msgs_sent, 5, '5 more messages pushed.');
    is($queue->size(), 8, '5 more messages pushed, queue size is 8.');
    diag('Total 8 messages pushed to queue.');

};

# Let's pull some messages.
my @msg_pulls;
subtest 'Pulled messages match with the sent messages.' => sub {
    plan tests => 16;
    @msg_pulls = $queue->reserve_messages( 'n' => 10, 'timeout' => 120 );
    is( scalar @msg_pulls, 8, 'Pulled 8 messages.');
    my $yaml_de = YAML::Tiny->new(); $yaml_de = $yaml_de->read_string($msg_pulls[1]->body());
    is_deeply($yaml_de->[0], \%msg_body_hash_02, '#2 message body after serialization matches with the sent message body.');
    is( $msg_pulls[0]->id(), $sent_msg_ids[0], 'Pulled two, ids match with the sent ids.');
    is( $msg_pulls[1]->id(), $sent_msg_ids[1], 'Pulled two, ids match with the sent ids.');
    is( $msg_pulls[2]->id(), $sent_msg_ids[2], 'Pulled two, ids match with the sent ids.');
    #is_deeply( [$msg_pulls[0..2]], \@sent_msg_ids, 'Ids match with sent messages.')
    # msg #7 has JSON encoded body
    require JSON::MaybeXS;
    my $json = JSON::MaybeXS->new(utf8 => 1, pretty => 1);
    is_deeply($json->decode($msg_pulls[6]->body()), \%msg_body_hash_02, '#7 message body after serialization matches with the sent message body.');
    # msg #8 has Storable encoded body
    require Storable;
    is_deeply(Storable::thaw($msg_pulls[7]->body()), \%msg_body_hash_02, '#8 message body after serialization matches with the sent message body.');

    is($queue->size(), 8, 'Three messages pulled in total; put queue size is still 8. (pull does not delete messages.)');
    diag('Pulled 8 messages from queue.');
    foreach (0..7) {
        my $pushed_body = $msg_pulls[$_]->body();
        my $pulled_body = $send_messages[$_]->body();
        #diag('Message number $_:\nPushed body: $pushed_body, dumped:\n' . Dumper($pushed_body) . ' is utf8:' . utf8::is_utf8($pushed_body) . ';\n Pulled body: $pulled_body, dumped:\n' . Dumper($pulled_body) . ' is utf8:' . utf8::is_utf8($pulled_body) . '.');
        is($msg_pulls[$_]->body(), $send_messages[$_]->body(), "Message number $_: pulled body matches pushed body.");
    }
};

subtest 'Clean up after us.' => sub {
    plan tests => 2;
    # Let's clear the queue
    $queue->clear_messages();
    is($queue->size(), 0, 'Cleared the queue, queue size is 0.');
    diag('Cleared the queue, queue size is 0.');

    # Delete queue. Confirm deletion.
    $iron_mq_client->delete_queue( 'name' => $queue_name);
    throws_ok {
        my $dummy = $iron_mq_client->get_queue( 'name' => $queue_name);
    } '/IronHTTPCallException: status_code=404/',
    # IronHTTPCallException: status_code=404 response_message=Queue not found
    # status code 404: server could not find what was requested! Response message can change, code remains 404!
            'Throw IronHTTPCallException when no message queue of given name.';
    diag('Definately deleted message queue ' . $queue->name() . q{'.});
    diag('All cleaned up.')
};