// Copyright 2014 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 "components/invalidation/push_client_channel.h" #include "base/stl_util.h" #include "components/invalidation/notifier_reason_util.h" #include "google/cacheinvalidation/client_gateway.pb.h" #include "google/cacheinvalidation/types.pb.h" #include "jingle/notifier/listener/push_client.h" namespace syncer { namespace { const char kBotJid[] = "tango@bot.talk.google.com"; const char kChannelName[] = "tango_raw"; } // namespace PushClientChannel::PushClientChannel( scoped_ptr<notifier::PushClient> push_client) : push_client_(push_client.Pass()), scheduling_hash_(0), sent_messages_count_(0) { push_client_->AddObserver(this); notifier::Subscription subscription; subscription.channel = kChannelName; subscription.from = ""; notifier::SubscriptionList subscriptions; subscriptions.push_back(subscription); push_client_->UpdateSubscriptions(subscriptions); } PushClientChannel::~PushClientChannel() { push_client_->RemoveObserver(this); } void PushClientChannel::UpdateCredentials( const std::string& email, const std::string& token) { push_client_->UpdateCredentials(email, token); } int PushClientChannel::GetInvalidationClientType() { #if defined(OS_IOS) return ipc::invalidation::ClientType::CHROME_SYNC_IOS; #else return ipc::invalidation::ClientType::CHROME_SYNC; #endif } void PushClientChannel::RequestDetailedStatus( base::Callback<void(const base::DictionaryValue&)> callback) { callback.Run(*CollectDebugData()); } void PushClientChannel::SendMessage(const std::string& message) { std::string encoded_message; EncodeMessage(&encoded_message, message, service_context_, scheduling_hash_); notifier::Recipient recipient; recipient.to = kBotJid; notifier::Notification notification; notification.channel = kChannelName; notification.recipients.push_back(recipient); notification.data = encoded_message; push_client_->SendNotification(notification); sent_messages_count_++; } void PushClientChannel::OnNotificationsEnabled() { NotifyNetworkStatusChange(true); NotifyChannelStateChange(INVALIDATIONS_ENABLED); } void PushClientChannel::OnNotificationsDisabled( notifier::NotificationsDisabledReason reason) { NotifyNetworkStatusChange(false); NotifyChannelStateChange(FromNotifierReason(reason)); } void PushClientChannel::OnIncomingNotification( const notifier::Notification& notification) { std::string message; std::string service_context; int64 scheduling_hash; if (!DecodeMessage( notification.data, &message, &service_context, &scheduling_hash)) { DLOG(ERROR) << "Could not parse ClientGatewayMessage"; return; } if (DeliverIncomingMessage(message)) { service_context_ = service_context; scheduling_hash_ = scheduling_hash; } } const std::string& PushClientChannel::GetServiceContextForTest() const { return service_context_; } int64 PushClientChannel::GetSchedulingHashForTest() const { return scheduling_hash_; } std::string PushClientChannel::EncodeMessageForTest( const std::string& message, const std::string& service_context, int64 scheduling_hash) { std::string encoded_message; EncodeMessage(&encoded_message, message, service_context, scheduling_hash); return encoded_message; } bool PushClientChannel::DecodeMessageForTest(const std::string& data, std::string* message, std::string* service_context, int64* scheduling_hash) { return DecodeMessage(data, message, service_context, scheduling_hash); } void PushClientChannel::EncodeMessage(std::string* encoded_message, const std::string& message, const std::string& service_context, int64 scheduling_hash) { ipc::invalidation::ClientGatewayMessage envelope; envelope.set_is_client_to_server(true); if (!service_context.empty()) { envelope.set_service_context(service_context); envelope.set_rpc_scheduling_hash(scheduling_hash); } envelope.set_network_message(message); envelope.SerializeToString(encoded_message); } bool PushClientChannel::DecodeMessage(const std::string& data, std::string* message, std::string* service_context, int64* scheduling_hash) { ipc::invalidation::ClientGatewayMessage envelope; if (!envelope.ParseFromString(data)) { return false; } *message = envelope.network_message(); if (envelope.has_service_context()) { *service_context = envelope.service_context(); } if (envelope.has_rpc_scheduling_hash()) { *scheduling_hash = envelope.rpc_scheduling_hash(); } return true; } scoped_ptr<base::DictionaryValue> PushClientChannel::CollectDebugData() const { scoped_ptr<base::DictionaryValue> status(new base::DictionaryValue); status->SetString("PushClientChannel.NetworkChannel", "Push Client"); status->SetInteger("PushClientChannel.SentMessages", sent_messages_count_); status->SetInteger("PushClientChannel.ReceivedMessages", SyncNetworkChannel::GetReceivedMessagesCount()); return status.Pass(); } } // namespace syncer