NAME
DBD::PO - DBI driver for PO files
$Id: PO.pm 238 2008-09-26 07:55:36Z steffenw $
$HeadURL: https://dbd-po.svn.sourceforge.net/svnroot/dbd-po/trunk/DBD-PO/lib/DBD/PO.pm $
VERSION
0.10
SYNOPSIS
connect
use Carp qw(croak);
use DBI ();
use Socket qw($LF);
my $dbh = DBI->connect(
'DBI:PO:'
. 'f_dir=dir_x;' # optional:
# The default database is './',
# here set to the directory 'dir_x'.
# The default value is often unuseful.
. "po_separator=\n;" # optional:
# The default 'po_separator' to set/get
# concatinated data is "\n",
# here set to "\n" unnecessary.
# The default value is mostly useful
# because the internal representation
# of line ending is \n (Windows too).
# This is different for binary IO.
. "po_eol=$LF;" # optional:
# The default 'po_eol' (po end of line)
# is network typical like 'use Socket qw($CRLF)',
# here set to $LF like 'use Socket qw($LF)'.
# The default value is mostly useful.
. 'po_charset=utf-8', # optional:
# Write nothing for 'iso-8859-1' files
# and use bytes internal.
# (It is not a good idea.)
# Write 'utf-8' for 'utf-8' files
# and use unicode chars internal.
# Write 'iso-8859-1' for 'iso-8859-1' files
# and use unicode chars internal
# and so on for other charsets.
# The default value is mostly not useful.
undef, # Username is not used.
undef, # Password is not used.
{
RaiseError => 1, # The easy way to handle exceptions.
PrintError => 0, # No extra console output.
},
) or croak 'Cannot connect: ' . DBI->errstr();
create table
Note that no column names will be stored. They are fixed. Thus all other information including column type (INTEGER or CHAR(x), for example), column attributes (NOT NULL, PRIMARY KEY, ...) will silently be discarded.
Table names cannot be arbitrary, due to restrictions of the SQL syntax. I recommend that table names are valid SQL identifiers: The first character is alphabetic, followed by an arbitrary number of alphanumeric characters. If you want to use other files, the file names must start with '/', './' or '../' and they must not contain white space.
For conditional execution use CREATE TABLE IF EXISTS statement.
Columns:
comment
translator comment text concatinated by 'po_separator'
automatic
automatic comment text concatinated by 'po_separator'
reference
where the text to translate is from, concatinated by 'po_separator'
obsolete
the translation is used (0) or not (1)
fuzzy
the translation is finished (0) or not (1)
c_format (c-format, no-c-format)
format flag, not set (0), set (1) or negative set (-1)
php_format (php-format, no-php-format)
format flag, not set (0), set (1) or negative set (-1)
msgid
the text to translate (emty string for header)
The 'msgid' can contain Locale::Maketext placeholders. They have to be stored in gettext format. To change the format, use the database handle function 'maketext_to_gettext'.
msgstr
the translation
The 'msgid' can contain Locale::Maketext placeholder. They have to be stored in gettext format. To change the format, use the database handle function 'maketext_to_gettext'.
$dbh->do(<<'EOT');
CREATE TABLE
table.po (
comment VARCHAR,
automatic VARCHAR,
reference VARCHAR,
obsolete INTEGER,
fuzzy INTEGER,
c_format INTEGER,
php_format INTEGER,
msgid VARCHAR,
msgstr VARCHAR
)
EOT
write the header
build msgstr
The charset will set to the as parameter 'po_charset' given value at the connect method. Note that the default encoding is nothing, not 'utf-8'.
minimized example
my $header_msgstr = $dbh->func(
undef,
# function name
'build_header_msgstr',
);
full example (but use the named one)
my $header_msgstr = $dbh->func(
[
# project
'Project name',
# ISO 8601 time format like [YYYY]-[MM]-[DD]T[hh]::[mm]:[ss]Z
'the POT creation date',
'the PO revision date',
# last translator name and mail address
[
'Steffen Winkler',
'steffenw@example.org',
],
# language team, name and mail address
[
'MyTeam',
'cpan@example.org',
],
# undef to accept the defaut settings
undef, # mime version (1.0)
undef, # arrayref of content type (text/plain) and charset
# 'iso-8859-1' or given as po_charset at the connect method
undef, # content transfer encoding (8bit)
# place here pairs for extra parameters
[qw(
X-Poedit-Language German
X-Poedit-Country GERMANY
X-Poedit-SourceCharset utf-8
)],
],
# function name
'build_header_msgstr',
);
full example using named parameters
my $header_msgstr = $dbh->func(
{
'Project-Id-Version' => 'Project name',
'POT-Creation-Date' => 'the POT creation date',
'PO-Revision-Date' => 'the PO revision date',
'Last-Translator-Name' => 'Steffen Winkler',
'Last-Translator-Mail' => 'steffenw@example.org',
'Language-Team-Name' => 'MyTeam',
'Language-Team-Mail' => 'cpan@example.org',
# Do not set the following values.
# They will be set automaticly.
'MIME-Version' => '1.0',
'Content-Type' => 'text/plain',
charset => $po_charset || 'iso-8859-1',
'Content-Transfer-Encoding' => '8bit',
# place here pairs for extra parameters
extended => [qw(
X-Poedit-Language German
X-Poedit-Country GERMANY
X-Poedit-SourceCharset utf-8
)],
},
# function name
'build_header_msgstr',
);
write header row
Write the header row always at first!
use Socket qw($CRLF);
my $separator = $CRLF; # But it is more easy
# to use the po_separator default \n
# and than the join is obsolete
# because the strings typical including \n.
my $header_comment = join(
$separator,
'This is a translator comment for the header.',
'And this is line 2 of.',
);
$dbh->do(<<'EOT', undef, $header_comment, $header_msgstr);
INSERT INTO table.po (
comment,
msgstr
) VALUES (?, ?)
EOT
write a row
without Locale::Maketext placeholders (not typical)
my $sth = $dbh->prepare(<<'EOT');
INSERT INTO table.po (
msgid,
msgstr,
reference
) VALUES (?, ?, ?)
EOT
$sth->execute(
join(
$separator,
'text to translate',
'2nd line of text',
),
join(
$separator,
'translation',
'2nd line of translation',
),
join(
$separator,
'my_program: 17',
'my_program: 269',
),
);
with Locale::Maketext placeholders
my $sth = $dbh->prepare(<<'EOT');
INSERT INTO table.po (
msgid,
msgstr,
reference
) VALUES (?, ?, ?)
EOT
$sth->execute(
$dbh->func(
# mapping:
# - scalar to scalar
# - or array to array
# - here 2 values given to 2 returns
join(
$separator,
'text to translate',
'2nd line of text',
),
join(
$separator,
'translation',
'2nd line of translation',
),
'maketext_to_gettext',
),
join(
$separator,
'my_program: 17',
'my_program: 269',
),
);
read the header
easy
read only 1 header information
Scalar to scalar mapping.
my $charset = $dhh->func(
{table => 'table_name'},
'charset',
'get_header_msgstr_data',
);
read more header informations
Arrayref to arrayref mapping.
my $array_ref = @{
$dbh->func(
{table => 'table_name'},
[qw(charset Project-Id-Version)],
'get_header_msgstr_data',
)
};
my ($charset, $project_id_version) = @{$array_ref};
not easy (do not use)
# read the header msgstr
$sth = $dbh->prepare(<<'EOT');
SELECT msgstr
FROM table.po
WHERE msgid = ''
EOT
$sth->execute();
($header_msgstr) = $sth->fetchrow_array();
# extract the header data
my $header_struct = $dbh->func(
$header_msgstr,
# function name
'split_header_msgstr',
);
# get values by name
my $charset = $dhh->func(
$header_struct,
'charset',
'get_header_msgstr_data',
);
not easy, implicit call of split_header_msgstr (do not use)
# read the header msgstr
$sth = $dbh->prepare(<<'EOT');
SELECT msgstr
FROM table.po
WHERE msgid = ''
EOT
$sth->execute();
($header_msgstr) = $sth->fetchrow_array();
# get values by name
my $charset = $dhh->func(
$header_msgstr,
'charset',
'get_header_msgstr_data',
);
not easy, implicit SQL call (do not use)
# extract the header data
my $header_struct = $dbh->func(
{table => 'table_name'},
# function name
'split_header_msgstr',
);
# get values by name
my $charset = $dhh->func(
$header_struct,
'charset',
'get_header_msgstr_data',
);
read a row
$sth = $dbh->prepare(<<'EOT');
SELECT msgstr
FROM table.po
WHERE msgid = ?
EOT
$sth->execute(
join(
$separator,
'text to translate',
'2nd line of text',
),
);
my ($msgstr) = $sth->fetchrow_array();
update rows
$dbh->do(<<'EOT');
UPDATE table.po
SET msgstr = '',
fuzzy = 1
WHERE msgid = 'my_id'
EOT
delete rows
$dbh->do(<<'EOT');
DELETE FROM table.po
WHERE obsolete = 1
EOT
drop table
For conditional execution use DROP TABLE IF EXISTS statement.
$dbh->do(<<'EOT');
DROP TABLE table.po
EOT
disconnect
$dbh->disconnect();
EXAMPLE
Inside of this Distribution is a directory named example. Run this *.pl files.
DESCRIPTION
The DBD::PO module is yet another driver for the DBI (Database independent interface for Perl). This one is based on the SQL 'engine' SQL::Statement and the abstract DBI driver DBD::File and implements access to so-called PO files (GNU gettext). Such files are readable by Locale::Maketext.
See DBI for details on DBI, SQL::Statement for details on SQL::Statement and DBD::File for details on the base class DBD::File.
---------------------
| DBI |
---------------------
|
---------------------
| DBD::File |
| (SQL::Statement) |
---------------------
|
---------------------
| DBD::PO |
---------------------
|
---------------------
| DBD::PO::Text::PO |
---------------------
|
---------------------
| DBD::PO::Locale::PO |
---------------------
|
table_file.po
SUBROUTINES/METHODS
nothing in this module
DIAGNOSTICS
see DBI
CONFIGURATION AND ENVIRONMENT
see DBI
DEPENDENCIES
Carp
Socket
parent
DBI
Prerequisites
The only system dependent feature that DBD::File uses, is the flock()
function. Thus the module should run (in theory) on any system with a working flock()
, in particular on all Unix machines and on Windows NT. Under Windows 95 and MacOS the use of flock()
is disabled, thus the module should still be usable,
Unlike other DBI drivers, you don't need an external SQL engine or a running server. All you need are the following Perl modules, available from any CPAN mirror.
SQL
The level of SQL support available depends on the version of SQL::Statement installed. Any version will support *basic* CREATE, INSERT, DELETE, UPDATE, and SELECT statements. Only versions of SQL::Statement 1.0 and above support additional features such as table joins, string functions, etc. See the documentation of the latest version of SQL::Statement for details.
INCOMPATIBILITIES
not known
BUGS AND LIMITATIONS
The module is using flock() internally. However, this function is not available on platforms. Using flock() is disabled on MacOS and Windows 95: There's no locking at all (perhaps not so important on these operating systems, as they are for single users anyways).
SEE ALSO
DBI
Locale::PO has bugs, more than documented
AUTHOR
Steffen Winkler
LICENSE AND COPYRIGHT
Copyright (c) 2008, Steffen Winkler <steffenw at cpan.org>
. All rights reserved.
This module is free software; you can redistribute it and/or modify it under the same terms as Perl itself.