File Coverage

blib/lib/List/GroupBy.pm
Criterion Covered Total %
statement 37 37 100.0
branch 10 10 100.0
condition 16 19 84.2
subroutine 6 6 100.0
pod 1 1 100.0
total 70 73 95.8


line stmt bran cond sub pod time code
1             package List::GroupBy;
2 2     2   86680 use 5.010001;
  2         16  
3 2     2   11 use strict;
  2         4  
  2         49  
4 2     2   9 use warnings;
  2         4  
  2         92  
5              
6             our $VERSION = "0.02";
7              
8 2     2   13 use Exporter qw( import );
  2         12  
  2         110  
9              
10             our @EXPORT_OK = qw( groupBy );
11              
12 2     2   12 use Carp;
  2         14  
  2         804  
13              
14             my $nop = sub { $_[0] };
15              
16             sub groupBy {
17 9     9 1 70261 my ( $options, @list ) = @_;
18              
19 9 100       29 $options = ref $options eq "ARRAY" ? { keys => $options } : $options;
20              
21 9 100       52 croak "missing grouping keys" unless ref $options->{keys} eq "ARRAY";
22              
23 7         12 my @keys = @{ $options->{keys} };
  7         16  
24            
25 7   100     25 my $default = $options->{defaults} // {};
26              
27 7 100       28 croak "defaults should be a hashref" unless ref $default eq "HASH";
28              
29 6   100     21 $options->{operations} //= {};
30              
31 6 100       23 croak "operations should be a hashref" unless ref $options->{operations} eq "HASH";
32              
33             my %op = map {
34 5   66     11 my $operation = $options->{operations}->{ $_ } // $nop;
  9         28  
35              
36 9 100       27 croak "operation defined should be an anonymous sub" unless ref $operation eq "CODE";
37              
38 8         22 $_ => $operation;
39             } @keys;
40              
41 4         9 my $groupings = {};
42              
43 4         7 my $leaf = pop @keys;
44              
45 4         18 foreach my $item (@list) {
46 20         30 my $current = $groupings;
47              
48 20         26 foreach my $key ( @keys ) {
49 20   100     79 $current = $current->{ $op{ $key }->( $item->{ $key } // $default->{ $key } // '' ) } //= {};
      100        
      100        
50             }
51              
52 20   66     42 push @{ $current->{ $op{ $leaf }->( $item->{$leaf} // $default->{ $leaf } // '' ) } }, $item;
  20   50     56  
53             }
54            
55 4         6 return %{$groupings};
  4         31  
56             }
57              
58              
59             1;
60             __END__