/[webpac]/trunk/all2xml.pl
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /trunk/all2xml.pl

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 108 by dpavlin, Mon Jul 14 18:20:27 2003 UTC revision 626 by dpavlin, Sun Jan 2 00:53:33 2005 UTC
# Line 1  Line 1 
1  #!/usr/bin/perl -w  #!/usr/bin/perl -w
2    
3  use strict;  use strict;
4  use OpenIsis;  use IsisDB;
5  use Getopt::Std;  use Getopt::Std;
6  use Data::Dumper;  use Data::Dumper;
7  use XML::Simple;  use XML::Simple;
 use Text::Unaccent 1.02;        # 1.01 won't compile on my platform,  
8  use Text::Iconv;  use Text::Iconv;
9  use Config::IniFiles;  use Config::IniFiles;
10  use Encode;  use Encode;
11    #use GDBM_File;
12    use Fcntl;      # for O_RDWR
13    use TDB_File;
14    
15  $|=1;  $|=1;
16    
17  my $config_file = $0;  my $config_file = $0;
18  $config_file =~ s/\.pl$/.conf/;  $config_file =~ s/\.pl$/.conf/;
19    $config_file = $ARGV[0] if ($ARGV[0] && -f $ARGV[0]);
20  die "FATAL: can't find configuration file '$config_file'" if (! -e $config_file);  die "FATAL: can't find configuration file '$config_file'" if (! -e $config_file);
21    
22  my $config;  my $config;
# Line 61  my %type2tag = ( Line 64  my %type2tag = (
64          'feed' => 'feed'          'feed' => 'feed'
65  );  );
66    
67    my $cache;      # for cacheing
68    
69    # lookup hash (tied to file)
70    my %lhash;
71    # this option will cache all lookup entries in memory.
72    # if you are tight on memory, turn this off
73    my $use_lhash_cache = 1;
74    
75    my $last_field_name;    # cache to prevent repeated fields
76    
77  sub data2xml {  sub data2xml {
78    
79          use xmlify;          use xmlify;
# Line 90  sub data2xml { Line 103  sub data2xml {
103                  return $va <=> $vb;                  return $va <=> $vb;
104          }          }
105    
106          foreach my $field (sort by_order keys %{$config->{indexer}}) {          my @sorted_tags;
107            if ($cache->{tags_by_order}) {
108                    @sorted_tags = @{$cache->{tags_by_order}};
109            } else {
110                    @sorted_tags = sort by_order keys %{$config->{indexer}};
111                    $cache->{tags_by_order} = \@sorted_tags;
112            }
113    
114            # lookup key
115            my $lookup_key;
116    
117            # cache for field in pages
118            delete $cache->{display_data};
119            delete $cache->{swish_data};
120            delete $cache->{swish_exact_data};
121            delete $cache->{index_data};
122            delete $cache->{index_delimiter};
123            my @page_fields;        # names of fields
124    
125    
126            # subs used to produce output
127    
128            sub get_field_name($$$) {
129                    my ($config,$field,$field_usage) = @_;
130    
131                    # find field name (signular, plural)
132                    my $field_name = "";
133                    if ($config->{indexer}->{$field}->{name_singular} && $field_usage == 1) {
134                            $field_name = $config->{indexer}->{$field}->{name_singular};
135                    } elsif ($config->{indexer}->{$field}->{name_plural}) {
136                            $field_name = $config->{indexer}->{$field}->{name_plural};
137                    } elsif ($config->{indexer}->{$field}->{name}) {
138                            $field_name = $config->{indexer}->{$field}->{name};
139                    } else {
140                            print STDERR "WARNING: field '$field' doesn't have 'name' attribute!";
141                    }
142    
143                    if ($field_name) {
144                            $field_name = x($field_name);
145                            if (! $last_field_name) {
146                                    $last_field_name = $field_name;
147                                    return $last_field_name;
148                            } elsif ($field_name ne $last_field_name) {
149                                    $last_field_name = $field_name;
150                                    return $last_field_name;
151                            }
152                    }
153            }
154    
155    
156            # init variables for different types
157            sub init_visible_type($) {
158                    my $type = shift;
159    
160                    # swish, swish_exact, display, index, index_lookup
161                    # swish and display defaults
162                    my ($s,$se,$d,$i,$il) = (1,0,1,0,0);
163                    if (lc($type) eq "display") {
164                            $s = 0;
165                    } elsif (lc($type) eq "swish") {
166                            $d = 0;
167                    } elsif (lc($type) eq "index") {
168                            ($s,$se,$d,$i) = (0,1,0,1);
169                    } elsif (lc($type) eq "swish_exact") {
170                            ($s,$se,$d,$i) = (0,1,0,0);
171                    } elsif (lc($type) =~ /^lookup/) {
172                            ($s,$se,$d,$i,$il) = (0,1,0,0,1);
173                    } elsif ($type) {
174                            print STDERR "WARNING: unknown type: $type\n";
175                    }
176                    return ($s,$se,$d,$i,$il);
177            }
178    
179    
180            # convert
181            #
182            # <tag>
183            #  <delimiter>, </delimiter>
184            #  <value>200a</value>
185            # </tag>
186            #
187            # to
188            #
189            # <tag delimiter=", ">200a</tag>
190            #
191            # but without loosing spaces in delimiter (becasue
192            # new XML::Simple strips spaces in attribute values
193            # as defined in XML specification)
194            #
195            sub unroll_x($) {
196                    my $x = shift;
197    
198                    if (defined $x->{value}) {
199                            my ($v,$d) = ($x->{value}->{content}, $x->{delimiter}->{content});
200                            delete $x->{value};
201                            delete $x->{delimiter};
202                            $x->{content} = $v;
203                            $x->{delimiter} = $d;
204                    }
205                    return $x;
206            }
207    
208            # begin real work: go field by field
209            foreach my $field (@sorted_tags) {
210    
211                  $field=x($field);                  $field=x($field);
212                  $field_usage{$field}++;                  $field_usage{$field}++;
213    
214                  my $swish_data = "";                  my $swish_data = "";
215                    my $swish_exact_data = "";
216                  my $display_data = "";                  my $display_data = "";
217                    my @index_data;
218                  my $line_delimiter;                  my $line_delimiter;
219    
220                  my ($swish,$display);                  my ($swish,$display);
221    
222                  my $tag = $type2tag{$type} || die "can't find which tag to use for type $type";                  my $tag = $type2tag{$type} || die "can't find which tag to use for type $type";
223    
224                    # is this field page-by-page?
225                    my $iterate_by_page = $config->{indexer}->{$field}->{iterate_by_page};
226                    push @page_fields,$field if ($iterate_by_page);
227                    my %page_max = ();
228                    # default line_delimiter if using
229                    my $page_line_delimiter = $config->{indexer}->{$field}->{page_line_delimiter} || '<br/>';
230                    $cache->{index_delimiter}->{$field} = $config->{indexer}->{$field}->{index_delimiter};
231    
232                    my $format_name = $config->{indexer}->{$field}->{format_name};
233                    my $format_delimiter = $config->{indexer}->{$field}->{format_delimiter};
234                    if ($format_name && $format_delimiter) {
235                            $cache->{format}->{$field}->{format_name} = $format_name;
236                            $cache->{format}->{$field}->{format_delimiter} = $format_delimiter;
237                    }
238    
239                  foreach my $x (@{$config->{indexer}->{$field}->{$tag}}) {                  foreach my $x (@{$config->{indexer}->{$field}->{$tag}}) {
240    
241                            $x = unroll_x($x);
242    
243                          my $format = x($x->{content});                          my $format = x($x->{content});
244                          my $delimiter = x($x->{delimiter}) || ' ';                          my $delimiter = x($x->{delimiter}) || ' ';
245    
246                          my $repeat_off = 0;             # repeatable offset                          my $repeat_off = 0;     # init repeatable offset
247    
248                          my ($s,$d,$i) = (1,1,0);        # swish, display default                          my ($s,$se,$d,$i,$il) = init_visible_type($x->{type});
                         $s = 0 if (lc($x->{type}) eq "display");  
                         $d = 0 if (lc($x->{type}) eq "swish");  
                         ($s,$d,$i) = (0,0,1) if (lc($x->{type}) eq "index");  
249    
250                          # what will separate last line from this one?                          # what will separate last line from this one?
251                          if ($display_data && $x->{append} && $x->{append} eq "1") {                          if ($display_data && $x->{append}) {
252                                  $line_delimiter = ' ';                                  $line_delimiter = $delimiter;
253                          } elsif ($display_data) {                          } elsif ($display_data) {
254                                  $line_delimiter = '<br/>';                                  $line_delimiter = '<br/>';
255                          }                          }
# Line 125  sub data2xml { Line 258  sub data2xml {
258                          ($swish,$display) = (1,1);                          ($swish,$display) = (1,1);
259    
260                          # placeholder for all repeatable entries for index                          # placeholder for all repeatable entries for index
261                          my @index_data;  
262                          my $index_filter;                          sub mkformat($$) {
263                                    my $x = shift || die "mkformat needs tag reference";
264                                    my $data = shift || return;
265                                    my $format_name = x($x->{format_name}) || return $data;
266                                    my $fmt = x($config->{format}->{$format_name}->{content}) || die "<format name=\"$format_name\"> is not defined!";
267                                    my $format_delimiter = x($x->{format_delimiter});
268                                    my @data;
269                                    if ($format_delimiter) {
270                                            @data = split(/$format_delimiter/,$data);
271                                    } else {
272                                            push @data,$data;
273                                    }
274    
275                                    if ($fmt) {
276                                            my $nr = scalar $fmt =~ s/%s/%s/g;
277                                            if (($#data+1) == $nr) {
278                                                    return sprintf($fmt,@data);
279                                            } else {
280                                                    #print STDERR "mkformat: [$data] can't be split on [$format_delimiter] to $nr fields!\n";
281                                                    return $data;
282                                            }
283                                    } else {
284                                            print STDERR "usage of link '$format_name' without defined format (<link> tag)\n";
285                                    }
286                            }
287    
288                          # while because of repeatable fields                          # while because of repeatable fields
289                          while ($swish || $display) {                          while ($swish || $display) {
290                                    my $page = $repeat_off;
291                                    $page_max{$field} = $page if ($iterate_by_page && $page > ($page_max{$field} || 0));
292                                  ($swish,$display) = parse_format($type, $format,$row,$repeat_off++,$import2cp);                                  ($swish,$display) = parse_format($type, $format,$row,$repeat_off++,$import2cp);
293                                  if ($repeat_off > 1000) {                                  if ($repeat_off > 1000) {
294                                          print STDERR "loop (more than 1000 repeatable fields) deteced in $row, $format\n";                                          print STDERR "loop (more than 1000 repeatable fields) deteced in $row, $format\n";
295                                          last;                                          last;
296                                  }                                  }
297                                    
298                                    # is this field is lookup?
299                                    if ($display && $x->{lookup}) {
300                                            my $null = "<!-- null -->";
301                                            if ($use_lhash_cache) {
302                                                    if (!defined($cache->{lhash}->{$display})) {
303                                                            my $new_display = $lhash{$display};
304                                                            if (defined($new_display)) {
305    #print STDERR "lookup cache store '$display' = '$new_display'\n";
306                                                                    $display = $new_display;
307                                                                    $cache->{lhash}->{$display} = $new_display;
308                                                            } else {
309    #                                                               print STDERR "WARNING: lookup for '$display' didn't find anything.\n";
310                                                                    $display = "";
311                                                                    $cache->{lhash}->{$display} = $null;
312                                                            }
313                                                    } else {
314                                                            $display = $cache->{lhash}->{$display};
315                                                    }
316                                            } else {
317                                                    $display = $lhash{$display} || $null;
318                                            }
319                                    }
320    
321                                  # filter="name" ; filter this field through                                  # filter="name" ; filter this field through
322                                  # filter/[name].pm                                  # filter/[name].pm
323                                  my $filter = $x->{filter};                                  my $filter = $x->{filter};
324                                  if ($filter) {                                  if ($filter && !$cache->{filter_loaded}->{$filter}) {
325                                          require "filter/".$filter.".pm";                                          require "filter/".$filter.".pm";
326                                            $cache->{filter_loaded}->{$filter}++;
327                                  }                                  }
328                                  # type="swish" ; field for swish                                  # type="swish" ; field for swish
329                                  if ($s && $swish) {                                  if ($swish) {
330                                          if ($filter) {                                          my $tmp = $swish;
331                                            if ($filter && ($s || $se)) {
332                                                  no strict 'refs';                                                  no strict 'refs';
333                                                  $swish_data .= join(" ",&$filter($swish));                                                  $tmp = join(" ",&$filter($tmp)) if ($s || $se);
                                         } else {  
                                                 $swish_data .= $swish;  
334                                          }                                          }
335    
336                                            $swish_data .= $tmp if ($s && $tmp);
337                                            $swish_exact_data .= "xxbxx $tmp xxexx " if ($tmp && $tmp ne "" && $se);
338                                  }                                  }
339    
340                                  # type="display" ; field for display                                  # type="display" ; field for display
341                                  if ($d && $display) {                                  if ($d && $display) {
342                                            my $ldel = $delimiter;
343                                          if ($line_delimiter && $display_data) {                                          if ($line_delimiter && $display_data) {
344                                                  $display_data .= $line_delimiter;                                                  $ldel = $line_delimiter;
                                                 undef $line_delimiter;  
345                                          }                                          }
346                                          if ($filter) {                                          if ($filter) {
347                                                  no strict 'refs';                                                  no strict 'refs';
348                                                  $display_data .= join($delimiter,&$filter($display));                                                  my @arr;
349                                          } else {                                                  foreach my $tmp (&$filter($display)) {
350                                                  if ($display_data) {                                                          my $tmp2 = mkformat($x,$tmp);
351                                                          $display_data .= $delimiter.$display;                                                          push @arr,$tmp2 if ($tmp2);
                                                 } else {  
                                                         $display_data .= $display;  
352                                                  }                                                  }
353                                                    $display_data .= $ldel if ($display_data && @arr);
354                                                    $display_data .= join($delimiter,@arr);
355                                            } else {
356                                                    $display_data .= $ldel if ($display_data);
357                                                    my $tmp = mkformat($x,$display);
358                                                    $display_data .= $tmp if ($tmp);
359                                          }                                          }
360                                  }                                  }
361                                                                                                    
362                                  # type="index" ; insert into index                                  # type="index" ; insert into index
363                                    my $idisplay;
364                                  if ($i && $display) {                                  if ($i && $display) {
365                                          push @index_data, $display;                                          $idisplay = $display;
366                                          $index_filter = $filter if ($filter);                                          if ($filter) {
367                                                    no strict 'refs';
368                                                    $idisplay = &$filter($idisplay);
369                                            }
370                                            push @index_data, $idisplay if ($idisplay && !$iterate_by_page);
371                                  }                                  }
                         }  
372    
373                          # fill data in index                                  # store fields in lookup
374                          if (@index_data) {                                  if ($il && $display) {
375                                  if ($index_filter) {                                          if (lc($x->{type}) eq "lookup_key") {
376                                          no strict 'refs';                                                  if ($lookup_key) {
377                                          foreach my $d (&$index_filter(@index_data)) {                                                          print STDERR "WARNING: try to redefine lookup_key (keys shouldn't be repeatable fields!)";
378                                                  $index->insert($field, $d, $path);                                                  } else {
379                                                            if ($filter) {
380                                                                    no strict 'refs';
381                                                                    $lookup_key = &$filter($display);
382                                                            } else {
383                                                                    $lookup_key = $display;
384                                                            }
385                                                    }
386                                            } elsif (lc($x->{type}) eq "lookup_val") {
387                                                    if ($lookup_key) {
388                                                            if ($filter) {
389                                                                    no strict 'refs';
390                                                                    $lhash{$lookup_key} = &$filter($display);
391                                                            } else {
392                                                                    $lhash{$lookup_key} = $display;
393                                                            }
394                                                    } else {
395                                                            print STDERR "WARNING: no lookup_key defined for  '$display'?";
396                                                    }
397                                          }                                          }
398                                  } else {  
399                                          foreach my $d (@index_data) {                                  }
400                                                  $index->insert($field, $d, $path);  
401                                    # store data for page-by-page repeatable fields
402                                    if ($iterate_by_page) {
403                                            sub iterate_fld($$$$$$) {
404                                                    my ($cache,$what,$field,$page,$data,$append) = @_;
405                                                    return if (!$data);
406    
407                                                    my $ldel = $page_line_delimiter;
408                                                    $ldel = " " if ($append);
409    #print STDERR "line delimiter: ",Dumper($ldel) if ($ldel);
410                                                    if (! $cache->{$what}->{$field}->[$page]) {
411                                                            $cache->{$what}->{$field}->[$page] = $data;
412                                                    } else {
413                                                            $cache->{$what}->{$field}->[$page] .= $ldel.$data;
414                                                    }
415                                            }
416    
417                                            if ($display_data) {
418                                                    iterate_fld($cache,'display_data',$field,$page,$display_data,$x->{append});
419                                            }
420                                                    $display_data = "";
421                                            if ($swish_data) {
422                                                    iterate_fld($cache,'swish_data',$field,$page,$swish_data,$x->{append});
423                                                    $swish_data = "";
424                                            }
425                                            if ($swish_exact_data) {
426                                                    iterate_fld($cache,'swish_exact_data',$field,$page,$swish_exact_data,$x->{append});
427                                                    $swish_exact_data = "";
428                                          }                                          }
429    
430                                            if ($idisplay) {
431                                                    my $ldel=$page_line_delimiter;
432                                                    my @index_data;
433                                                    if ($cache->{index_data}->{$field}->[$page]) {
434    
435                                                            @index_data = @{$cache->{index_data}->{$field}->[$page]};
436                                                    }
437                                                    if ($x->{append}) {
438                                                            if (@index_data) {
439                                                                    $index_data[$#index_data] .= $idisplay;
440                                                            } else {
441                                                                    push @index_data, $idisplay;
442                                                            }
443                                                    } else {
444                                                            push @index_data, $idisplay;
445                                                    }
446                                                    $idisplay = "";
447                                                    @{$cache->{index_data}->{$field}->[$page]} = @index_data;
448                                            }
449                                    }
450                            }
451    
452                            if (! $iterate_by_page) {
453                                    my $idel = $x->{index_delimiter};
454                                    # fill data in index
455                                    foreach my $tmp (@index_data) {
456                                            my $i = $d = $tmp;
457                                            if ($idel && $tmp =~ m/$idel/) {
458                                                    ($i,$d) = split(/$idel/,$tmp);
459                                            }
460                                            $index->insert($field, $i, $d, $path);
461                                  }                                  }
462                                    @index_data = ();
463                          }                          }
464                  }                  }
465    
466                  # now try to parse variables from configuration file                  # now try to parse variables from configuration file
467                  foreach my $x (@{$config->{indexer}->{$field}->{'config'}}) {                  foreach my $x (@{$config->{indexer}->{$field}->{'config'}}) {
468    
469                            $x = unroll_x($x);
470    
471                          my $delimiter = x($x->{delimiter}) || ' ';                          my $delimiter = x($x->{delimiter}) || ' ';
472                          my $val = $cfg->val($database, x($x->{content}));                          my $val = $cfg->val($database, x($x->{content}));
473    
474                          my ($s,$d,$i) = (1,1,0);        # swish, display default                          # FIXME index_lookup is not supported!
475                          $s = 0 if (lc($x->{type}) eq "display");                          my ($s,$se,$d,$i,$il) = init_visible_type($x->{type});
                         $d = 0 if (lc($x->{type}) eq "swish");  
                         ($s,$d,$i) = (0,0,1) if (lc($x->{type}) eq "index");  
476    
477                          if ($val) {                          if ($val) {
478                                  $display_data .= $delimiter.$val if ($d);                                  $display_data .= $delimiter.$val if ($d);
479                                  $swish_data .= $val if ($s);                                  $swish_data .= " ".$val if ($s);
480                                  $index->insert($field, $val, $path) if ($i);                                  $index->insert($field, $val, $path) if ($i);
481                          }                          }
482    
483                            if ($iterate_by_page) {
484                                    # FIXME data from config tag will appear just
485                                    # on first page!!!
486                                    my $page = 0;
487                                    if ($display_data) {
488                                            $cache->{display_data}->{$field}->[$page] = $display_data;
489                                            $display_data = "";
490                                    }
491                                    if ($swish_data) {
492                                            $cache->{swish_data}->{$field}->[$page] = $swish_data;
493                                            $swish_data = "";
494                                    }
495                                    if ($swish_exact_data) {
496                                            $cache->{swish_exact_data}->{$field}->[$page] = $swish_exact_data;
497                                            $swish_exact_data = "";
498                                    }
499                            }
500                  }                  }
501    
502                    # save data page-by-page
503                  if ($display_data) {                  foreach my $field (@page_fields) {
504                            my $nr_pages = $page_max{$field} || next;
505                          if ($field eq "headline") {  #print STDERR "field '$field' iterate over ",($nr_pages || 0)," pages...\n";
506                                  $xml .= xmlify("headline", $display_data);  #print STDERR Dumper($cache->{display_data});
507                          } else {                          for (my $page=0; $page <= $nr_pages; $page++) {
508                                    my $display_data;
509                                  # find field name (signular, plural)                                  if ($cache->{format}->{$field}) {
510                                  my $field_name = "";                                          my $tmp = mkformat($cache->{format}->{$field},$cache->{display_data}->{$field}->[$page]);
511                                  if ($config->{indexer}->{$field}->{name_singular} && $field_usage{$field} == 1) {                                          $display_data=$tmp if ($tmp);
                                         $field_name = $config->{indexer}->{$field}->{name_singular}."#-#";  
                                 } elsif ($config->{indexer}->{$field}->{name_plural}) {  
                                         $field_name = $config->{indexer}->{$field}->{name_plural}."#-#";  
                                 } elsif ($config->{indexer}->{$field}->{name}) {  
                                         $field_name = $config->{indexer}->{$field}->{name}."#-#";  
512                                  } else {                                  } else {
513                                          print STDERR "WARNING: field '$field' doesn't have 'name' attribute!";                                          $display_data = $cache->{display_data}->{$field}->[$page];
514                                    }
515                                    if ($display_data) { # default
516                                            if ($field eq "headline") {
517                                                    $xml .= xmlify("headline", $display_data);
518                                            } else {
519    
520                                                    # fallback to empty field name if needed
521                                                    $html .= get_field_name($config,$field,$field_usage{$field}) || '';
522                                                    $html .= "#-#".$display_data."###\n";
523                                            }
524                                    }
525                                    
526                                    my $swish_data = $cache->{swish_data}->{$field}->[$page];
527                                    if ($swish_data) {
528                                            # remove extra spaces
529                                            $swish_data =~ s/ +/ /g;
530                                            $swish_data =~ s/ +$//g;
531    
532                                            $xml .= xmlify($field."_swish", my_unac_string($codepage,$swish_data));
533                                    }
534    
535                                    my $swish_exact_data = $cache->{swish_exact_data}->{$field}->[$page];
536                                    if ($swish_exact_data) {
537                                            $swish_exact_data =~ s/ +/ /g;
538                                            $swish_exact_data =~ s/ +$//g;
539    
540                                            # add delimiters before and after word.
541                                            # That is required to produce exact match
542                                            $xml .= xmlify($field."_swish_exact", my_unac_string($codepage,$swish_exact_data));
543                                  }                                  }
544                                  if ($field_name) {                                  
545                                          $html .= x($field_name);                                  my $idel = $cache->{index_delimiter}->{$field};
546                                    foreach my $tmp (@{$cache->{index_data}->{$field}->[$page]}) {
547                                            my $i = $tmp;
548                                            my $d = $tmp;
549                                            if ($idel && $tmp =~ m/$idel/) {
550                                                    ($i,$d) = split(/$idel/,$tmp);
551                                            }
552                                            $index->insert($field, $i, $d, $path);
553    #print STDERR "index [$idel] $field: $i --> $d [$path]\n";
554                                  }                                  }
                                 $html .= $display_data."###\n";  
555                          }                          }
556            
557                  }                  }
558                  if ($swish_data) {                  
559                          # remove extra spaces                  if (! $iterate_by_page) {
560                          $swish_data =~ s/ +/ /g;                          if ($display_data) {
561                          $swish_data =~ s/ +$//g;                                  if ($field eq "headline") {
562                                            $xml .= xmlify("headline", $display_data);
563                                    } else {
564    
565                          $xml .= xmlify($field."_swish", unac_string($codepage,$swish_data));                                          # fallback to empty field name if needed
566                  }                                          $html .= get_field_name($config,$field,$field_usage{$field}) || '';
567                                            $html .= "#-#".$display_data."###\n";
568                                    }
569                            }
570                            if ($swish_data) {
571                                    # remove extra spaces
572                                    $swish_data =~ s/ +/ /g;
573                                    $swish_data =~ s/ +$//g;
574    
575                                    $xml .= xmlify($field."_swish", my_unac_string($codepage,$swish_data));
576                            }
577    
578                            if ($swish_exact_data) {
579                                    $swish_exact_data =~ s/ +/ /g;
580                                    $swish_exact_data =~ s/ +$//g;
581    
582                                    # add delimiters before and after word.
583                                    # That is required to produce exact match
584                                    $xml .= xmlify($field."_swish_exact", my_unac_string($codepage,$swish_exact_data));
585                            }
586                    }
587          }          }
588    
589          # dump formatted output in <html>          # dump formatted output in <html>
# Line 278  $index = new index_DBI( Line 618  $index = new index_DBI(
618    
619  my $show_progress = $cfg_global->val('global', 'show_progress');  my $show_progress = $cfg_global->val('global', 'show_progress');
620    
621    my $my_unac_filter = $cfg_global->val('global', 'my_unac_filter');
622    if ($my_unac_filter) {
623            print STDERR "using $my_unac_filter to filter characters for search\n";
624            require $my_unac_filter;
625    } else {
626            print STDERR "### fallback to default my_unac_string!\n";
627            eval q{
628            sub main::my_unac_string($$) {
629                    my ($charset, $string) = (@_);
630                    return $string;
631            }
632            };
633    }
634    
635  foreach my $database ($cfg->Sections) {  foreach my $database ($cfg->Sections) {
636    
637          my $type = lc($cfg -> val($database, 'type')) || die "$database doesn't have 'type' defined";          my $type = lc($cfg -> val($database, 'type')) || die "$database doesn't have 'type' defined";
638          my $add_xml = $cfg -> val($database, 'xml');    # optional          my $add_xml = $cfg -> val($database, 'xml');    # optional
639    
640            # create new lookup file
641            my $lookup_file = $cfg -> val($database, 'lookup_newfile'); # optional
642            if ($lookup_file) {
643                    #tie %lhash, 'GDBM_File', $lookup_file, &GDBM_NEWDB, 0644;
644                    if (! -e $lookup_file) {
645                            open(LOOKUP, "> $lookup_file") || die "can't create $lookup_file': $!";
646                            close(LOOKUP);
647                    }
648                    tie %lhash, 'TDB_File', $lookup_file, TDB_CLEAR_IF_FIRST, O_RDWR, 0644;
649                    print STDERR "creating lookup file '$lookup_file'\n";
650                    # delete memory cache for lookup file
651                    delete $cache->{lhash};
652            }
653    
654            # open existing lookup file
655            $lookup_file = $cfg -> val($database, 'lookup_open'); # optional
656            if ($lookup_file) {
657                    #tie %lhash, 'GDBM_File', $lookup_file, &GDBM_READER, 0644;
658                    tie %lhash, 'TDB_File', $lookup_file, TDB_DEFAULT, O_RDWR, 0644;
659                    print STDERR "opening lookup file '$lookup_file'\n";
660            }
661    
662  print STDERR "reading ./import_xml/$type.xml\n";  print STDERR "reading ./import_xml/$type.xml\n";
663    
664          # extract just type basic          # extract just type basic
665          my $type_base = $type;          my $type_base = $type;
666          $type_base =~ s/_.+$//g;          $type_base =~ s/_.+$//g;
667    
668          $config=XMLin("./import_xml/$type.xml", forcearray => [ $type2tag{$type_base}, 'config' ], forcecontent => 1);          $config=XMLin("./import_xml/$type.xml", ForceArray => [ $type2tag{$type_base}, 'config', 'format' ], ForceContent => 1 );
669    
670            # helper for progress bar
671            sub fmt_time {
672                    my $t = shift || 0;
673                    my $out = "";
674    
675                    my ($ss,$mm,$hh) = gmtime($t);
676                    $out .= "${hh}h" if ($hh);
677                    $out .= sprintf("%02d:%02d", $mm,$ss);
678                    $out .= "  " if ($hh == 0);
679                    return $out;
680            }
681    
682          # output current progress indicator          # output current progress indicator
683          my $last_p = 0;          my $last_p = 0;
684            my $start_t = time();
685          sub progress {          sub progress {
686                  return if (! $show_progress);                  return if (! $show_progress);
687                  my $current = shift;                  my $current = shift;
688                  my $total = shift || 1;                  my $total = shift || 1;
689                  my $p = int($current * 100 / $total);                  my $p = int($current * 100 / $total);
690                  if ($p != $last_p) {                  if ($p < $last_p) {
691                          printf STDERR ("%5d / %5d [%-51s] %-2d %% \r",$current,$total,"=" x ($p/2).">", $p );                          $start_t = time();
692                    } elsif ($p != $last_p) {
693                            my $rate = ($current / (time() - $start_t || 1));
694                            my $eta = ($total-$current) / ($rate || 1);
695                            printf STDERR ("%5d [%-38s] %-5d %0.1f/s %s\r",$current,"=" x ($p/3)."$p%>", $total, $rate, fmt_time($eta));
696                          $last_p = $p;                          $last_p = $p;
697                  }                  }
698          }          }
# Line 321  print STDERR "reading ./import_xml/$type Line 714  print STDERR "reading ./import_xml/$type
714          # now read database          # now read database
715  print STDERR "using: $type...\n";  print STDERR "using: $type...\n";
716    
717            # erase cache for tags by order in this database
718            delete $cache->{tags_by_order};
719    
720          if ($type_base eq "isis") {          if ($type_base eq "isis") {
721    
722                  my $isis_db = $cfg -> val($database, 'isis_db') || die "$database doesn't have 'isis_db' defined!";                  my $isis_db = $cfg -> val($database, 'isis_db') || die "$database doesn't have 'isis_db' defined!";
723    
724                  $import2cp = Text::Iconv->new($config->{isis_codepage},$codepage);                  $import2cp = Text::Iconv->new($config->{isis_codepage},$codepage);
725                  my $db = OpenIsis::open( $isis_db );                  my $db = new IsisDB( isisdb => $isis_db );
   
                 # check if .txt database for OpenIsis is zero length,  
                 # if so, erase it and re-open database  
                 sub check_txt_db {  
                         my $isis_db = shift || die "need isis database name";  
                         if (-e $isis_db.".TXT") {  
                                 print STDERR "WARNING: removing .txt OpenIsis database...\n";  
                                 unlink $isis_db.".TXT" || warn "FATAL: unlink error on '$isis_db.TXT': $!";  
                                 my $db = OpenIsis::open( $isis_db );  
                                 return $db;  
                         }  
                 }  
   
                 # EOF error  
                 if ($db == -1) {  
                         $db = check_txt_db($isis_db);  
                         if ($db == -1) {  
                                 print STDERR "FATAL: OpenIsis can't open zero size file $isis_db\n";  
                                 next;  
                         }  
                 }  
726    
727                  # OpenIsis::ERR_BADF                  my $max_rowid = $db->{'maxmfn'} || die "can't find maxmfn";
                 if ($db == -4) {  
                         print STDERR "FATAL: OpenIsis can't find file $isis_db\n";  
                         next;  
                 # OpenIsis::ERR_IO  
                 } elsif ($db == -5) {  
                         print STDERR "FATAL: OpenIsis can't access file $isis_db\n";  
                         next;  
                 } elsif ($db < 0) {  
                         print STDERR "FATAL: OpenIsis unknown error $db with file $isis_db\n";  
                         next;  
                 }  
   
                 my $max_rowid = OpenIsis::maxRowid( $db );  
   
                 # if 0 records, try to rease isis .txt database  
                 if ($max_rowid == 0) {  
                         # force removal of database  
                         $db = check_txt_db($isis_db);  
                         $max_rowid = OpenIsis::maxRowid( $db );  
                 }  
728    
729                  print STDERR "Reading database: $isis_db [$max_rowid rows]\n";                  print STDERR "Reading database: $isis_db [$max_rowid rows]\n";
730    
731                  my $path = $database;                  my $path = $database;
732    
733                  for (my $row_id = 1; $row_id <= $max_rowid; $row_id++ ) {                  for (my $row_id = 1; $row_id <= $max_rowid; $row_id++ ) {
734                          my $row = OpenIsis::read( $db, $row_id );                          my $row = $db->to_hash( $row_id );
735                          if ($row && $row->{mfn}) {                          if ($row) {
736            
737                                    $row->{mfn} = $row_id;
738                                    $row->{record} = $db->{record};
739    
740                                  progress($row->{mfn}, $max_rowid);                                  progress($row->{mfn}, $max_rowid);
741    
742                                  my $swishpath = $path."#".int($row->{mfn});                                  my $swishpath = $path."#".int($row->{mfn});
# Line 392  print STDERR "using: $type...\n"; Line 750  print STDERR "using: $type...\n";
750                                  }                                  }
751                          }                          }
752                  }                  }
                 # for this to work with current version of OpenIsis (0.9.0)  
                 # you might need my patch from  
                 # http://www.rot13.org/~dpavlin/projects/openisis-0.9.0-perl_close.diff  
                 OpenIsis::close($db);  
753                  print STDERR "\n";                  print STDERR "\n";
754    
755          } elsif ($type_base eq "excel") {          } elsif ($type_base eq "excel") {
756                  use Spreadsheet::ParseExcel;                  require Spreadsheet::ParseExcel;
757                  use Spreadsheet::ParseExcel::Utility qw(int2col);                  require Spreadsheet::ParseExcel::Utility;
758                    import Spreadsheet::ParseExcel::Utility qw(int2col);
759                                    
760                  $import2cp = Text::Iconv->new($config->{excel_codepage},$codepage);                  $import2cp = Text::Iconv->new($config->{excel_codepage},$codepage);
761                  my $excel_file = $cfg -> val($database, 'excel_file') || die "$database doesn't have 'excel_file' defined!";                  my $excel_file = $cfg -> val($database, 'excel_file') || die "$database doesn't have 'excel_file' defined!";
# Line 428  print STDERR "using: $type...\n"; Line 783  print STDERR "using: $type...\n";
783                          for(my $iC = $oWorksheet->{MinCol} ; defined $oWorksheet->{MaxCol} && $iC <= $oWorksheet->{MaxCol} ; $iC++) {                          for(my $iC = $oWorksheet->{MinCol} ; defined $oWorksheet->{MaxCol} && $iC <= $oWorksheet->{MaxCol} ; $iC++) {
784                                  my $cell = $oWorksheet->{Cells}[$iR][$iC];                                  my $cell = $oWorksheet->{Cells}[$iR][$iC];
785                                  if ($cell) {                                  if ($cell) {
786                                          $row->{int2col($iC)} = $cell->Value;                                          # this conversion is a cludge.
787                                            # Files from Excell could have
788                                            # characters which don't fit into
789                                            # destination encoding.
790                                            $row->{int2col($iC)} = $utf2cp->convert($cell->Value) || $cell->Value;
791                                  }                                  }
792                          }                          }
793    
# Line 454  print STDERR "using: $type...\n"; Line 813  print STDERR "using: $type...\n";
813                  }                  }
814          } elsif ($type_base eq "marc") {          } elsif ($type_base eq "marc") {
815    
816                  use MARC;                  require MARC::File::USMARC;
817                                    
818                  $import2cp = Text::Iconv->new($config->{marc_codepage},$codepage);                  $import2cp = Text::Iconv->new($config->{marc_codepage},$codepage);
819                  my $marc_file = $cfg -> val($database, 'marc_file') || die "$database doesn't have 'marc_file' defined!";                  my $marc_file = $cfg -> val($database, 'marc_file') || die "$database doesn't have 'marc_file' defined!";
820    
821                  # optional argument is format                  # optional argument is format
822                  my $format = x($config->{format}) || 'usmarc';                  warn "marc_format is no longer used!" if ($config->{marc_format});
   
823                  print STDERR "Reading MARC file '$marc_file'\n";                  print STDERR "Reading MARC file '$marc_file'\n";
824    
825                  my $marc = new MARC;                  my $marc = MARC::File::USMARC->in( $marc_file )
826                  my $nr = $marc->openmarc({                          || die "Can't open MARC file '$marc_file': ".$MARC::File::ERROR;
827                                  file=>$marc_file, format=>$format  
828                          }) || die "Can't open MARC file '$marc_file'";                  # count records in MARC file
829                    sub marc_count {
830                            my $filename = shift || die;
831                            my $file = MARC::File::USMARC->in($filename) || die $MARC::File::ERROR;
832                            my $count = 0;
833                            while ($file->skip()) {
834                                    $count++;
835                            }
836                            return $count;
837                    }
838    
839                  my $i=0;        # record nr.                  my $count = marc_count($marc_file) || warn "no records in '$marc_file'?";
840    
841                  my $rec;                  my $i = 0;
842    
843                  while ($marc->nextmarc(1)) {                  while( my $rec = $marc->next() ) {
844    
845                          # XXX                          progress($i++,$count);
                         fakeprogress($i++);  
846    
847                          my $swishpath = $database."#".$i;                          my $swishpath = $database."#".$i;
848    
849                          if (my $xml = data2xml($type_base,$marc,$add_xml,$cfg,$database)) {                          if (my $xml = data2xml($type_base,$rec,$add_xml,$cfg,$database)) {
850                                  $xml = $cp2utf->convert($xml);                                  $xml = $cp2utf->convert($xml);
851                                  use bytes;      # as opposed to chars                                  use bytes;      # as opposed to chars
852                                  print "Path-Name: $swishpath\n";                                  print "Path-Name: $swishpath\n";
# Line 488  print STDERR "using: $type...\n"; Line 854  print STDERR "using: $type...\n";
854                                  print "Document-Type: XML\n\n$xml\n";                                  print "Document-Type: XML\n\n$xml\n";
855                          }                          }
856                  }                  }
857    
858                    print STDERR "\n";
859    
860          } elsif ($type_base eq "feed") {          } elsif ($type_base eq "feed") {
861    
862                  $import2cp = Text::Iconv->new($config->{feed_codepage},$codepage);                  $import2cp = Text::Iconv->new($config->{feed_codepage},$codepage);
# Line 526  print STDERR "using: $type...\n"; Line 895  print STDERR "using: $type...\n";
895                          fakeprogress($i);                          fakeprogress($i);
896    
897                  }                  }
898                    # close lookup
899                    untie %lhash if (%lhash);
900          }          }
901  }  }
902    
# Line 540  __END__ Line 911  __END__
911    
912  all2xml.pl - read various file formats and dump XML for SWISH-E  all2xml.pl - read various file formats and dump XML for SWISH-E
913    
914    =head1 SYNOPSYS
915    
916     $ all2xml.pl [test.conf]
917    
918  =head1 DESCRIPTION  =head1 DESCRIPTION
919    
920  This command will read ISIS data file using OpenIsis perl module, MARC  This command will read ISIS data file using IsisDB perl module, MARC
921  records using MARC module and optionally Micro$oft Excel files to  records using MARC module and optionally Micro$oft Excel files to
922  create one XML file for usage with I<SWISH-E> indexer. Dispite it's name,  create one XML file for usage with I<SWISH-E> indexer. Dispite it's name,
923  this script B<isn't general xml generator> from isis files (isis allready  this script B<isn't general xml generator> from isis files (isis allready
924  has something like that). Output of this script is tailor-made for SWISH-E.  has something like that). Output of this script is tailor-made for SWISH-E.
925    
926    If no configuration file is specified, it will use default one called
927    C<all2xml.conf>.
928    
929  =head1 BUGS  =head1 BUGS
930    
931  Documentation is really lacking. However, in true Open Source spirit, source  Documentation is really lacking. However, in true Open Source spirit, source

Legend:
Removed from v.108  
changed lines
  Added in v.626

  ViewVC Help
Powered by ViewVC 1.1.26