716cb28430
* chore: bump chromium in DEPS to 1e9f9a24aa12bea9cf194a82a7e249bd1242ec4f * chore: update patches * Make WebContents' theme color a base::Optional<SkColor> https://chromium-review.googlesource.com/c/chromium/src/+/1540022 * update autofill patch for incorrect header includes * Move Shell messages to web_test and rename to BlinkTest. https://chromium-review.googlesource.com/c/chromium/src/+/1525181 * Make PlatformNotificationServiceImpl a KeyedService. https://chromium-review.googlesource.com/c/chromium/src/+/1336150 * Move MediaPlayerId to its own file. https://chromium-review.googlesource.com/c/chromium/src/+/1547057 * Remove net/base/completion_callback.h, which is no longer used https://chromium-review.googlesource.com/c/chromium/src/+/1552821 * AW NS: support file scheme cookies https://chromium-review.googlesource.com/c/chromium/src/+/1533486 * Remove SecurityInfo and adapt remaining consumers https://chromium-review.googlesource.com/c/chromium/src/+/1509455 * Remove deprecated type-specific number to string conversion functions https://chromium-review.googlesource.com/c/chromium/src/+/1545881 * DevTools: Adding new performance histograms for launch of top 4 tools https://chromium-review.googlesource.com/c/chromium/src/+/1506388 * Update include paths for //base/hash/hash.h https://chromium-review.googlesource.com/c/chromium/src/+/1544630 * build: Disable ensure_gn_version gclient hook for mac CI checkout * update patches * use maybe version of v8::String::NewFromTwoByte * bump appveyor image version * fix mac ci hopefully * Convert enum to enum class for MenuAnchorPosition https://chromium-review.googlesource.com/c/chromium/src/+/1530508 * use maybe version of ToObject * RenderViewHost::GetProcess is no longer const * Unrefcount AuthChallengeInfo https://chromium-review.googlesource.com/c/chromium/src/+/1550631 * MenuButtonController takes Button rather than MenuButton https://chromium-review.googlesource.com/c/chromium/src/+/1500935 * add //ui/views_bridge_mac to deps to fix link error * forward declare views::Button in atom::MenuDelegate * more v8 patches * base/{=> hash}/md5.h https://chromium-review.googlesource.com/c/chromium/src/+/1535124 * gfx::{PlatformFontWin => win}::* https://chromium-review.googlesource.com/c/chromium/src/+/1534178 * fix v8 patches * [base] Rename TaskScheduler to ThreadPool https://chromium-review.googlesource.com/c/chromium/src/+/1561552 * use internal_config_base for bytecode_builtins_list_generator avoids windows link errors * FIXME: temporarily disable v8/breakpad integration * FIXME: temporarily disable prevent-will-redirect test * FIXME: disable neon on aarch64 pending crbug.com/953815 * update to account for WebCursor refactor https://chromium-review.googlesource.com/c/chromium/src/+/1562755 * enable stack dumping on appveyor * Revert "FIXME: disable neon on aarch64 pending crbug.com/953815" This reverts commit 57f082026be3d83069f2a2814684abf4dc9e7b53. * fix: remove const qualifiers to match upstream * fix: remove const qualifiers to match upstream in cc files as well * don't throw an error when testing if an object is an object * use non-deprecated Buffer constructor * Remove net::CookieSameSite::DEFAULT_MODE enum value https://chromium-review.googlesource.com/c/chromium/src/+/1567955 * depend on modded dbus-native to work around buffer deprecation https://github.com/sidorares/dbus-native/pull/262 * revert clang roll to fix arm build on linux * fixup! depend on modded dbus-native to work around buffer deprecation need more coffee * update coffee-script * robustify verify-mksnapshot w.r.t. command-line parameters * Revert "robustify verify-mksnapshot w.r.t. command-line parameters" This reverts commit a49af01411f684f6025528d604895c3696e0bc57. * fix mksnapshot by matching args * update patches * TMP: enable rdp on appveyor * Changed ContentBrowserClient::CreateQuotaPermissionContext() to return scoped_refptr. https://chromium-review.googlesource.com/c/chromium/src/+/1569376 * Make content::ResourceType an enum class. https://chromium-review.googlesource.com/c/chromium/src/+/1569345 * fixup! Make content::ResourceType an enum class. * turn off rdp * use net::CompletionRepeatingCallback instead of base::Callback<void(int)> * remove disable_ensure_gn_version_gclient_hook.patch * copy repeating callback instead of std::move * fix lint * add completion_repeating_callback.h include
214 lines
7.2 KiB
C++
214 lines
7.2 KiB
C++
// Copyright (c) 2016 GitHub, Inc.
|
|
// Use of this source code is governed by the MIT license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#ifndef ATOM_BROWSER_API_ATOM_API_URL_REQUEST_H_
|
|
#define ATOM_BROWSER_API_ATOM_API_URL_REQUEST_H_
|
|
|
|
#include <array>
|
|
#include <string>
|
|
|
|
#include "atom/browser/api/event_emitter.h"
|
|
#include "atom/browser/api/trackable_object.h"
|
|
#include "base/memory/weak_ptr.h"
|
|
#include "native_mate/dictionary.h"
|
|
#include "native_mate/handle.h"
|
|
#include "native_mate/wrappable_base.h"
|
|
#include "net/base/auth.h"
|
|
#include "net/base/io_buffer.h"
|
|
#include "net/http/http_response_headers.h"
|
|
#include "net/url_request/url_request_context.h"
|
|
|
|
namespace atom {
|
|
|
|
class AtomURLRequest;
|
|
|
|
namespace api {
|
|
|
|
//
|
|
// The URLRequest class implements the V8 binding between the JavaScript API
|
|
// and Chromium native net library. It is responsible for handling HTTP/HTTPS
|
|
// requests.
|
|
//
|
|
// The current class provides only the binding layer. Two other JavaScript
|
|
// classes (ClientRequest and IncomingMessage) in the net module provide the
|
|
// final API, including some state management and arguments validation.
|
|
//
|
|
// URLRequest's methods fall into two main categories: command and event
|
|
// methods. They are always executed on the Browser's UI thread.
|
|
// Command methods are called directly from JavaScript code via the API defined
|
|
// in BuildPrototype. A command method is generally implemented by forwarding
|
|
// the call to a corresponding method on AtomURLRequest which does the
|
|
// synchronization on the Browser IO thread. The latter then calls into Chromium
|
|
// net library. On the other hand, net library events originate on the IO
|
|
// thread in AtomURLRequest and are synchronized back on the UI thread, then
|
|
// forwarded to a corresponding event method in URLRequest and then to
|
|
// JavaScript via the EmitRequestEvent/EmitResponseEvent helpers.
|
|
//
|
|
// URLRequest lifetime management: we followed the Wrapper/Wrappable pattern
|
|
// defined in native_mate. However, we augment that pattern with a pin/unpin
|
|
// mechanism. The main reason is that we want the JS API to provide a similar
|
|
// lifetime guarantees as the XMLHttpRequest.
|
|
// https://xhr.spec.whatwg.org/#garbage-collection
|
|
//
|
|
// The primary motivation is to not garbage collect a URLInstance as long as the
|
|
// object is emitting network events. For instance, in the following JS code
|
|
//
|
|
// (function() {
|
|
// let request = new URLRequest(...);
|
|
// request.on('response', (response)=>{
|
|
// response.on('data', (data) = > {
|
|
// console.log(data.toString());
|
|
// });
|
|
// });
|
|
// })();
|
|
//
|
|
// we still want data to be logged even if the response/request objects are n
|
|
// more referenced in JavaScript.
|
|
//
|
|
// Binding by simply following the native_mate Wrapper/Wrappable pattern will
|
|
// delete the URLRequest object when the corresponding JS object is collected.
|
|
// The v8 handle is a private member in WrappableBase and it is always weak,
|
|
// there is no way to make it strong without changing native_mate.
|
|
// The solution we implement consists of maintaining some kind of state that
|
|
// prevents collection of JS wrappers as long as the request is emitting network
|
|
// events. At initialization, the object is unpinned. When the request starts,
|
|
// it is pinned. When no more events would be emitted, the object is unpinned
|
|
// and lifetime is again managed by the standard native mate Wrapper/Wrappable
|
|
// pattern.
|
|
//
|
|
// pin/unpin: are implemented by constructing/reseting a V8 strong persistent
|
|
// handle.
|
|
//
|
|
// The URLRequest/AtmURLRequest interaction could have been implemented in a
|
|
// single class. However, it implies that the resulting class lifetime will be
|
|
// managed by two conflicting mechanisms: JavaScript garbage collection and
|
|
// Chromium reference counting. Reasoning about lifetime issues become much
|
|
// more complex.
|
|
//
|
|
// We chose to split the implementation into two classes linked via a
|
|
// reference counted/raw pointers. A URLRequest instance is deleted if it is
|
|
// unpinned and the corresponding JS wrapper object is garbage collected. On the
|
|
// other hand, an AtmURLRequest instance lifetime is totally governed by
|
|
// reference counting.
|
|
//
|
|
class URLRequest : public mate::EventEmitter<URLRequest> {
|
|
public:
|
|
static mate::WrappableBase* New(mate::Arguments* args);
|
|
|
|
static void BuildPrototype(v8::Isolate* isolate,
|
|
v8::Local<v8::FunctionTemplate> prototype);
|
|
|
|
// Methods for reporting events into JavaScript.
|
|
void OnReceivedRedirect(
|
|
int status_code,
|
|
const std::string& method,
|
|
const GURL& url,
|
|
scoped_refptr<net::HttpResponseHeaders> response_headers);
|
|
void OnAuthenticationRequired(const net::AuthChallengeInfo& auth_info);
|
|
void OnResponseStarted(
|
|
scoped_refptr<net::HttpResponseHeaders> response_headers);
|
|
void OnResponseData(scoped_refptr<const net::IOBufferWithSize> data);
|
|
void OnResponseCompleted();
|
|
void OnError(const std::string& error, bool isRequestError);
|
|
mate::Dictionary GetUploadProgress(v8::Isolate* isolate);
|
|
|
|
protected:
|
|
explicit URLRequest(v8::Isolate* isolate, v8::Local<v8::Object> wrapper);
|
|
~URLRequest() override;
|
|
|
|
private:
|
|
template <typename Flags>
|
|
class StateBase {
|
|
public:
|
|
void SetFlag(Flags flag);
|
|
|
|
protected:
|
|
explicit StateBase(Flags initialState);
|
|
bool operator==(Flags flag) const;
|
|
bool IsFlagSet(Flags flag) const;
|
|
|
|
private:
|
|
Flags state_;
|
|
};
|
|
|
|
enum class RequestStateFlags {
|
|
kNotStarted = 0x0,
|
|
kStarted = 0x1,
|
|
kFinished = 0x2,
|
|
kCanceled = 0x4,
|
|
kFailed = 0x8,
|
|
kClosed = 0x10
|
|
};
|
|
|
|
class RequestState : public StateBase<RequestStateFlags> {
|
|
public:
|
|
RequestState();
|
|
bool NotStarted() const;
|
|
bool Started() const;
|
|
bool Finished() const;
|
|
bool Canceled() const;
|
|
bool Failed() const;
|
|
bool Closed() const;
|
|
};
|
|
|
|
enum class ResponseStateFlags {
|
|
kNotStarted = 0x0,
|
|
kStarted = 0x1,
|
|
kEnded = 0x2,
|
|
kFailed = 0x4
|
|
};
|
|
|
|
class ResponseState : public StateBase<ResponseStateFlags> {
|
|
public:
|
|
ResponseState();
|
|
bool NotStarted() const;
|
|
bool Started() const;
|
|
bool Ended() const;
|
|
bool Canceled() const;
|
|
bool Failed() const;
|
|
bool Closed() const;
|
|
};
|
|
|
|
bool NotStarted() const;
|
|
bool Finished() const;
|
|
bool Canceled() const;
|
|
bool Failed() const;
|
|
bool Write(scoped_refptr<const net::IOBufferWithSize> buffer, bool is_last);
|
|
void Cancel();
|
|
void FollowRedirect();
|
|
bool SetExtraHeader(const std::string& name, const std::string& value);
|
|
void RemoveExtraHeader(const std::string& name);
|
|
void SetChunkedUpload(bool is_chunked_upload);
|
|
void SetLoadFlags(int flags);
|
|
|
|
int StatusCode() const;
|
|
std::string StatusMessage() const;
|
|
net::HttpResponseHeaders* RawResponseHeaders() const;
|
|
uint32_t ResponseHttpVersionMajor() const;
|
|
uint32_t ResponseHttpVersionMinor() const;
|
|
|
|
void Close();
|
|
void Pin();
|
|
void Unpin();
|
|
template <typename... Args>
|
|
void EmitRequestEvent(Args... args);
|
|
template <typename... Args>
|
|
void EmitResponseEvent(Args... args);
|
|
|
|
scoped_refptr<AtomURLRequest> atom_request_;
|
|
RequestState request_state_;
|
|
ResponseState response_state_;
|
|
|
|
// Used to implement pin/unpin.
|
|
v8::Global<v8::Object> wrapper_;
|
|
scoped_refptr<net::HttpResponseHeaders> response_headers_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(URLRequest);
|
|
};
|
|
|
|
} // namespace api
|
|
|
|
} // namespace atom
|
|
|
|
#endif // ATOM_BROWSER_API_ATOM_API_URL_REQUEST_H_
|