1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
<?php
/**
* Contains a class for querying external translation service.
*
* @file
* @author Carsten Schmitz / LimeSurvey GmbH
* @license GPL-2.0-or-later
*/
/**
* Implements support for Google Translate API
* @see https://cloud.google.com/translate/docs/reference/rest
* @ingroup TranslationWebService
* @since 2020.05
*/
class GoogleTranslateWebService extends TranslationWebService {
private const PUBLIC_API = 'https://translation.googleapis.com/language/translate/v2';
/** @inheritDoc */
public function getType() {
return 'mt';
}
/** @inheritDoc */
protected function mapCode( $code ) {
/** @phpcs-require-sorted-array */
$map = [
'be-tarask' => 'be',
'zh-cn' => 'zh-CN',
'zh-hans' => 'zh-CN',
'zh-hant' => 'zh-TW',
'zh-tw' => 'zh-TW',
];
return $map[$code] ?? $code;
}
/** @inheritDoc */
public function isSupportedLanguagePair( $from, $to ) {
$pairs = $this->getSupportedLanguagePairs();
$from = $this->mapCode( $from );
$to = $this->mapCode( $to );
// As long as the source & target language exist at Google it is fine
return isset( $pairs[$from] ) && isset( $pairs[$to] ) && $from !== $to;
}
/** @inheritDoc */
protected function doPairs() {
if ( !isset( $this->config['key'] ) ) {
throw new TranslationWebServiceConfigurationException( 'API key is not set' );
}
$api = $this->config['pairs'] ?? self::PUBLIC_API . '/languages';
$params = [
'key' => $this->config['key'],
];
$json = Http::get(
wfAppendQuery( $api, wfArrayToCgi( $params ) ),
[ 'timeout' => $this->config['timeout'] ],
__METHOD__
);
$response = FormatJson::decode( $json );
if ( !is_object( $response ) ) {
$exception = 'Malformed reply from remote server: ' . (string)$json;
throw new TranslationWebServiceException( $exception );
}
$pairs = [];
foreach ( $response->data->languages as $language ) {
// Google can translate from any language to any language
$pairs[$language->language] = true;
}
return $pairs;
}
/** @inheritDoc */
protected function getQuery( $text, $from, $to ) {
if ( !isset( $this->config['key'] ) ) {
throw new TranslationWebServiceConfigurationException( 'API key is not set' );
}
# https://cloud.google.com/translate/docs/reference/translate
if ( strlen( $text ) > 10000 ) {
// There is no limitation but we don't want the translation service to be abused, don't we?
throw new TranslationWebServiceInvalidInputException( 'Source text too long' );
}
$url = $this->config['url'] ?? self::PUBLIC_API;
$text = trim( $text );
$text = $this->wrapUntranslatable( $text );
return TranslationQuery::factory( $url )
->timeout( $this->config['timeout'] ?? 3 )
->postWithData( wfArrayToCgi( [
'key' => $this->config['key'],
'q' => $text,
'target' => $to,
'source' => $from,
'format' => 'html',
] ) );
}
/** @inheritDoc */
protected function parseResponse( TranslationQueryResponse $reply ) {
$body = $reply->getBody();
$response = FormatJson::decode( $body );
if ( !is_object( $response ) ) {
throw new TranslationWebServiceException( 'Invalid json: ' . serialize( $body ) );
}
$text = Sanitizer::decodeCharReferences( $response->data->translations[0]->translatedText );
$text = $this->unwrapUntranslatable( $text );
return trim( $text );
}
}
|