1 package SL::Helper::Csv::Dispatcher;
7 use Scalar::Util qw(weaken);
8 use List::MoreUtils qw(all pairwise);
9 use Rose::Object::MakeMethods::Generic scalar => [ qw(
10 _specs _row_class _row_spec _errors
13 use SL::Helper::Csv::Error;
16 my ($class, $parent) = @_;
17 my $self = bless { }, $class;
19 weaken($self->{_csv} = $parent);
26 my ($self, $line) = @_;
28 my $class = $self->_class_by_line($line);
29 croak 'no class given' unless $class;
31 eval "require " . $class;
32 my $obj = $class->new;
34 my $specs = $self->_specs_by_line($line);
35 for my $spec (@{ $specs }) {
36 $self->apply($obj, $spec, $line->{$spec->{key}});
43 my ($self, $line) = @_;
45 # initialize lookup hash if not already done
46 if ($self->_csv->is_multiplexed && ! defined $self->_row_class ) {
47 $self->_row_class({ map { $_->{row_ident} => $_->{class} } @{ $self->_csv->profile } });
50 if ($self->_csv->is_multiplexed) {
51 return $self->_row_class->{$line->{datatype}};
53 return $self->_csv->profile->[0]->{class};
58 my ($self, $line) = @_;
60 # initialize lookup hash if not already done
61 if ($self->_csv->is_multiplexed && ! defined $self->_row_spec ) {
62 $self->_row_spec({ pairwise { no warnings 'once'; $a->{row_ident} => $b } @{ $self->_csv->profile }, @{ $self->_specs } });
65 if ($self->_csv->is_multiplexed) {
66 return $self->_row_spec->{$line->{datatype}};
68 return $self->_specs->[0];
74 my ($self, $obj, $spec, $value) = @_;
77 for my $step (@{ $spec->{steps} }) {
78 my ($acc, $class, $index) = @$step;
83 if (! $obj->$acc || !$obj->$acc->[$index]) {
84 my @objects = $obj->$acc;
85 $obj->$acc(@objects, map { $class->new } 0 .. $index - @objects);
87 $obj = $obj->$acc->[$index];
90 $obj->$acc($class->new);
102 my ($self, $col, $row) = @_;
103 return grep { $col eq $_->{key} } @{ $self->_specs->[$row // 0] };
107 my ($self, %params) = @_;
111 my $csv_profile = $self->_csv->profile;
112 my $h_aref = ($self->_csv->is_multiplexed)? $self->_csv->header : [ $self->_csv->header ];
114 foreach my $header (@{ $h_aref }) {
115 my $spec = $self->_parse_profile(profile => $csv_profile->[$i]->{profile},
116 mapping => $csv_profile->[$i]->{mapping},
117 class => $csv_profile->[$i]->{class},
123 $self->_specs(\@specs);
125 $self->_csv->_push_error($self->errors);
127 return ! $self->errors;
131 my ($self, %params) = @_;
133 my $profile = $params{profile} // {};
134 my $class = $params{class};
135 my $header = $params{header};
136 my $mapping = $params{mapping};
140 for my $col (@$header) {
142 if (exists $mapping->{$col} && $profile->{$mapping->{$col}}) {
143 push @specs, $self->make_spec($col, $profile->{$mapping->{$col}}, $class);
144 } elsif (exists $mapping->{$col} && !%{ $profile }) {
145 push @specs, $self->make_spec($col, $mapping->{$col}, $class);
146 } elsif (exists $profile->{$col}) {
147 push @specs, $self->make_spec($col, $profile->{$col}, $class);
149 if ($self->_csv->strict_profile) {
150 $self->unknown_column($col, undef);
152 push @specs, $self->make_spec($col, $col, $class);
161 my ($self, $col, $path, $cur_class) = @_;
163 my $spec = { key => $col, path => $path, steps => [] };
167 return unless $cur_class;
169 for my $step_index ( split /\.(?!\d)/, $path ) {
170 my ($step, $index) = split /\./, $step_index;
171 if ($cur_class->can($step)) {
172 if (my $rel = $cur_class->meta->relationship($step)) { #a
173 if ($index && ! $rel->isa('Rose::DB::Object::Metadata::Relationship::OneToMany')) {
177 "Profile path error. Indexed relationship is not OneToMany around here: '$step_index'",
183 my $next_class = $cur_class->meta->relationship($step)->class;
184 push @{ $spec->{steps} }, [ $step, $next_class, $index ];
185 $cur_class = $next_class;
186 eval "require $cur_class; 1" or die "could not load class '$cur_class'";
188 } else { # simple dispatch
189 push @{ $spec->{steps} }, [ $step ];
193 $self->unknown_column($col, $path);
201 my ($self, $col, $path) = @_;
202 return if $self->_csv->ignore_unknown_columns;
207 "header field '$col' is not recognized",
222 my ($self, @errors) = @_;
223 my @new_errors = ($self->errors, map { SL::Helper::Csv::Error->new(@$_) } @errors);
224 $self->_errors(\@new_errors);