123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423 |
- <?php
- /**
- * @copyright Copyright (c) 2016, ownCloud, Inc.
- * @copyright 2016 Roeland Jago Douma <roeland@famdouma.nl>
- * @copyright 2016 Lukas Reschke <lukas@statuscode.ch>
- *
- * @author Bart Visscher <bartv@thisnet.nl>
- * @author Joas Schilling <coding@schilljs.com>
- * @author Lukas Reschke <lukas@statuscode.ch>
- * @author Morris Jobke <hey@morrisjobke.de>
- * @author Robin Appelman <robin@icewind.nl>
- * @author Robin McCorkell <robin@mccorkell.me.uk>
- * @author Roeland Jago Douma <roeland@famdouma.nl>
- *
- * @license AGPL-3.0
- *
- * This code is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License, version 3,
- * as published by the Free Software Foundation.
- *
- * 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 Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License, version 3,
- * along with this program. If not, see <http://www.gnu.org/licenses/>
- *
- */
- namespace OC\L10N;
- use OCP\IConfig;
- use OCP\IRequest;
- use OCP\IUserSession;
- use OCP\L10N\IFactory;
- /**
- * A factory that generates language instances
- */
- class Factory implements IFactory {
- /** @var string */
- protected $requestLanguage = '';
- /**
- * cached instances
- * @var array Structure: Lang => App => \OCP\IL10N
- */
- protected $instances = [];
- /**
- * @var array Structure: App => string[]
- */
- protected $availableLanguages = [];
- /**
- * @var array Structure: string => callable
- */
- protected $pluralFunctions = [];
- /** @var IConfig */
- protected $config;
- /** @var IRequest */
- protected $request;
- /** @var IUserSession */
- protected $userSession;
- /** @var string */
- protected $serverRoot;
- /**
- * @param IConfig $config
- * @param IRequest $request
- * @param IUserSession $userSession
- * @param string $serverRoot
- */
- public function __construct(IConfig $config,
- IRequest $request,
- IUserSession $userSession,
- $serverRoot) {
- $this->config = $config;
- $this->request = $request;
- $this->userSession = $userSession;
- $this->serverRoot = $serverRoot;
- }
- /**
- * Get a language instance
- *
- * @param string $app
- * @param string|null $lang
- * @return \OCP\IL10N
- */
- public function get($app, $lang = null) {
- $app = \OC_App::cleanAppId($app);
- if ($lang !== null) {
- $lang = str_replace(array('\0', '/', '\\', '..'), '', (string) $lang);
- }
- $forceLang = $this->config->getSystemValue('force_language', false);
- if (is_string($forceLang)) {
- $lang = $forceLang;
- }
- if ($lang === null || !$this->languageExists($app, $lang)) {
- $lang = $this->findLanguage($app);
- }
- if (!isset($this->instances[$lang][$app])) {
- $this->instances[$lang][$app] = new L10N(
- $this, $app, $lang,
- $this->getL10nFilesForApp($app, $lang)
- );
- }
- return $this->instances[$lang][$app];
- }
- /**
- * Find the best language
- *
- * @param string|null $app App id or null for core
- * @return string language If nothing works it returns 'en'
- */
- public function findLanguage($app = null) {
- $forceLang = $this->config->getSystemValue('force_language', false);
- if (is_string($forceLang)) {
- $this->requestLanguage = $forceLang;
- }
- if ($this->requestLanguage !== '' && $this->languageExists($app, $this->requestLanguage)) {
- return $this->requestLanguage;
- }
- /**
- * At this point Nextcloud might not yet be installed and thus the lookup
- * in the preferences table might fail. For this reason we need to check
- * whether the instance has already been installed
- *
- * @link https://github.com/owncloud/core/issues/21955
- */
- if($this->config->getSystemValue('installed', false)) {
- $userId = !is_null($this->userSession->getUser()) ? $this->userSession->getUser()->getUID() : null;
- if(!is_null($userId)) {
- $userLang = $this->config->getUserValue($userId, 'core', 'lang', null);
- } else {
- $userLang = null;
- }
- } else {
- $userId = null;
- $userLang = null;
- }
- if ($userLang) {
- $this->requestLanguage = $userLang;
- if ($this->languageExists($app, $userLang)) {
- return $userLang;
- }
- }
- try {
- // Try to get the language from the Request
- $lang = $this->getLanguageFromRequest($app);
- if ($userId !== null && $app === null && !$userLang) {
- $this->config->setUserValue($userId, 'core', 'lang', $lang);
- }
- return $lang;
- } catch (LanguageNotFoundException $e) {
- // Finding language from request failed fall back to default language
- $defaultLanguage = $this->config->getSystemValue('default_language', false);
- if ($defaultLanguage !== false && $this->languageExists($app, $defaultLanguage)) {
- return $defaultLanguage;
- }
- }
- // We could not find any language so fall back to english
- return 'en';
- }
- /**
- * Find all available languages for an app
- *
- * @param string|null $app App id or null for core
- * @return array an array of available languages
- */
- public function findAvailableLanguages($app = null) {
- $key = $app;
- if ($key === null) {
- $key = 'null';
- }
- // also works with null as key
- if (!empty($this->availableLanguages[$key])) {
- return $this->availableLanguages[$key];
- }
- $available = ['en']; //english is always available
- $dir = $this->findL10nDir($app);
- if (is_dir($dir)) {
- $files = scandir($dir);
- if ($files !== false) {
- foreach ($files as $file) {
- if (substr($file, -5) === '.json' && substr($file, 0, 4) !== 'l10n') {
- $available[] = substr($file, 0, -5);
- }
- }
- }
- }
- // merge with translations from theme
- $theme = $this->config->getSystemValue('theme');
- if (!empty($theme)) {
- $themeDir = $this->serverRoot . '/themes/' . $theme . substr($dir, strlen($this->serverRoot));
- if (is_dir($themeDir)) {
- $files = scandir($themeDir);
- if ($files !== false) {
- foreach ($files as $file) {
- if (substr($file, -5) === '.json' && substr($file, 0, 4) !== 'l10n') {
- $available[] = substr($file, 0, -5);
- }
- }
- }
- }
- }
- $this->availableLanguages[$key] = $available;
- return $available;
- }
- /**
- * @param string|null $app App id or null for core
- * @param string $lang
- * @return bool
- */
- public function languageExists($app, $lang) {
- if ($lang === 'en') {//english is always available
- return true;
- }
- $languages = $this->findAvailableLanguages($app);
- return array_search($lang, $languages) !== false;
- }
- /**
- * @param string|null $app
- * @return string
- * @throws LanguageNotFoundException
- */
- private function getLanguageFromRequest($app) {
- $header = $this->request->getHeader('ACCEPT_LANGUAGE');
- if ($header) {
- $available = $this->findAvailableLanguages($app);
- // E.g. make sure that 'de' is before 'de_DE'.
- sort($available);
- $preferences = preg_split('/,\s*/', strtolower($header));
- foreach ($preferences as $preference) {
- list($preferred_language) = explode(';', $preference);
- $preferred_language = str_replace('-', '_', $preferred_language);
- foreach ($available as $available_language) {
- if ($preferred_language === strtolower($available_language)) {
- return $available_language;
- }
- }
- // Fallback from de_De to de
- foreach ($available as $available_language) {
- if (substr($preferred_language, 0, 2) === $available_language) {
- return $available_language;
- }
- }
- }
- }
- throw new LanguageNotFoundException();
- }
- /**
- * Checks if $sub is a subdirectory of $parent
- *
- * @param string $sub
- * @param string $parent
- * @return bool
- */
- private function isSubDirectory($sub, $parent) {
- // Check whether $sub contains no ".."
- if(strpos($sub, '..') !== false) {
- return false;
- }
- // Check whether $sub is a subdirectory of $parent
- if (strpos($sub, $parent) === 0) {
- return true;
- }
- return false;
- }
- /**
- * Get a list of language files that should be loaded
- *
- * @param string $app
- * @param string $lang
- * @return string[]
- */
- // FIXME This method is only public, until OC_L10N does not need it anymore,
- // FIXME This is also the reason, why it is not in the public interface
- public function getL10nFilesForApp($app, $lang) {
- $languageFiles = [];
- $i18nDir = $this->findL10nDir($app);
- $transFile = strip_tags($i18nDir) . strip_tags($lang) . '.json';
- if (($this->isSubDirectory($transFile, $this->serverRoot . '/core/l10n/')
- || $this->isSubDirectory($transFile, $this->serverRoot . '/lib/l10n/')
- || $this->isSubDirectory($transFile, $this->serverRoot . '/settings/l10n/')
- || $this->isSubDirectory($transFile, \OC_App::getAppPath($app) . '/l10n/')
- )
- && file_exists($transFile)) {
- // load the translations file
- $languageFiles[] = $transFile;
- }
- // merge with translations from theme
- $theme = $this->config->getSystemValue('theme');
- if (!empty($theme)) {
- $transFile = $this->serverRoot . '/themes/' . $theme . substr($transFile, strlen($this->serverRoot));
- if (file_exists($transFile)) {
- $languageFiles[] = $transFile;
- }
- }
- return $languageFiles;
- }
- /**
- * find the l10n directory
- *
- * @param string $app App id or empty string for core
- * @return string directory
- */
- protected function findL10nDir($app = null) {
- if (in_array($app, ['core', 'lib', 'settings'])) {
- if (file_exists($this->serverRoot . '/' . $app . '/l10n/')) {
- return $this->serverRoot . '/' . $app . '/l10n/';
- }
- } else if ($app && \OC_App::getAppPath($app) !== false) {
- // Check if the app is in the app folder
- return \OC_App::getAppPath($app) . '/l10n/';
- }
- return $this->serverRoot . '/core/l10n/';
- }
- /**
- * Creates a function from the plural string
- *
- * Parts of the code is copied from Habari:
- * https://github.com/habari/system/blob/master/classes/locale.php
- * @param string $string
- * @return string
- */
- public function createPluralFunction($string) {
- if (isset($this->pluralFunctions[$string])) {
- return $this->pluralFunctions[$string];
- }
- if (preg_match( '/^\s*nplurals\s*=\s*(\d+)\s*;\s*plural=(.*)$/u', $string, $matches)) {
- // sanitize
- $nplurals = preg_replace( '/[^0-9]/', '', $matches[1] );
- $plural = preg_replace( '#[^n0-9:\(\)\?\|\&=!<>+*/\%-]#', '', $matches[2] );
- $body = str_replace(
- array( 'plural', 'n', '$n$plurals', ),
- array( '$plural', '$n', '$nplurals', ),
- 'nplurals='. $nplurals . '; plural=' . $plural
- );
- // add parents
- // important since PHP's ternary evaluates from left to right
- $body .= ';';
- $res = '';
- $p = 0;
- for($i = 0; $i < strlen($body); $i++) {
- $ch = $body[$i];
- switch ( $ch ) {
- case '?':
- $res .= ' ? (';
- $p++;
- break;
- case ':':
- $res .= ') : (';
- break;
- case ';':
- $res .= str_repeat( ')', $p ) . ';';
- $p = 0;
- break;
- default:
- $res .= $ch;
- }
- }
- $body = $res . 'return ($plural>=$nplurals?$nplurals-1:$plural);';
- $function = create_function('$n', $body);
- $this->pluralFunctions[$string] = $function;
- return $function;
- } else {
- // default: one plural form for all cases but n==1 (english)
- $function = create_function(
- '$n',
- '$nplurals=2;$plural=($n==1?0:1);return ($plural>=$nplurals?$nplurals-1:$plural);'
- );
- $this->pluralFunctions[$string] = $function;
- return $function;
- }
- }
- }
|