423 lines
14 KiB
PHP
423 lines
14 KiB
PHP
|
<?php
|
||
|
// {{{ICINGA_LICENSE_HEADER}}}
|
||
|
/**
|
||
|
* This file is part of Icinga Web 2.
|
||
|
*
|
||
|
* Icinga Web 2 - Head for multiple monitoring backends.
|
||
|
* Copyright (C) 2014 Icinga Development Team
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License
|
||
|
* as published by the Free Software Foundation; either version 2
|
||
|
* 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
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
*
|
||
|
* @copyright 2014 Icinga Development Team <info@icinga.org>
|
||
|
* @license http://www.gnu.org/licenses/gpl-2.0.txt GPL, version 2
|
||
|
* @author Icinga Development Team <info@icinga.org>
|
||
|
*
|
||
|
*/
|
||
|
// {{{ICINGA_LICENSE_HEADER}}}
|
||
|
|
||
|
namespace Icinga\Module\Translation\Util;
|
||
|
|
||
|
use \Exception;
|
||
|
use Icinga\Application\Modules\Manager;
|
||
|
use Icinga\Application\ApplicationBootstrap;
|
||
|
|
||
|
/**
|
||
|
* This class provides some useful utility functions to handle gettext translations
|
||
|
*/
|
||
|
class GettextTranslationHelper
|
||
|
{
|
||
|
/**
|
||
|
* All project files are supposed to have the same/this encoding
|
||
|
*/
|
||
|
const FILE_ENCODING = 'UTF-8';
|
||
|
|
||
|
/**
|
||
|
* The source files to parse
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private $sourceExtensions = array(
|
||
|
'php',
|
||
|
'phtml'
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* The module manager of the application's bootstrap
|
||
|
*
|
||
|
* @var Manager
|
||
|
*/
|
||
|
private $moduleMgr;
|
||
|
|
||
|
/**
|
||
|
* The current version of IcingaWeb2
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $version;
|
||
|
|
||
|
/**
|
||
|
* The locale used by this helper
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $locale;
|
||
|
|
||
|
/**
|
||
|
* The path to the Zend application root
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $appDir;
|
||
|
|
||
|
/**
|
||
|
* The path to the module, if any
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $moduleDir;
|
||
|
|
||
|
/**
|
||
|
* The path to the file catalog
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $catalogPath;
|
||
|
|
||
|
/**
|
||
|
* The path to the *.pot file
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $templatePath;
|
||
|
|
||
|
/**
|
||
|
* The path to the *.po file
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $tablePath;
|
||
|
|
||
|
/**
|
||
|
* Create a new TranslationHelper object
|
||
|
*
|
||
|
* @param ApplicationBootstrap $bootstrap The application's bootstrap object
|
||
|
* @param string $locale The locale to be used by this helper
|
||
|
*/
|
||
|
public function __construct(ApplicationBootstrap $bootstrap, $locale)
|
||
|
{
|
||
|
$this->version = $bootstrap->getConfig()->app()->global->get('version', '0.1');
|
||
|
$this->moduleMgr = $bootstrap->getModuleManager();
|
||
|
$this->appDir = $bootstrap->getApplicationDir();
|
||
|
$this->locale = $locale;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the translation table for the main application
|
||
|
*/
|
||
|
public function updateIcingaTranslations()
|
||
|
{
|
||
|
$this->catalogPath = tempnam(sys_get_temp_dir(), 'IcingaTranslation_');
|
||
|
$this->templatePath = tempnam(sys_get_temp_dir(), 'IcingaPot_');
|
||
|
|
||
|
$this->moduleDir = null;
|
||
|
$this->tablePath = implode(
|
||
|
DIRECTORY_SEPARATOR,
|
||
|
array(
|
||
|
$this->appDir,
|
||
|
'locale',
|
||
|
$this->locale,
|
||
|
'LC_MESSAGES',
|
||
|
'icinga.po'
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->createFileCatalog();
|
||
|
$this->createTemplateFile();
|
||
|
$this->updateTranslationTable();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the translation table for a particular module
|
||
|
*
|
||
|
* @param string $module The name of the module for which to update the translation table
|
||
|
*/
|
||
|
public function updateModuleTranslations($module)
|
||
|
{
|
||
|
$this->catalogPath = tempnam(sys_get_temp_dir(), 'IcingaTranslation_');
|
||
|
$this->templatePath = tempnam(sys_get_temp_dir(), 'IcingaPot_');
|
||
|
|
||
|
$this->moduleDir = $this->moduleMgr->getModuleDir($module);
|
||
|
$this->tablePath = implode(
|
||
|
DIRECTORY_SEPARATOR,
|
||
|
array(
|
||
|
$this->moduleDir,
|
||
|
'application',
|
||
|
'locale',
|
||
|
$this->locale,
|
||
|
'LC_MESSAGES',
|
||
|
$module . '.po'
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->createFileCatalog();
|
||
|
$this->createTemplateFile();
|
||
|
$this->updateTranslationTable();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile the translation table for the main application
|
||
|
*/
|
||
|
public function compileIcingaTranslation()
|
||
|
{
|
||
|
$this->tablePath = implode(
|
||
|
DIRECTORY_SEPARATOR,
|
||
|
array(
|
||
|
$this->appDir,
|
||
|
'locale',
|
||
|
$this->locale,
|
||
|
'LC_MESSAGES',
|
||
|
'icinga.po'
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->compileTranslationTable();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile the translation table for a particular module
|
||
|
*
|
||
|
* @param string $module The name of the module for which to compile the translation table
|
||
|
*/
|
||
|
public function compileModuleTranslation($module)
|
||
|
{
|
||
|
$this->moduleDir = $this->moduleMgr->getModuleDir($module);
|
||
|
$this->tablePath = implode(
|
||
|
DIRECTORY_SEPARATOR,
|
||
|
array(
|
||
|
$this->moduleDir,
|
||
|
'application',
|
||
|
'locale',
|
||
|
$this->locale,
|
||
|
'LC_MESSAGES',
|
||
|
$module . '.po'
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->compileTranslationTable();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update any existing or create a new translation table using the gettext tools
|
||
|
*
|
||
|
* @throws Exception In case the translation table does not yet exist and cannot be created
|
||
|
*/
|
||
|
private function updateTranslationTable()
|
||
|
{
|
||
|
if (is_file($this->tablePath)) {
|
||
|
shell_exec(sprintf('/usr/bin/msgmerge --update %s %s 2>&1', $this->tablePath, $this->templatePath));
|
||
|
} else {
|
||
|
if ((!is_dir(dirname($this->tablePath)) && !@mkdir(dirname($this->tablePath), 0755, true)) ||
|
||
|
!rename($this->templatePath, $this->tablePath)) {
|
||
|
throw new Exception('Unable to create ' . $this->tablePath);
|
||
|
}
|
||
|
}
|
||
|
$this->updateHeader($this->tablePath);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create the template file using the gettext tools
|
||
|
*/
|
||
|
private function createTemplateFile()
|
||
|
{
|
||
|
shell_exec(
|
||
|
implode(
|
||
|
' ',
|
||
|
array(
|
||
|
'/usr/bin/xgettext',
|
||
|
'--language=PHP',
|
||
|
'--keyword=translate',
|
||
|
'--keyword=mt:2',
|
||
|
'--keyword=t',
|
||
|
'--sort-output',
|
||
|
'--force-po',
|
||
|
'--omit-header',
|
||
|
'--from-code=' . self::FILE_ENCODING,
|
||
|
'--files-from="' . $this->catalogPath . '"',
|
||
|
'--output="' . $this->templatePath . '"'
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create or update a gettext conformant header in the given file
|
||
|
*
|
||
|
* @param string $path The path to the file
|
||
|
*/
|
||
|
private function updateHeader($path)
|
||
|
{
|
||
|
$headerInfo = array(
|
||
|
'title' => 'Icinga Web 2 - Head for multiple monitoring backends',
|
||
|
'copyright_holder' => 'Icinga Development Team',
|
||
|
'copyright_year' => date('Y'),
|
||
|
'author_name' => 'FIRST AUTHOR',
|
||
|
'author_mail' => 'EMAIL@ADDRESS',
|
||
|
'author_year' => 'YEAR',
|
||
|
'project_name' => 'Icinga Web 2',
|
||
|
'project_version' => $this->version,
|
||
|
'project_bug_mail' => 'dev@icinga.org',
|
||
|
'pot_creation_date' => date('Y-m-d H:iO'),
|
||
|
'po_revision_date' => 'YEAR-MO-DA HO:MI+ZONE',
|
||
|
'translator_name' => 'FULL NAME',
|
||
|
'translator_mail' => 'EMAIL@ADDRESS',
|
||
|
'language_team_name' => 'LANGUAGE',
|
||
|
'language_team_url' => 'LL@li.org',
|
||
|
'charset' => self::FILE_ENCODING
|
||
|
);
|
||
|
|
||
|
$content = file_get_contents($path);
|
||
|
if (strpos($content, '# ') === 0) {
|
||
|
$authorInfo = array();
|
||
|
if (preg_match('@# (.+) <(.+)>, (\d+|YEAR)\.@', $content, $authorInfo)) {
|
||
|
$headerInfo['author_name'] = $authorInfo[1];
|
||
|
$headerInfo['author_mail'] = $authorInfo[2];
|
||
|
$headerInfo['author_year'] = $authorInfo[3];
|
||
|
}
|
||
|
$revisionInfo = array();
|
||
|
if (preg_match('@Revision-Date: (\d{4}-\d{2}-\d{2} \d{2}:\d{2}\+\d{4})@', $content, $revisionInfo)) {
|
||
|
$headerInfo['po_revision_date'] = $revisionInfo[1];
|
||
|
}
|
||
|
$translatorInfo = array();
|
||
|
if (preg_match('@Last-Translator: (.+) <(.+)>@', $content, $translatorInfo)) {
|
||
|
$headerInfo['translator_name'] = $translatorInfo[1];
|
||
|
$headerInfo['translator_mail'] = $translatorInfo[2];
|
||
|
}
|
||
|
$languageInfo = array();
|
||
|
if (preg_match('@Language-Team: (.+) <(.+)>@', $content, $languageInfo)) {
|
||
|
$headerInfo['language_team_name'] = $languageInfo[1];
|
||
|
$headerInfo['language_team_url'] = $languageInfo[2];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
file_put_contents(
|
||
|
$path,
|
||
|
implode(
|
||
|
PHP_EOL,
|
||
|
array(
|
||
|
'# ' . $headerInfo['title'] . '.',
|
||
|
'# Copyright (C) ' . $headerInfo['copyright_year'] . ' ' . $headerInfo['copyright_holder'],
|
||
|
'# This file is distributed under the same license as ' . $headerInfo['project_name'] . '.',
|
||
|
'# ' . $headerInfo['author_name'] . ' <' . $headerInfo['author_mail']
|
||
|
. '>, ' . $headerInfo['author_year'] . '.',
|
||
|
'# ',
|
||
|
'#, fuzzy',
|
||
|
'msgid ""',
|
||
|
'msgstr ""',
|
||
|
'"Project-Id-Version: ' . $headerInfo['project_name'] . ' ('
|
||
|
. $headerInfo['project_version'] . ')\n"',
|
||
|
'"Report-Msgid-Bugs-To: ' . $headerInfo['project_bug_mail'] . '\n"',
|
||
|
'"POT-Creation-Date: ' . $headerInfo['pot_creation_date'] . '\n"',
|
||
|
'"PO-Revision-Date: ' . $headerInfo['po_revision_date'] . '\n"',
|
||
|
'"Last-Translator: ' . $headerInfo['translator_name'] . ' <'
|
||
|
. $headerInfo['translator_mail'] . '>\n"',
|
||
|
'"Language-Team: ' . $headerInfo['language_team_name'] . ' <'
|
||
|
. $headerInfo['language_team_url'] . '>\n"',
|
||
|
'"MIME-Version: 1.0\n"',
|
||
|
'"Content-Type: text/plain; charset=' . $headerInfo['charset'] . '\n"',
|
||
|
'"Content-Transfer-Encoding: 8bit\n"',
|
||
|
''
|
||
|
)
|
||
|
) . PHP_EOL . substr($content, strpos($content, '#: '))
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create the file catalog
|
||
|
*
|
||
|
* @throws Exception In case the catalog-file cannot be created
|
||
|
*/
|
||
|
private function createFileCatalog()
|
||
|
{
|
||
|
$catalogHandle = fopen($this->catalogPath, 'w');
|
||
|
if (!$catalogHandle) {
|
||
|
throw new Exception('Unable to create ' . $this->catalogPath);
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
if ($this->moduleDir) {
|
||
|
$this->getSourceFileNames($this->moduleDir, $catalogHandle);
|
||
|
} else {
|
||
|
$this->getSourceFileNames($this->appDir, $catalogHandle);
|
||
|
$this->getSourceFileNames(realpath($this->appDir . '/../library/Icinga'), $catalogHandle);
|
||
|
}
|
||
|
} catch (Exception $error) {
|
||
|
fclose($catalogHandle);
|
||
|
throw $error;
|
||
|
}
|
||
|
|
||
|
fclose($catalogHandle);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Recursively scan the given directory for translatable source files
|
||
|
*
|
||
|
* @param string $directory The directory where to search for sources
|
||
|
* @param resource $fileHandle The file where to write the results
|
||
|
* @param array $blacklist A list of directories to omit
|
||
|
*
|
||
|
* @throws Exception In case the given directory is not readable
|
||
|
*/
|
||
|
private function getSourceFileNames($directory, &$fileHandle)
|
||
|
{
|
||
|
$directoryHandle = opendir($directory);
|
||
|
if (!$directoryHandle) {
|
||
|
throw new Exception('Unable to read files from ' . $directory);
|
||
|
}
|
||
|
|
||
|
$subdirs = array();
|
||
|
while (($filename = readdir($directoryHandle)) !== false) {
|
||
|
$filepath = $directory . DIRECTORY_SEPARATOR . $filename;
|
||
|
if (preg_match('@^[^\.].+\.(' . implode('|', $this->sourceExtensions) . ')$@', $filename)) {
|
||
|
fwrite($fileHandle, $filepath . PHP_EOL);
|
||
|
} elseif (is_dir($filepath) && !preg_match('@^(\.|\.\.)$@', $filename)) {
|
||
|
$subdirs[] = $filepath;
|
||
|
}
|
||
|
}
|
||
|
closedir($directoryHandle);
|
||
|
|
||
|
foreach ($subdirs as $subdir) {
|
||
|
$this->getSourceFileNames($subdir, $fileHandle);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile the translation table
|
||
|
*/
|
||
|
private function compileTranslationTable()
|
||
|
{
|
||
|
$targetPath = substr($this->tablePath, 0, strrpos($this->tablePath, '.')) . '.mo';
|
||
|
shell_exec(
|
||
|
implode(
|
||
|
' ',
|
||
|
array(
|
||
|
'/usr/bin/msgfmt',
|
||
|
'-o ' . $targetPath,
|
||
|
$this->tablePath
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
}
|