tao-test/app/tao/scripts/class.TaoTranslate.php

1519 lines
65 KiB
PHP
Raw Normal View History

2022-08-29 20:14:13 +02:00
<?php
use oat\generis\model\OntologyRdfs;
use oat\tao\helpers\ApplicationHelper;
use oat\tao\model\menu\MenuService;
/**
* 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; under version 2
* of the License (non-upgradable).
*
* 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 (c) 2002-2008 (original work) Public Research Centre Henri Tudor & University of Luxembourg (under the project TAO & TAO2);
* 2008-2010 (update and modification) Deutsche Institut für Internationale Pädagogische Forschung (under the project TAO-TRANSFER);
* 2009-2012 (update and modification) Public Research Centre Henri Tudor (under the project TAO-SUSTAIN & TAO-DEV);
*
*/
/**
* The TaoTranslate script aims at providing command line tools to manage
* of tao. It enables you to manage the i18n of the messages found in the source
* (gettext) but also i18n of RDF Models.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @package tao
*/
class tao_scripts_TaoTranslate extends tao_scripts_Runner
{
// --- ASSOCIATIONS ---
// --- ATTRIBUTES ---
/**
* Short description of attribute DEF_INPUT_DIR
*
* @access public
* @var string
*/
const DEF_INPUT_DIR = '.';
/**
* Short description of attribute DEF_OUTPUT_DIR
*
* @access public
* @var string
*/
const DEF_OUTPUT_DIR = 'locales';
/**
* Short description of attribute DEF_PO_FILENAME
*
* @access public
* @var string
*/
const DEF_PO_FILENAME = 'messages.po';
/**
* Short description of attribute DEF_JS_FILENAME
*
* @access public
* @var string
*/
const DEF_JS_FILENAME = 'messages_po.js';
/**
* Short description of attribute options
*
* @access protected
* @var array
*/
protected $options = [];
/**
* Short description of attribute DEF_LANG_FILENAME
*
* @access public
* @var string
*/
const DEF_LANG_FILENAME = 'lang.rdf';
private static $WHITE_LIST = [
'actions',
'helpers',
'models',
'views',
'helper',
'controller',
'model',
'scripts',
];
protected $verbose = false;
// --- OPERATIONS ---
/**
* keys - action names from user input
* value - base name for method to call
* @return array
*/
protected function getAllowedActions()
{
return [
'create' => 'Create',
'update' => 'Update',
'delete' => 'Delete',
'updateall' => 'UpdateAll',
'deleteall' => 'DeleteAll',
'enable' => 'Enable',
'disable' => 'Disable',
'compile' => 'Compile',
'compileall' => 'CompileAll',
'changecode' => 'ChangeCode',
'getallextensions' => 'GetExt',
];
}
/**
* Things that must happen before script execution.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function preRun()
{
$this->options = ['verbose' => false,
'action' => null,
'extension' => null];
$this->options = array_merge($this->options, $this->parameters);
if ($this->options['verbose'] == true) {
$this->verbose = true;
}
// The 'action' parameter is always required.
if ($this->options['action'] == null) {
$this->err("Please enter the 'action' parameter.", true);
} else {
$this->options['action'] = strtolower($this->options['action']);
if (!in_array($this->options['action'], array_keys($this->getAllowedActions()))) {
$this->err("'" . $this->options['action'] . "' is not a valid action.", true);
} else {
// The 'action' parameter is ok.
// Let's check additional inputs depending on the value of the 'action' parameter.
$this->checkInput();
}
}
}
/**
* Main script implementation.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function run()
{
// Select the action to perform depending on the 'action' parameter.
// Verification of the value of 'action' performed in self::preRun().
$actions = $this->getAllowedActions();
$pendingActionName = 'action' . $actions[$this->options['action']];
if (method_exists($this, $pendingActionName)) {
return $this->$pendingActionName();
}
}
/**
* Things that must happen after the run() method.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function postRun()
{
}
/**
* Checks the inputs for the current script call.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkInput()
{
$actions = $this->getAllowedActions();
$pendingActionName = 'check' . $actions[$this->options['action']] . 'Input';
if (method_exists($this, $pendingActionName)) {
return $this->$pendingActionName();
}
}
/**
* Checks the inputs for the 'create' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkCreateInput()
{
$defaults = ['language' => null,
'languageLabel' => null,
'extension' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR,
'build' => true, // Build translation files by having a look in source code, models.
'force' => false]; // Do not force rebuild if locale already exist.
$this->options = array_merge($defaults, $this->options);
if (is_null($this->options['language'])) {
$this->err("Please provide a 'language' identifier such as en-US, fr-CA, IT, ...", true);
} else {
if (is_null($this->options['extension'])) {
$this->err("Please provide an 'extension' for which the 'language' will be created", true);
} else {
// Check if the extension(s) exists.
$extensionsToCreate = explode(',', $this->options['extension']);
$extensionsToCreate = array_unique($extensionsToCreate);
foreach ($extensionsToCreate as $etc) {
$this->options['input'] = dirname(__FILE__) . '/../../' . $etc . '/' . self::DEF_INPUT_DIR;
$this->options['output'] = dirname(__FILE__) . '/../../' . $etc . '/' . self::DEF_OUTPUT_DIR;
$extensionDir = dirname(__FILE__) . '/../../' . $etc;
if (!is_dir($extensionDir)) {
$this->err("The extension '" . $etc . "' does not exist.", true);
} elseif (!is_readable($extensionDir)) {
$this->err("The '" . $etc . "' directory is not readable. Please check permissions on this directory.", true);
} elseif (!is_writable($extensionDir)) {
$this->err("The '" . $etc . "' directory is not writable. Please check permissions on this directory.", true);
}
// The input 'parameter' is optional.
// (and only used if the 'build' parameter is set to true)
$this->checkInputOption();
// The 'output' parameter is optional.
$this->checkOutputOption();
}
}
}
}
/**
* Checks the inputs for the 'update' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkUpdateInput()
{
$defaults = ['language' => null,
'extension' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR];
$this->options = array_merge($defaults, $this->options);
if (is_null($this->options['language'])) {
$this->err("Please provide a 'language' identifier such as en-US, fr-CA, IT, ...", true);
} else {
// Check if the language folder exists and is readable/writable.
$languageDir = $this->buildLanguagePath($this->options['extension'], $this->options['language']);
if (!is_dir($languageDir)) {
$this->err("The 'language' directory ${languageDir} does not exist.", true);
} elseif (!is_readable($languageDir)) {
$this->err("The 'language' directory ${languageDir} is not readable. Please check permissions on this directory.");
} elseif (!is_writable($languageDir)) {
$this->err("The 'language' directory ${languageDir} is not writable. Please check permissions on this directory.");
} else {
if (is_null($this->options['extension'])) {
$this->err("Please provide an 'extension' for which the 'language' will be created", true);
} else {
// Check if the extension exists.
$extensionDir = dirname(__FILE__) . '/../../' . $this->options['extension'];
if (!is_dir($extensionDir)) {
$this->err("The extension '" . $this->options['extension'] . "' does not exist.", true);
} elseif (!is_readable($extensionDir)) {
$this->err("The '" . $this->options['extension'] . "' directory is not readable. Please check permissions on this directory.", true);
} elseif (!is_writable($extensionDir)) {
$this->err("The '" . $this->options['extension'] . "' directory is not writable. Please check permissions on this directory.", true);
} else {
// And can we read the messages.po file ?
if (!file_exists($languageDir . '/' . self::DEF_PO_FILENAME)) {
$this->err("Cannot find " . self::DEF_PO_FILENAME . " for extension '" . $this->options['extension'] . "' and language '" . $this->options['language'] . "'.", true);
} elseif (!is_readable($languageDir . '/' . self::DEF_PO_FILENAME)) {
$this->err(self::DEF_PO_FILENAME . " is not readable for '" . $this->options['extension'] . "' and language '" . $this->options['language'] . "'. Please check permissions for this file.", true);
} else {
// The input 'parameter' is optional.
$this->checkInputOption();
// The 'output' parameter is optional.
$this->checkOutputOption();
}
}
}
}
}
}
/**
* checks the input for the 'updateAll' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkUpdateAllInput()
{
$defaults = ['input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR,
'extension' => null];
$this->options = array_merge($defaults, $this->options);
// The input 'parameter' is optional.
$this->checkInputOption();
// The 'output' parameter is optional.
$this->checkOutputOption();
}
/**
* Checks the inputs for the 'delete' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void/
*/
private function checkDeleteInput()
{
$defaults = ['language' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR,
'extension' => null];
$this->options = array_merge($defaults, $this->options);
if (is_null($this->options['extension'])) {
$this->err("Please provide an 'extension' identifier.", true);
} else {
if (is_null($this->options['language'])) {
$this->err("Please provide a 'language' identifier such as en-US, fr-CA, IT, ...", true);
} else {
$this->checkInputOption();
}
}
}
/**
* Checks inputs for the 'deleteAll' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkDeleteAllInput()
{
$defaults = ['input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR,
'extension' => null];
$this->options = array_merge($defaults, $this->options);
// The input 'parameter' is optional.
if (!is_null($this->options['extension'])) {
$this->checkInputOption();
} else {
$this->err("Please provide an 'extension' identifier.", true);
}
}
private function checkChangeCodeInput()
{
$defaults = ['input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR,
'extension' => null,
'language' => null,
'targetLanguage' => null];
$this->options = array_merge($defaults, $this->options);
if (empty($this->options['language'])) {
$this->err("Please provide a source 'language' identifier such as en-US, fr-CA, IT, ...", true);
} elseif (empty($this->options['targetLanguage'])) {
$this->err("Please provide a 'targetLanguage' identifier such as en-US, fr-CA, IT, ...", true);
} elseif (empty($this->options['extension'])) {
$this->err("Please provide an 'extension' identifier.", true);
} elseif (!is_readable($this->options['output'] . DIRECTORY_SEPARATOR . $this->options['language'])) {
$this->err("The '" . $this->options['language'] . "' locale directory is not readable.", true);
} elseif (!is_writable($this->options['output'])) {
$this->err("The locales directory of extension '" . $this->options['extension'] . "' is not writable.");
}
}
/**
* Implementation of the 'create' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionCreate()
{
$extensionsToCreate = explode(',', $this->options['extension']);
$extensionsToCreate = array_unique($extensionsToCreate);
foreach ($extensionsToCreate as $etc) {
$this->options['extension'] = $etc;
$this->options['input'] = dirname(__FILE__) . '/../../' . $etc . '/' . self::DEF_INPUT_DIR;
$this->options['output'] = dirname(__FILE__) . '/../../' . $etc . '/' . self::DEF_OUTPUT_DIR;
$this->outVerbose("Creating language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "' ...");
// We first create the directory where locale files will go.
$dir = $this->buildLanguagePath($this->options['extension'], $this->options['language']);
$dirExists = false;
if (file_exists($dir) && is_dir($dir) && $this->options['force'] == true) {
$dirExists = true;
$this->outVerbose("Language '" . $this->options['language'] . "' exists for extension '" . $this->options['extension'] . "'. Creation will be forced.");
// Clean it up.
foreach (scandir($dir) as $d) {
if ($d !== '.' && $d !== '..' && $d !== '.svn') {
if (!tao_helpers_File::remove($dir . '/' . $d, true)) {
$this->err("Unable to clean up 'language' directory '" . $dir . "'.", true);
}
}
}
} elseif (file_exists($dir) && is_dir($dir) && $this->options['force'] == false) {
$this->err("The 'language' " . $this->options['language'] . " already exists in the file system. Use the 'force' parameter to overwrite it.", true);
}
// If we are still here... it means that we have to create the language directory.
if (!$dirExists && !@mkdir($dir)) {
$this->err("Unable to create 'language' directory '" . $this->options['language'] . "'.", true);
} else {
if ($this->options['build'] == true) {
$sortingMethod = tao_helpers_translation_TranslationFile::SORT_ASC_I;
$this->outVerbose("Building language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "' ...");
// Let's populate the language with raw PO files containing sources but no targets.
// Source code extraction.
$fileExtensions = ['php', 'tpl', 'js', 'ejs'];
$filePaths = [];
foreach (self::$WHITE_LIST as $subFolder) {
$filePaths[] = $this->options['input'] . DIRECTORY_SEPARATOR . $subFolder;
}
$sourceExtractor = new tao_helpers_translation_SourceCodeExtractor($filePaths, $fileExtensions);
$sourceExtractor->extract();
$translationFile = new tao_helpers_translation_POFile();
$translationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$translationFile->setTargetLanguage($this->options['language']);
$translationFile->addTranslationUnits($sourceExtractor->getTranslationUnits());
$file = MenuService::getStructuresFilePath($this->options['extension']);
if (!is_null($file)) {
$structureExtractor = new tao_helpers_translation_StructureExtractor([$file]);
$structureExtractor->extract();
$translationFile->addTranslationUnits($structureExtractor->getTranslationUnits());
}
$sortedTus = $translationFile->sortBySource($sortingMethod);
$sortedTranslationFile = new tao_helpers_translation_POFile();
$sortedTranslationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$sortedTranslationFile->setTargetLanguage($this->options['language']);
$sortedTranslationFile->addTranslationUnits($sortedTus);
$this->preparePOFile($sortedTranslationFile, true);
$poPath = $dir . '/' . self::DEF_PO_FILENAME;
$writer = new tao_helpers_translation_POFileWriter(
$poPath,
$sortedTranslationFile
);
$writer->write();
$this->outVerbose("PO Translation file '" . basename($poPath) . "' in '" . $this->options['language'] . "' created for extension '" . $this->options['extension'] . "'.");
$writer->write();
// Writing JS files
$jsPath = $dir . '/' . self::DEF_JS_FILENAME;
$writer = new tao_helpers_translation_JSFileWriter(
$jsPath,
$sortedTranslationFile
);
$writer->write(false);
$this->outVerbose("JavaScript Translation file '" . basename($jsPath) . "' in '" . $this->options['language'] . "' created for extension '" . $this->options['extension'] . "'.");
$writer->write();
// Now that PO files & JS files are created, we can create the translation models
// if we find RDF models to load for this extension.
$translatableProperties = [OntologyRdfs::RDFS_LABEL, OntologyRdfs::RDFS_COMMENT];
foreach ($this->getOntologyFiles() as $f) {
common_Logger::d('reading rdf ' . $f);
$translationFile = $this->extractPoFileFromRDF($f, $translatableProperties);
$writer = new tao_helpers_translation_POFileWriter($dir . '/' . $this->getOntologyPOFileName($f), $translationFile);
$writer->write();
$this->outVerbose("PO Translation file '" . $this->getOntologyPOFileName($f) . "' in '" . $this->options['language'] . "' created for extension '" . $this->options['extension'] . "'.");
}
$this->outVerbose("Language '" . $this->options['language'] . "' created for extension '" . $this->options['extension'] . "'.");
} else {
// Only build virgin files.
// (Like a virgin... woot !)
$translationFile = new tao_helpers_translation_POFile();
$translationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$translationFile->setTargetLanguage($this->options['language']);
$this->preparePOFile($translationFile, true);
foreach ($this->getOntologyFiles() as $f) {
common_Logger::d('reading rdf ' . $f);
$translationFile = new tao_helpers_translation_POFile();
$translationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$translationFile->setTargetLanguage($this->options['language']);
$translationFile->setExtensionId($this->options['extension']);
$writer = new tao_helpers_translation_POFileWriter($dir . '/' . $this->getOntologyPOFileName($f), $translationFile);
$writer->write();
}
$this->outVerbose("Language '" . $this->options['language'] . "' created for extension '" . $this->options['extension'] . "'.");
}
// Create the language manifest in RDF.
if ($this->options['extension'] == 'tao') {
$langDescription = tao_helpers_translation_RDFUtils::createLanguageDescription(
$this->options['language'],
$this->options['languageLabel']
);
$langDescription->save($dir . '/' . self::DEF_LANG_FILENAME);
}
}
}
}
/**
* Implementation of the 'update' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionUpdate()
{
$this->outVerbose("Updating language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "'...");
$sortingMethod = tao_helpers_translation_TranslationFile::SORT_ASC_I;
// Get virgin translations from the source code and manifest.
$filePaths = [];
foreach (self::$WHITE_LIST as $subFolder) {
$filePaths[] = $this->options['input'] . DIRECTORY_SEPARATOR . $subFolder;
}
$extensions = ['php', 'tpl', 'js', 'ejs'];
$sourceCodeExtractor = new tao_helpers_translation_SourceCodeExtractor($filePaths, $extensions);
$sourceCodeExtractor->extract();
$translationFile = new tao_helpers_translation_POFile();
$translationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$translationFile->setTargetLanguage($this->options['language']);
$translationFile->addTranslationUnits($sourceCodeExtractor->getTranslationUnits());
$file = MenuService::getStructuresFilePath($this->options['extension']);
if (!is_null($file)) {
$structureExtractor = new tao_helpers_translation_StructureExtractor([$file]);
$structureExtractor->extract();
$structureUnits = $structureExtractor->getTranslationUnits();
$this->outVerbose(count($structureUnits) . ' units extracted from structures.xml.');
$translationFile->addTranslationUnits($structureUnits);
}
// For each TU that was recovered, have a look in an older version
// of the translations.
$oldFilePath = $this->buildLanguagePath($this->options['extension'], $this->options['language']) . '/' . self::DEF_PO_FILENAME;
$translationFileReader = new tao_helpers_translation_POFileReader($oldFilePath);
$translationFileReader->read();
$oldTranslationFile = $translationFileReader->getTranslationFile();
foreach ($oldTranslationFile->getTranslationUnits() as $oldTu) {
if (($newTu = $translationFile->getBySource($oldTu)) !== null && $oldTu->getTarget() != '') {
// No duplicates in TFs so I simply add it whatever happens.
// If it already has the same one, it means we will update it.
$newTu->setTarget($oldTu->getTarget());
}
}
$sortedTranslationFile = new tao_helpers_translation_POFile();
$sortedTranslationFile->setSourceLanguage($translationFile->getSourceLanguage());
$sortedTranslationFile->setTargetLanguage($translationFile->getTargetLanguage());
$sortedTranslationFile->addTranslationUnits($translationFile->sortBySource($sortingMethod));
$this->preparePOFile($sortedTranslationFile, true);
// Write the new ones.
$poFileWriter = new tao_helpers_translation_POFileWriter($oldFilePath, $sortedTranslationFile);
$poFileWriter->write();
$this->outVerbose("PO translation file '" . basename($oldFilePath) . "' in '" . $this->options['language'] . "' updated for extension '" . $this->options['extension'] . "'.");
$translatableProperties = [OntologyRdfs::RDFS_LABEL, OntologyRdfs::RDFS_COMMENT];
// We now deal with RDF models.
foreach ($this->getOntologyFiles() as $f) {
// Loop on 'master' models.
$translationFile = $this->extractPoFileFromRDF($f, $translatableProperties);
// The slave RDF file is the translation of the ontology that we find in /extId/Locales/langCode.
$slavePOFilePath = $this->buildLanguagePath($this->options['extension'], $this->options['language']) . '/' . $this->getOntologyPOFileName($f);
if (file_exists($slavePOFilePath)) {
// Read the existing RDF Translation file for this RDF model.
$poReader = new tao_helpers_translation_POFileReader($slavePOFilePath);
$poReader->read();
$slavePOFile = $poReader->getTranslationFile();
// Try to update translation units found in the master PO file with
// targets found in the old translation of the ontology.
foreach ($slavePOFile->getTranslationUnits() as $oTu) {
$translationFile->addTranslationUnit($oTu);
}
// Remove Slave PO file. It will be overwritten by the modified Master PO file.
tao_helpers_File::remove($slavePOFilePath);
}
// Write Master PO file as the new Slave PO file.
$rdfWriter = new tao_helpers_translation_POFileWriter($slavePOFilePath, $translationFile);
$rdfWriter->write();
$this->outVerbose("Translation model {$this->getOntologyPOFileName($f)} in '" . $this->options['language'] . "' updated for extension '" . $this->options['extension'] . "'.");
}
$this->outVerbose("Language '" . $this->options['language'] . "' updated for extension '" . $this->options['extension'] . "'.");
}
/**
* Implementation of the 'updateAll' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionUpdateAll()
{
// Scan the locales folder for languages in the wwextension and
// launch actionUpdate for each of them.
// Get the list of languages that will be updated.
$locales = $this->getLanguageList();
// We now identified locales to be updated.
$this->outVerbose("Languages '" . implode(',', $locales) . "' will be updated for extension '" . $this->options['extension'] . "'.");
foreach ($locales as $l) {
$this->options['language'] = $l;
$this->checkUpdateInput();
$this->actionUpdate();
$this->outVerbose("");
}
}
/**
* Implementation of the 'delete' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionDelete()
{
$this->outVerbose("Deleting language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "' ...");
$dir = $this->buildLanguagePath($this->options['extension'], $this->options['language']);
if (!tao_helpers_File::remove($dir, true)) {
$this->err("Could not delete language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "'.", true);
}
$this->outVerbose("Language '" . $this->options['language'] . "' for extension '" . $this->options['extension'] . "' successfully deleted.");
}
/**
* Implementation of the 'deleteAll' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionDeleteAll()
{
// Get the list of languages that will be deleted.
$this->outVerbose("Deleting all languages for extension '" . $this->options['extension'] . "'...");
$locales = $this->getLanguageList();
foreach ($locales as $l) {
$this->options['language'] = $l;
$this->checkDeleteInput();
$this->actionDelete();
$this->outVerbose("");
}
}
public function actionChangeCode()
{
$this->outVerbose("Changing code of locale '" . $this->options['language'] . "' to '" . $this->options['targetLanguage'] . "' for extension '" . $this->options['extension'] . "'...");
// First we copy the old locale to a new directory named as 'targetLanguage'.
$sourceLocaleDir = $this->options['output'] . DIRECTORY_SEPARATOR . $this->options['language'];
$destLocaleDir = $this->options['output'] . DIRECTORY_SEPARATOR . $this->options['targetLanguage'];
if (!tao_helpers_File::copy($sourceLocaleDir, $destLocaleDir, true, true)) {
$this->err("Locale '" . $this->options['language'] . "' could not be copied to locale '" . $this->options['targetLanguage'] . "'.");
}
// We now apply transformations to the new locale.
foreach (scandir($destLocaleDir) as $f) {
$sourceLang = $this->options['language'];
$destLang = $this->options['targetLanguage'];
$qSourceLang = preg_quote($sourceLang);
$qDestLang = preg_quote($destLang);
if (!is_dir($f) && $f[0] != '.') {
if ($f == 'messages.po') {
// Change the language tag in the PO file.
$pattern = "/Language: ${qSourceLang}/u";
$count = 0;
$content = file_get_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'messages.po');
$newFileContent = preg_replace($pattern, "Language: ${destLang}", $content, -1, $count);
if ($count == 1) {
$this->outVerbose("Language tag '${destLang}' applied to messages.po.");
file_put_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'messages.po', $newFileContent);
} else {
$this->err("Could not change language tag in messages.po.");
}
} elseif ($f == 'messages_po.js') {
// Change the language tag in comments.
// Change the langCode JS variable.
$pattern = "/var langCode = '${qSourceLang}';/u";
$count1 = 0;
$content = file_get_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'messages_po.js');
$newFileContent = preg_replace($pattern, "var langCode = '${destLang}';", $content, -1, $count1);
$pattern = "|/\\* lang: ${qSourceLang} \\*/|u";
$count2 = 0;
$newFileContent = preg_replace($pattern, "/* lang: ${destLang} */", $newFileContent, -1, $count2);
if ($count1 + $count2 == 2) {
$this->outVerbose("Language tag '${destLang}' applied to messages_po.js");
file_put_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'messages_po.js', $newFileContent);
} else {
$this->err("Could not change language tag in messages_po.js");
}
} elseif ($f == 'lang.rdf') {
// Change <![CDATA[XX]]>
// Change http://www.tao.lu/Ontologies/TAO.rdf#LangXX
$pattern = "/<!\\[CDATA\\[${qSourceLang}\\]\\]>/u";
$count1 = 0;
$content = file_get_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'lang.rdf');
$newFileContent = preg_replace($pattern, "<![CDATA[${destLang}]]>", $content, -1, $count1);
$pattern = "|http://www.tao.lu/Ontologies/TAO.rdf#Lang${qSourceLang}|u";
$count2 = 0;
$newFileContent = preg_replace($pattern, "http://www.tao.lu/Ontologies/TAO.rdf#Lang${destLang}", $newFileContent, -1, $count2);
$pattern = '/xml:lang="EN"/u';
$count3 = 0;
$newFileContent = preg_replace($pattern, 'xml:lang="en-US"', $newFileContent, -1, $count3);
if ($count1 + $count2 + $count3 == 3) {
$this->outVerbose("Language tag '${destLang}' applied to lang.rdf");
file_put_contents($destLocaleDir . DIRECTORY_SEPARATOR . 'lang.rdf', $newFileContent);
} else {
$this->err("Could not change language tag in lang.rdf");
}
} else {
// Check for a .rdf extension.
$infos = pathinfo($destLocaleDir . DIRECTORY_SEPARATOR . $f);
if (isset($infos['extension']) && $infos['extension'] == 'rdf') {
// Change annotations @sourceLanguage and @targetLanguage
// Change xml:lang
$pattern = "/@sourceLanguage EN/u";
$content = file_get_contents($destLocaleDir . DIRECTORY_SEPARATOR . $f);
$newFileContent = preg_replace($pattern, "@sourceLanguage en-US", $content);
$pattern = "/@targetLanguage ${qSourceLang}/u";
$newFileContent = preg_replace($pattern, "@targetLanguage ${destLang}", $newFileContent);
$pattern = '/xml:lang="' . $qSourceLang . '"/u';
$newFileContent = preg_replace($pattern, 'xml:lang="' . $destLang . '"', $newFileContent);
$this->outVerbose("Language tag '${destLang}' applied to ${f}");
file_put_contents($destLocaleDir . DIRECTORY_SEPARATOR . $f, $newFileContent);
}
}
}
}
}
/**
* Builds the path to files dedicated to a given language (locale) for a
* extension ID.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @param string extension
* @param string language
* @return string
*/
private function buildLanguagePath($extension, $language)
{
$returnValue = (string) '';
$returnValue = dirname(__FILE__) . '/../../' . $extension . '/' . self::DEF_OUTPUT_DIR . '/' . $language;
return (string) $returnValue;
}
/**
* Find a structure.xml manifest in a given directory.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @param string directory
* @return mixed
*/
public function findStructureManifest($directory = null)
{
$returnValue = null;
if ($directory == null) {
$actionsDir = $this->options['input'] . '/actions';
} else {
$actionsDir = $directory . '/actions';
}
$dirEntries = scandir($actionsDir);
if ($dirEntries === false) {
$returnValue = false;
} else {
$structureFile = null;
foreach ($dirEntries as $f) {
if (preg_match("/(.*)structure\.xml$/", $f)) {
$structureFile = $f;
break;
}
}
if ($structureFile === null) {
$returnValue = false;
} else {
$returnValue = $structureFile;
}
}
return $returnValue;
}
/**
* Prepare a PO file before output by adding headers to it.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @param tao_helpers_translation_POFile $poFile
* @param bool $poEditorReady
* @return void
*/
public function preparePOFile(tao_helpers_translation_POFile $poFile, $poEditorReady = false)
{
$poFile->addHeader('Project-Id-Version', PRODUCT_NAME . ' ' . ApplicationHelper::getVersionName());
$poFile->addHeader('PO-Revision-Date', date('Y-m-d') . 'T' . date('H:i:s'));
$poFile->addHeader('Last-Translator', 'TAO Translation Team <translation@tao.lu>');
$poFile->addHeader('MIME-Version', '1.0');
$poFile->addHeader('Language', $poFile->getTargetLanguage());
$poFile->addHeader('sourceLanguage', $poFile->getSourceLanguage());
$poFile->addHeader('targetLanguage', $poFile->getTargetLanguage());
$poFile->addHeader('Content-Type', 'text/plain; charset=utf-8');
$poFile->addHeader('Content-Transfer-Encoding', '8bit');
if ($poEditorReady) {
$poFile->addHeader('X-Poedit-Basepath', '../../');
$poFile->addHeader('X-Poedit-KeywordsList', '__');
$poFile->addHeader('X-Poedit-SearchPath-0', '.');
}
}
/**
* Determines if an given directory actually contains a TAO extension.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @param string directory
* @return boolean
*/
public function isExtension($directory)
{
$returnValue = (bool) false;
$hasStructure = $this->findStructureManifest($directory) !== false;
$hasPHPManifest = false;
$files = scandir($this->options['input']);
if ($files !== false) {
foreach ($files as $f) {
if (is_file($this->options['input'] . '/' . $f) && is_readable($this->options['input'] . '/' . $f)) {
if ($f == 'manifest.php') {
$hasPHPManifest = true;
}
}
}
}
$returnValue = $hasStructure || $hasPHPManifest;
return (bool) $returnValue;
}
/**
* Add translations as translation units found in a structure.xml manifest
* a given PO file.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @param tao_helpers_translation_POFile $poFile
* @throws Exception
* @return void
*/
public function addManifestsTranslations(tao_helpers_translation_POFile $poFile)
{
$this->outVerbose("Adding all manifests messages to extension '" . $this->options['extension'] . "'");
$rootDir = dirname(__FILE__) . '/../../';
$directories = scandir($rootDir);
$exceptions = ['generis', 'tao', '.*'];
if (false === $directories) {
$this->err("The TAO root directory is not readable. Please check permissions on this directory.", true);
} else {
foreach ($directories as $dir) {
if (is_dir($rootDir . $dir) && !in_array($dir, $exceptions)) {
// Maybe it should be read.
if (in_array('.*', $exceptions) && $dir[0] == '.') {
continue;
} else {
// Is this a TAO extension ?
$file = MenuService::getStructuresFilePath($this->options['extension']);
if (!is_null($file)) {
$structureExtractor = new tao_helpers_translation_StructureExtractor([$file]);
$structureExtractor->extract();
$poFile->addTranslationUnits($structureExtractor->getTranslationUnits());
$this->outVerbose("Manifest of extension '" . $dir . "' added to extension '" . $this->options['extension'] . "'");
}
}
}
}
}
}
/**
* Add the requested language in the ontology. It will used the parameters
* the command line for logic.
*
* @access protected
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
protected function addLanguageToOntology()
{
$this->outVerbose("Importing RDF language description '" . $this->options['language'] . "' to ontology...");
// RDF Language Descriptions are stored in the tao meta-extension locales.
$expectedDescriptionPath = $this->buildLanguagePath('tao', $this->options['language']) . '/lang.rdf';
if (file_exists($expectedDescriptionPath)) {
if (is_readable($expectedDescriptionPath)) {
// Let's remove any instance of the language description before inserting the new one.
$taoNS = 'http://www.tao.lu/Ontologies/TAO.rdf#';
$expectedLangUri = $taoNS . 'Lang' . $this->options['language'];
$lgDescription = new core_kernel_classes_Resource($expectedLangUri);
if ($lgDescription->exists()) {
$lgDescription->delete();
$this->outVerbose("Existing RDF Description language '" . $this->options['language'] . "' deleted.");
}
$generisAdapterRdf = new tao_helpers_data_GenerisAdapterRdf();
if (true === $generisAdapterRdf->import($expectedDescriptionPath, null, LOCAL_NAMESPACE)) {
$this->outVerbose("RDF language description '" . $this->options['language'] . "' successfully imported.");
} else {
$this->err("An error occured while importing the RDF language description '" . $this->options['language'] . "'.", true);
}
} else {
$this->err("RDF language description (lang.rdf) cannot be read in meta-extension 'tao' for language '" . $this->options['language'] . "'.", true);
}
} else {
$this->err("RDF language description (lang.rdf) not found in meta-extension 'tao' for language '" . $this->options['language'] . "'.", true);
}
$this->outVerbose("RDF language description '" . $this->options['language'] . "' added to ontology.");
}
/**
* Removes a language from the Ontology and all triples with the related
* tag. Will use command line parameters for logic.
*
* @access protected
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
protected function removeLanguageFromOntology()
{
$this->outVerbose("Removing RDF language description '" . $this->options['language'] . "' from ontology...");
$taoNS = 'http://www.tao.lu/Ontologies/TAO.rdf#';
$expectedDescriptionUri = $taoNS . 'Lang' . $this->options['language'];
$lgResource = new core_kernel_classes_Resource($expectedDescriptionUri);
if (true === $lgResource->exists()) {
$lgResource->delete();
$this->outVerbose("RDF language description '" . $this->options['language'] . "' successfully removed.");
} else {
$this->outVerbose("RDF language description '" . $this->options['language'] . "' not found but considered removed.");
}
}
/**
* Checks authentication parameters for the TAO API.
*
* @access protected
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
protected function checkAuthInput()
{
$defaults = ['user' => null,
'password' => null];
$this->options = array_merge($defaults, $this->options);
if ($this->options['user'] == null) {
$this->err("Please provide a value for the 'user' parameter.", true);
} elseif ($this->options['password'] == null) {
$this->err("Please provide a value for the 'password' parameter.", true);
}
}
/**
* Get the ontology file paths for a given extension, sorted by target name
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return array
*/
private function getOntologyFiles()
{
$returnValue = [];
$ext = common_ext_ExtensionsManager::singleton()->getExtensionById($this->options['extension']);
$returnValue = $ext->getManifest()->getInstallModelFiles();
return (array) $returnValue;
}
/**
* Check inputs for the 'enable' action.
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkEnableInput()
{
$this->checkAuthInput();
$defaults = ['language' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR];
$this->options = array_merge($defaults, $this->options);
if ($this->options['language'] == null) {
$this->err("Please provide the 'language' parameter.", true);
}
}
/**
* Short description of method checkDisableInput
*
* @access private
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkDisableInput()
{
$this->checkAuthInput();
$defaults = ['language' => null];
$this->options = array_merge($defaults, $this->options);
if ($this->options['language'] == null) {
$this->err("Please provide the 'language' parameter.", true);
}
}
/**
* Short description of method actionEnable
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionEnable()
{
$userService = tao_models_classes_UserService::singleton();
$this->outVerbose("Connecting to TAO as '" . $this->options['user'] . "' ...");
if ($userService->loginUser($this->options['user'], $this->options['password'])) {
$this->outVerbose("Connected to TAO as '" . $this->options['user'] . "'.");
$this->addLanguageToOntology();
$userService->logout();
$this->outVerbose("Disconnected from TAO.");
} else {
$this->err("Unable to connect to TAO as '" . $this->options['user'] . "'. Please check user name and password.", true);
}
}
/**
* Implementation of the 'disable' action. When this action is called, a
* Language Description ('language' param) is removed from the Knowledge
* The language is at this time not available anymore to end-users. However,
* Triples that had a corresponding language tag are not remove from the
* If the language is enabled again via the 'enable' action of this script,
* having corresponding languages will be reachable again.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionDisable()
{
$userService = tao_models_classes_UserService::singleton();
$this->outVerbose("Connecting to TAO as '" . $this->options['user'] . "' ...");
if ($userService->loginUser($this->options['user'], $this->options['password'])) {
$this->outVerbose("Connected to TAO as '" . $this->options['user'] . "'.");
$this->removeLanguageFromOntology();
$userService->logout();
$this->outVerbose("Disconnected from TAO.");
} else {
$this->err("Unable to connect to TAO as '" . $this->options['user'] . "'. Please check user name and password.", true);
}
}
/**
* Implementation of the 'compile' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionCompile()
{
$extensionsToCreate = explode(',', $this->options['extension']);
$extensionsToCreate = array_unique($extensionsToCreate);
foreach ($extensionsToCreate as $extension) {
$language = $this->options['language'];
$compiledTranslationFile = new tao_helpers_translation_TranslationFile();
$compiledTranslationFile->setTargetLanguage($this->options['language']);
$this->outVerbose("Compiling language '${language}' for extension '${extension}'...");
// Get the dependencies of the target extension.
// @todo Deal with dependencies at compilation time.
$dependencies = [];
if ($extension !== 'tao') {
$dependencies[] = 'tao';
}
$this->outVerbose("Resolving Dependencies...");
foreach ($dependencies as $depExtId) {
$this->outVerbose("Adding messages from extension '${depExtId}' in '${language}'...");
// Does the locale exist for $depExtId?
$depPath = $this->buildLanguagePath($depExtId, $language) . '/' . self::DEF_PO_FILENAME;
if (!file_exists($depPath) || !is_readable($depPath)) {
$this->outVerbose("Dependency on extension '${depExtId}' in '${language}' does not exist. Trying to resolve default language...");
$depPath = $this->buildLanguagePath($depExtId, tao_helpers_translation_Utils::getDefaultLanguage() . '/' . self::DEF_PO_FILENAME);
if (!file_exists($depPath) || !is_readable($depPath)) {
$this->outVerbose("Dependency on extension '${depExtId}' in '${language}' does not exist.");
continue;
}
}
// Recompile the dependent extension (for the moment 'tao' meta-extension only).
$oldVerbose = $this->options['verbose'];
$this->parameters['verbose'] = false;
$this->options['extension'] = $depExtId;
$this->actionCompile();
$this->options['extension'] = $extension;
$this->parameters['verbose'] = $oldVerbose;
$poFileReader = new tao_helpers_translation_POFileReader($depPath);
$poFileReader->read();
$poFile = $poFileReader->getTranslationFile();
$poCount = $poFile->count();
$compiledTranslationFile->addTranslationUnits($poFile->getTranslationUnits());
$this->outVerbose("${poCount} messages added.");
}
if (($extDirectories = scandir(ROOT_PATH)) !== false) {
// Get all public messages accross extensions.
foreach ($extDirectories as $extDir) {
$extPath = ROOT_PATH . '/' . $extDir;
if (is_dir($extPath) && is_readable($extPath) && $extDir[0] != '.' && !in_array($extDir, $dependencies) && $extDir != $extension && $extDir != 'generis') {
$this->outVerbose("Adding public messages from extension '${extDir}' in '${language}'...");
$poPath = $this->buildLanguagePath($extDir, $language) . '/' . self::DEF_PO_FILENAME;
if (!file_exists($poPath) || !is_readable($poPath)) {
$this->outVerbose("Extension '${extDir}' is not translated in language '${language}'. Trying to retrieve default language...");
$poPath = $this->buildLanguagePath($extDir, tao_helpers_translation_Utils::getDefaultLanguage()) . '/' . self::DEF_PO_FILENAME;
if (!file_exists($poPath) || !is_readable($poPath)) {
$this->outVerbose("Extension '${extDir}' in '${language}' does not exist.");
continue;
}
}
$poFileReader = new tao_helpers_translation_POFileReader($poPath);
$poFileReader->read();
$poFile = $poFileReader->getTranslationFile();
$poUnits = $poFile->getByFlag('tao-public');
$poCount = count($poUnits);
$compiledTranslationFile->addTranslationUnits($poUnits);
$this->outVerbose("${poCount} public messages added.");
}
}
// Finally, add the translation units related to the target extension.
$path = $this->buildLanguagePath($extension, $language) . '/' . self::DEF_PO_FILENAME;
if (file_exists($path) && is_readable($path)) {
$poFileReader = new tao_helpers_translation_POFileReader($path);
$poFileReader->read();
$poFile = $poFileReader->getTranslationFile();
$compiledTranslationFile->addTranslationUnits($poFile->getTranslationUnits());
// Sort the TranslationUnits.
$sortingMethod = tao_helpers_translation_TranslationFile::SORT_ASC_I;
$compiledTranslationFile->setTranslationUnits($compiledTranslationFile->sortBySource($sortingMethod));
$jsPath = $this->buildLanguagePath($extension, $language) . '/' . self::DEF_JS_FILENAME;
$jsFileWriter = new tao_helpers_translation_JSFileWriter($jsPath, $compiledTranslationFile);
$jsFileWriter->write();
$this->outVerbose("JavaScript compiled translations for extension '${extension}' with '${language}' written.");
} else {
$this->err("PO file '${path}' for extension '${extension}' with language '${language}' cannot be read.", true);
}
} else {
$this->err("Cannot list TAO Extensions from root path. Check your system rights.", true);
}
$this->outVerbose("Translations for '${extension}' with language '${language}' gracefully compiled.");
}
}
/**
* Checks the input for the 'compile' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkCompileInput()
{
$defaults = ['extension' => null,
'language' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR];
$this->options = array_merge($defaults, $this->options);
if ($this->options['extension'] == null) {
$this->err("Please provide the 'extension' parameter.", true);
} elseif ($this->options['language'] == null) {
$this->err("Please provide the 'language' parameter.", true);
}
}
/**
* Implementation of the 'compileAll' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
public function actionCompileAll()
{
// Get the list of languages that will be compiled.
$this->outVerbose("Compiling all languages for extension '" . $this->options['extension'] . "'...");
$rootDir = ROOT_PATH;
$extensionDir = $rootDir . '/' . $this->options['extension'];
$localesDir = $extensionDir . '/locales';
$locales = [];
$directories = scandir($localesDir);
if ($directories === false) {
$this->err("The locales directory of extension '" . $this->options['extension'] . "' cannot be read.", true);
} else {
foreach ($directories as $dir) {
if ($dir[0] !== '.') {
// It is a language directory.
$locales[] = $dir;
}
}
}
foreach ($locales as $l) {
$this->options['language'] = $l;
$this->checkCompileInput();
$this->actionCompile();
$this->outVerbose("");
}
}
/**
* Checks the input of the 'compileAll' action.
*
* @access public
* @author Joel Bout, <joel.bout@tudor.lu>
* @return void
*/
private function checkCompileAllInput()
{
$defaults = ['extension' => null,
'input' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_INPUT_DIR,
'output' => dirname(__FILE__) . '/../../' . $this->options['extension'] . '/' . self::DEF_OUTPUT_DIR];
$this->options = array_merge($defaults, $this->options);
if ($this->options['extension'] == null) {
$this->err("Please provide the 'extension' parameter.", true);
}
}
private function actionGetExt()
{
$rootDir = ROOT_PATH;
$extensions = [];
$extensionsList = '';
$language = $this->options['language'];
// create initial folders array
if ($dir = opendir($rootDir)) {
$j = 0;
while (($file = readdir($dir)) !== false) {
if ($file[0] !== '.' && $file != '.' && $file != '..' && is_dir($rootDir . $file)) {
$j++;
$directories[$j] = $file;
}
}
}
if ($directories === false) {
$this->err("The locales directory of extension '" . $this->options['extension'] . "' cannot be read.", true);
} else {
foreach ($directories as $dir) {
// folder where these .po files should be
$extensionLocalesDir = $dir . '/' . self::DEF_OUTPUT_DIR . '/' . $language;
$poFile = $rootDir . $extensionLocalesDir . '/' . self::DEF_PO_FILENAME;
// check .po file existency
if (file_exists($poFile) && is_readable($poFile)) {
$extensions[] = $dir;
$extensionsList .= (!$extensionsList ? "" : ",") . $dir;
}
}
sort($extensions);
print_r($extensions);
echo ( count($extensions) . ' extensions with translations: ' . $extensionsList . "\n");
}
}
/**
* @param $f
* @param $translatableProperties
* @return tao_helpers_translation_POFile
* @throws tao_helpers_translation_TranslationException
*/
protected function extractPoFileFromRDF($f, $translatableProperties)
{
$modelExtractor = new tao_helpers_translation_POExtractor([$f]);
$modelExtractor->setTranslatableProperties($translatableProperties);
$modelExtractor->extract();
$translationFile = new tao_helpers_translation_POFile();
$translationFile->setSourceLanguage(tao_helpers_translation_Utils::getDefaultLanguage());
$translationFile->setTargetLanguage($this->options['language']);
$translationFile->addTranslationUnits($modelExtractor->getTranslationUnits());
$translationFile->setExtensionId($this->options['extension']);
$this->preparePOFile($translationFile);
return $translationFile;
}
protected function checkInputOption()
{
if (!is_null($this->options['input'])) {
if (!is_dir($this->options['input'])) {
$this->err("The 'input' parameter you provided is not a directory.", true);
} else {
if (!is_readable($this->options['input'])) {
$this->err("The 'input' directory is not readable.", true);
}
}
}
}
protected function checkOutputOption()
{
if (!is_null($this->options['output'])) {
if (!is_dir($this->options['output'])) {
$this->err("The 'output' parameter you provided is not a directory.", true);
} else {
if (!is_writable($this->options['output'])) {
$this->err("The 'output' directory is not writable.", true);
}
}
}
}
/**
* @return array
* @throws Exception
*/
protected function getLanguageList()
{
$rootDir = dirname(__FILE__) . '/../..';
$extensionDir = $rootDir . '/' . $this->options['extension'];
$localesDir = $extensionDir . '/locales';
$locales = [];
$directories = scandir($localesDir);
if ($directories === false) {
$this->err("The locales directory of extension '" . $this->options['extension'] . "' cannot be read.", true);
return $locales;
} else {
foreach ($directories as $dir) {
if ($dir[0] !== '.') {
// It is a language directory.
$locales[] = $dir;
}
}
return $locales;
}
}
/**
*
* @param $f filename
* @return string
*/
protected function getOntologyPOFileName($f)
{
return basename($f) . '.po';
}
}