package LIMS::Controller::PrintRun; use base 'LIMS::Base'; use LIMS::Local::Utils; use Data::Dumper; use Moose; with ( 'LIMS::Controller::Roles::DataMap', 'LIMS::Controller::Roles::RecordHandler', 'LIMS::Controller::Roles::ResultHandler', ); has reports => ( is => 'ro', isa => 'ArrayRef[HashRef]', default => sub { [] }, lazy => 1, traits => ['Array'], handles => { add_to_reports => 'push', }, ); __PACKAGE__->meta->make_immutable(inline_constructor => 0); #------------------------------------------------------------------------------- sub default : StartRunmode { my $self = shift; $self->_debug_path($self->get_current_runmode); return $self->forbidden() unless $self->user_can('print_all'); my $last_working_day = $self->_get_last_working_day(); return $self->tt_process({ date => $last_working_day }); } #------------------------------------------------------------------------------- sub do : Runmode { my $self = shift; $self->_debug_path($self->get_current_runmode); # get start & end datetime objects: my $start_date = $self->_get_start_date; my $end_date = $self->_get_end_date; my %args = ( start_date => $start_date, end_date => $end_date, ); # exclude any locations which don't require paper copies: if ( my $cfg = $self->get_yaml_file('skip_paper_reports') ) { $args{skip_reports} = $cfg; } # first find out how many have been requested: my $count = $self->model('PrintRun')->get_print_run_size(\%args); if (! $count) { $self->flash( info => $self->messages('search')->{no_records_found} ); return $self->redirect( $self->query->url . '/printrun' ); } # if more than max print_run size (defined in settings), split load: elsif ($count > $self->cfg('settings')->{print_run_max}) { my %args = ( count => $count, start_date => $start_date, # $end defaults to 00:00:00 on current_date (unless use_today selected) end_date => $end_date->subtract(seconds => 1), # so subtract 1 sec for .tt param ); return $self->tt_process(\%args); } else { $self->stash( date_span => { start => $start_date, end => $end_date } ); return $self->forward('_do_print'); # forward so tt picks up 'print' in runmode } } #------------------------------------------------------------------------------- # converts form start & end date strings back to DT objects, forwards to _do_print(): sub do_offset : Runmode { my $self = shift; $self->_debug_path($self->get_current_runmode); my $p = $self->query->Vars(); # to_datetime_using_parsedate() seems to work: # my $start = LIMS::Local::Utils::to_datetime_using_parsedate($p->{start}); # my $end = LIMS::Local::Utils::to_datetime_using_parsedate($p->{end}); # to_datetime_using_strptime() config'd to use supplied pattern: my $dt = sub { # datetime string eg 2010-01-01T00:00:00 LIMS::Local::Utils::to_datetime_using_strptime(@_, '%FT%T'); }; my $start = &$dt($p->{start}); # warn $start->datetime; my $end = &$dt($p->{end}); # warn $end->datetime; $self->stash( date_span => { start => $start, end => $end } ); return $self->forward('_do_print'); # forward so tt picks up 'print' in runmode } #------------------------------------------------------------------------------- sub _do_print : Runmode { # rm so template picks up 'print' in get_current_runmode my $self = shift; $self->_debug_path($self->get_current_runmode); my @sort_by = qw(referrers.name requests.year requests.request_number); my $dates = $self->stash->{date_span}; my %args = ( start_date => $dates->{start}, end_date => $dates->{end}, sort_by => \@sort_by, ); # exclude any locations which don't require paper copies: if ( my $cfg = $self->get_yaml_file('skip_paper_reports') ) { $args{skip_reports} = $cfg; } # warn Dumper \%args; # offset supplied if total > max print_run size: $args{offset} = $self->query->param('offset') || 0; # default to beginning $args{limit} = $self->cfg('settings')->{print_run_max} || 50; # sensible default # get request id's for records amended between start & end datetimes: my $request_ids = $self->model('PrintRun')->get_print_run_request_ids(\%args); unless (@$request_ids) { # should never get here - count checked in do() $self->flash( info => $self->messages('search')->{no_records_found} ); return $self->redirect( $self->query->url . '/printrun' ); } # my $i; foreach my $request_id (@$request_ids) { # last if $i++ > 1; # for testing # format request data for template: my $request_data = $self->get_single_request_data($request_id); # add processed lab test data to $request_data: $self->process_raw_lab_test_data($request_data); $self->add_to_reports($request_data); # warn Dumper $request_data; { # has copy_to been requested? my $request_options = $request_data->{request_options}; # warn Dumper $request_options; if ( $request_options->{copy_to}->{is_selected} ) { my $copy = $self->_generate_copy($request_data); $self->add_to_reports($copy); } } } my $tmpl = 'record/default.tt'; return $self->render_view($tmpl, { reports => $self->reports }); } =begin sub _do_print_old { my ($self, $start, $end) = @_; $self->_debug_path(); my @sort_by = qw(referrers.name requests.year requests.request_number); my %args = ( start_date => $start, end_date => $end, sort_by => \@sort_by, ); # exclude any locations which don't require paper copies: if ( my $cfg = $self->get_yaml_file('skip_paper_reports') ) { $args{skip_reports} = $cfg; } # warn Dumper \%args; # offset supplied if total > max print_run size (maybe '0' so use defined): # if ( defined $self->query->param('offset') ) { # always submitted now $args{offset} = $self->query->param('offset') || 0; # default to beginning $args{limit} = $self->cfg('settings')->{print_run_max} || 50; # sensible default # } # get request id's for records amended between start & end datetimes: my $request_ids = $self->model('PrintRun')->get_print_run_request_ids(\%args); unless (@$request_ids) { # should never get here - count checked in do() $self->flash( info => $self->messages('search')->{no_records_found} ); return $self->redirect( $self->query->url . '/printrun' ); } # allowed report error_codes: my $report_error_codes = $self->report_error_codes_map; # diagnosis_context_warnings: my $context_warning = $self->diagnosis_context_warning_map; # categories exempt from 'use NHS number': my $nhs_number_exempt = $self->nhs_number_exempt; # preserve 'app_url' & 'active_link' tt_params for use in loop: my %required_tt_params = map { $_ => $self->tt_params->{$_}; } qw(app_url active_link); # my $i; foreach my $request_id (@$request_ids) { # last if $i++ > 1; # for testing # format request data for template: my $request_data = $self->format_print_record_request_data($request_id); # warn Dumper $request_id; # add data maps for template: $self->tt_params( context_warning_map => $context_warning, nhs_number_exempt => $nhs_number_exempt, report_error_codes => $report_error_codes, request => $request_data, ); # template depends on whether it's outreach data: my $tmpl = $request_data->{outreach} ? 'outreach/report_body.tt' : 'record/body.tt'; # warn $tmpl; # render template & add to 'reports' attr: my $html = $self->render_view($tmpl); $self->add_to_reports($$html); # dereference first { # has copy_to been requested? my $request_options = $request_data->{request_options}; # warn Dumper $request_options; if ( $request_options->{copy_to}->{is_selected} ) { my $cc = $self->_generate_copy($html); $self->add_to_reports($cc); # already deref'ed } } $self->tt_clear_params; # clear all template params for next loop $self->tt_params(\%required_tt_params); # add required ones back } return $self->tt_process('printrun/wrapper.tt', { reports => $self->reports }); } =cut #------------------------------------------------------------------------------- sub _generate_copy { my ($self, $ref) = @_; $self->_debug_path(); # clone() of $ref generates thousands of warning msgs in Devel::Cycle # (ie dev server), caused by presence of a LIMS::DB object (OK if as_tree'd): # my $clone = LIMS::Local::Utils::clone($ref); $self->debug($clone); # take hash copy of $ref so original unaffected: my %h = %$ref; # take %h LIMS::DB::Request 'data' object into a hashref and clone it: my $data = $h{data}; # warn ref $data; # LIMS::DB::Request object # clone hashref NOT object - vital to prevent Devel::Cycle warnings: my $cloned = LIMS::Local::Utils::clone( $data->as_tree(deflate => 0) ); # replace referrer name with 'cc' flag for template: $cloned->{referrer_department}->{referrer}->{name} = 'cc'; # warn Dumper $clone; # replace original $h{data} DB::Request object with cloned hashref (OK for tt): $h{data} = $cloned; # return %h as hashref, same as original $ref, but with referrer name changed: return \%h; } #------------------------------------------------------------------------------- sub _get_start_date { my $self = shift; $self->_debug_path(); my $q = $self->query; my $datetime; # expect params 'last_working_day' OR 'date', and optional 'today': if ( $q->param('last_working_day') ) { $datetime = $self->_get_last_working_day(); } elsif ( my $date = $q->param('date') ) { $datetime = LIMS::Local::Utils::to_datetime_using_datecalc($date) || return $self->error("cannot decode date-string $date"); } # warn Dumper $datetime; # in case start date not supplied: $datetime ||= LIMS::Local::Utils::today(); # beginning of day (00:00:00) return $datetime; } #------------------------------------------------------------------------------- sub _get_end_date { my $self = shift; $self->_debug_path(); my $datetime = $self->query->param('today') ? LIMS::Local::Utils::time_now() # up to current time (BST-aware) : LIMS::Local::Utils::today(); # beginning of day (00:00:00) return $datetime; } #------------------------------------------------------------------------------- sub _get_last_working_day { # doesn't consider public holidays - does now !! my $self = shift; $self->_debug_path(); # function moved to LLU (returns DT object): return LIMS::Local::Utils::last_working_date(); # considers public hols =begin my $dt = DateTime->today->subtract( days => 1 ); # yesterday at 00:00:00 # day 6 is Saturday, day 7 is Sunday while ( $dt->day_of_week >= 6 ) { $dt->subtract( days => 1 ); } # warn Dumper $dt; return $dt; =cut } 1;