Bug 23846: Add a check to the data inconsistencies script
[koha.git] / catalogue / MARCdetail.pl
1 #!/usr/bin/perl
2
3 # Copyright 2000-2002 Katipo Communications
4 # Copyright 2010 BibLibre
5 #
6 # This file is part of Koha.
7 #
8 # Koha is free software; you can redistribute it and/or modify it
9 # under the terms of the GNU General Public License as published by
10 # the Free Software Foundation; either version 3 of the License, or
11 # (at your option) any later version.
12 #
13 # Koha is distributed in the hope that it will be useful, but
14 # WITHOUT ANY WARRANTY; without even the implied warranty of
15 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 # GNU General Public License for more details.
17 #
18 # You should have received a copy of the GNU General Public License
19 # along with Koha; if not, see <http://www.gnu.org/licenses>.
20
21 =head1 NAME
22
23 MARCdetail.pl : script to show a biblio in MARC format
24
25 =head1 SYNOPSIS
26
27 =cut
28
29 =head1 DESCRIPTION
30
31 This script needs a biblionumber as parameter
32
33 It shows the biblio in a (nice) MARC format depending on MARC
34 parameters tables.
35
36 The template is in <templates_dir>/catalogue/MARCdetail.tt.
37 this template must be divided into 11 "tabs".
38
39 The first 10 tabs present the biblio, the 11th one presents
40 the items attached to the biblio
41
42 =head1 FUNCTIONS
43
44 =cut
45
46 use Modern::Perl;
47 use CGI qw ( -utf8 );
48 use HTML::Entities;
49
50 use C4::Auth;
51 use C4::Context;
52 use C4::Output;
53 use C4::Koha;
54 use MARC::Record;
55 use C4::Biblio;
56 use C4::Items;
57 use C4::Acquisition;
58 use C4::Serials;    #uses getsubscriptionsfrombiblionumber GetSubscriptionsFromBiblionumber
59 use C4::Search;         # enabled_staff_search_views
60
61 use Koha::Biblios;
62 use Koha::BiblioFrameworks;
63 use Koha::Patrons;
64
65 use List::MoreUtils qw( uniq );
66
67 my $query        = new CGI;
68 my $dbh          = C4::Context->dbh;
69 my $biblionumber = $query->param('biblionumber');
70 $biblionumber = HTML::Entities::encode($biblionumber);
71 my $frameworkcode = $query->param('frameworkcode') // GetFrameworkCode( $biblionumber );
72 my $popup        =
73   $query->param('popup')
74   ;    # if set to 1, then don't insert links, it's just to show the biblio
75 my $subscriptionid = $query->param('subscriptionid');
76
77 # open template
78 my ( $template, $loggedinuser, $cookie ) = get_template_and_user(
79     {
80         template_name   => "catalogue/MARCdetail.tt",
81         query           => $query,
82         type            => "intranet",
83         authnotrequired => 0,
84         flagsrequired   => { catalogue => 1 },
85         debug           => 1,
86     }
87 );
88
89 my $record = GetMarcBiblio({
90     biblionumber => $biblionumber,
91     embed_items  => 1 });
92
93 if ( not defined $record ) {
94     # biblionumber invalid -> report and exit
95     $template->param( unknownbiblionumber => 1,
96                 biblionumber => $biblionumber
97     );
98     output_html_with_http_headers $query, $cookie, $template->output;
99     exit;
100 }
101
102 my $biblio_object = Koha::Biblios->find( $biblionumber ); # FIXME Should replace $biblio
103 my $tagslib = &GetMarcStructure(1,$frameworkcode);
104 my $biblio = GetBiblioData($biblionumber);
105
106 if($query->cookie("holdfor")){ 
107     my $holdfor_patron = Koha::Patrons->find( $query->cookie("holdfor") );
108     $template->param(
109         holdfor => $query->cookie("holdfor"),
110         holdfor_surname => $holdfor_patron->surname,
111         holdfor_firstname => $holdfor_patron->firstname,
112         holdfor_cardnumber => $holdfor_patron->cardnumber,
113     );
114 }
115
116 $template->param( ocoins => $biblio_object->get_coins );
117
118 #count of item linked
119 my $itemcount = $biblio_object->items->count;
120 $template->param( count => $itemcount,
121                                         bibliotitle => $biblio->{title}, );
122
123 my $frameworks = Koha::BiblioFrameworks->search( {}, { order_by => ['frameworktext'] } );
124 $template->param(
125     frameworks    => $frameworks,
126     frameworkcode => $frameworkcode,
127 );
128 # fill arrays
129 my @loop_data = ();
130
131 # loop through each tab 0 through 9
132 for ( my $tabloop = 0 ; $tabloop <= 10 ; $tabloop++ ) {
133
134     # loop through each tag
135     my @fields    = $record->fields();
136     my @loop_data = ();
137     my @subfields_data;
138
139     # deal with leader
140     unless ( $tagslib->{'000'}->{'@'}->{tab} ne $tabloop )
141     {    #  or ($tagslib->{'000'}->{'@'}->{hidden} =~ /-7|-4|-3|-2|2|3|5|8/ )) {
142         my %subfield_data;
143         $subfield_data{marc_lib}      = $tagslib->{'000'}->{'@'}->{lib};
144         $subfield_data{marc_value}    = $record->leader();
145         $subfield_data{marc_subfield} = '@';
146         $subfield_data{marc_tag}      = '000';
147         push( @subfields_data, \%subfield_data );
148         my %tag_data;
149         $tag_data{tag} = '000';
150         $tag_data{tag_desc} = $tagslib->{'000'}->{lib};
151         my @tmp = @subfields_data;
152         $tag_data{subfield} = \@tmp;
153         push( @loop_data, \%tag_data );
154         undef @subfields_data;
155     }
156     @fields = $record->fields();
157     for ( my $x_i = 0 ; $x_i <= $#fields ; $x_i++ ) {
158
159         # if tag <10, there's no subfield, use the "@" trick
160         if ( $fields[$x_i]->tag() < 10 ) {
161             next
162               if (
163                 $tagslib->{ $fields[$x_i]->tag() }->{'@'}->{tab} ne $tabloop );
164             next if ( $tagslib->{ $fields[$x_i]->tag() }->{'@'}->{hidden} =~ /-7|-4|-3|-2|2|3|5|8/);
165             my %subfield_data;
166             $subfield_data{marc_lib} =
167               $tagslib->{ $fields[$x_i]->tag() }->{'@'}->{lib};
168             $subfield_data{marc_value}    = $fields[$x_i]->data();
169             $subfield_data{marc_subfield} = '@';
170             $subfield_data{marc_tag}      = $fields[$x_i]->tag();
171             push( @subfields_data, \%subfield_data );
172         }
173         else {
174             my @subf = $fields[$x_i]->subfields;
175
176             # loop through each subfield
177             for my $i ( 0 .. $#subf ) {
178                 $subf[$i][0] = "@" unless defined $subf[$i][0];
179                 next
180                   if (
181                     $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }->{tab}
182                     ne $tabloop );
183                 next
184                   if ( $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }
185                     ->{hidden} =~ /-7|-4|-3|-2|2|3|5|8/);
186                 my %subfield_data;
187                 $subfield_data{short_desc} = $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }->{lib};
188                 $subfield_data{long_desc} =
189                   $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }->{lib};
190                 $subfield_data{link} =
191                   $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }->{link};
192
193 #                 warn "tag : ".$tagslib->{$fields[$x_i]->tag()}." subfield :".$tagslib->{$fields[$x_i]->tag()}->{$subf[$i][0]}."lien koha? : "$subfield_data{link};
194                 if ( $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }
195                     ->{isurl} )
196                 {
197                     $subfield_data{marc_value} = $subf[$i][1];
198                                         $subfield_data{is_url} = 1;
199                 }
200                 elsif ( $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }
201                     ->{kohafield} eq "biblioitems.isbn" )
202                 {
203
204 #                    warn " tag : ".$tagslib->{$fields[$x_i]->tag()}." subfield :".$tagslib->{$fields[$x_i]->tag()}->{$subf[$i][0]}. "ISBN : ".$subf[$i][1]."PosttraitementISBN :".DisplayISBN($subf[$i][1]);
205                     $subfield_data{marc_value} = $subf[$i][1];
206                 }
207                 else {
208                     if ( $tagslib->{ $fields[$x_i]->tag() }->{ $subf[$i][0] }
209                         ->{authtypecode} )
210                     {
211                         $subfield_data{authority} = $fields[$x_i]->subfield(9);
212                     }
213                     $subfield_data{marc_value} =
214                       GetAuthorisedValueDesc( $fields[$x_i]->tag(),
215                         $subf[$i][0], $subf[$i][1], '', $tagslib) || $subf[$i][1];
216
217                 }
218                 $subfield_data{marc_subfield} = $subf[$i][0];
219                 $subfield_data{marc_tag}      = $fields[$x_i]->tag();
220                 push( @subfields_data, \%subfield_data );
221             }
222         }
223         if ( $#subfields_data == 0 ) {
224             $subfields_data[0]->{marc_lib}      = '';
225 #            $subfields_data[0]->{marc_subfield} = '';
226         }
227         if ( $#subfields_data >= 0) {
228             my %tag_data;
229             if ( $fields[$x_i]->tag() eq $fields[ $x_i - 1 ]->tag() && (C4::Context->preference('LabelMARCView') eq 'economical')) {
230                 $tag_data{tag} = "";
231             }
232             else {
233                 if ( C4::Context->preference('hide_marc') ) {
234                     $tag_data{tag} = $tagslib->{ $fields[$x_i]->tag() }->{lib};
235                 }
236                 else {
237                     $tag_data{tag} = $fields[$x_i]->tag();
238             $tag_data{tag_ind} = C4::Koha::display_marc_indicators($fields[$x_i]);
239             $tag_data{tag_desc} = $tagslib->{ $fields[$x_i]->tag() }->{lib};
240                 }
241             }
242             my @tmp = @subfields_data;
243             $tag_data{subfield} = \@tmp;
244             push( @loop_data, \%tag_data );
245             undef @subfields_data;
246         }
247     }
248     $template->param( "tab" . $tabloop . "XX" => \@loop_data );
249 }
250
251 # now, build item tab !
252 # the main difference is that datas are in lines and not in columns : thus, we build the <th> first, then the values...
253 # loop through each tag
254 # warning : we may have differents number of columns in each row. Thus, we first build a hash, complete it if necessary
255 # then construct template.
256 my @fields = $record->fields();
257 my %witness
258   ; #---- stores the list of subfields used at least once, with the "meaning" of the code
259 my @item_subfield_codes;
260 my @item_loop;
261 my $norequests = 1;
262 foreach my $field (@fields) {
263     next if ( $field->tag() < 10 );
264     my @subf = $field->subfields;
265     my $item;
266
267     # loop through each subfield
268     for my $i ( 0 .. $#subf ) {
269         next if ( $tagslib->{ $field->tag() }->{ $subf[$i][0] }->{tab} ne 10 );
270         next if ( $tagslib->{ $field->tag() }->{ $subf[$i][0] }->{hidden} =~ /-7|-4|-3|-2|2|3|5|8/);
271         push @item_subfield_codes, $subf[$i][0];
272         $witness{ $subf[$i][0] } =
273         $tagslib->{ $field->tag() }->{ $subf[$i][0] }->{lib};
274         $item->{ $subf[$i][0] } = GetAuthorisedValueDesc( $field->tag(),
275                         $subf[$i][0], $subf[$i][1], '', $tagslib) || $subf[$i][1];
276         $norequests = 0
277           if  $tagslib->{ $field->tag() }->{ $subf[$i][0] }->{kohafield} eq 'items.notforloan'
278           and $subf[$i][1] == 0;
279     }
280     push @item_loop, $item if $item;
281 }
282
283 my ($holdingbrtagf,$holdingbrtagsubf) = &GetMarcFromKohaField( "items.holdingbranch" );
284 @item_loop = sort {$a->{$holdingbrtagsubf} cmp $b->{$holdingbrtagsubf}} @item_loop;
285
286 @item_subfield_codes = uniq @item_subfield_codes;
287 # fill item info
288 my @item_header_loop;
289 for my $subfield_code ( @item_subfield_codes ) {
290     push @item_header_loop, $witness{$subfield_code};
291     for my $item_data ( @item_loop ) {
292         $item_data->{$subfield_code} ||= "&nbsp;"
293     }
294 }
295
296 my $subscriptionscount = CountSubscriptionFromBiblionumber($biblionumber);
297
298 if ($subscriptionscount) {
299     my $subscriptions = GetSubscriptionsFromBiblionumber($biblionumber);
300     my $subscriptiontitle = $subscriptions->[0]{'bibliotitle'};
301     $template->param(
302         subscriptiontitle   => $subscriptiontitle,
303         subscriptionsnumber => $subscriptionscount,
304     );
305 }
306
307 $template->param (
308     norequests              => $norequests,
309     item_loop               => \@item_loop,
310     item_header_loop        => \@item_header_loop,
311     item_subfield_codes     => \@item_subfield_codes,
312     biblionumber            => $biblionumber,
313     popup                   => $popup,
314     hide_marc               => C4::Context->preference('hide_marc'),
315         marcview => 1,
316         z3950_search_params             => C4::Search::z3950_search_args($biblio),
317         C4::Search::enabled_staff_search_views,
318     searchid                => scalar $query->param('searchid'),
319     biblio                  => $biblio_object->unblessed,
320 );
321
322 my @allorders_using_biblio = GetOrdersByBiblionumber ($biblionumber);
323 my @deletedorders_using_biblio;
324 my @orders_using_biblio;
325 my @baskets_orders;
326 my @baskets_deletedorders;
327
328 foreach my $myorder (@allorders_using_biblio) {
329     my $basket = $myorder->{'basketno'};
330     if ((defined $myorder->{'datecancellationprinted'}) and  ($myorder->{'datecancellationprinted'} ne '0000-00-00') ){
331         push @deletedorders_using_biblio, $myorder;
332         unless (grep(/^$basket$/, @baskets_deletedorders)){
333             push @baskets_deletedorders,$myorder->{'basketno'};
334         }
335     }
336     else {
337         push @orders_using_biblio, $myorder;
338         unless (grep(/^$basket$/, @baskets_orders)){
339             push @baskets_orders,$myorder->{'basketno'};
340             }
341     }
342 }
343
344 my $count_orders_using_biblio = scalar @orders_using_biblio ;
345 $template->param (countorders => $count_orders_using_biblio);
346
347 my $count_deletedorders_using_biblio = scalar @deletedorders_using_biblio ;
348 $template->param (countdeletedorders => $count_deletedorders_using_biblio);
349
350 $biblio = Koha::Biblios->find( $biblionumber );
351 my $holds = $biblio->holds;
352 $template->param( holdcount => $holds->count );
353
354 output_html_with_http_headers $query, $cookie, $template->output;