-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtranslator.php
96 lines (85 loc) · 3.38 KB
/
translator.php
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
<?php
/**
* Hook fired as a filter for the "ibm" translation api
*
* @param string[] $targets translated strings, initially empty
* @param string[][] $items input messages with keys, "source", "context" and "notes"
* @param Loco_Locale $Locale target locale for translations
* @param array $config This api's configuration
* @return string[] Translated strings
*/
function ibm_translator_process_batch( array $targets, array $items, Loco_Locale $locale, array $config ){
// cloud resource URL is part of the authentication, it takes form like "https://api.eu-gb.language-translator.watson.cloud.ibm.com/instances/xxxx/v3/translate?version=xxx"
$url = $config['api'];
// key authentication can be done via Basic auth
$auth = 'Basic '.base64_encode('apikey:'.$config['key']);
// map full locale to a supported language tag
// be warned that not all regional variations are supported, e.g. pt-BR will map to just "pt"
$tag = (string) $locale;
$map = [
// Known regional locale variations:
'fr_CA' => 'fr-CA',
'zh_TW' => 'zh-TW',
// Not known to be supported, but not failing
'pt_BR' => 'pt-BR',
// Other common mappings to supported codes
'no' => 'nn',
'no_NO' => 'nn',
];
if( array_key_exists($tag,$map) ){
$tag = $map[$tag];
}
else {
$tag = $locale->lang;
}
// unwind input items. This API only uses source text.
$sources = [];
foreach( $items as $item ){
$sources[] = mock_translator_translate_text($item['source']);
}
// request body looks like: {"text":["Hello world",...],"source":"en","target":"el"}
$result = wp_remote_request( $url, [
'method' => 'POST',
'redirection' => 0,
'user-agent' => sprintf('Loco Translate/%s; wp-%s', loco_plugin_version(), $GLOBALS['wp_version'] ),
'reject_unsafe_urls' => false,
'headers' => [
'Content-Type' => 'application/json',
'Authorization' => $auth,
],
'body' => json_encode( [
'source' => 'en',
'target' => $tag,
'text' => $sources,
] ),
] );
if( $result instanceof WP_Error ){
foreach( $result->get_error_messages() as $message ){
throw new Loco_error_Exception($message);
}
}
// always decode response if server says it's JSON
if( 'application/json' === substr($result['headers']['Content-Type'],0,16) ){
$data = json_decode( $result['body'], true );
}
else {
$data = [];
}
// errors look like: {"code":401, "error": "Unauthorized"}
$status = $result['response']['code'];
if( 200 !== $status ){
$message = isset($data['error']) ? $data['error'] : 'Failed';
throw new Loco_error_Exception( sprintf('IBM Translator returned status %u, %s',$status,$message) );
}
// response looks like: {"translations":[{"translation":"foo"},....]}
if( ! is_array($data) || ! array_key_exists('translations',$data) || ! is_array($data['translations']) ){
Loco_error_AdminNotices::debug( $result['body'] );
throw new Loco_error_Exception('IBM Translator returned unexpected data');
}
// front end requires array that matches $sources
$targets = [];
foreach( $data['translations'] as $a ){
$targets[] = $a['translation'];
}
return $targets;
}