blob: 5b427666a49c6b3bc0b1f434407d525a657f8ca5 [file] [log] [blame]
// 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.
#ifndef EXTENSIONS_RENDERER_REQUEST_SENDER_H_
#define EXTENSIONS_RENDERER_REQUEST_SENDER_H_
#include <map>
#include <string>
#include "base/macros.h"
#include "base/memory/linked_ptr.h"
#include "v8/include/v8.h"
struct ExtensionHostMsg_Request_Params;
namespace base {
class ListValue;
}
namespace content {
class RenderFrame;
}
namespace extensions {
class ScriptContext;
struct PendingRequest;
// Responsible for sending requests for named extension API functions to the
// extension host and routing the responses back to the caller.
class RequestSender {
public:
// Source represents a user of RequestSender. Every request is associated with
// a Source object, which will be notified when the corresponding response
// arrives. When a Source object is going away and there are pending requests,
// it should call InvalidateSource() to make sure no notifications are sent to
// it later.
class Source {
public:
virtual ~Source() {}
virtual ScriptContext* GetContext() = 0;
virtual void OnResponseReceived(const std::string& name,
int request_id,
bool success,
const base::ListValue& response,
const std::string& error) = 0;
};
// Helper class to (re)set the |source_tab_id_| below.
class ScopedTabID {
public:
ScopedTabID(RequestSender* request_sender, int tab_id);
~ScopedTabID();
private:
RequestSender* const request_sender_;
const int tab_id_;
const int previous_tab_id_;
DISALLOW_COPY_AND_ASSIGN(ScopedTabID);
};
RequestSender();
virtual ~RequestSender();
// In order to avoid collision, all |request_id|s passed into StartRequest()
// should be generated by this method.
int GetNextRequestId() const;
// Makes a call to the API function |name| that is to be handled by the
// extension host. The response to this request will be received in
// HandleResponse().
// TODO(koz): Remove |request_id| and generate that internally.
// There are multiple of these per render view though, so we'll
// need to vend the IDs centrally.
// Returns true if the request is successfully sent.
bool StartRequest(Source* source,
const std::string& name,
int request_id,
bool has_callback,
bool for_io_thread,
base::ListValue* value_args);
// Sends the IPC to extension host for the API function that is described
// in |params|.
virtual void SendRequest(content::RenderFrame* render_frame,
bool for_io_thread,
ExtensionHostMsg_Request_Params& params);
// Handles responses from the extension host to calls made by StartRequest().
void HandleResponse(int request_id,
bool success,
const base::ListValue& response,
const std::string& error);
// Notifies this that a request source is no longer valid.
// TODO(kalman): Do this in a generic/safe way.
void InvalidateSource(Source* source);
private:
friend class ScopedTabID;
typedef std::map<int, linked_ptr<PendingRequest> > PendingRequestMap;
void InsertRequest(int request_id, PendingRequest* pending_request);
linked_ptr<PendingRequest> RemoveRequest(int request_id);
PendingRequestMap pending_requests_;
int source_tab_id_; // Id of the tab sending the request, or -1 if no tab.
DISALLOW_COPY_AND_ASSIGN(RequestSender);
};
} // namespace extensions
#endif // EXTENSIONS_RENDERER_REQUEST_SENDER_H_