164 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /*
 | |
|  * This file is part of the Symfony package.
 | |
|  *
 | |
|  * (c) Fabien Potencier <fabien@symfony.com>
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Symfony\Component\Translation\Loader;
 | |
| 
 | |
| use Symfony\Component\Translation\MessageCatalogue;
 | |
| use Symfony\Component\Translation\Exception\InvalidResourceException;
 | |
| use Symfony\Component\Translation\Exception\NotFoundResourceException;
 | |
| use Symfony\Component\Config\Resource\FileResource;
 | |
| 
 | |
| /**
 | |
|  * XliffFileLoader loads translations from XLIFF files.
 | |
|  *
 | |
|  * @author Fabien Potencier <fabien@symfony.com>
 | |
|  *
 | |
|  * @api
 | |
|  */
 | |
| class XliffFileLoader implements LoaderInterface
 | |
| {
 | |
|     /**
 | |
|      * {@inheritdoc}
 | |
|      *
 | |
|      * @api
 | |
|      */
 | |
|     public function load($resource, $locale, $domain = 'messages')
 | |
|     {
 | |
|         if (!stream_is_local($resource)) {
 | |
|             throw new InvalidResourceException(sprintf('This is not a local file "%s".', $resource));
 | |
|         }
 | |
| 
 | |
|         if (!file_exists($resource)) {
 | |
|             throw new NotFoundResourceException(sprintf('File "%s" not found.', $resource));
 | |
|         }
 | |
| 
 | |
|         list($xml, $encoding) = $this->parseFile($resource);
 | |
|         $xml->registerXPathNamespace('xliff', 'urn:oasis:names:tc:xliff:document:1.2');
 | |
| 
 | |
|         $catalogue = new MessageCatalogue($locale);
 | |
|         foreach ($xml->xpath('//xliff:trans-unit') as $translation) {
 | |
|             $attributes = $translation->attributes();
 | |
| 
 | |
|             if (!(isset($attributes['resname']) || isset($translation->source)) || !isset($translation->target)) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             $source = isset($attributes['resname']) && $attributes['resname'] ? $attributes['resname'] : $translation->source;
 | |
|             $target = (string) $translation->target;
 | |
| 
 | |
|             // If the xlf file has another encoding specified, try to convert it because
 | |
|             // simple_xml will always return utf-8 encoded values
 | |
|             if ('UTF-8' !== $encoding && !empty($encoding)) {
 | |
|                 if (function_exists('mb_convert_encoding')) {
 | |
|                     $target = mb_convert_encoding($target, $encoding, 'UTF-8');
 | |
|                 } elseif (function_exists('iconv')) {
 | |
|                     $target = iconv('UTF-8', $encoding, $target);
 | |
|                 } else {
 | |
|                     throw new \RuntimeException('No suitable convert encoding function (use UTF-8 as your encoding or install the iconv or mbstring extension).');
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             $catalogue->set((string) $source, $target, $domain);
 | |
|         }
 | |
|         $catalogue->addResource(new FileResource($resource));
 | |
| 
 | |
|         return $catalogue;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Validates and parses the given file into a SimpleXMLElement
 | |
|      *
 | |
|      * @param string $file
 | |
|      *
 | |
|      * @throws \RuntimeException
 | |
|      *
 | |
|      * @return \SimpleXMLElement
 | |
|      *
 | |
|      * @throws InvalidResourceException
 | |
|      */
 | |
|     private function parseFile($file)
 | |
|     {
 | |
|         $internalErrors = libxml_use_internal_errors(true);
 | |
|         $disableEntities = libxml_disable_entity_loader(true);
 | |
|         libxml_clear_errors();
 | |
| 
 | |
|         $dom = new \DOMDocument();
 | |
|         $dom->validateOnParse = true;
 | |
|         if (!@$dom->loadXML(file_get_contents($file), LIBXML_NONET | (defined('LIBXML_COMPACT') ? LIBXML_COMPACT : 0))) {
 | |
|             libxml_disable_entity_loader($disableEntities);
 | |
| 
 | |
|             throw new InvalidResourceException(implode("\n", $this->getXmlErrors($internalErrors)));
 | |
|         }
 | |
| 
 | |
|         libxml_disable_entity_loader($disableEntities);
 | |
| 
 | |
|         foreach ($dom->childNodes as $child) {
 | |
|             if ($child->nodeType === XML_DOCUMENT_TYPE_NODE) {
 | |
|                 libxml_use_internal_errors($internalErrors);
 | |
| 
 | |
|                 throw new InvalidResourceException('Document types are not allowed.');
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $location = str_replace('\\', '/', __DIR__).'/schema/dic/xliff-core/xml.xsd';
 | |
|         $parts = explode('/', $location);
 | |
|         if (0 === stripos($location, 'phar://')) {
 | |
|             $tmpfile = tempnam(sys_get_temp_dir(), 'sf2');
 | |
|             if ($tmpfile) {
 | |
|                 copy($location, $tmpfile);
 | |
|                 $parts = explode('/', str_replace('\\', '/', $tmpfile));
 | |
|             }
 | |
|         }
 | |
|         $drive = '\\' === DIRECTORY_SEPARATOR ? array_shift($parts).'/' : '';
 | |
|         $location = 'file:///'.$drive.implode('/', array_map('rawurlencode', $parts));
 | |
| 
 | |
|         $source = file_get_contents(__DIR__.'/schema/dic/xliff-core/xliff-core-1.2-strict.xsd');
 | |
|         $source = str_replace('http://www.w3.org/2001/xml.xsd', $location, $source);
 | |
| 
 | |
|         if (!@$dom->schemaValidateSource($source)) {
 | |
|             throw new InvalidResourceException(implode("\n", $this->getXmlErrors($internalErrors)));
 | |
|         }
 | |
| 
 | |
|         $dom->normalizeDocument();
 | |
| 
 | |
|         libxml_use_internal_errors($internalErrors);
 | |
| 
 | |
|         return array(simplexml_import_dom($dom), strtoupper($dom->encoding));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the XML errors of the internal XML parser
 | |
|      *
 | |
|      * @param Boolean $internalErrors
 | |
|      *
 | |
|      * @return array An array of errors
 | |
|      */
 | |
|     private function getXmlErrors($internalErrors)
 | |
|     {
 | |
|         $errors = array();
 | |
|         foreach (libxml_get_errors() as $error) {
 | |
|             $errors[] = sprintf('[%s %s] %s (in %s - line %d, column %d)',
 | |
|                 LIBXML_ERR_WARNING == $error->level ? 'WARNING' : 'ERROR',
 | |
|                 $error->code,
 | |
|                 trim($error->message),
 | |
|                 $error->file ? $error->file : 'n/a',
 | |
|                 $error->line,
 | |
|                 $error->column
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         libxml_clear_errors();
 | |
|         libxml_use_internal_errors($internalErrors);
 | |
| 
 | |
|         return $errors;
 | |
|     }
 | |
| }
 |