| // Copyright 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "cloud_print/gcp20/prototype/cloud_print_response_parser.h" |
| |
| #include "base/json/json_reader.h" |
| #include "base/json/json_writer.h" |
| #include "base/logging.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/values.h" |
| |
| namespace cloud_print_response_parser { |
| |
| Job::Job() { |
| } |
| |
| Job::~Job() { |
| } |
| |
| // Parses json base::Value to base::DictionaryValue. |
| // If |success| value in JSON dictionary is |false| then |message| will |
| // contain |message| from the JSON dictionary. |
| // Returns |true| if no parsing error occurred. |
| bool GetJsonDictinaryAndCheckSuccess(base::Value* json, |
| std::string* error_description, |
| bool* json_success, |
| std::string* message, |
| base::DictionaryValue** dictionary) { |
| base::DictionaryValue* response_dictionary = NULL; |
| if (!json || !json->GetAsDictionary(&response_dictionary)) { |
| *error_description = "No JSON dictionary response received."; |
| return false; |
| } |
| |
| bool response_json_success = false; |
| if (!response_dictionary->GetBoolean("success", &response_json_success)) { |
| *error_description = "Cannot parse success state."; |
| return false; |
| } |
| |
| if (!response_json_success) { |
| std::string response_message; |
| if (!response_dictionary->GetString("message", &response_message)) { |
| *error_description = "Cannot parse message from response."; |
| return false; |
| } |
| *message = response_message; |
| } |
| |
| *json_success = response_json_success; |
| *dictionary = response_dictionary; |
| return true; |
| } |
| |
| bool ParseRegisterStartResponse(const std::string& response, |
| std::string* error_description, |
| std::string* polling_url, |
| std::string* registration_token, |
| std::string* complete_invite_url, |
| std::string* device_id) { |
| scoped_ptr<base::Value> json(base::JSONReader::Read(response)); |
| base::DictionaryValue* response_dictionary = NULL; |
| bool json_success; |
| std::string message; |
| if (!GetJsonDictinaryAndCheckSuccess(json.get(), error_description, |
| &json_success, &message, |
| &response_dictionary)) { |
| return false; |
| } |
| if (!json_success) { |
| *error_description = message; |
| return false; |
| } |
| |
| std::string response_registration_token; |
| if (!response_dictionary->GetString("registration_token", |
| &response_registration_token)) { |
| *error_description = "No registration_token specified."; |
| return false; |
| } |
| |
| std::string response_complete_invite_url; |
| if (!response_dictionary->GetString("complete_invite_url", |
| &response_complete_invite_url)) { |
| *error_description = "No complete_invite_url specified."; |
| return false; |
| } |
| |
| std::string response_polling_url; |
| if (!response_dictionary->GetString("polling_url", &response_polling_url)) { |
| *error_description = "No polling_url specified."; |
| return false; |
| } |
| |
| base::ListValue* list = NULL; |
| if (!response_dictionary->GetList("printers", &list)) { |
| *error_description = "No printers list specified."; |
| return false; |
| } |
| |
| base::DictionaryValue* printer = NULL; |
| if (!list->GetDictionary(0, &printer)) { |
| *error_description = "Printers list is empty or printer is not dictionary."; |
| return false; |
| } |
| |
| std::string id; |
| if (!printer->GetString("id", &id)) { |
| *error_description = "No id specified."; |
| return false; |
| } |
| |
| if (id.empty()) { |
| *error_description = "id is empty."; |
| return false; |
| } |
| |
| *polling_url = response_polling_url; |
| *registration_token = response_registration_token; |
| *complete_invite_url = response_complete_invite_url; |
| *device_id = id; |
| return true; |
| } |
| |
| bool ParseRegisterCompleteResponse(const std::string& response, |
| std::string* error_description, |
| std::string* authorization_code, |
| std::string* xmpp_jid) { |
| scoped_ptr<base::Value> json(base::JSONReader::Read(response)); |
| base::DictionaryValue* response_dictionary = NULL; |
| bool json_success; |
| std::string message; |
| if (!GetJsonDictinaryAndCheckSuccess(json.get(), error_description, |
| &json_success, &message, |
| &response_dictionary)) { |
| return false; |
| } |
| if (!json_success) { |
| *error_description = message; |
| return false; |
| } |
| |
| std::string response_authorization_code; |
| if (!response_dictionary->GetString("authorization_code", |
| &response_authorization_code)) { |
| *error_description = "Cannot parse authorization_code."; |
| return false; |
| } |
| |
| std::string response_xmpp_jid; |
| if (!response_dictionary->GetString("xmpp_jid", &response_xmpp_jid)) { |
| *error_description = "Cannot parse xmpp jid."; |
| return false; |
| } |
| |
| *authorization_code = response_authorization_code; |
| *xmpp_jid = response_xmpp_jid; |
| return true; |
| } |
| |
| bool ParseFetchResponse(const std::string& response, |
| std::string* error_description, |
| std::vector<Job>* list) { |
| scoped_ptr<base::Value> json(base::JSONReader::Read(response)); |
| base::DictionaryValue* response_dictionary = NULL; |
| bool json_success; |
| std::string message; |
| if (!GetJsonDictinaryAndCheckSuccess(json.get(), error_description, |
| &json_success, &message, |
| &response_dictionary)) { |
| return false; |
| } |
| |
| if (!json_success) { // Let's suppose we have no jobs to proceed. |
| list->clear(); |
| return true; |
| } |
| |
| base::ListValue* jobs = NULL; |
| if (!response_dictionary->GetList("jobs", &jobs)) { |
| *error_description = "Cannot parse jobs list."; |
| return false; |
| } |
| |
| std::vector<Job> job_list(jobs->GetSize()); |
| std::string create_time_str; |
| for (size_t idx = 0; idx < job_list.size(); ++idx) { |
| base::DictionaryValue* job = NULL; |
| jobs->GetDictionary(idx, &job); |
| if (!job->GetString("id", &job_list[idx].job_id) || |
| !job->GetString("createTime", &create_time_str) || |
| !job->GetString("fileUrl", &job_list[idx].file_url) || |
| !job->GetString("ticketUrl", &job_list[idx].ticket_url) || |
| !job->GetString("title", &job_list[idx].title)) { |
| *error_description = "Cannot parse job info."; |
| return false; |
| } |
| int64 create_time_ms = 0; |
| if (!base::StringToInt64(create_time_str, &create_time_ms)) { |
| *error_description = "Cannot convert time."; |
| return false; |
| } |
| job_list[idx].create_time = |
| base::Time::UnixEpoch() + |
| base::TimeDelta::FromMilliseconds(create_time_ms); |
| } |
| |
| *list = job_list; |
| return true; |
| } |
| |
| bool ParseLocalSettingsResponse(const std::string& response, |
| std::string* error_description, |
| LocalSettings::State* state, |
| LocalSettings* settings) { |
| scoped_ptr<base::Value> json(base::JSONReader::Read(response)); |
| base::DictionaryValue* response_dictionary = NULL; |
| bool json_success; |
| std::string message; |
| if (!GetJsonDictinaryAndCheckSuccess(json.get(), error_description, |
| &json_success, &message, |
| &response_dictionary)) { |
| return false; |
| } |
| |
| if (!json_success) { // Let's suppose our printer was deleted. |
| *state = LocalSettings::PRINTER_DELETED; |
| return true; |
| } |
| |
| base::ListValue* list = NULL; |
| if (!response_dictionary->GetList("printers", &list)) { |
| *error_description = "No printers list specified."; |
| return false; |
| } |
| |
| base::DictionaryValue* printer = NULL; |
| if (!list->GetDictionary(0, &printer)) { |
| *error_description = "Printers list is empty or printer is not dictionary."; |
| return false; |
| } |
| |
| base::DictionaryValue* local_settings_dict = NULL; |
| if (!printer->GetDictionary("local_settings", &local_settings_dict)) { |
| *error_description = "No local_settings found."; |
| return false; |
| } |
| |
| base::DictionaryValue* current = NULL; |
| if (!local_settings_dict->GetDictionary("current", ¤t)) { |
| *error_description = "No *current* local settings found."; |
| return false; |
| } |
| |
| LocalSettings::State settings_state; |
| base::DictionaryValue* pending = NULL; |
| base::DictionaryValue* settings_to_parse = NULL; |
| if (local_settings_dict->GetDictionary("pending", &pending)) { |
| settings_to_parse = pending; |
| settings_state = LocalSettings::PENDING; |
| } else { |
| settings_to_parse = current; |
| settings_state = LocalSettings::CURRENT; |
| } |
| |
| LocalSettings local_settings; |
| if (!settings_to_parse->GetBoolean("local_discovery", |
| &local_settings.local_discovery) || |
| !settings_to_parse->GetBoolean("access_token_enabled", |
| &local_settings.access_token_enabled) || |
| !settings_to_parse->GetBoolean("printer/local_printing_enabled", |
| &local_settings.local_printing_enabled) || |
| !settings_to_parse->GetInteger("xmpp_timeout_value", |
| &local_settings.xmpp_timeout_value)) { |
| *error_description = "Cannot parse local_settings info."; |
| return false; |
| } |
| |
| *state = settings_state; |
| *settings = local_settings; |
| return true; |
| } |
| |
| } // namespace cloud_print_response_parser |
| |