Home | History | Annotate | Download | only in gcm
      1 // Copyright 2013 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "chrome/browser/extensions/api/gcm/gcm_api.h"
      6 
      7 #include <algorithm>
      8 #include <map>
      9 #include <vector>
     10 
     11 #include "base/sha1.h"
     12 #include "base/strings/string_number_conversions.h"
     13 #include "chrome/browser/extensions/extension_system.h"
     14 #include "chrome/browser/profiles/profile.h"
     15 #include "chrome/browser/profiles/profile.h"
     16 #include "chrome/browser/services/gcm/gcm_profile_service.h"
     17 #include "chrome/browser/services/gcm/gcm_profile_service_factory.h"
     18 #include "chrome/common/extensions/api/gcm.h"
     19 #include "extensions/browser/event_router.h"
     20 #include "extensions/common/extension.h"
     21 
     22 namespace {
     23 
     24 const size_t kMaximumMessageSize = 4096;  // in bytes.
     25 const char kGoogDotRestrictedPrefix[] = "goog.";
     26 const size_t kGoogDotPrefixLength = arraysize(kGoogDotRestrictedPrefix) - 1;
     27 const char kGoogleRestrictedPrefix[] = "google";
     28 const size_t kGooglePrefixLength = arraysize(kGoogleRestrictedPrefix) - 1;
     29 
     30 // Error messages.
     31 const char kInvalidParameter[] =
     32     "Function was called with invalid parameters.";
     33 const char kAsyncOperationPending[] =
     34     "Asynchronous operation is pending.";
     35 const char kNetworkError[] = "Network error occured.";
     36 const char kServerError[] = "Server error occured.";
     37 const char kTtlExceeded[] = "Time-to-live exceeded.";
     38 const char kUnknownError[] = "Unknown error occured.";
     39 
     40 std::string SHA1HashHexString(const std::string& str) {
     41   std::string hash = base::SHA1HashString(str);
     42   return base::HexEncode(hash.data(), hash.size());
     43 }
     44 
     45 const char* GcmResultToError(gcm::GCMClient::Result result) {
     46   switch (result) {
     47     case gcm::GCMClient::SUCCESS:
     48       return "";
     49     case gcm::GCMClient::INVALID_PARAMETER:
     50       return kInvalidParameter;
     51     case gcm::GCMClient::ASYNC_OPERATION_PENDING:
     52       return kAsyncOperationPending;
     53     case gcm::GCMClient::NETWORK_ERROR:
     54       return kNetworkError;
     55     case gcm::GCMClient::SERVER_ERROR:
     56       return kServerError;
     57     case gcm::GCMClient::TTL_EXCEEDED:
     58       return kTtlExceeded;
     59     case gcm::GCMClient::UNKNOWN_ERROR:
     60       return kUnknownError;
     61     default:
     62       NOTREACHED() << "Unexpected value of result cannot be converted: "
     63                    << result;
     64   }
     65 
     66   // Never reached, but prevents missing return statement warning.
     67   return "";
     68 }
     69 
     70 bool IsMessageKeyValid(const std::string& key) {
     71   return !key.empty() &&
     72       key.compare(0, kGooglePrefixLength, kGoogleRestrictedPrefix) != 0 &&
     73       key.compare(0, kGoogDotPrefixLength, kGoogDotRestrictedPrefix) != 0;
     74 }
     75 
     76 }  // namespace
     77 
     78 namespace extensions {
     79 
     80 bool GcmApiFunction::RunImpl() {
     81   if (!IsGcmApiEnabled())
     82     return false;
     83 
     84   return DoWork();
     85 }
     86 
     87 bool GcmApiFunction::IsGcmApiEnabled() const {
     88   return gcm::GCMProfileService::IsGCMEnabled() &&
     89       !GetExtension()->public_key().empty();
     90 }
     91 
     92 gcm::GCMProfileService* GcmApiFunction::GCMProfileService() const {
     93   return gcm::GCMProfileServiceFactory::GetForProfile(
     94       Profile::FromBrowserContext(context()));
     95 }
     96 
     97 GcmRegisterFunction::GcmRegisterFunction() {}
     98 
     99 GcmRegisterFunction::~GcmRegisterFunction() {}
    100 
    101 bool GcmRegisterFunction::DoWork() {
    102   scoped_ptr<api::gcm::Register::Params> params(
    103       api::gcm::Register::Params::Create(*args_));
    104   EXTENSION_FUNCTION_VALIDATE(params.get());
    105 
    106   // Caching the values so that it is possbile to safely pass them by reference.
    107   sender_ids_ = params->sender_ids;
    108   cert_ = SHA1HashHexString(GetExtension()->public_key());
    109 
    110   GCMProfileService()->Register(
    111       GetExtension()->id(),
    112       sender_ids_,
    113       cert_,
    114       base::Bind(&GcmRegisterFunction::CompleteFunctionWithResult, this));
    115 
    116   return true;
    117 }
    118 
    119 void GcmRegisterFunction::CompleteFunctionWithResult(
    120     const std::string& registration_id,
    121     gcm::GCMClient::Result result) {
    122   SetResult(base::Value::CreateStringValue(registration_id));
    123   SetError(GcmResultToError(result));
    124   SendResponse(gcm::GCMClient::SUCCESS == result);
    125 }
    126 
    127 GcmSendFunction::GcmSendFunction() {}
    128 
    129 GcmSendFunction::~GcmSendFunction() {}
    130 
    131 bool GcmSendFunction::DoWork() {
    132   scoped_ptr<api::gcm::Send::Params> params(
    133       api::gcm::Send::Params::Create(*args_));
    134   EXTENSION_FUNCTION_VALIDATE(params.get());
    135   EXTENSION_FUNCTION_VALIDATE(
    136       ValidateMessageData(params->message.data.additional_properties));
    137 
    138   // Caching the values so that it is possbile to safely pass them by reference.
    139   outgoing_message_.id = params->message.message_id;
    140   outgoing_message_.data = params->message.data.additional_properties;
    141   if (params->message.time_to_live.get())
    142     outgoing_message_.time_to_live = *params->message.time_to_live;
    143   destination_id_ = params->message.destination_id;
    144 
    145   GCMProfileService()->Send(
    146       GetExtension()->id(),
    147       params->message.destination_id,
    148       outgoing_message_,
    149       base::Bind(&GcmSendFunction::CompleteFunctionWithResult, this));
    150 
    151   return true;
    152 }
    153 
    154 void GcmSendFunction::CompleteFunctionWithResult(
    155     const std::string& message_id,
    156     gcm::GCMClient::Result result) {
    157   SetResult(base::Value::CreateStringValue(message_id));
    158   SetError(GcmResultToError(result));
    159   SendResponse(gcm::GCMClient::SUCCESS == result);
    160 }
    161 
    162 bool GcmSendFunction::ValidateMessageData(
    163     const gcm::GCMClient::MessageData& data) const {
    164   size_t total_size = 0u;
    165   for (std::map<std::string, std::string>::const_iterator iter = data.begin();
    166        iter != data.end(); ++iter) {
    167     total_size += iter->first.size() + iter->second.size();
    168 
    169     if (!IsMessageKeyValid(iter->first) ||
    170         kMaximumMessageSize < iter->first.size() ||
    171         kMaximumMessageSize < iter->second.size() ||
    172         kMaximumMessageSize < total_size)
    173       return false;
    174   }
    175 
    176   return total_size != 0;
    177 }
    178 
    179 GcmJsEventRouter::GcmJsEventRouter(Profile* profile) : profile_(profile) {}
    180 
    181 GcmJsEventRouter::~GcmJsEventRouter() {}
    182 
    183 void GcmJsEventRouter::OnMessage(
    184     const std::string& app_id,
    185     const gcm::GCMClient::IncomingMessage& message) {
    186   api::gcm::OnMessage::Message message_arg;
    187   message_arg.data.additional_properties = message.data;
    188 
    189   scoped_ptr<Event> event(new Event(
    190       api::gcm::OnMessage::kEventName,
    191       api::gcm::OnMessage::Create(message_arg).Pass(),
    192       profile_));
    193   ExtensionSystem::Get(profile_)->event_router()->DispatchEventToExtension(
    194       app_id, event.Pass());
    195 }
    196 
    197 void GcmJsEventRouter::OnMessagesDeleted(const std::string& app_id) {
    198   scoped_ptr<Event> event(new Event(
    199       api::gcm::OnMessagesDeleted::kEventName,
    200       api::gcm::OnMessagesDeleted::Create().Pass(),
    201       profile_));
    202   ExtensionSystem::Get(profile_)->event_router()->DispatchEventToExtension(
    203       app_id, event.Pass());
    204 }
    205 
    206 void GcmJsEventRouter::OnSendError(const std::string& app_id,
    207                                    const std::string& message_id,
    208                                    gcm::GCMClient::Result result) {
    209   api::gcm::OnSendError::Error error;
    210   error.message_id.reset(new std::string(message_id));
    211   error.error_message = GcmResultToError(result);
    212 
    213   scoped_ptr<Event> event(new Event(
    214       api::gcm::OnSendError::kEventName,
    215       api::gcm::OnSendError::Create(error).Pass(),
    216       profile_));
    217   ExtensionSystem::Get(profile_)->event_router()->DispatchEventToExtension(
    218       app_id, event.Pass());
    219 }
    220 
    221 }  // namespace extensions
    222