Translate extension for MediaWiki
 
Loading...
Searching...
No Matches
GoogleTranslateWebService.php
1<?php
2declare( strict_types = 1 );
3
4namespace MediaWiki\Extension\Translate\WebService;
5
6use FormatJson;
7use MediaWiki\Http\HttpRequestFactory;
8use Sanitizer;
9
19 private const PUBLIC_API = 'https://translation.googleapis.com/language/translate/v2';
20 private HttpRequestFactory $httpRequestFactory;
21
22 public function __construct(
23 HttpRequestFactory $httpRequestFactory,
24 string $serviceName,
25 array $config
26 ) {
27 parent::__construct( $serviceName, $config );
28 $this->httpRequestFactory = $httpRequestFactory;
29 }
30
32 public function getType(): string {
33 return 'mt';
34 }
35
37 protected function mapCode( string $code ): string {
39 $map = [
40 'be-tarask' => 'be',
41 'nb' => 'no',
42 'tw' => 'ak',
43 'zh-cn' => 'zh-CN',
44 'zh-hans' => 'zh-CN',
45 'zh-hant' => 'zh-TW',
46 'zh-tw' => 'zh-TW',
47 ];
48
49 return $map[$code] ?? $code;
50 }
51
53 public function isSupportedLanguagePair( string $sourceLanguage, string $targetLanguage ): bool {
54 $pairs = $this->getSupportedLanguagePairs();
55 $from = $this->mapCode( $sourceLanguage );
56 $to = $this->mapCode( $targetLanguage );
57
58 // As long as the source & target language exist at Google it is fine
59 return isset( $pairs[$from] ) && isset( $pairs[$to] ) && $from !== $to;
60 }
61
63 protected function doPairs(): array {
64 if ( !isset( $this->config['key'] ) ) {
65 throw new TranslationWebServiceConfigurationException( 'API key is not set' );
66 }
67
68 $api = $this->config['pairs'] ?? self::PUBLIC_API . '/languages';
69 $json = $this->httpRequestFactory->get(
70 wfAppendQuery( $api, [ 'key' => $this->config['key'], ] ),
71 [ 'timeout' => $this->config['timeout'] ],
72 __METHOD__
73 );
74 if ( $json === null ) {
75 throw new TranslationWebServiceException( 'Failure encountered when contacting remote server' );
76 }
77
78 $response = FormatJson::decode( $json );
79 if ( !is_object( $response ) ) {
80 throw new TranslationWebServiceException( 'Malformed reply from remote server: ' . $json );
81 }
82
83 $pairs = [];
84 foreach ( $response->data->languages as $language ) {
85 // Google can translate from any language to any language
86 $pairs[$language->language] = true;
87 }
88
89 return $pairs;
90 }
91
93 protected function getQuery( string $text, string $sourceLanguage, string $targetLanguage ): TranslationQuery {
94 if ( !isset( $this->config['key'] ) ) {
95 throw new TranslationWebServiceConfigurationException( 'API key is not set' );
96 }
97 # https://cloud.google.com/translate/docs/reference/translate
98 if ( strlen( $text ) > 10000 ) {
99 // There is no limitation but we don't want the translation service to be abused, don't we?
100 throw new TranslationWebServiceInvalidInputException( 'Source text too long' );
101 }
102
103 $url = $this->config['url'] ?? self::PUBLIC_API;
104 $text = trim( $text );
105 $text = $this->wrapUntranslatable( $text );
106
107 return TranslationQuery::factory( $url )
108 ->timeout( intval( $this->config['timeout'] ?? 3 ) )
109 ->postWithData( wfArrayToCgi( [
110 'key' => $this->config['key'],
111 'q' => $text,
112 'target' => $targetLanguage,
113 'source' => $sourceLanguage,
114 'format' => 'html',
115 ] ) );
116 }
117
119 protected function parseResponse( TranslationQueryResponse $response ): string {
120 $body = $response->getBody();
121 $responseBody = FormatJson::decode( $body );
122 if ( !is_object( $responseBody ) ) {
123 throw new TranslationWebServiceException( 'Invalid json: ' . serialize( $body ) );
124 }
125 $text = Sanitizer::decodeCharReferences( $responseBody->data->translations[0]->translatedText );
126 $text = $this->unwrapUntranslatable( $text );
127
128 return trim( $text );
129 }
130}
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
isSupportedLanguagePair(string $sourceLanguage, string $targetLanguage)
@inheritDoc
getQuery(string $text, string $sourceLanguage, string $targetLanguage)
@inheritDoc
Value object that represents a HTTP(S) query response.
Mutable objects that represents an HTTP(S) query.
Used to signal that the requested input is rejected and cannot be used with an external web service.