Data-Transform-ExplicitMetadata

 view release on metacpan or  search on metacpan

t/recursive.t  view on Meta::CPAN

use strict;
use warnings;

use Data::Transform::ExplicitMetadata qw(encode decode);

use Scalar::Util qw(refaddr);
use Test::More tests => 6;

subtest recurse_array => sub {
    my $idx_2 = [ 2 ];
    push @$idx_2, $idx_2;
    my $original = [ 0, 1, $idx_2 ];

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'ARRAY',
        __value => [
            0,
            1,
            {
                __refaddr => refaddr($idx_2),
                __reftype => 'ARRAY',
                __value => [
                     2,
                    {
                        __refaddr => refaddr($idx_2),
                        __reftype => 'ARRAY',
                        __recursive => 1,
                        __value => '$VAR->[2]',
                    },
                ],
            },
        ],
    };

    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode recursive data structure');

    my $decoded = decode($encoded);
    is_deeply($decoded, $original, 'decode recursive data structure');
};

subtest recurse_hash => sub {
    my $nested = { bar => 'bar' };
    $nested->{nested} = $nested;
    my $original = { foo => 'foo', nested => $nested };

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'HASH',
        __value => {
            foo => 'foo',
            nested => {
                __refaddr => refaddr($nested),
                __reftype => 'HASH',
                __value => {
                    bar => 'bar',
                    nested => {
                        __refaddr => refaddr($nested),
                        __reftype => 'HASH',
                        __recursive => 1,
                        __value => '$VAR->{nested}'
                    }
                }
            }
        }
    };
    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode recursive hash');

    my $decoded = decode($encoded);
    is_deeply($decoded, $original, 'decode recursive hash');
};

subtest recurse_ref1 => sub {
    my $a = 1;
    my $b = \$a;
    my $original = \$b;
    $a = \$original;

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'REF',
        __value => {
            __refaddr => refaddr($b),
            __reftype => 'REF',
            __value => {
                __refaddr => refaddr($a),
                __reftype => 'REF',
                __value => {
                    __refaddr => refaddr($original),
                    __reftype => 'REF',
                    __recursive => 1,
                    __value => '$VAR',
                }
            }
        }
    };
    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode ref reference');

    my $decoded = decode($encoded);
    is_deeply($decoded, $original, 'decode ref reference');

    undef($a); # break the cycle
};

subtest recurse_ref2 => sub {
    my $c = 1;
    my $b = \$c;
    $c = \$b;
    my $a = \$b;
    my $original = \$a;

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'REF',
        __value => {
            __refaddr => refaddr($a),
            __reftype => 'REF',
            __value => {
                __refaddr => refaddr($b),
                __reftype => 'REF',
                __value => {
                    __refaddr => refaddr($c),
                    __reftype => 'REF',
                    __recursive => 1,
                    __value => '${$VAR}',
                },
            }
        }
    };
    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode ref, circularity not at root');

    my $decoded = decode($encoded);
    is_deeply($decoded, $original, 'decode ref, circularity not at root');

    undef($a);
};

subtest recurse_glob => sub {
    use vars '@typeglob','$typeglob';

    @typeglob = (\@typeglob);
    my $original = \*typeglob;

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'GLOB',
        __value => {
            NAME => 'typeglob',
            PACKAGE => 'main',
            ARRAY => {
                __refaddr => refaddr(\@typeglob),
                __reftype => 'ARRAY',
                __value => [
                    {
                        __refaddr => refaddr(\@typeglob),
                        __reftype => 'ARRAY',
                        __recursive => 1,
                        __value => '*{$VAR}{ARRAY}'
                    }
                ],
            },
            SCALAR => {
                __refaddr => refaddr(\$typeglob),
                __reftype => 'SCALAR',
                __value => undef,
            },
        },
    };
    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode glob');

    my $decoded = decode($encoded);
    is(ref($decoded), 'GLOB', 'decode glob');
    my $decoded_array = *{$decoded}{ARRAY};
    is_deeply($decoded_array, $decoded_array, 'decoded array from glob');
};

subtest blessed => sub {
    my $original = bless [ ], 'BlessedArray';
    push @$original, (1, $original);

    my $expected = {
        __refaddr => refaddr($original),
        __reftype => 'ARRAY',
        __blessed => 'BlessedArray',
        __value => [
            1,
            {
                __refaddr => refaddr($original),
                __reftype => 'ARRAY',
                __blessed => 'BlessedArray',
                __recursive => 1,
                __value => '$VAR',
            },
        ],
    };

    my $encoded = encode($original);
    is_deeply($encoded, $expected, 'encode recursive with blessed item');

    my $decoded = decode($encoded);
    is_deeply($decoded, $original, 'decode from encoded');

    isa_ok($decoded, 'BlessedArray', 'decoded blessed properly');
    isa_ok($decoded->[1], 'BlessedArray', 'recursed decoded blessed properly');
};



( run in 1.008 second using v1.01-cache-2.11-cpan-2398b32b56e )