From 222f6e2e23a878b5c598fa189a8c385fe6f0ebf3 Mon Sep 17 00:00:00 2001 From: Andy Staudacher Date: Wed, 18 Mar 2009 00:53:44 +0000 Subject: Functional l10n_client / server interaction: - Get / verify API Key from l10n server - Submit translations - Fetch translations / updates Reference: Tasks: 75, 76, 55 TODO: Move out of core (and a series of other tasks). --- core/helpers/l10n_client.php | 209 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 209 insertions(+) create mode 100644 core/helpers/l10n_client.php (limited to 'core/helpers/l10n_client.php') diff --git a/core/helpers/l10n_client.php b/core/helpers/l10n_client.php new file mode 100644 index 00000000..da841116 --- /dev/null +++ b/core/helpers/l10n_client.php @@ -0,0 +1,209 @@ + $version, + "client_token" => self::client_token(), + "signature" => $signature, + "uid" => self::server_uid($api_key))); + if (!remote::success($response_status)) { + return false; + } + return true; + } + + static function fetch_updates() { + $request->locales = array(); + $request->messages = new stdClass(); + + $locales = locale::installed(); + foreach ($locales as $locale => $locale_data) { + $request->locales[] = $locale; + } + + // TODO: Batch requests (max request size) + + foreach (Database::instance() + ->select("key", "locale", "revision", "translation") + ->from("incoming_translations") + ->get() + ->as_array() as $row) { + if (!isset($request->messages->{$row->key})) { + $request->messages->{$row->key} = 1; + } + if (!empty($row->revision) && !empty($row->translation)) { + if (!is_object($request->messages->{$row->key})) { + $request->messages->{$row->key} = new stdClass(); + } + $request->messages->{$row->key}->{$row->locale} = $row->revision; + } + } + // TODO: Include messages from outgoing_translations? + + $request_data = json_encode($request); + $url = self::server_url() . "?q=translations/fetch"; + list ($response_data, $response_status) = remote::post($url, array("data" => $request_data)); + if (!remote::success($response_status)) { + throw new Exception("Translations fetch request failed with: " . $response_status); + } + if (empty($response_data)) { + log::info(t("translations"), t("Translations fetch request resulted in an emptyu response.")); + return; + } + + $response = json_decode($response_data); + + /* + * Response format (JSON payload): + * [{key:, translation: , rev:, locale:}, + * {key:, ...} + * ] + */ + log::info(t("translations"), t2("Installed 1 new / updated translation message.", + "Installed %count new / updated translation messages.", + count($response))); + + foreach ($response as $message_data) { + // TODO: Better input validation + if (empty($message_data->key) || empty($message_data->translation) || + empty($message_data->locale) || empty($message_data->rev)) { + throw new Exception("Translations fetch request resulted in Invalid response data"); + } + $key = $message_data->key; + $locale = $message_data->locale; + $revision = $message_data->rev; + $translation = serialize(json_decode($message_data->translation)); + // TODO: Should we normalize the incoming_translations table into messages(id, key, message) + // and incoming_translations(id, translation, locale, revision)? + // Or just allow incoming_translations.message to be NULL? + $locale = $message_data->locale; + $entry = ORM::factory("incoming_translation") + ->where(array("key" => $key, "locale" => $locale)) + ->find(); + if (!$entry->loaded) { + // TODO: Load a message key -> message (text) dict into memory outside of this loop + $root_entry = ORM::factory("incoming_translation") + ->where(array("key" => $key, "locale" => "root")) + ->find(); + $entry->key = $key; + $entry->message = $root_entry->message; + $entry->locale = $locale; + } + $entry->revision = $revision; + $entry->translation = $translation; + $entry->save(); + } + } + + static function submit_translations() { + /* + * Request format (HTTP POST): + * client_token = + * uid = + * signature = md5(user_api_key($uid, $client_token) . $data . $client_token)) + * data = // JSON payload + * + * {: {message: + * translations: {: , + * : ...}}, + * : {...} + * } + */ + + // TODO: Batch requests (max request size) + // TODO: include base_revision in submission / how to handle resubmissions / edit fights? + foreach (Database::instance() + ->select("key", "message", "locale", "base_revision", "translation") + ->from("outgoing_translations") + ->get() + ->as_array() as $row) { + $key = $row->key; + if (!isset($request->{$key})) { + $request->{$key}->message = json_encode(unserialize($row->message)); + } + $request->{$key}->translations->{$row->locale} = json_encode(unserialize($row->translation)); + } + + // TODO: reduce memory consumpotion, e.g. free $request + $request_data = json_encode($request); + $url = self::server_url() . "?q=translations/submit"; + $signature = self::_sign($request_data); + + list ($response_data, $response_status) = remote::post($url, array("data" => $request_data, + "client_token" => self::client_token(), + "signature" => $signature, + "uid" => self::server_uid())); + + if (!remote::success($response_status)) { + throw new Exception("Translations submission failed with: " . $response_status); + } + if (empty($response_data)) { + return; + } + + $response = json_decode($response_data); + /* + * Response format (JSON payload): + * [{key:, locale:, rev:, status:}, + * {key:, ...} + * ] + * + */ + // TODO: Move messages out of outgoing into incoming, using new rev? + // TODO: show which messages have been rejected / are pending? + } +} \ No newline at end of file -- cgit v1.2.3