Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
QueryManageMessageGroupsActionApi.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\MessageGroupProcessing;
5
6use ApiQuery;
7use ApiQueryBase;
12use Title;
13use Wikimedia\ParamValidator\ParamValidator;
14
22class QueryManageMessageGroupsActionApi extends ApiQueryBase {
23 private const RIGHT = 'translate-manage';
24
25 public function __construct( ApiQuery $query, string $moduleName ) {
26 parent::__construct( $query, $moduleName, 'mmg' );
27 }
28
29 public function execute() {
30 $params = $this->extractRequestParams();
31 $groupId = $params['groupId'];
32 $msgKey = $params['messageKey'];
33 $name = $params['changesetName'] ?? MessageChangeStorage::DEFAULT_NAME;
34
35 $user = $this->getUser();
36 $allowed = $user->isAllowed( self::RIGHT );
37
38 if ( !$allowed ) {
39 $this->dieWithError( 'apierror-permissiondenied-generic', 'permissiondenied' );
40 }
41
42 $group = MessageGroups::getGroup( $groupId );
43 if ( !$group ) {
44 $this->dieWithError( 'apierror-translate-invalidgroup', 'invalidgroup' );
45 }
46
47 if ( !MessageChangeStorage::isValidCdbName( $name ) ) {
48 $this->dieWithError(
49 [ 'apierror-translate-invalid-changeset-name', wfEscapeWikiText( $name ) ],
50 'invalidchangeset'
51 );
52 }
53 $cdbPath = MessageChangeStorage::getCdbPath( $name );
54
55 $sourceChanges = MessageChangeStorage::getGroupChanges( $cdbPath, $groupId );
56
57 if ( $sourceChanges->getAllModifications() === [] ) {
58 $this->dieWithError( [ 'apierror-translate-smg-nochanges' ] );
59 }
60
61 $messages = $this->getPossibleRenames(
62 $sourceChanges, $group->getNamespace(), $msgKey, $group->getSourceLanguage()
63 );
64
65 $result = $this->getResult();
66 $result->addValue( [ 'query', $this->getModuleName() ], null, $messages );
67 }
68
70 protected function getPossibleRenames(
71 MessageSourceChange $sourceChanges,
72 int $groupNamespace,
73 string $msgKey,
74 string $languageCode
75 ): array {
76 $deletions = $sourceChanges->getDeletions( $languageCode );
77 $targetMsg = $sourceChanges->findMessage(
78 $languageCode, $msgKey, [ MessageSourceChange::ADDITION, MessageSourceChange::RENAME ]
79 );
80 $stringComparator = new SimpleStringComparator();
81 $renameList = [];
82
83 // compare deleted messages with the target message and get the similarity.
84 foreach ( $deletions as $deletion ) {
85 if ( $deletion['content'] === null ) {
86 continue;
87 }
88
89 $similarity = $stringComparator->getSimilarity(
90 $deletion['content'],
91 // @phan-suppress-next-line PhanTypeArraySuspiciousNullable
92 $targetMsg['content']
93 );
94
95 $title = Title::makeTitle(
96 $groupNamespace,
97 Utilities::title( $deletion['key'], $languageCode, $groupNamespace )
98 );
99
100 $renameList[] = [
101 'key' => $deletion['key'],
102 'content' => $deletion['content'],
103 'similarity' => $similarity,
104 'link' => $title->getFullURL(),
105 'title' => $title->getPrefixedText()
106 ];
107 }
108
109 // sort them based on similarity
110 usort( $renameList, static function ( $a, $b ) {
111 return -( $a['similarity'] <=> $b['similarity'] );
112 } );
113
114 return $renameList;
115 }
116
117 protected function getAllowedParams(): array {
118 $params = parent::getAllowedParams();
119 $params['groupId'] = [
120 ParamValidator::PARAM_TYPE => 'string',
121 ParamValidator::PARAM_REQUIRED => true,
122 ];
123
124 $params['messageKey'] = [
125 ParamValidator::PARAM_TYPE => 'string',
126 ParamValidator::PARAM_REQUIRED => true,
127 ];
128
129 $params['changesetName'] = [
130 ParamValidator::PARAM_TYPE => 'string',
131 ParamValidator::PARAM_DEFAULT => MessageChangeStorage::DEFAULT_NAME
132 ];
133
134 return $params;
135 }
136
137 protected function getExamplesMessages(): array {
138 return [
139 'action=query&meta=managemessagegroup&mmggroupId=hello
140 &mmgchangesetName=default&mmgmessageKey=world' => 'apihelp-query+managemessagegroups-example-1',
141 ];
142 }
143}
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
static getGroup(string $id)
Fetch a message group by id.
getPossibleRenames(MessageSourceChange $sourceChanges, int $groupNamespace, string $msgKey, string $languageCode)
Fetches the messages that can be used as possible renames for a given message.
Class is used to track the changes made when importing messages from the remote sources using importE...
findMessage( $language, $key, $possibleStates=[], &$modificationType=null)
Finds a message with the given key across different types of modifications.
A simple string comparator, that compares two strings and determines if they are an exact match.
Essentially random collection of helper functions, similar to GlobalFunctions.php.
Definition Utilities.php:31