relationships
This commit is contained in:
parent
fe516b6b32
commit
8639858b60
@ -38,6 +38,7 @@ Adds a data type to the registry.
|
|||||||
|
|
||||||
sub add_type($self,$type,$definition={}) {
|
sub add_type($self,$type,$definition={}) {
|
||||||
$self->{types}{$type} = Dancer2::Plugin::JsonApi::Registry::Schema->new(
|
$self->{types}{$type} = Dancer2::Plugin::JsonApi::Registry::Schema->new(
|
||||||
|
registry => $self,
|
||||||
type => $type,
|
type => $type,
|
||||||
%$definition
|
%$definition
|
||||||
);
|
);
|
||||||
|
@ -35,6 +35,9 @@ has id => (
|
|||||||
has links => (is => 'ro');
|
has links => (is => 'ro');
|
||||||
has top_level_links => (is => 'ro');
|
has top_level_links => (is => 'ro');
|
||||||
has top_level_meta => (is => 'ro');
|
has top_level_meta => (is => 'ro');
|
||||||
|
has relationships => (is => 'ro', default => sub { +{} });
|
||||||
|
|
||||||
|
has registry => ( is => 'ro' );
|
||||||
|
|
||||||
=head1 METHODS
|
=head1 METHODS
|
||||||
|
|
||||||
@ -51,15 +54,25 @@ sub serialize ( $self, $data, $extra_data = {} ) {
|
|||||||
|
|
||||||
$serial->{jsonapi} = { version => '1.0' };
|
$serial->{jsonapi} = { version => '1.0' };
|
||||||
|
|
||||||
$serial->{data} = $self->serialize_data($data,$extra_data);
|
my @included;
|
||||||
|
|
||||||
|
$serial->{data} = $self->serialize_data($data,$extra_data,\@included);
|
||||||
|
|
||||||
$serial->{links} = gen_links($self->top_level_links,$data,$extra_data)
|
$serial->{links} = gen_links($self->top_level_links,$data,$extra_data)
|
||||||
if $self->top_level_links;
|
if $self->top_level_links;
|
||||||
$serial->{meta} = gen_links($self->top_level_meta,$data,$extra_data)
|
$serial->{meta} = gen_links($self->top_level_meta,$data,$extra_data)
|
||||||
if $self->top_level_meta;
|
if $self->top_level_meta;
|
||||||
|
|
||||||
return $serial;
|
$serial->{included} = [ dedupe_included( @included )] if @included;
|
||||||
|
|
||||||
|
return $serial;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub dedupe_included {
|
||||||
|
my %seen;
|
||||||
|
return grep {
|
||||||
|
not $seen{$_->{type}}{$_->{id}}++
|
||||||
|
} @_;
|
||||||
}
|
}
|
||||||
|
|
||||||
=head2 serialize_data($data,$extra_data)
|
=head2 serialize_data($data,$extra_data)
|
||||||
@ -68,9 +81,9 @@ Serializes the inner C<$data>.
|
|||||||
|
|
||||||
=cut
|
=cut
|
||||||
|
|
||||||
sub serialize_data ( $self, $data, $extra_data = {} ) {
|
sub serialize_data ( $self, $data, $extra_data = {}, $included = undef ) {
|
||||||
|
|
||||||
return [ map { $self->serialize_data($_,$extra_data) } @$data ] if ref $data eq 'ARRAY';
|
return [ map { $self->serialize_data($_,$extra_data, $included) } @$data ] if ref $data eq 'ARRAY';
|
||||||
|
|
||||||
my $s = {
|
my $s = {
|
||||||
type => $self->type,
|
type => $self->type,
|
||||||
@ -82,10 +95,38 @@ sub serialize_data ( $self, $data, $extra_data = {} ) {
|
|||||||
|
|
||||||
$s->{attributes} = +{ pairgrep { $a ne $self->id } %$data };
|
$s->{attributes} = +{ pairgrep { $a ne $self->id } %$data };
|
||||||
|
|
||||||
|
my %relationships = $self->relationships->%*;
|
||||||
|
|
||||||
|
for my $key ( keys %relationships ) {
|
||||||
|
my $attr = delete $s->{attributes}{$key} or next;
|
||||||
|
|
||||||
|
my @inc;
|
||||||
|
|
||||||
|
my $t = $self->registry->serialize(
|
||||||
|
$relationships{$key}{type}, $attr, \@inc
|
||||||
|
);
|
||||||
|
|
||||||
|
$s->{relationships}{ $key }{data} = obj_ref($t->{data},\@inc);
|
||||||
|
|
||||||
|
push @$included, @inc if $included;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete $s->{attributes} unless $s->{attributes}->%*;
|
||||||
|
|
||||||
return $s;
|
return $s;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub obj_ref($data, $included) {
|
||||||
|
return [ map { obj_ref($_,$included) } @$data ] if ref $data eq 'ARRAY';
|
||||||
|
|
||||||
|
return $data if keys %$data == 2;
|
||||||
|
|
||||||
|
push @$included, $data;
|
||||||
|
|
||||||
|
return +{ $data->%{qw/ id type/} };
|
||||||
|
}
|
||||||
|
|
||||||
sub gen_id($self,$data) {
|
sub gen_id($self,$data) {
|
||||||
my $id = $self->id;
|
my $id = $self->id;
|
||||||
|
|
||||||
|
46
t/relationships.t
Normal file
46
t/relationships.t
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
use Test2::V0;
|
||||||
|
|
||||||
|
use Dancer2::Plugin::JsonApi::Registry;
|
||||||
|
|
||||||
|
use experimental qw/ signatures /;
|
||||||
|
|
||||||
|
my $registry = Dancer2::Plugin::JsonApi::Registry->new;
|
||||||
|
|
||||||
|
$registry->add_type( 'thing',
|
||||||
|
{ relationships => { subthings => { type => 'subthing' }, } } );
|
||||||
|
$registry->add_type('subthing');
|
||||||
|
|
||||||
|
subtest basic => sub {
|
||||||
|
my $s = $registry->serialize(
|
||||||
|
'thing',
|
||||||
|
{ id => 1,
|
||||||
|
subthings => [ { id => 2, x => 10 }, { id => 3 } ] } );
|
||||||
|
|
||||||
|
ok not $s->{data}{attributes};
|
||||||
|
|
||||||
|
like $s,
|
||||||
|
{ data => {
|
||||||
|
id => 1,
|
||||||
|
type => 'thing',
|
||||||
|
relationships =>
|
||||||
|
{ subthings => { data => [ { id => 2 }, { id => 3 } ] } }
|
||||||
|
},
|
||||||
|
included =>
|
||||||
|
[ { type => 'subthing', id => 2, attributes => { x => 10 } }, ] };
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
subtest "don't repeat includes" => sub {
|
||||||
|
my $s = $registry->serialize(
|
||||||
|
'thing', [
|
||||||
|
{ id => 1,
|
||||||
|
subthings => [ { id => 2, x => 10 }, { id => 3, y => 20 } ] },
|
||||||
|
{ id => 2,
|
||||||
|
subthings => [ { id => 3, y => 20 }, { id => 2, x => 10 } ] }
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
is $s->{ included }->@* + 0, 2;
|
||||||
|
};
|
||||||
|
|
||||||
|
done_testing;
|
Loading…
Reference in New Issue
Block a user