Provided by: libmarc-perl_1.07-6.1_all bug

NAME

       MARC.pm - Perl extension to manipulate MAchine Readable Cataloging records.

SYNOPSIS

         use MARC;

               # constructors
         $x=MARC->new();
         $x=MARC->new("filename","fileformat");
         $x->openmarc({file=>"makrbrkr.mrc",'format'=>"marcmaker",
                       increment=>"5", lineterm=>"\n",
                       charset=>\%char_hash});
         $record_num=$x->createrecord({leader=>"00000nmm  2200000 a 4500"});

               # input/output operations
         $y=$x->nextmarc(10);                  # increment
         $x->closemarc();
         print $x->marc_count();
         $x->deletemarc({record=>'2',field=>'110'});
         $y=$x->selectmarc(['4','21-50','60']);

               # character translation
         my %inc = %{$x->usmarc_default()};    # MARCMaker input charset
         my %outc = %{$x->ustext_default()};   # MARCBreaker output charset

               # data queries
         @records = $x->searchmarc({field=>"245"});
         @records = $x->searchmarc({field=>"260",subfield=>"c",
                                    regex=>"/19../"});
         @records = $x->searchmarc({field=>"245",notregex=>"/huckleberry/i"});
         @results = $x->getvalue({record=>'12',field=>'856',subfield=>'u'});

               # header and control field operations
         $rldr = $x->unpack_ldr($record);
         print "Desc is $rldr->{Desc}";
         next if ($x->bib_format($record) eq 'SERIALS');
         $rff = $x->unpack_008($record);
         last if ($rff->{'Date1'}=~/00/ or $rff->{'Date2'}=~/00/);

               # data modifications
         $x->addfield({record=>"2", field=>"245",
                       i1=>"1", i2=>"4", ordered=>'y', value=>
                       [a=>"The adventures of Huckleberry Finn /",
                        c=>"Mark Twain ; illustrated by E.W. Kemble."]});

         my $update245 = {field=>'245',record=>2,ordered=>'y'};
         my @u245 = $x->getupdate($update245);
         $x->deletemarc($update245);
         $x->addfield($update245, @u245_modified);

               # outputs
         $y = $x->output({'format'=>"marcmaker", charset=>\%outc});
         $x->output({file=>">>my_text.txt",'format'=>"ascii",record=>2});
         $x->output({file=>">my_marcmaker.mkr",'format'=>"marcmaker",
                     nolinebreak=>'y',lineterm=>'\n'});
         $x->output({file=>">titles.html",'format'=>"html", 245=>"Title: "});

               # manipulation of individual marc records.
         @recs = $x[1..$#$x];
         grep {$_->unpack_ldr() && 0} @recs;
         @LCs = grep {$_->unp_ldr{Desc} eq 'a' &&
                      $_->getvalue({field=>'040'}) =~/DLC\c_.DLC/} @recs;
         foreach my $rec (@LCs) {
                 print $rec->output({format=>'usmarc'});
         }

               # manipulation as strings.
         foreach my $rec (@LCs) {
                 my $stringvar = $rec->as_string();
                 $stringvar=~s[^(
                                 100\s # main entries of this stripe..
                                 ..\s # (don't care about indicators)
                                 \c_.\s*
                                 )(\S) # take the first letter..
                               ] [
                               ${1}uc($2) # and upcase it. All authors have
                                          # upcase first letters in my library.
                               ]xm; # x means 'ignore whitespace and allow
                                    # embedded comments'.
                $rec->from_string($stringvar);
                my ($i2,$article) = $stringvar =~/245 .(.) \c_.(.{0,9})/;
                $article = substr($article,0,$i2) if $i2=~/\d/;
                print "article $article is not common" unless $COMMON_ARTS{$article};
         }

DESCRIPTION

       MARC.pm is a Perl 5 module for reading in, manipulating, and outputting bibliographic
       records in the USMARC format. You will need to have Perl 5.004 or greater for MARC.pm to
       work properly. Since it is a Perl module you use MARC.pm from one of your own Perl
       scripts. To see what sorts of conversions are possible you can try out a web interface to
       MARC.pm which will allow you to upload MARC files and retrieve the results (for details
       see the section below entitled "Web Interface").

       However, to get the full functionality you will probably want to install MARC.pm on your
       server or PC. MARC.pm can handle both single and batches of MARC  records. The limit on
       the number of records in a batch is determined by the memory capacity of the machine you
       are running. If memory is an issue for you MARC.pm will allow you to read in records from
       a batch gradually. MARC.pm also includes a variety of tools for searching, removing, and
       even creating records from scratch.

   Types of Conversions:
       ·   MARC -> ASCII : separates the MARC fields out into separate lines

       ·   MARC <-> MARCMaker : The MARCMaker format is a format that was developed by the
           Library of Congress for use with their DOS based MARCMaker and MARCBreaker utilities.
           This format is particularly useful for making global changes (ie. with a text editor's
           search and replace) and then converting back to MARC (MARC.pm will read properly
           formatted MARCMaker records). For more information about the MARCMaker format see
           http://lcweb.loc.gov/marc/marcsoft.html

       ·   MARC -> HTML : The MARC to HTML conversion creates an HTML file from the fields and
           field labels that you supply. You could possibly use this to create HTML
           bibliographies from a batch of MARC records.

       ·   MARC <-> XML : XML support is handled by MARC::XML which is a subclass of MARC.pm and
           is also available for download from the CPAN.

       ·   MARC -> URLS : This conversion will extract URLs from a batch of MARC records. The
           URLs are found in the 856 field, subfield u. The HTML page that is generated can then
           be used with link-checking software to determine which URLs need to be repaired.
           Hopefully library system vendors will soon support this activity soon and make this
           conversion unnecessary!

   Downloading and Installing
       Download
           The module is provided in standard CPAN distribution format. It will extract into a
           directory MARC-version with any necessary subdirectories.  Change into the MARC top
           directory. Download the latest version from
           http://www.cpan.org/modules/by-module/MARC/

       Unix
               perl Makefile.PL
               make
               make test
               make install

       Win9x/WinNT/Win2000
               perl Makefile.PL
               perl test.pl
               perl install.pl

       Test
           Once you have installed, you can check if Perl can find it. Change to some other
           directory and execute from the command line:

               perl -e "use MARC"

           If you do not get any response that means everything is OK! If you get an error like
           Can't locate method "use" via package MARC.  then Perl is not able to find MARC.pm -
           double check that the file copied it into the right place during the install.

   Todo
       ·   Support for other MARC formats (UKMARC, FINMARC, etc).

       ·   Create a map and instructions for using and extending the MARC.pm data structure.

       ·   Develop better error catching mechanisms.

       ·   Support for MARC <-> Unicode character conversions.

       ·   MARC <-> EAD (Encoded Archival Description) conversion?

       ·   MARC <-> DC/RDF (Dublin Core Metadata encoded in the Resource Description Framework)?

   Web Interface
       A web interface to MARC.pm is available at http://libstaff.lib.odu.edu/cgi-bin/marc.cgi
       where you can upload records and observe the results. If you'd like to check out the cgi
       script take a look at
       http://libstaff.lib.odu.edu/depts/systems/iii/scripts/MARCpm/marc-cgi.txt However, to get
       the full functionality you will want to install MARC.pm on your server or PC.

   Option Templates
       A MARC record is a complex structure. Hence, most of the methods have a number of options.
       Since a series of operations frequently uses many the same options for each method, you
       can create a single variable that forms a "template" for the desired options. The variable
       points to a hash - and the hash keys have been selected so the same hash works for all of
       the related methods.

           my $loc852 = {record=>1, field=>'852', ordered=>'y'};
           my ($found) = $x->searchmarc($loc852);
           if (defined $found) {
               my @m852 = $x->getupdate($loc852);
               $x->deletemarc($loc852);
                   # change @m852 as desired
               $x->updaterecord($loc852, @m852fix);
           }
           else {
               $x->addfield($loc852, @m852new);
           }

       The following methods are specifically designed to work together using Option Templates.
       The required options are shown as bold. Any "(default)" options are shown in parentheses.
       Although deletemarc() permits an array for the record option, a single record should be
       used in a Template. The subfield option must not be used in a Template that uses both
       deletemarc and one of the methods that acts on a complete field like addfield(). The value
       option must not be used with updaterecord().

           deletemarc() - field (all), record (all), subfield [supplemental]

           searchmarc() - field, regex, notregex, subfield [supplemental]

           getvalue() - record, field, subfield, delimiter [supplemental]

           getupdate() - record, field

           addfield() - record, field, i1 (' '), i2 (' '), value, ordered ('y')

           updaterecord() - record, field, i1 (' '), i2 (' '), ordered ('y')

       The methods that accept a subfield option also accept specifying it as a supplemental
       parameter. Supplemental parameters append/overwrite the hash values specified in the
       template.

           $x->deletemarc($loc852, 'subfield','k');

           my $f260 = {field=>"260",regex=>"/19../"};
           my @records=$x->searchmarc($f260,'subfield','c');
           foreach $found (@records) {
               $value = $x->getvalue($f260,'record',"$found",'field',"245");
               print "TITLE: $value\n";
           }

METHODS

       Here is a list of the methods in MARC.pm that are available to you for reading in,
       manipulating and outputting MARC data.

   new()
       Creates a new MARC object.

           $x = MARC->new();

       You can also use the optional file and format parameters to create and populate the object
       with data from a file. If a file is specified it will read in the entire file. If you wish
       to read in only portions of the file see openmarc(), nextmarc(), and closemarc() below.
       The format defaults to 'usmarc' if not specified. It is only used when a file is given.

           $x = MARC->new("mymarc.dat","usmarc");
           $x = MARC->new("mymarcmaker.mkr","marcmaker");

       Creates a new MARC::Rec object.

           $rec=MARC::Rec->new();
           $rec=MARC::Rec->new($filehandle,"usmarc");

       MARC::Rec objects are typically created by reading from a filehandle using nextrec() and a
       proto MARC::Rec object or by directly stuffing the @{$rec->{'array'}} array.

   openmarc()
       Opens a specified file for reading data into a MARC object. If no format is specified
       openmarc() will default to USMARC. The increment parameter defines how many records you
       would like to read from the file. If no increment is defined then the file will just be
       opened, and no records will be read in. If increment is set to -1 then the entire file
       will be read in.

           $x = new MARC;
           $x->openmarc({file=>"mymarc.dat",'format'=>"usmarc",
                         increment=>"1"});
           $x->openmarc({file=>"mymarcmaker.mkr",'format'=>"marcmaker",
                         increment=>"5"});

       note: openmarc() will return the number of records read in. If the file opens
       successfully, but no records are read, it returns "0 but true". For example:

           $y=$x->openmarc({file=>"mymarc.dat",'format'=>"usmarc",
                            increment=>"5"});
           print "Read in $y records!";

       When the MARCMaker format is specified, the lineterm parameter can be used to override the
       CRLF line-ending default (the format was originally released for MS-DOS). A charset
       parameter accepts a hash-reference to a user supplied character translation table. The
       "usmarc.txt" table supplied with the LoC. MARCMaker utility is used internally as the
       default. You can use the usmarc_default method to get a hash-reference to it if you only
       want to modify a couple of characters. See example below.

           $x->openmarc({file=>"makrbrkr.mrc",'format'=>"marcmaker",
                         increment=>"5",lineterm=>"\n",
                         charset=>\%char_hash});

   nextmarc()
       Once a file is open nextmarc() can be used to read in the next group of records. The
       increment can be passed to change the number of records read in if necessary. An increment
       of -1 will read in the rest of the file. Specifying the increment will change the value
       set with openmarc(). Otherwise, that value is the default.

           $x->nextmarc();
           $x->nextmarc(10);
           $x->nextmarc(-1);

       note: Similar to openmarc(), nextmarc() will return the number of records read in.

           $y=$x->nextmarc();
           print "$y more records read in!";

   nextrec()
       MARC:Rec instances can read from a filehandle and produce a new MARC::Rec instance.  If
       nextrec is passed a string, it will read from that instead. The string should be formatted
       according to the {format} field of the instance.

       Cases where a new instance cannot be created are classified by a status value:

           my ($newrec,$status) = $rec->nextrec();

       $status is undefined if we are at the end of the filehandle. If the data read from the
       filehandle cannot be made into a marc record, $status will be negative.  For example,
       $status is -1 if there is a distinction between recsize and leader definition of recsize,
       and -2 if the leader is not numeric.

       An idiom for reading records incrementally with MARC::Recs is:

           my $proto=MARC::Rec->new($filehandle,$format);
           while (1) {
                 my ($rec,$status)=$proto->nextrec();
                 last unless $status;
                 die "Bad record, bad, bad record: error $status"
                     if $status <0;
                 print $rec->output({$format=>'ascii'});
                 # or replace print and output with your own functions/methods.
           }
           close $filehandle or die "File $filehandle is not happy on close\n";

       If you are getting records from an external source as strings, the idiom is:

           my $proto=MARC::Rec->new($filehandle,$format);
           while (1) {
                 my $string = get_external_marc();
                 last unless $string;
                 my ($rec,$status)=$proto->nextrec($string);
                 last unless $status;
                 die "Bad record, bad, bad record: error $status"
                     if $status <0;
                 print $rec->output({$format=>'ascii'});
                 # or replace print and output with your own functions/methods.
           }

   closemarc()
       If you are finished reading in records from a file you should close it immediately.

           $x->closemarc();

   add_map()
       add_map() takes a recnum and a ref to a field in ($tag, $i1,$i2,a=>"bar",...) or ($tag,
       $field) formats and will append to the various indices that we have hanging off that
       record.  It is intended for use in creating records de novo and as a component for
       rebuild_map(). It carefully does not copy subfield values or entire fields, maintaining
       some reference relationships.  What this means for indices created with add_map that you
       can directly edit subfield values in $marc->[recnum]{array} and the index will adjust
       automatically. Vice-versa, if you edit subfield values in
       $marc->{recnum}{tag}{subfield_code} the fields in $marc->[recnum]{array} will adjust. If
       you change structural information in the array with such an index, you must rebuild the
       part of the index related to the current tag (and possibly the old tag if you change the
       tag).

          use MARC 1.02;
          while (<>) {
               chomp;
               my ($author,$title) = split(/\t/);
               my $rnum = $x->createrecord({leader=>
                                              "00000nmm  2200000 a 4500"});

               my @auth = (100, ' ', ' ', a=>$author);
               my @title = (245, ' ', ' ', a=>$title);
               push @{$x->[$rnum]{array}}, \@auth;
               $x->add_map($rnum,\@auth);
               push @{$x->[$rnum]{array}}, \@title;
               $x->add_map($rnum,\@title);
          }

       MARC::Rec::add_map($rfield) does not need the record specification and has the same effect
       as add_map.

   rebuild_map
       rebuild_map takes a recnum and a tag and will synchronise the index with the array
       elements of the marc record at the recnum with that tag.

             #Gonna change all 099's to 092's since this is a music collection.
             grep {$->[0] =~s/099/092} @{$x->[$recnum]{array}};

             #Oops, now the index is out of date on the 099's...
             $x->rebuild_map($recnum,099);
             #... and the 092's since we now have new ones.
             $x->rebuild_map($recnum,092);
             #All fixed.

       MARC::Rec::rebuild_map($tag) does not need the record number and has the same effect as
       rebuild_map.

   rebuild_map_all
       rebuild_map takes a recnum and will synchronise the index with the array elements of the
       marc record at the recnum.

       MARC::Rec::rebuild_map_all() does not need the record number and has the same effect as
       rebuild_map_all.

   getfields
       getfields takes a template and returns an array of fieldrefs from the record number
       implied by that template. The fields referred are fields from the $marc->[$recnum]{array}
       group. The fields are all fields from the first one with the tag from the template to the
       last with that tag. Some marc records (e.g. cjk) may have fields with other tags mixed in.
       Consecutive calls to updatefields with a different tag and the same record are probably a
       bad idea unless you have assurance that fields with the same tag are always together.

       MARC::Rec::getfields is identical to getfields, but ignores any record specification in
       the template.

   marc_count()
       Returns the total number of records in a MARC object. This method was previously named
       length(), but that conflicts with the Perl built-in of the same name. Use the new name,
       the old one is deprecated and will disappear shortly.

           $length=$x->marc_count();

   getfirstvalue()
       getfirstvalue will return the first value of a field or subfield or indicator or i12 in a
       particular record found in the MARC object. It does not depend on the index being up to
       date.

       MARC::Rec::getfirstvalue is identical to getfields, but ignores any record specification
       in the template.

   getvalue()
       This method will retrieve MARC field data from a specific record in the MARC object.
       getvalue() takes four parameters: record, field, subfield, and delimiter. Since a single
       MARC record could contain several of the fields or subfields the results are returned to
       you as an array. If you only pass record and field you will be returned the entire field
       without subfield delimiters. Optionally you can use delimiter to specify what character to
       use for the delimiter, and you will also get the subfield delimiters. If you also specify
       subfield your results will be limited to just the contents of that subfield. Repeated
       subfield occurances will end up in separate array elements in the order in which they were
       read in. The subfield designations 'i1', 'i2' and 'i12' can be used to get indicator(s).

               #get the 650 field(s)
           @results = $x->getvalue({record=>'1',field=>'650'});

               #get the 650 field(s) with subfield delimiters (ie. |x |v etc)
           @results = $x->getvalue({record=>'1',field=>'650',delimiter=>'|'});

               #get all of the subfield u's from the 856 field
           @results = $x->getvalue({record=>'12',field=>'856',subfield=>'u'});

       MARC::Rec::getvalue($template) is identical to getvalue, but ignores any record
       specification.

   unpack_ldr($record)
       Returns a ref to a hash version of the record'th LDR.  Installs the ref in $marc as
       $marc->[$record]{unp_ldr}

           my $rldr = $x->unpack_ldr(1);
           print "Desc is $rldr{Desc}";
           my ($m040) = $x->getvalues({record=>'1',field=>'040'});
           print "First record is LC, let's leave it alone"
                 if $rldr->{'Desc'} eq 'a' && $m040=~/DLC\s*\c_c\s*DLC/;

       The hash version contains the following information:

               Key             000-Pos length  Function [standard value]
               ---             ------- ------  --------
               rec_len         00-04      5    Logical Record Length
               RecStat         05         1    Record Status
               Type            06         1    Type of Record
               BLvl            07         1    Bibliographic Level
               Ctrl            08         1
               Undefldr        09-11      3    [x22]
               base_addr       12-16      5    Base Address of Data
               ELvl            17         1    Encoding Level
               Desc            18         1    Descriptive Cataloging Form
               ln_rec          19         1    Linked-Record Code
               len_len_field   20         1    Length "length of field" [4]
               len_start_char  21         1    Length "start char pos" [5]
               len_impl        22         1    Length "implementation dep" [0]
               Undef2ldr       23         1    [0]

       MARC::Rec::unpack_ldr() is identical to unpack_ldr, but does not need the record number.

   get_hash_ldr($record)
       Takes a record number. Returns a ref to the cached version of the hash ldr if it exists.
       Does this *without* overwriting the hash ldr. Allows external code to safely manipulate
       hash versions of the ldr.

            my $rhldr = $marc->get_hash_ldr($record);
            return undef unless $rhldr;
            $rhldr->{'Desc'} =~ s/a/b/;
            $ldr = $x->pack_ldr($record);

       MARC::Rec::get_hash_ldr() is identical to get_hash_ldr, but does not need the record
       number.

   pack_ldr($record)
       Takes a record number. Updates the appropriate ldr.

            $marc->[$record]{'unp_ldr'}{'Desc'} =~ s/a/b/;
            my $ldr = $x->pack_ldr($record);
            return undef unless $ldr;

       MARC::Rec::pack_ldr() is identical to pack_ldr, but does not need the record number.

   bib_format($record)
       Takes a record number. Returns the "format" used in determining the meanings of the fixed
       fields in 008. Will force update of the ldr based on any existing hash version.

             foreach $record (1..$#$x) {
                   next if $x->bib_format($record) eq 'SERIALS';
                       # serials are hard
                   do_something($x->[record]);
             }

       MARC::Rec::bib_format() is identical to bib_format, but does not need the record number.

   unpack_008($record)
       Returns a ref to hash version of the 008 field, based on the field's value.  Installs the
       ref as $marc->[$record]{unp_008}

             foreach $record (1..$#$x) {
                   my $rff = $x->unpack_008($record);
                   print "Record $record: Y2K problem possible"
                       if ($rff->{'Date1'}=~/00/ or $rff->{'Date2'}=~/00/);
             }

       MARC::Rec::unpack_008() is identical to unpack_008, but does not need the record number.

   get_hash_008($record)
       Takes a record number. Returns a ref to the cached version of the hash 008 if it exists.
       Does this *without* overwriting the hash 008. Allows external code to safely manipulate
       hash versions of the 008.

            my $rh008 = $marc->get_hash_008($record);
            return undef unless $rh008;
            $rh008->{'Date1'} =~ s/00/01/;
            my $m008 = $x->pack_008($record);
            return undef unless $m008;

       MARC::Rec::get_hash_008() is identical to get_hash_008, but does not need the record
       number.

   pack_008($record)
       Takes a record number and updates the appropriate 008. Will force update of the ldr based
       on any existing hash version.

             foreach $record (1..$#$x) {
                   my $rff = $x->unpack_008($record);
                   $rff->{'Date1'}='2000';
                   print "Record:$record Y2K problem created";
                   $x->pack_008($record);
                   # New value is in the 008 field of $record'th marc
             }

       MARC::Rec::pack_008() is identical to pack_008, but does not need the record number.

   deletefirst()
       deletefirst() takes a template. It deletes the field data for a first match, using the
       template and leaves the rest alone. If the template has a subfield element it deletes
       based on the subfield information in the template. If the last subfield of a field is
       deleted, deletefirst() also deletes the field.  It complains about attempts to delete
       indicators.  If there is no match, it does nothing. Deletefirst also rebuilds the map if
       the template asks for that $do_rebuild_map. Deletefirst returns the number of matches
       deleted (that would be 0 or 1), or undef if it feels grumpy (i.e. carps).

       MARC::Rec::deletefirst($template) is identical to deletefirst, but ignores any record
       number specified by $template.

       Most use of deletefirst is expected to be by MARC::Tie.

   deletemarc()
       This method will allow you to remove a specific record, fields or subfields from a MARC
       object. Accepted parameters include: record, field and subfield. Note: you can use the ..
       operator to delete a range of records. deletemarc() will return the number of items
       deleted (be they records, fields or subfields). The record parameter is optional. It
       defaults to all user records [1..$#marc] if not specified.

               #delete all the records in the object
           $x->deletemarc();

               #delete records 1-5 and 7
           $x->deletemarc({record=>[1..5,7]});

               #delete all of the 650 fields from all of the records
           $x->deletemarc({field=>'650'});

               #delete the 110 field in record 2
           $x->deletemarc({record=>'2',field=>'110'});

               #delete all of the subfield h's in the 245 fields
           $x->deletemarc({field=>'245',subfield=>'h'});

   updatefirst()
       updatefirst() takes a template, and an array from $marc->[recnum]{array}. It
       replaces/creates the field data for a first match, using the template and the array, and
       leaves the rest alone. If the template has a subfield element, (this includes indicators)
       it ignores all other information in the array and only updates/creates based on the
       subfield information in the array. If the template has no subfield information then
       indicators are left untouched unless a new field needs to be created, in which case they
       are left blank.

       MARC::Rec::updatefirst($template) is identical to deletefirst, but ignores any record
       number specified by $template.

       Most use of updatefirst() is expected to be from MARC::Tie.  It does not currently provide
       a useful return value.

   updatefields()
       updatefields() takes a template which specifies recnum, a $do_rebuild_map and a field
       (needs the field in case $rafields->[0] is empty). It also takes a ref to an array of
       fieldrefs formatted like the output of getfields(), and replaces/creates the field data.
       It assumes that it should replace the fields with the first tag in the fieldrefs. It calls
       rebuild_map() if $do_rebuild_map.

           #Let's kill the *last* 500 field.
           my $loc500 = {record=>1,field=>500,rebuild_map=>1};
           my @rfields = $x->getfields($loc500);
           pop @rfields;
           $x->updatefields($loc500,\@rfields);

   getmatch()
       getmatch() takes a subfield code (can be an indicator) and a fieldref.  Returns 0 or a ref
       to the value to be updated.

           #Let's update the value of i2 for the *last* 500
           my $loc500 = {record=>1,field=>500,rebuild_map=>1};
           my @rfields = $x->getfields($loc500);
           my $rvictim = pop @rfields;
           my $rval = getmatch('i2',$rvictim);
           $$rval = "4" if $rval;

       MARC::Rec::getmatch($subf,$rfield) is identical to getmatch;

   insertpos()
       insertpos() takes a subfield code (can not be an indicator), a value, and a fieldref.
       Updates the fieldref with the first place that the fieldref can match. Assumes there is no
       exact subfield match in $fieldref.

           #Let's update the value of subfield 'a' for the *last* 500
           my $value = "new info";
           my $loc500 = {record=>1,field=>500,rebuild_map=>1};
           my @rfields = $x->getfields($loc500);
           my $rvictim = pop @rfields;
           my $rval = getmatch('a',$rvictim);
           if ($rval) {
               $$rval = $value ;
           } else {
               $x->insertpos('a',$value,$rvictim);
           }

       MARC::Rec::insertpos($subf,$value,$rfield) is identical to insertpos;

   selectmarc()
       This method will select specific records from a MARC object and delete the rest. You can
       specify both individual records and ranges of records in the same way as deletemarc().
       selectmarc() will also return the number of records deleted.

           $x->selectmarc(['3']);
           $y=$x->selectmarc(['4','21-50','60']);
           print "$y records selected!";

   searchmarc()
       This method will allow you to search through a MARC object, and retrieve record numbers
       for records that matched your criteria. You can search for: 1) records that contain a
       particular field, or field and subfield ; 2) records that have fields or subfields that
       match a regular expression ; 3) and records that have fields or subfields that do not
       match a regular expression. The record numbers are returned to you in an array which you
       can then use with deletemarc(), selectmarc() and output() if you want.

       ·   1) Field/Subfield Presence:

               @records=$x->searchmarc({field=>"245"});
               @records=$x->searchmarc({field=>"245",subfield=>"a"});

       ·   2) Field/Subfield Match:

               @records=$x->searchmarc({field=>"245",
                                        regex=>"/huckleberry/i"});
               @records=$x->searchmarc({field=>"260",subfield=>"c",
                                        regex=>"/19../"});

       ·   3) Field/Subfield NotMatch:

               @records=$x->searchmarc({field=>"245",
                                        notregex=>"/huckleberry/i"});
               @records=$x->searchmarc({field=>"260",
                                        subfield=>"c",notregex=>"/19../"});

   createrecord()
       You can use this method to initialize a new record. It only takes one optional parameter,
       leader which sets the 24 characters in the record leader: see
       http://lcweb.loc.gov/marc/bibliographic/ecbdhome.html for more details on the leader.
       Note: you do not need to pass character positions 00-04 or 12-16 since these are
       calculated by MARC.pm if outputting to MARC you can assign 0 to each position. If no
       leader is passed a default USMARC leader will be created of "00000nam  2200000 a 4500".
       createrecord() will return the record number for the record that was created, which you
       will need to use later when adding fields with addfield(). Createrecord now makes the new
       record an instance of an appropriate MARC::Rec subclass.

           use MARC;
           my $x = new MARC;
           $record_number = $x->createrecord();
           $record_number = $x->createrecord({leader=>
                                              "00000nmm  2200000 a 4500"});

       MARC::Rec::createrecord($leader) returns an instance of a suitable subclass of MARC::Rec.

   getupdate()
       The getupdate() method returns an array that contains the contents of a fieldin a defined
       order that permits restoring the field after deleting it. This permits changing only
       individual subfields while keeping other data intact. If a field is repeated in the
       record, the resulting array separates the field infomation with an element containing
       "\036" - the internal field separator which can never occur in real MARC data parameters.
       A non-existing field returns "undef". An example will make the structure clearer. The next
       two MARC fields (shown in ASCII) will be described in the following array:

                       246  30  $aPhoto archive
                       246  3   $aAssociated Press photo archive

           my $update246 = {field=>'246',record=>2,ordered=>'y'};
               # next two statements are equivalent
           my @u246 = $x->getupdate($update246);
               # or
           my @u246 = ('i1','3','i2','0',
                       'a','Photo archive',"\036",
                       'i1','3','i2',' ',
                       'a','Associated Press photo archive',"\036");

       After making any desired modifications to the data, the existing field can be replaced
       using the following sequence (for non-repeating fields):

           $x->deletemarc($update246));
           my @records = ();
           foreach my $y1 (@u246) {
               last if ($y1 eq "\036");
               push @records, $y1;
           }
           $x->addfield($update246, @records);

   updaterecord()
       The updaterecord() method is a more complete version of the preceding sequence with error
       checking and the ability to split the update array into multiple addfield() commands when
       given repeating fields. It takes an array of key/value pairs, formatted like the output of
       getupdate(), and replaces/creates the field data. For repeated tags, a "\036" element is
       used to delimit data into separate addfield() commands. It returns the number of
       successful addfield() commands or "undef" on failure.

           $repeats = $x->updaterecord($update246, @u246);     # same as above

   addfield()
       This method will allow you to addfields to a specified record. The syntax may look
       confusing at first, but once you understand it you will be able to add fields to records
       that you have read in, or to records that you have created with createrecord(). addfield()
       takes six parameters: record which indicates the record number to add the field to, field
       which indicates the field you wish to create (ie. 245), i1 which holds one character for
       the first indicator, i2 which holds one character for the second indicator, and value
       which holds the subfield data that you wish to add to the field. addfield() will
       automatically try to insert your new field in tag order (ie. a 500 field before a 520
       field), however you can turn this off if you set ordered to "no" which will add the field
       to the end. Here are some examples:

           $y = $x->createrecord(); # $y will store the record number created

           $x->addfield({record=>"$y", field=>"100", i1=>"1", i2=>"0",
                         value=> [a=>"Twain, Mark, ", d=>"1835-1910."]});

           $x->addfield({record=>"$y", field=>"245",
                         i1=>"1", i2=>"4", value=>
                        [a=>"The adventures of Huckleberry Finn /",
                         c=>"Mark Twain ; illustrated by E.W. Kemble."]});

       This example intitalized a new record, and added a 100 field and a 245 field. For some
       more creative uses of the addfield() function take a look at the EXAMPLES section. The
       value parameters, including i1 and i2, can be specified using a separate array. This
       permits restoring field(s) from the array returned by the getupdate() method - either as-
       is or with modifications. The i1 and i2 key/value pairs must be first and in that order if
       included.

               # same as "100" example above
           my @v100 = 'i1','1','i2',"0",'a',"Twain, Mark, ",
                      'd',"1835-1910.";
           $x->addfield({record=>"$y", field=>"100"}, @v100);

   add_005s()
       Add_005s takes a specification of records (defaults to everything) and updates the
       indicated records with updated 005 fields (date of last transaction).

   output()
       Output is a multifunctional method for creating formatted output from a MARC object. There
       are three parameters file, format, records. If file is specified the output will be
       directed to that file. It is important to specify with > and >> whether you want to create
       or append the file! If no file is specified then the results of the output will be
       returned to a variable (both variations are listed below).

       The MARC standard includes a control field (005) that records the date of last automatic
       processing. This is implemented as a side-effect of output() to a file or if explicitly
       requested via a add_005s field of the template. The current time is stamped on the records
       indicated by the template.

       Valid format values currently include usmarc, marcmaker, ascii, html, urls, and isbd. The
       optional records parameter allows you to pass an array of record numbers which you wish to
       output. You must pass the array as a reference, hence the forward-slash in \@records
       below. If you do not include records the output will default to all the records in the
       object.

       The lineterm parameter can be used to override the line-ending default for any of the
       formats. MARCMaker defaults to CRLF (the format was originally released for MS-DOS). The
       others use '\n' as the default.

       With the MARCMaker format, a charset parameter accepts a hash-reference to a user supplied
       character translation table. The "ustext.txt" table supplied with the LoC. MARCBreaker
       utility is used internally as the default. You can use the ustext_default method to get a
       hash-reference to it if you only want to modify a couple of characters. See example below.

       The MARCMaker Specification requires that long lines be split to less than 80 columns.
       While that behavior is the default, the nolinebreak parameter can override it and the
       resulting output will be much like the ascii format.

       MARC::Rec::output($template) is the same as output except that ignores record number(s)
       and only outputs its caller. (E.g., with $format eq 'urls' it does not output html header
       and footer information.)

       ·   MARC

               $x->output({file=>">mymarc.dat",'format'=>"usmarc"});
               $x->output({file=>">mymarc.dat",'format'=>"usmarc",
                           records=>\@records});
               $y=$x->output({'format'=>"usmarc"}); #put the output into $y

       ·   MARCMaker

               $x->output({file=>">mymarcmaker.mkr",'format'=>"marcmaker"});
               $x->output({file=>">mymarcmaker.mkr",'format'=>"marcmaker",
                           records=>\@records});
               $y=$x->output({'format'=>"marcmaker"}); #put the output into $y

               $x->output({file=>"brkrtest.mkr",'format'=>"marcmaker",
                           nolinebreak=>"1", lineterm=>"\n",
                           charset=>\%char_hash});

       ·   ASCII

               $x->output({file=>">myascii.txt",'format'=>"ascii"});
               $x->output({file=>">myascii.txt",'format'=>"ascii",
                           records=>\@records});
               $y=$x->output({'format'=>"ascii"}); #put the output into $y

       ·   HTML

           The HTML output method has some additional parameters. fields which if set to "all"
           will output all of the fields. Or you can pass the tag number and a label that you
           want to use for that tag. This will result in HTML output that only contains the
           specified tags, and will use the label in place of the MARC code.

               $x->output({file=>">myhtml.html",'format'=>"html",
                           fields=>"all"});

                   #this will only output the 100 and 245 fields, with the
                   #labels "Title: " and "Author: "
               $x->output({file=>">myhtml.html",'format'=>"html",
                           245=>"Title: ",100=>"Author: "});

               $y=$x->output({'format'=>"html"});

           If you want to build the HTML file in stages, there are four other format values
           available to you: 1) "html_header", 2) "html_start", 3) "html_body", and 4)
           "html_footer". Be careful to use the >> append when adding to a file though!

               $x->output({file=>">myhtml.html",
                           'format'=>"html_header"}); # Content-type
               $x->output({file=>">>myhtml.html",
                           'format'=>"html_start"});  # <BODY>
               $x->output({file=>">>myhtml.html",
                           'format'=>"html_body",fields=>"all"});
               $x->output({file=>">>myhtml.html",
                           'format'=>"html_footer"});

       ·   URLS

               $x->output({file=>"urls.html",'format'=>"urls"});
               $y=$x->output({'format'=>"urls"});

       ·   ISBD

           An experimental output format that attempts to mimic the ISBD.

               $x->output({file=>"isbd.txt",'format'=>"isbd"});
               $y=$x->output({'format'=>"isbd"});

       ·   XML

           Roundtrip conversion between MARC and XML is handled by the subclass MARC::XML.
           MARC::XML is available for download from the CPAN.

   usmarc_default()
       This method returns a hash reference to a translation table between mnemonics delimited by
       curly braces and single-byte character codes in the MARC record.  Multi-byte characters
       are not currently supported. The hash has keys of the form '{esc}' and values of the form
       chr(0x1b). It is used during MARCMaker input.

           my %inc = %{$x->usmarc_default()};
           printf "dollar = %s\n", $inc{'dollar'};     # prints '$'
           $inc{'yen'} = 'Y';
           $x->openmarc({file=>"makrbrkr.mrc",'format'=>"marcmaker",
                         charset=>\%inc});

       MARC::Rec::usmarc_default is identical to usmarc_default;

   ustext_default()
       This method returns a hash reference to a translation table between single-byte character
       codes and mnemonics delimited by curly braces. Multi-byte characters are not currently
       supported. The hash has keys of the form chr(0x1b) and values of the form '{esc}'. It is
       used during MARCMaker output.

           my %outc = %{$x->ustext_default()};
           printf "dollar = %s\n", $outc{'$'}; # prints '{dollar}'
           $outc{'$'} = '{uscash}';
           printf "dollar = %s\n", $outc{'$'}; # prints '{uscash}'
           $y = $x->output({'format'=>"marcmaker", charset=>\%outc});

       MARC::Rec::ustext_default is identical to ustext_default;

   as_string()
       As_string() takes no parameters and returns a (Unix) newline separated version of the
       record.

         Format is: $tag<SPACE>$i1$i2<SPACE>$subfields
         where $subfields are separated by "\c_" binary subfield indicators.
         Tag 000 is ldr.

       Subclasses may need to override this format. If so, they should override from_string.

   from_string()
       From_string() takes a string parameter and updates the calling record's {array}
       information.  It assumes the string is formatted like the output of as_string().

EXAMPLES

       Here are a few examples to fire your imagination.

       ·   This example will read in the complete contents of a MARC file called "mymarc.dat" and
           then output it as a MARCMaker file called "mymkr.mkr".

               #!/usr/bin/perl
               use MARC;
               $x = MARC->new("mymarc.dat","marcmaker");
               $x->output({file=>"mymkr.mkr",'format'=>"marcmaker");

       ·   The MARC object occupies a fair number of working memory, and you may want to do
           conversions on very large files. In this case you will want to use the openmarc(),
           nextmarc(), deletemarc(), and closemarc() methods to read in portions of the MARC
           file, do something with the record(s), remove them from the object, and then read in
           the next record(s). This example will read in one record at a time from a MARC file
           called "mymarc.dat" and convert it to a MARC Maker file called "myfile.mkr".

               #!/usr/bin/perl
               use MARC;
               $x = new MARC;
               $x->openmarc({file=>"mymarc.dat",'format'=>"usmarc"});
               while ($x->nextmarc(1)) {
                   $x->output({file=>">>myfile.mkr",'format'=>"marcmaker"});
                   $x->deletemarc(); #empty the object for reading in another
               }

       ·   Perhaps you have a tab delimited text file of data for online journals you have access
           to from Dow Jones Interactive, and you would like to create a batch of MARC records to
           load into your catalog. In this case you can use createrecord(), addfield() and
           output() to create records as you read in your delimited file. When you are done, you
           then output to a file in USMARC.

               #!/usr/bin/perl
               use MARC;
               $x = new MARC;
               open (INPUT_FILE, "delimited_file");
               while ($line=<INPUT_FILE>) {
                   ($journaltitle,$issn) = split /\t/,$line;
                   $num=$x->createrecord();
                   $x->addfield({record=>$num,
                                 field=>"022",
                                 i1=>" ", i2=>" ",
                                 value=>$issn});
                   $x->addfield({record=>$num,
                                 field=>"245",
                                 i1=>"0", i2=>" ",
                                 value=>[a=>$journaltitle]});
                   $x->addfield({record=>$num,
                                 field=>"260",
                                 i1=>" ", i2=>" ",
                                 value=>[a=>"New York (N.Y.) :",
                                         b=>"Dow Jones & Company"]});
                   $x->addfield({record=>$num,
                                 field=>"710",
                                 i1=>"2", i2=>" ",
                                 value=>[a=>"Dow Jones Interactive."]});
                   $x->addfield({record=>$num,
                                 field=>"856",
                                 i1=>"4", i2=>" ",
                                 value=>[u=>"http://www.djnr.com",
                                         z=>"Connect"]});
               }
               close INPUT_FILE;
               $x->output({file=>">dowjones.mrc",'format'=>"usmarc"})

       ·   Perhaps you have periodicals coming in that you want to order by location and then
           title. MARC::Rec's get you out of some array indexing.

           #!/usr/bin//perl use MARC 1.03;

           my @newmarcs=@$marc[1..$#$marc]; # array slice.  my @sortmarcs = sort by_loc_oclc
           @newmarcs; @marc[1..$#$marc] = @sortmarcs;

           sub by_loc_title {
               my ($aloc,$atitle) = loc_title($a);
               my ($bloc,$btitle) = loc_title($b);
               return  $aloc cmp $bloc                ||        $atitle cmp $btitle; }

           sub loc_title {
               my ($rec)=@_;
               my $n049 = $rec->getfirstvalue({field=>040});
               my ($loc) = $n049=~/(ND\S+)/; # Or the first two letters of your OCLC
                                             # location.

               my $title = $rec->getfirstvalue({field=>100,delimiter=>" "});

               return ($loc,$title);
           }

NOTES

       Please let us know if you run into any difficulties using MARC.pm - we'd be happy to try
       to help. Also, please contact us if you notice any bugs, or if you would like to suggest
       an improvement/enhancement. Email addresses are listed at the bottom of this page.

       Development of MARC.pm and other library oriented Perl utilities is conducted on the
       Perl4Lib listserv. Perl4Lib is an open list and is an ideal place to ask questions about
       MARC.pm. Subscription information is available at http://www.vims.edu/perl4lib

       Two global boolean variables are reserved for test and debugging. Both are "0" (off) by
       default. The $TEST variable disables internal error messages generated using Carp. It also
       overrides the date_stamp in the "005" field with a constant "19960221075055.7". It should
       only be used in the automatic test suite. The $DEBUG variable adds verbose diagnostic
       messages. Since both variables are used only in testing, MARC::Rec uses $MARC::TEST and
       $MARC::DEBUG rather than define a second pair.

AUTHORS

       Chuck Bearden cbearden@rice.edu

       Bill Birthisel wcbirthisel@alum.mit.edu

       Derek Lane dereklane@pobox.com

       Charles McFadden chuck@vims.edu

       Ed Summers ed@cheetahmail.com

SEE ALSO

       perl(1), http://lcweb.loc.gov/marc

COPYRIGHT

       Copyright (C) 1999,2000, Bearden, Birthisel, Lane, McFadden, and Summers.  All rights
       reserved. This module is free software; you can redistribute it and/or modify it under the
       same terms as Perl itself. 23 April 2000.  Portions Copyright (C) 1999,2000, Duke
       University, Lane.