Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
QueryMessageCollectionActionApi.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\MessageLoading;
5
6use ApiBase;
7use ApiPageSet;
8use ApiQuery;
9use ApiQueryGeneratorBase;
10use ApiResult;
15use MediaWiki\Languages\LanguageNameUtils;
18use Title;
19use Wikimedia\ParamValidator\ParamValidator;
20use Wikimedia\ParamValidator\TypeDef\EnumDef;
21use Wikimedia\ParamValidator\TypeDef\IntegerDef;
22use Wikimedia\Rdbms\ILoadBalancer;
23
30class QueryMessageCollectionActionApi extends ApiQueryGeneratorBase {
31 private ConfigHelper $configHelper;
32 private LanguageNameUtils $languageNameUtils;
33 private ILoadBalancer $loadBalancer;
34 private MessageGroupReviewStore $groupReviewStore;
35
36 public function __construct(
37 ApiQuery $query,
38 string $moduleName,
39 ConfigHelper $configHelper,
40 LanguageNameUtils $languageNameUtils,
41 ILoadBalancer $loadBalancer,
42 MessageGroupReviewStore $groupReviewStore
43 ) {
44 parent::__construct( $query, $moduleName, 'mc' );
45 $this->configHelper = $configHelper;
46 $this->languageNameUtils = $languageNameUtils;
47 $this->loadBalancer = $loadBalancer;
48 $this->groupReviewStore = $groupReviewStore;
49 }
50
51 public function execute(): void {
52 $this->run();
53 }
54
56 public function getCacheMode( $params ): string {
57 return 'public';
58 }
59
61 public function executeGenerator( $resultPageSet ): void {
62 $this->run( $resultPageSet );
63 }
64
65 private function validateLanguageCode( string $code ): void {
66 if ( !Utilities::isSupportedLanguageCode( $code ) ) {
67 $this->dieWithError( [ 'apierror-translate-invalidlanguage', $code ] );
68 }
69 }
70
71 private function run( ApiPageSet $resultPageSet = null ): void {
72 $params = $this->extractRequestParams();
73
74 $group = MessageGroups::getGroup( $params['group'] );
75 if ( !$group ) {
76 $this->dieWithError( [ 'apierror-badparameter', 'mcgroup' ] );
77 }
78
79 $languageCode = $params[ 'language' ];
80 $this->validateLanguageCode( $languageCode );
81 $sourceLanguageCode = $group->getSourceLanguage();
82
83 // Even though translation to source language maybe disabled, we still want to
84 // fetch the message collections for the source language.
85 if ( $sourceLanguageCode === $languageCode ) {
86 $name = $this->getLanguageName( $languageCode );
87 $this->addWarning( [ 'apiwarn-translate-language-disabled-source', wfEscapeWikiText( $name ) ] );
88 } else {
89 $languages = $group->getTranslatableLanguages();
90 if ( $languages === null ) {
91 $checks = [
92 $group->getId(),
93 strtok( $group->getId(), '-' ),
94 '*'
95 ];
96
97 $disabledLanguages = $this->configHelper->getDisabledTargetLanguages();
98 foreach ( $checks as $check ) {
99 if ( isset( $disabledLanguages[ $check ][ $languageCode ] ) ) {
100 $name = $this->getLanguageName( $languageCode );
101 $reason = $disabledLanguages[ $check ][ $languageCode ];
102 $this->dieWithError( [ 'apierror-translate-language-disabled-reason', $name, $reason ] );
103 }
104 }
105 } elseif ( !isset( $languages[ $languageCode ] ) ) {
106 // Not a translatable language
107 $name = $this->getLanguageName( $languageCode );
108 $this->dieWithError( [ 'apierror-translate-language-disabled', $name ] );
109 }
110
111 // A check for cases where the source language of group messages
112 // is a variant of the target language being translated into.
113 if ( strtok( $sourceLanguageCode, '-' ) === strtok( $languageCode, '-' ) ) {
114 $sourceLanguageName = $this->getLanguageName( $sourceLanguageCode );
115 $targetLanguageName = $this->getLanguageName( $languageCode );
116 $this->addWarning( [
117 'apiwarn-translate-language-targetlang-variant-of-source',
118 wfEscapeWikiText( $targetLanguageName ),
119 wfEscapeWikiText( $sourceLanguageName ) ]
120 );
121 }
122 }
123
124 if ( MessageGroups::isDynamic( $group ) ) {
126 // @phan-suppress-next-line PhanUndeclaredMethod
127 $group->setLanguage( $params['language'] );
128 }
129
130 $messages = $group->initCollection( $params['language'] );
131
132 foreach ( $params['filter'] as $filter ) {
133 if ( $filter === '' || $filter === null ) {
134 continue;
135 }
136
137 $value = null;
138 if ( str_contains( $filter, ':' ) ) {
139 [ $filter, $value ] = explode( ':', $filter, 2 );
140 }
141 /* The filtering params here are swapped wrt MessageCollection.
142 * There (fuzzy) means do not show fuzzy, which is the same as !fuzzy
143 * here and fuzzy here means (fuzzy, false) there. */
144 try {
145 $value = $value === null ? $value : (int)$value;
146 if ( str_starts_with( $filter, '!' ) ) {
147 $messages->filter( substr( $filter, 1 ), true, $value );
148 } else {
149 $messages->filter( $filter, false, $value );
150 }
151 } catch ( InvalidFilterException $e ) {
152 $this->dieWithError(
153 [ 'apierror-translate-invalidfilter', wfEscapeWikiText( $e->getMessage() ) ],
154 'invalidfilter'
155 );
156 }
157 }
158
159 $resultSize = count( $messages );
160 $offsets = $messages->slice( $params['offset'], $params['limit'] );
161 $batchSize = count( $messages );
162 [ /*$backwardsOffset*/, $forwardsOffset, $startOffset ] = $offsets;
163
164 $result = $this->getResult();
165 $result->addValue(
166 [ 'query', 'metadata' ],
167 'state',
168 $this->groupReviewStore->getWorkflowState( $group->getId(), $params['language'] )
169 );
170
171 $result->addValue( [ 'query', 'metadata' ], 'resultsize', $resultSize );
172 $result->addValue(
173 [ 'query', 'metadata' ],
174 'remaining',
175 $resultSize - $startOffset - $batchSize
176 );
177
178 $messages->loadTranslations();
179
180 $pages = [];
181
182 if ( $forwardsOffset !== false ) {
183 $this->setContinueEnumParameter( 'offset', $forwardsOffset );
184 }
185
186 $props = array_flip( $params['prop'] );
187
189 foreach ( $messages->keys() as $mkey => $titleValue ) {
190 $title = Title::newFromLinkTarget( $titleValue );
191
192 if ( $resultPageSet === null ) {
193 $data = $this->extractMessageData( $result, $props, $messages[$mkey] );
194 $data['title'] = $title->getPrefixedText();
195 $data['targetLanguage'] = $messages->getLanguage();
196
197 $handle = new MessageHandle( $title );
198
199 if ( $handle->isValid() ) {
200 $data['primaryGroup'] = $handle->getGroup()->getId();
201 }
202
203 $result->addValue( [ 'query', $this->getModuleName() ], null, $data );
204 } else {
205 $pages[] = $title;
206 }
207 }
208
209 if ( $resultPageSet === null ) {
210 $result->addIndexedTagName(
211 [ 'query', $this->getModuleName() ],
212 'message'
213 );
214 } else {
215 $resultPageSet->populateFromTitles( $pages );
216 }
217 }
218
219 private function getLanguageName( string $languageCode ): string {
220 return $this
221 ->languageNameUtils
222 ->getLanguageName( $languageCode, $this->getLanguage()->getCode() );
223 }
224
225 private function extractMessageData(
226 ApiResult $result,
227 array $props,
228 Message $message
229 ): array {
230 $data = [ 'key' => $message->key() ];
231
232 if ( isset( $props['definition'] ) ) {
233 $data['definition'] = $message->definition();
234 }
235 if ( isset( $props['translation'] ) ) {
236 // Remove !!FUZZY!! from translation if present.
237 $translation = $message->translation();
238 if ( $translation !== null ) {
239 $translation = str_replace( TRANSLATE_FUZZY, '', $translation );
240 }
241 $data['translation'] = $translation;
242 }
243 if ( isset( $props['tags'] ) ) {
244 $data['tags'] = $message->getTags();
245 $result->setIndexedTagName( $data['tags'], 'tag' );
246 }
247 // BC
248 if ( isset( $props['revision'] ) ) {
249 $data['revision'] = $message->getProperty( 'revision' );
250 }
251 if ( isset( $props['properties'] ) ) {
252 foreach ( $message->getPropertyNames() as $prop ) {
253 $data['properties'][$prop] = $message->getProperty( $prop );
254 ApiResult::setIndexedTagNameRecursive( $data['properties'], 'val' );
255 }
256 }
257
258 return $data;
259 }
260
262 protected function getAllowedParams(): array {
263 return [
264 'group' => [
265 ParamValidator::PARAM_TYPE => 'string',
266 ParamValidator::PARAM_REQUIRED => true,
267 ],
268 'language' => [
269 ParamValidator::PARAM_TYPE => 'string',
270 ParamValidator::PARAM_DEFAULT => 'en',
271 ],
272 'limit' => [
273 ParamValidator::PARAM_DEFAULT => 500,
274 ParamValidator::PARAM_TYPE => 'limit',
275 IntegerDef::PARAM_MIN => 1,
276 IntegerDef::PARAM_MAX => ApiBase::LIMIT_BIG2,
277 IntegerDef::PARAM_MAX2 => ApiBase::LIMIT_BIG2,
278 ],
279 'offset' => [
280 ParamValidator::PARAM_DEFAULT => '',
281 ParamValidator::PARAM_TYPE => 'string',
282 ApiBase::PARAM_HELP_MSG => 'api-help-param-continue',
283 ],
284 'filter' => [
285 ParamValidator::PARAM_TYPE => 'string',
286 ParamValidator::PARAM_DEFAULT => '!optional|!ignored',
287 ParamValidator::PARAM_ISMULTI => true,
288 ],
289 'prop' => [
290 ParamValidator::PARAM_TYPE => [
291 'definition',
292 'translation',
293 'tags',
294 'properties',
295 'revision',
296 ],
297 ParamValidator::PARAM_DEFAULT => 'definition|translation',
298 ParamValidator::PARAM_ISMULTI => true,
299 ApiBase::PARAM_HELP_MSG_PER_VALUE => [
300 'translation' => [ 'apihelp-query+messagecollection-paramvalue-prop-translation', TRANSLATE_FUZZY ],
301 ],
302 EnumDef::PARAM_DEPRECATED_VALUES => [
303 'revision' => true,
304 ],
305 ],
306 ];
307 }
308
310 protected function getExamplesMessages(): array {
311 return [
312 'action=query&meta=siteinfo&siprop=languages'
313 => 'apihelp-query+messagecollection-example-1',
314 'action=query&list=messagecollection&mcgroup=page-Example'
315 => 'apihelp-query+messagecollection-example-2',
316 'action=query&list=messagecollection&mcgroup=page-Example&mclanguage=fi&' .
317 'mcprop=definition|translation|tags&mcfilter=optional'
318 => 'apihelp-query+messagecollection-example-3',
319 'action=query&generator=messagecollection&gmcgroup=page-Example&gmclanguage=nl&prop=revisions'
320 => 'apihelp-query+messagecollection-example-4',
321 ];
322 }
323}
return[ 'Translate:ConfigHelper'=> static function():ConfigHelper { return new ConfigHelper();}, 'Translate:CsvTranslationImporter'=> static function(MediaWikiServices $services):CsvTranslationImporter { return new CsvTranslationImporter( $services->getWikiPageFactory());}, 'Translate:EntitySearch'=> static function(MediaWikiServices $services):EntitySearch { return new EntitySearch($services->getMainWANObjectCache(), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), MessageGroups::singleton(), $services->getNamespaceInfo(), $services->get( 'Translate:MessageIndex'), $services->getTitleParser(), $services->getTitleFormatter());}, 'Translate:ExternalMessageSourceStateImporter'=> static function(MediaWikiServices $services):ExternalMessageSourceStateImporter { return new ExternalMessageSourceStateImporter($services->getMainConfig(), $services->get( 'Translate:GroupSynchronizationCache'), $services->getJobQueueGroup(), LoggerFactory::getInstance( 'Translate.GroupSynchronization'), $services->get( 'Translate:MessageIndex'));}, 'Translate:FileFormatFactory'=> static function(MediaWikiServices $services):FileFormatFactory { return new FileFormatFactory( $services->getObjectFactory());}, 'Translate:GroupSynchronizationCache'=> static function(MediaWikiServices $services):GroupSynchronizationCache { return new GroupSynchronizationCache( $services->get( 'Translate:PersistentCache'));}, 'Translate:HookRunner'=> static function(MediaWikiServices $services):HookRunner { return new HookRunner( $services->getHookContainer());}, 'Translate:MessageBundleStore'=> static function(MediaWikiServices $services):MessageBundleStore { return new MessageBundleStore($services->get( 'Translate:RevTagStore'), $services->getJobQueueGroup(), $services->getLanguageNameUtils(), $services->get( 'Translate:MessageIndex'));}, 'Translate:MessageGroupReviewStore'=> static function(MediaWikiServices $services):MessageGroupReviewStore { return new MessageGroupReviewStore($services->getDBLoadBalancer(), $services->get( 'Translate:HookRunner'));}, 'Translate:MessageGroupStatsTableFactory'=> static function(MediaWikiServices $services):MessageGroupStatsTableFactory { return new MessageGroupStatsTableFactory($services->get( 'Translate:ProgressStatsTableFactory'), $services->getDBLoadBalancer(), $services->getLinkRenderer(), $services->get( 'Translate:MessageGroupReviewStore'), $services->getMainConfig() ->get( 'TranslateWorkflowStates') !==false);}, 'Translate:MessageIndex'=> static function(MediaWikiServices $services):MessageIndex { $params=$services->getMainConfig() ->get( 'TranslateMessageIndex');if(is_string( $params)) { $params=(array) $params;} $class=array_shift( $params);return new $class( $params);}, 'Translate:MessagePrefixStats'=> static function(MediaWikiServices $services):MessagePrefixStats { return new MessagePrefixStats( $services->getTitleParser());}, 'Translate:ParsingPlaceholderFactory'=> static function():ParsingPlaceholderFactory { return new ParsingPlaceholderFactory();}, 'Translate:PersistentCache'=> static function(MediaWikiServices $services):PersistentCache { return new PersistentDatabaseCache($services->getDBLoadBalancer(), $services->getJsonCodec());}, 'Translate:ProgressStatsTableFactory'=> static function(MediaWikiServices $services):ProgressStatsTableFactory { return new ProgressStatsTableFactory($services->getLinkRenderer(), $services->get( 'Translate:ConfigHelper'));}, 'Translate:RevTagStore'=> static function(MediaWikiServices $services):RevTagStore { return new RevTagStore($services->getDBLoadBalancerFactory());}, 'Translate:SubpageListBuilder'=> static function(MediaWikiServices $services):SubpageListBuilder { return new SubpageListBuilder($services->get( 'Translate:TranslatableBundleFactory'), $services->getLinkBatchFactory());}, 'Translate:TranslatableBundleExporter'=> static function(MediaWikiServices $services):TranslatableBundleExporter { return new TranslatableBundleExporter($services->get( 'Translate:SubpageListBuilder'), $services->getWikiExporterFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslatableBundleFactory'=> static function(MediaWikiServices $services):TranslatableBundleFactory { return new TranslatableBundleFactory($services->get( 'Translate:TranslatablePageStore'), $services->get( 'Translate:MessageBundleStore'));}, 'Translate:TranslatableBundleImporter'=> static function(MediaWikiServices $services):TranslatableBundleImporter { return new TranslatableBundleImporter($services->getWikiImporterFactory(), $services->get( 'Translate:TranslatablePageParser'), $services->getRevisionLookup());}, 'Translate:TranslatableBundleMover'=> static function(MediaWikiServices $services):TranslatableBundleMover { return new TranslatableBundleMover($services->getMovePageFactory(), $services->getJobQueueGroup(), $services->getLinkBatchFactory(), $services->get( 'Translate:TranslatableBundleFactory'), $services->get( 'Translate:SubpageListBuilder'), $services->getMainConfig() ->get( 'TranslatePageMoveLimit'));}, 'Translate:TranslatableBundleStatusStore'=> static function(MediaWikiServices $services):TranslatableBundleStatusStore { return new TranslatableBundleStatusStore($services->getDBLoadBalancer() ->getConnection(DB_PRIMARY), $services->getCollationFactory() ->makeCollation( 'uca-default-u-kn'), $services->getDBLoadBalancer() ->getMaintenanceConnectionRef(DB_PRIMARY));}, 'Translate:TranslatablePageParser'=> static function(MediaWikiServices $services):TranslatablePageParser { return new TranslatablePageParser($services->get( 'Translate:ParsingPlaceholderFactory'));}, 'Translate:TranslatablePageStore'=> static function(MediaWikiServices $services):TranslatablePageStore { return new TranslatablePageStore($services->get( 'Translate:MessageIndex'), $services->getJobQueueGroup(), $services->get( 'Translate:RevTagStore'), $services->getDBLoadBalancer(), $services->get( 'Translate:TranslatableBundleStatusStore'), $services->get( 'Translate:TranslatablePageParser'),);}, 'Translate:TranslationStashReader'=> static function(MediaWikiServices $services):TranslationStashReader { $db=$services->getDBLoadBalancer() ->getConnection(DB_REPLICA);return new TranslationStashStorage( $db);}, 'Translate:TranslationStatsDataProvider'=> static function(MediaWikiServices $services):TranslationStatsDataProvider { return new TranslationStatsDataProvider(new ServiceOptions(TranslationStatsDataProvider::CONSTRUCTOR_OPTIONS, $services->getMainConfig()), $services->getObjectFactory(), $services->getDBLoadBalancer());}, 'Translate:TranslationUnitStoreFactory'=> static function(MediaWikiServices $services):TranslationUnitStoreFactory { return new TranslationUnitStoreFactory( $services->getDBLoadBalancer());}, 'Translate:TranslatorActivity'=> static function(MediaWikiServices $services):TranslatorActivity { $query=new TranslatorActivityQuery($services->getMainConfig(), $services->getDBLoadBalancer());return new TranslatorActivity($services->getMainObjectStash(), $query, $services->getJobQueueGroup());}, 'Translate:TtmServerFactory'=> static function(MediaWikiServices $services):TtmServerFactory { $config=$services->getMainConfig();$default=$config->get( 'TranslateTranslationDefaultService');if( $default===false) { $default=null;} return new TtmServerFactory( $config->get( 'TranslateTranslationServices'), $default);}]
@phpcs-require-sorted-array
Provides methods to get and change the state of a message group.
Factory class for accessing message groups individually by id or all of them as a list.
A helper class added to work with configuration values of the Translate Extension.
Essentially random collection of helper functions, similar to GlobalFunctions.php.
Definition Utilities.php:31
Class for pointing to messages, like Title class is for titles.