NAME
File::Tabular - searching and editing flat tabular files
SYNOPSIS
use File::Tabular;
my $f = new File::Tabular($filename);
my $row = $f->fetchrow;
print $row->{field1}, $row->{field2};
$row = $f->fetchrow(where => 'someWord');
$row = $f->fetchrow(where => 'field1 > 4 AND field2 >= "01.01.2001"');
$row = $f->fetchrow(where => qr/some\s+(complex\s*)?(regex|regular expression)/i);
$f->rewind;
my $rows = $f->fetchall(where => 'someField =~ ^[abc]+');
print $_->{someField} foreach @$rows;
$f->rewind;
$rows = $f->fetchall(where => '+field1:someWord -field2:otherWord',
orderBy => 'field3, field6:num, field5:-alpha');
$f->rewind;
my $hashRows = $f->fetchall(where => 'foo AND NOT bar',
key => 'someField');
print $hashRows->{someKey}{someOtherField};
# open for updates, and remember the updates in a journal file
$f = new File::Tabular("+<$filename", {journal => ">>$journalFile"});
# updates at specific positions (line numbers)
$f->splices(4 => 2, undef, # delete 2 lines from position 4
7 => 1, {f1 => $v1, f2 => $v2, ...}, # replace line 7
9 => 0, { ...}, # insert 1 new line at position 9
22 => 0, [{...}, {...}, ...] # insert several lines at pos. 22
...
-1 => 0, [{...}, {...}, ...] # append at the end
);
# shorthand to add new data at the end
$f->append({f1 => $v1, f2 => $v2, ...});
# same thing, but use the "Hash::Type" associated to the file
$f->append($f->ht->new($v1, $v2, ...));
$f->clear; # removes all data (but keeps the header line)
# updates at specific keys, corresponding to @keyFields
$f->writeKeys({key1 => {f1 => $v1, f2 => $v2, ...}, # add or update
key2 => undef, # remove
...
}, @keyFields);
# replay the updates on a backup file
my $bck = new File::Tabular("+<$backupFile");
$bck->playJournal($journalFile);
DESCRIPTION
A tabular file is a flat text file containing data organised in rows (records) and columns (fields).
This module provides database-like functionalities for managing tabular files : retrieving, searching, writing, autonumbering, journaling. However, unlike other modules like DBD::CSV, it doesn't try to make it look like a database : rather, the API was designed specifically for work with tabular files. Instead of SQL, search queries are specified in a web-like fashion, with support for regular expressions and cross-field searches. Queries are compiled internally into perl closures before being applied to every data record, which makes it quite fast.
Write operations take a list of modifications as argument; then they apply the whole list atomically in a single rewrite of the data file.
Here are some of the reasons why you might choose to work with a tabular file rather than a regular database :
no need to install a database system (not even buy one)!
easy portability and data exchange with external tools (text editor, spreadsheet, etc.)
search queries immediately ready for a web application
good search performance, even with several thousand records
On the other hand, tabular files will probably be inappropriate if you need very large volumes of data, complex multi-table data models or frequent write operations.
METHODS
new (open1, open2, ..., {opt1 => v1, opt2 => v2, ...})
-
Creates a new tabular file object. The list of arguments
open1, open2, ...
is fed directly to "open" in perlfunc for opening the associated file. Can also be a reference to an already opened filehandle.The final hash ref is a collection of optional parameters, taken from the following list :
- fieldSep
-
field separator : any character except '%' ('|' by default).
- recordSep
-
record separator ('\n' by default).
- fieldSepRepl
-
string to substitute if fieldSep is met in the data. (by default, url encoding of fieldSep, i.e. '%7C' )
- recordSepRepl
-
string to substitute if recordSep is met in the data (by default, url encoding of recordSep, i.e. '%0A' )
- autoNumField
-
name of field for which autonumbering is turned on (none by default). This is useful to generate keys : when you write a record, the character '#' in that field will be replaced by a fresh number, incremented automatically. This number will be 1 + the largest number read so far (it is your responsability to read all records before the first write operation).
- autoNum
-
initial value of the counter for autonumbering (1 by default).
- autoNumChar
-
character that will be substituted by an autonumber when writing records ('#' by default).
- flockMode
-
mode for locking the file, see "flock" in perlfunc. By default, this will be LOCK_EX if open1 contains '>' or '+<', LOCK_SH otherwise.
- flockAttempts
-
Number of attempts to lock the file, at 1 second intervals, before returning an error. Zero by default. If nonzero, LOCK_NB is added to flockMode; if zero, a single locking attempt will be made, blocking until the lock is available.
- headers
-
reference to an array of field names. If not present, headers will be read from the first line of the file.
- printHeaders
-
if true, the headers will be printed to the file. If not specified, treated as 'true' if open1 contains '>'.
- journal
-
name of journaling file, or reference to a list of arguments for "open" in perlfunc. The journaling file will log all write operations. If specified as a simple file name, it will be be opened in '>>' mode.
A journal file can then be replayed through method "playJournal" (this is useful to recover after a crash, by playing the journal on a backup copy of your data).
- rxDate
-
Regular expression for matching a date. Default value is
qr/^\d\d?\.\d\d?\.\d\d\d?\d?$/
. This will be used by "compileFilter" to perform appropriate comparisons. - date2str
-
Ref to a function for transforming dates into strings suitable for sorting (i.e. year-month-day). Default is :
sub {my ($d, $m, $y) = split /\./, $_[0]; $y += ($y > 50) ? 1900 : 2000 if $y < 100; return "$y$m$d"; }
- rxNum
-
Regular expression for matching a number. Default value is
qr/^[-+]?\d+(?:\.\d*)?$/
. This will be used by "compileFilter" to perform appropriate comparisons. - preMatch/postMatch
-
Strings to insert before or after a match when applying a filter to fetch rows.
fetchrow(where => filter)
-
returns the next record matching the (optional) filter. If there is no filter, just returns the next record.
The filter is either a code reference generated by "compileFilter", or a string which will be automatically fed as argument to "compileFilter"; this string can contain just a word, a regular expression, a complex boolean query involving field names and operators, etc., as explained below.
fetchall(where => filter, orderBy => cmp)
fetchall(where => filter, key => keySpecif)
-
finds all next records matching the (optional) filter. If there is no filter, finds all remaining records.
The filter is either a code reference generated by "compileFilter", or a string which will be automatically fed as argument to "compileFilter".
The return value depends on context and on arguments :
if no key parameter is given, and we are in a scalar context, then
fetchall
returns a reference to an array of records.The optional orderBy parameter can be a field name, a ref to a list of field names, a string like
"field1: -alpha, field2:-num, ..."
, or, more generally, a user-provided comparison function; see "cmp" in Hash::Type for a fully detailed explanation.Otherwise, the resulting array is in data source order.
if no key parameter is given, and we are in a list context, then
fetchall
returns a pair : the first item is a reference to an array of records as explained above ; the second item is a reference to an array of line numbers corresponding to those records (first data line has number 0). These line numbers might be useful later if you update the records through the "splices" method. No orderBy is allowed iffetchall
is called in list context.if a key parameter is given, then
fetchall
returns a reference to a hash, whose values are the retrieved records, and whose keys are built according to the keySpecif argument. This must be either a single field name (scalar), or a a list of field names (ref to an array of scalars). Values corresponding to those field names will form the key for each entry of the hash; if necessary, multiple values are joined together through $;. No orderBy argument is allowed, because hashes have no ordering.
rewind
-
Rewinds the file to the first data line (after the headers)
ht
-
Returns the instance of Hash::Type associated with the file.
headers
-
returns the list of field names
splices
-
splices(pos1 => 2, undef, # delete 2 lines pos2 => 1, row, # replace 1 line pos3 => 0, [row1, row2 ...] # insert lines ... -1 => 0, [row1, ... ] # append lines ); # special case : autonum if pos== -1
Updates the data, in a spirit similar to "splice" in perlfunc (hence the name of the method). The whole file is rewritten in an atomic operation, deleting, replacing or appending data lines as specified by the "splice instructions". Returns the number of "splice instructions" performed.
A splice instruction is a triple composed of :
a position (line number) that specifies the place where modifications will occur. Line numbers start at 0. Position -1 means end of data.
a number of lines to delete (might be zero).
a ref to a hash or to a list of hashes containing new data to insert (or
undef
if there is no new data).
If there are several splice instructions, their positions must be sorted in increasing order (except of course position -1, meaning "end of data", which must appear last).
Positions always refer to line numbers in the original file, before any modifications. Therefore, it makes no sense to write
splices(10 => 5, undef, 12 => 0, $myRow)
because after deleting 5 rows at line 10, we cannot insert a new row at line 12.
The whole collection of splice instructions may also be passed as an array ref instead of a list.
append(row1, row2, ...)
-
This appends new records at the end of data, i.e. it is a shorthand for
splices(-1 => 0, [row1, row2, ...])
clear
-
removes all data (but keeps the header line)
writeKeys({key1 => row1, key2 => ...}, @keyFields)
-
Rewrites the whole file, applying modifications as specified in the hash ref passed as first argument. Keys in this hash are compared to keys built from the original data, according to
@keyFields
. Therefore,row1
may replace an existing row, if the key corresponding tokey1
was found ; otherwise, a new row is added. Ifrow1
isundef
, the corresponding row is deleted from the file.@keyFields
must contain the name of one or several fields that build up the primary key. For each data record, the values corresponding to those fields are taken and joined together through $;, and then compared tokey1
,key2
, etc. playJournal(open1, open2, ...)
-
Reads a sequence of update instructions from a journal file and applies them to the current tabular file. Arguments
open1, open2, ...
will be passed to perl open for opening the journal file ; in most cases, just give the filename.The journal file must contain a sequence of instructions as encoded by the automatic journaling function of this module ; to activate journaling, see the
journal
parameter of the "new" method. compileFilter(query)
-
Compiles a query into a filter (code reference) that can be passed to "fetchrow" or "fetchall".
The query can be
a regular expression compiled through
qr/.../
. The regex will be applied to whole data lines, and therefore covers all fields at once. This is the fastest way to filter lines, because it avoids systematic splitting into data records.a string that will be analyzed through
Search::QueryParser
, and then compiled into a filter function. The query string can contain boolean combinators, parenthesis, comparison operators, etc., as documented in Search::QueryParser. Notice that in addition to usual comparison operators, you can also use regular expressions in queries like+field1=~'^[abc]+' +field2!~'foobar$'
The query compiler needs to distinguish between word and non-word characters ; therefore it is important to
use locale
in your scripts (see perllocale). The compiler tries to be clever about a number of details :- looking for complete words
-
Words in queries become regular expressions enclosed by
\b
(word boundaries) ; so a query forfoo OR bar
will not matchfoobar
. - supports * for word completion
-
A '*' in a word is compiled into regular expression
\w*
; so queriesfoo*
or*bar
will both matchfoobar
. - case insensitive, accent-insensitive
-
Iso-latin-1 accented characters are translated into character classes, so for example
hétaïre
becomesqr/h[ée]ta[ïi]re/i
. Furthermore, as shown in this example, thei
flag is turned on (case-insensitive). Therefore this query will also matchHETAIRE
. - numbers and dates in operators
-
When compiling a subquery like
fieldname >= 'value'
, the compiler checks the value againstrxNum
andrxDate
(as specified in the "new" method). Depending on these tests, the subquery is translated into a string comparison, a numerical comparison, or a date comparison (more precisely,{date2str($a) cmp date2str($b)}
). - pre/postMatch
-
Words matched by a query can be highlighted; see parameters
preMatch
andpostMatch
in the "new" method.
AUTHOR
Laurent Dami, <laurent.dami AT etat ge ch>
COPYRIGHT AND LICENSE
Copyright (C) 2005 by Laurent Dami.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
1 POD Error
The following errors were encountered while parsing the POD:
- Around line 900:
Non-ASCII character seen before =encoding in 'C<hétaïre>'. Assuming CP1252