Files
luos/gulliver/system/class.i18n_po.php

419 lines
12 KiB
PHP
Raw Normal View History

2010-12-02 23:34:41 +00:00
<?php
2010-12-02 23:34:41 +00:00
/**
* class.dbtable.php
*
* @package gulliver.system
2010-12-02 23:34:41 +00:00
*
* ProcessMaker Open Source Edition
2011-01-24 21:07:14 +00:00
* Copyright (C) 2004 - 2011 Colosa Inc.
2010-12-02 23:34:41 +00:00
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2010-12-02 23:34:41 +00:00
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
2010-12-02 23:34:41 +00:00
*
* For more information, contact Colosa Inc, 2566 Le Jeune Rd.,
* Coral Gables, FL, 33134, USA, or email info@colosa.com.
*
*/
/**
* i18n_PO
2010-12-02 23:34:41 +00:00
* This class build biggers PO files without size limit and this not use much memory that the allowed
*
2010-12-02 23:34:41 +00:00
* @package gulliver.system
* @author Erik Amaru Ortiz <erik@colosa.com>
2011-01-14 11:51:34 +00:00
* date Aug 31th, 2010
2010-12-02 23:34:41 +00:00
* @copyright (C) 2002 by Colosa Development Team.
*/
2010-12-02 23:34:41 +00:00
class i18n_PO
{
private $_file = null;
private $_string = '';
private $_meta;
private $_fp;
private $_fileComments;
protected $_editingHeader;
protected $_fileLine;
protected $flagEndHeaders;
protected $flagError;
protected $flagInit;
protected $lineNumber;
public $translatorComments;
public $extractedComments;
public $references;
public $flags;
public $previousUntranslatedStrings;
function __construct ($file)
{
$this->file = $file;
}
function buildInit ()
{
$this->_fp = fopen( $this->file, 'w' );
if (! is_resource( $this->_fp )) {
throw new Exception( 'Could\'t open ' . $this->file . ' file' );
}
// lock PO file exclusively
if (! flock( $this->_fp, LOCK_EX )) {
fclose( $this->_fp );
return false;
}
$this->_meta = 'msgid ""';
$this->_writeLine( $this->_meta );
$this->_meta = 'msgstr ""';
$this->_writeLine( $this->_meta );
$this->_editingHeader = true;
}
function readInit ()
{
$this->_fp = fopen( $this->file, 'r' );
if (! is_resource( $this->_fp )) {
throw new Exception( 'Could\'t open ' . $this->file . ' file' );
}
//skipping comments
$this->skipCommets();
//deaing headers
$this->readHeaders();
$this->translatorComments = Array ();
$this->extractedComments = Array ();
$this->references = Array ();
$this->flags = Array ();
$this->previousUntranslatedStrings = Array ();
}
function addHeader ($id, $value)
{
if ($this->_editingHeader) {
$meta = '"' . trim( $id ) . ': ' . trim( $value ) . '\n"';
$this->_writeLine( $meta );
}
}
function addTranslatorComment ($str)
{
$this->headerStroke();
$comment = '# ' . trim( $str );
$this->_writeLine( $comment );
2010-12-02 23:34:41 +00:00
}
function addExtractedComment ($str)
{
$this->headerStroke();
$comment = '#. ' . trim( $str );
$this->_writeLine( $comment );
2010-12-02 23:34:41 +00:00
}
function addReference ($str)
{
$this->headerStroke();
$reference = '#: ' . trim( $str );
$this->_writeLine( $reference );
2010-12-02 23:34:41 +00:00
}
function addFlag ($str)
{
$this->headerStroke();
$flag = '#, ' . trim( $str );
$this->_writeLine( $flag );
2010-12-02 23:34:41 +00:00
}
function addPreviousUntranslatedString ($str)
{
$this->headerStroke();
$str = '#| ' . trim( $str );
$this->_writeLine( $str );
2010-12-02 23:34:41 +00:00
}
function addTranslation ($msgid, $msgstr)
{
$this->headerStroke();
$this->_writeLine( 'msgid "' . $this->prepare( $msgid, true ) . '"' );
$this->_writeLine( 'msgstr "' . $this->prepare( $msgstr, true ) . '"' );
$this->_writeLine( '' );
2010-12-02 23:34:41 +00:00
}
function _writeLine ($str)
{
$this->_write( $str . "\n" );
}
function _write ($str)
{
fwrite( $this->_fp, $str );
}
function prepare ($string, $reverse = false)
{
//$string = str_replace('\"', '"', $string);
//$string = stripslashes($string);
if ($reverse) {
$smap = array ('"',"\n","\t","\r"
);
$rmap = array ('\"','\\n"' . "\n" . '"','\\t','\\r'
);
return (string) str_replace( $smap, $rmap, $string );
} else {
$string = preg_replace( '/"\s+"/', '', $string );
$smap = array ('\\n','\\r','\\t','\"'
);
$rmap = array ("\n","\r","\t",'"'
);
return (string) str_replace( $smap, $rmap, $string );
2010-12-02 23:34:41 +00:00
}
}
function headerStroke ()
{
if ($this->_editingHeader) {
$this->_editingHeader = false;
$this->_writeLine( '' );
;
}
2010-12-02 23:34:41 +00:00
}
/**
* read funtions *
*/
private function skipCommets ()
{
$this->_fileComments = '';
do {
$lastPos = ftell( $this->_fp );
$line = fgets( $this->_fp );
$this->_fileComments .= $line;
} while ((substr( $line, 0, 1 ) == '#' || trim( $line ) == '') && ! feof( $this->_fp ));
fseek( $this->_fp, $lastPos );
2010-12-02 23:34:41 +00:00
}
private function readHeaders ()
{
$this->flagEndHeaders = false;
$this->flagError = false;
$this->flagInit = true;
$this->lineNumber = 0;
$errMsg = '';
while (! $this->flagError && ! $this->flagEndHeaders) {
if ($this->flagInit) { //in first instance
$this->flagInit = false; //unset init flag
//read the first and second line of the file
$firstLine = fgets( $this->_fp );
$secondLine = fgets( $this->_fp );
//verifying the file head
if (strpos( $firstLine, 'msgid ""' ) === false || strpos( $secondLine, 'msgstr ""' ) === false) {
$this->flagError = true;
$errMsg = 'Misplace for firts msgid "" and msgstr "" in the header';
}
continue;
}
//getting the new line
$this->_fileLine = trim( fgets( $this->_fp ) );
//set line number
$this->lineNumber ++;
//verifying that is not end of file and applying a restriction for to read just the twenty firsts lines
if (trim( $this->_fileLine ) == '' || ! $this->_fileLine || $this->lineNumber >= 20) {
$this->flagEndHeaders = true; // set ending to read the headers
continue;
}
//verify if has a valid mask header line
preg_match( '/^"([a-z0-9\._-]+)\s*:\s*([\W\w]+)\\\n"$/i', $this->_fileLine, $match );
//for a valid header line the $match size should three
if (sizeof( $match ) == 3) {
$key = trim( $match[1] ); //getting the key of the header
$value = trim( $match[2] ); //getting the value of the header
$this->_meta[$key] = $value; //setting a new header
} else {
$this->flagEndHeaders = true; //otherwise set the ending to read the headers
break;
}
} //end looking for headeers
//verifying the headers data
if (! isset( $this->_meta['X-Poedit-Language'] )) {
if (! isset( $this->_meta['Language'] )) {
$this->flagError = true;
$errMsg = "X-Poedit-Language and Language meta doesn't exist";
} else if ($this->_meta['Language'] == '') {
$this->flagError = true;
$errMsg = "Language meta is empty";
} else {
$this->_meta['X-Poedit-Language'] = $this->_meta['Language'];
unset($this->_meta['Language']);
$this->flagError = false;
}
} else if ($this->_meta['X-Poedit-Language'] == '') {
2010-12-02 23:34:41 +00:00
$this->flagError = true;
$errMsg = "X-Poedit-Language meta is empty";
}
//if the country is not present in metadata
if (! isset( $this->_meta['X-Poedit-Country'] )) {
$this->_meta['X-Poedit-Country'] = '.';
} else if ($this->_meta['X-Poedit-Country'] == '') {
$this->_meta['X-Poedit-Country'] = '.';
}
//thowing the exception if is necesary
if ($this->flagError) {
throw new Exception( "This file is not a valid PO file. ($errMsg)" );
}
}
function getHeaders ()
{
return $this->_meta;
}
public function getTranslation ()
{
$flagReadingComments = true;
$this->translatorComments = Array ();
$this->extractedComments = Array ();
$this->references = Array ();
$this->flags = Array ();
//getting the new line
while ($flagReadingComments && ! $this->flagError) {
$this->_fileLine = trim( fgets( $this->_fp ) );
//set line number
$this->lineNumber ++;
if (! $this->_fileLine) {
return false;
}
$prefix = substr( $this->_fileLine, 0, 2 );
switch ($prefix) {
case '# ':
$lineItem = str_replace( '# ', '', $this->_fileLine );
$this->translatorComments[] = $lineItem;
break;
case '#.':
if (substr_count( $this->_fileLine, '#. ' ) == 0) {
$this->flagError = true;
} else {
$lineItem = str_replace( '#. ', '', $this->_fileLine );
$this->extractedComments[] = $lineItem;
}
break;
case '#:':
if (substr_count( $this->_fileLine, '#: ' ) == 0) {
$this->flagError = true;
} else {
$lineItem = str_replace( '#: ', '', $this->_fileLine );
$this->references[] = $lineItem;
}
break;
case '#,':
if (substr_count( $this->_fileLine, '#, ' ) == 0) {
$this->flagError = true;
} else {
$lineItem = str_replace( '#, ', '', $this->_fileLine );
$this->flags[] = $lineItem;
}
break;
case '#|':
if (substr_count( $this->_fileLine, '#| ' ) == 0) {
$this->flagError = true;
} else {
$lineItem = str_replace( '#| ', '', $this->_fileLine );
$this->previousUntranslatedStrings[] = $lineItem;
}
break;
default:
$flagReadingComments = false;
}
}
if (! $this->_fileLine) {
return false;
}
//Getting the msgid
preg_match( '/\s*msgid\s*"(.*)"\s*/s', $this->_fileLine, $match );
if (sizeof( $match ) != 2) {
throw new Exception( 'Invalid PO file format1' );
}
$msgid = '';
do {
//g::pr($match);
$msgid .= $match[1];
$this->_fileLine = trim( fgets( $this->_fp ) );
preg_match( '/^"(.*)"\s*/s', $this->_fileLine, $match );
} while (sizeof( $match ) == 2);
//Getting the msgstr
preg_match( '/\s*msgstr\s*"(.*)"\s*/s', $this->_fileLine, $match );
if (sizeof( $match ) != 2) {
throw new Exception( 'Invalid PO file format2' );
}
$msgstr = '';
do {
//g::pr($match);
$msgstr .= $match[1] . "\n";
$this->_fileLine = trim( fgets( $this->_fp ) );
preg_match( '/^"(.*)"\s*/s', $this->_fileLine, $match );
} while (sizeof( $match ) == 2);
/*g::pr($this->translatorComments);
g::pr($this->references);
g::pr($match);
die;*/
return Array ('msgid' => trim( $msgid ),'msgstr' => trim( $msgstr )
);
}
//garbage
function __destruct ()
{
if ($this->_fp) {
fclose( $this->_fp );
}
2010-12-02 23:34:41 +00:00
}
}