GettextMessageSource.php 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\i18n;
  8. use Yii;
  9. use yii\base\InvalidArgumentException;
  10. /**
  11. * GettextMessageSource represents a message source that is based on GNU Gettext.
  12. *
  13. * Each GettextMessageSource instance represents the message translations
  14. * for a single domain. And each message category represents a message context
  15. * in Gettext. Translated messages are stored as either a MO or PO file,
  16. * depending on the [[useMoFile]] property value.
  17. *
  18. * All translations are saved under the [[basePath]] directory.
  19. *
  20. * Translations in one language are kept as MO or PO files under an individual
  21. * subdirectory whose name is the language ID. The file name is specified via
  22. * [[catalog]] property, which defaults to 'messages'.
  23. *
  24. * @author Qiang Xue <qiang.xue@gmail.com>
  25. * @since 2.0
  26. */
  27. class GettextMessageSource extends MessageSource
  28. {
  29. const MO_FILE_EXT = '.mo';
  30. const PO_FILE_EXT = '.po';
  31. /**
  32. * @var string base directory of messages files
  33. */
  34. public $basePath = '@app/messages';
  35. /**
  36. * @var string sub-directory of messages files
  37. */
  38. public $catalog = 'messages';
  39. /**
  40. * @var bool whether to use generated MO files
  41. */
  42. public $useMoFile = true;
  43. /**
  44. * @var bool whether to use big-endian when reading and writing an integer
  45. */
  46. public $useBigEndian = false;
  47. /**
  48. * Loads the message translation for the specified $language and $category.
  49. * If translation for specific locale code such as `en-US` isn't found it
  50. * tries more generic `en`. When both are present, the `en-US` messages will be merged
  51. * over `en`. See [[loadFallbackMessages]] for details.
  52. * If the $language is less specific than [[sourceLanguage]], the method will try to
  53. * load the messages for [[sourceLanguage]]. For example: [[sourceLanguage]] is `en-GB`,
  54. * $language is `en`. The method will load the messages for `en` and merge them over `en-GB`.
  55. *
  56. * @param string $category the message category
  57. * @param string $language the target language
  58. * @return array the loaded messages. The keys are original messages, and the values are translated messages.
  59. * @see loadFallbackMessages
  60. * @see sourceLanguage
  61. */
  62. protected function loadMessages($category, $language)
  63. {
  64. $messageFile = $this->getMessageFilePath($language);
  65. $messages = $this->loadMessagesFromFile($messageFile, $category);
  66. $fallbackLanguage = substr($language, 0, 2);
  67. $fallbackSourceLanguage = substr($this->sourceLanguage, 0, 2);
  68. if ($fallbackLanguage !== '' && $fallbackLanguage !== $language) {
  69. $messages = $this->loadFallbackMessages($category, $fallbackLanguage, $messages, $messageFile);
  70. } elseif ($fallbackSourceLanguage !== '' && $language === $fallbackSourceLanguage) {
  71. $messages = $this->loadFallbackMessages($category, $this->sourceLanguage, $messages, $messageFile);
  72. } elseif ($messages === null) {
  73. Yii::error("The message file for category '$category' does not exist: $messageFile", __METHOD__);
  74. }
  75. return (array) $messages;
  76. }
  77. /**
  78. * The method is normally called by [[loadMessages]] to load the fallback messages for the language.
  79. * Method tries to load the $category messages for the $fallbackLanguage and adds them to the $messages array.
  80. *
  81. * @param string $category the message category
  82. * @param string $fallbackLanguage the target fallback language
  83. * @param array $messages the array of previously loaded translation messages.
  84. * The keys are original messages, and the values are the translated messages.
  85. * @param string $originalMessageFile the path to the file with messages. Used to log an error message
  86. * in case when no translations were found.
  87. * @return array the loaded messages. The keys are original messages, and the values are the translated messages.
  88. * @since 2.0.7
  89. */
  90. protected function loadFallbackMessages($category, $fallbackLanguage, $messages, $originalMessageFile)
  91. {
  92. $fallbackMessageFile = $this->getMessageFilePath($fallbackLanguage);
  93. $fallbackMessages = $this->loadMessagesFromFile($fallbackMessageFile, $category);
  94. if (
  95. $messages === null && $fallbackMessages === null
  96. && $fallbackLanguage !== $this->sourceLanguage
  97. && strpos($this->sourceLanguage, $fallbackLanguage) !== 0
  98. ) {
  99. Yii::error("The message file for category '$category' does not exist: $originalMessageFile "
  100. . "Fallback file does not exist as well: $fallbackMessageFile", __METHOD__);
  101. } elseif (empty($messages)) {
  102. return $fallbackMessages;
  103. } elseif (!empty($fallbackMessages)) {
  104. foreach ($fallbackMessages as $key => $value) {
  105. if (!empty($value) && empty($messages[$key])) {
  106. $messages[$key] = $value;
  107. }
  108. }
  109. }
  110. return (array) $messages;
  111. }
  112. /**
  113. * Returns message file path for the specified language and category.
  114. *
  115. * @param string $language the target language
  116. * @return string path to message file
  117. */
  118. protected function getMessageFilePath($language)
  119. {
  120. $language = (string) $language;
  121. if ($language !== '' && !preg_match('/^[a-z0-9_-]+$/i', $language)) {
  122. throw new InvalidArgumentException(sprintf('Invalid language code: "%s".', $language));
  123. }
  124. $messageFile = Yii::getAlias($this->basePath) . '/' . $language . '/' . $this->catalog;
  125. if ($this->useMoFile) {
  126. $messageFile .= self::MO_FILE_EXT;
  127. } else {
  128. $messageFile .= self::PO_FILE_EXT;
  129. }
  130. return $messageFile;
  131. }
  132. /**
  133. * Loads the message translation for the specified language and category or returns null if file doesn't exist.
  134. *
  135. * @param string $messageFile path to message file
  136. * @param string $category the message category
  137. * @return array|null array of messages or null if file not found
  138. */
  139. protected function loadMessagesFromFile($messageFile, $category)
  140. {
  141. if (is_file($messageFile)) {
  142. if ($this->useMoFile) {
  143. $gettextFile = new GettextMoFile(['useBigEndian' => $this->useBigEndian]);
  144. } else {
  145. $gettextFile = new GettextPoFile();
  146. }
  147. $messages = $gettextFile->load($messageFile, $category);
  148. if (!is_array($messages)) {
  149. $messages = [];
  150. }
  151. return $messages;
  152. }
  153. return null;
  154. }
  155. }