mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 05:37:11 +00:00
e39f9f88f7
- Bug 1152171 part 2 - Rename AnimationTimeline to DocumentTimeline; r=smaug (26c118319) - Bug 1152171 part 3 - Update web-platform-tests expectations; r=jgraham (b7b4032aa) - Bug 1153734 part 1 - Remove AnimationEffect; r=smaug (9cf67a02e) - Bug 1153734 part 2 - Rename Animation to KeyframeEffectReadonly; r=smaug (b69556ee6) - Bug 1153734 part 3 - Rename AnimationPlayer.source to AnimationPlayer.effect; r=smaug (50d3130ee) - Bug 1153734 part 4 - Rename other uses of 'source' and 'source content'; r=jwatt (b02c4ba36) - Bug 1153734 part 5 - Add AnimationEffectReadonly as a superinterface of KeyframeEffectReadonly; r=smaug (c3395d3f5) - Bug 1149990 - Support replaying of finished CSS transitions by supporting setting of currentTime/startTime. r=birtles (3fb2cb401) - Bug 1154615 part 1 - Rename AnimationPlayer to Animation in WebIDL; r=smaug (6c2125b49) - Bug 1154615 part 2 - Rename PendingPlayerTracker to PendingAnimationTracker; r=jwatt (8d6804def) - Bug 1154615 part 3 - Rename internal members of PendingAnimationTracker; r=jwatt (f348f6355) - Bug 1154615 part 4 - Rename references to players in dom/animation; r=jwatt (0250572e8) - Bug 1117603 part 1 - Don't assume style rules have been refreshed in GetAnimationRule; r=dbaron (a5d340d0f) - remove kungFuDeathGrip (49df758e6) - Bug 1117603 part 2 - Don't unregister from the refresh driver unless we are also queueing events; r=dbaron (715c9caa1) - Bug 1154615 part 5 - Rename AnimationPlayerCollection to AnimationCollection; r=jwatt (4c596f089) - Bug 1154615 part 6 - Rename references to players within layout/; r=jwatt (42405f3fc) - Bug 1154615 part 7 - Rename CSSAnimationPlayer and CSSTransitionPlayer; r=jwatt (49ab272ed) - Bug 1154615 part 8 - Rename references to players in animation observers; r=jwatt (c3fa26d7a) - Bug 1154615 part 9 - Rename test files; r=jwatt (9d9f03e7b) - Bug 1145439 (Part 1) - Throttle requestAnimationFrame for non-visible iframes. r=mstange,mchang (be7d183d6) - Bug 1145439 (Part 2) - Make test_scroll_event_ordering.html wait for rAF to unthrottle. r=roc (9ac8317c9) - Bug 1144324 - Try to register for, and handle, touch events when APZ is enabled. r=dvander,jimm (fb75d1665) - Bug 1144324 - Remove the codepaths that conditionally enable touch events based on touch the presence of touch listeners. r=smaug,jimm (710617e6b) - Bug 1003991 - Disable https:// only load for ServiceWorkers when Developer Tools are open. r=nsm, r=miker (9d6669814) - Bug 1153267 - part 1 - use smart-pointer .forget() instead of NS_ADDREF+assign; r=ehsan (e4555c90c) - Bug 1153267 - part 2 - use smart pointers instead of manual NS_ADDREF'ing outparams; r=ehsan (ae8b60d5a) - Bug 1153267 - fix typo that broke OS X builds on a CLOSED TREE; r=bustage (08fdb3c4f) - Bug 1146843 - Revert part of cset 33c30e283fa8 because the code is used in Fennec. r=snorp (407248257) - Bug 1151940 part 1. Make some readonly properties defined on Window by CSSOM-view replaceable. r=smaug (5cb9b91f0) - Bug 1151940 part 2. Add a convenience function in nsGlobalWindow for replacing a property on the window with a new value. r=smaug (2ba39331c) - Bug 1151940 part 3. Make some writable cssom-view attributes that we only allow setting from chrome act the way readonly replaceables would when called from content. r=smaug (b485e1b44) - Goanna -> Gecko (2c539d7be) - Goanna -> Gecko (25d34e213) - Bug 1148962 - Use TakeOwnershipOfErrorReporting in CPOW code (r=bholley) (96c997639) - pointer style (a07fbffaa) - Bug 1152577: Add 'aReason' argument to AutoEntryScript constructor, and provide plausible names for its instantiations. r=bholley (512fa27e2) - bug 1155691 - Expose WindowRoot to chrome from window in webidl. r=smaug (235281924) - Bug 404828 - No need to assert that the top window isn't reachable. r=smaug (d73154fa0) - Bug 404828 - Followup: remove assertion expectations on a CLOSED TREE. a=tomcat (a5dabe1b7) - Bug 1156102 - Mark nsGlobalWindowObserver::mWindow as MOZ_NON_OWNING_REF; r=baku (c0d4208b7) - Bug 1107801 - Improve gamepad support on MacOS. r=ted (c591bd5ac) - Goanna -> Gecko (d9b81bc9e) - Bug 852944 - Gamepad API IPC; r=ted, r=baku (521892538) - Bug 1143529 part 1. Stop manually calling WrapObject in DataStoreService::GetDataStoresResolve. r=baku (056ad6bfe) - Bug 1143529 part 2. Tighten up the assert in binding Wrap methods. r=peterv (765a13325) - Bug 1152169 - DataStoreService should check if the first revision exists, r=bent (ee371cc5d) - Bug 1152169 followup: Mark FirstRevisionIdCallback methods Run() and HandleEvent() as 'override'. rs=ehsan (8186c4168) - Bug 1143651 - don't use CallQueryInterface when the compiler can do the cast for us; r=ehsan (a50f0a54b) - Bug 1144322 - Handle tabindex in overridden IsInteractiveHTMLContent methods. r=smaug (fd4b9beed) - Bug 1086684 - Stash the full path for file inputs to avoid doing IPC at inopportune times. r=ehsan/bent/gps (b843b1efc) - Bug 1143934 - Disallow mozSetFileNameArray in content processes. r=ehsan (42e5c8c6d) - Bug 1143934 - Fix assorted forms mochitests for e10s-compatibility. r=smaug (7a3babfed) - Bug 1143934 - Work around SessionStore dependency on current brokenness. r=ttaubert (5b0fcb5ce) - Bug 956530 - Clear the delayed caret data when clicking on a selected part of a text control if the focus event handler selects the control; r=roc (2859f07b4) - Bug 956530 follow-up: Fix the test failure on Windows 8 caused by the text box having a glowing outline as a result of being clicked on (d34e8da1a) - Bug 1157898 part 1. Make code of the form "return rv.ErrorCode();" where rv is an ErrorResult use StealNSResult instead. r=peterv (800da50e2) - Bug 1157898 part 2. Make code of the form "NS_ENSURE_SUCCESS(rv.ErrorCode(), rv.ErrorCode());" use Failed and StealNSResult instead. r=peterv (472432a83) - Bug 1157898 part 3. Fix the remaining consumers of rv.ErrorCode() in NS_ENSURE_* expressions to not do that. r=peterv (d452807e7) - Bug 1122238 part 1. Switch to using the new stackframe APIs in JSStackFrame. r=bholley (9d87b261a) - Bug 1122238 part 2. Stop caching things in JSStackFrame when we're called over Xrays. r=bholley (83eda7275) - Bug 1122238 part 3. Drop all the DOMException-cloning and sanitization gunk we added in bug 1107592 and bug 1107953 and bug 1117242 . r=bholley (f237aa948) - add support for NetBSD/SPARC64 (065783b70) - Bug 1153484 - Fetch should ignore invalid headers, but still process later headers. r=nsm (8925ddd77) - Bug 1157754 part 2. Convert consumers of ErrorResult::ClearMessage() to the new better APIs we have for suppressing exceptions on ErrorResult. r=bkelly (6519fbd5e) - Bug 1157754 part 3. Make ClearMessage private on ErrorResult. r=peterv (3fb218692) - Bug 1157898 part 4. Add ErrorResult::ErrorCodeIs() and use it in various places to get rid of ErrorCode(). r=peterv (bed7bfb4c) - Bug 1130686 - Refactor PromiseHolder in the service worker clients code. r=nsm (b3dbdcbfe) - Bug 1130686 - Implement client.focus. r=baku (5dee6d850) - Bug 1149163 part 1 - Clean up nsHTMLEditRules::GetInnerContent; r=froydnj (cc8f65b54) - Bug 1149163 part 2 - Make nsDOMIterator infallible; r=froydnj (d975f6c62) - Bug 1149163 part 3 - Clean up nsHTMLEditRules::BustUpInlinesAtBRs; r=froydnj (58155adad) - Bug 1149163 part 4 - Allow use of temporary nsBoolDomIterFunctor; r=froydnj (dbafec00f) - Bug 1149163 part 5 - Clean up nsHTMLEditRules::GetNodesForOperation; r=froydnj (41179d810) - Bug 1149163 part 6 - Clean up nsHTMLEditRules::LookInsideDivBQandList; r=froydnj (0b757bf14) - Bug 1149163 part 7 - Clean up nsHTMLEditRules::PromoteRange; r=froydnj (c49c714b1) - Bug 1149163 part 8 - Clean up nsHTMLEditRules::GetPromotedRanges; r=froydnj (5163a0026) - Bug 1148228 - Stop checking ul twice (43a22088c) - Bug 1141017 - resurrect serif and monospace. r=ehsan (95a1b6fcf) - Bug 1147412 part 1 - Make methods take nsINode*, not just nsIContent*; r=ehsan (7f762cdbe) - Bug 1147412 part 2 - Clean up nsHTMLEditor::SetInlinePropertyOnTextNode; r=ehsan (faf805587) - Bug 1147412 part 3 - Fix completely broken nsHTMLCSSUtils::IsCSSEquivalentToHTMLInlineStyleSet implementation; r=ehsan (73fea67c1) - Bug 1147412 part 4 - Clean up nsHTMLEditor::GetInlinePropertyBase; r=ehsan (3265bfbce) - Bug 1147412 part 5 - Clean up nsHTMLEditor::RemoveInlinePropertyImpl; r=ehsan (0f402bd7e) - Bug 1147412 part 6 - Remove nsHTMLCSSUtils::IsCSSEditableProperty(nsIDOMNode*,...); r=ehsan (100e4038a) - Bug 1147412 part 7 - Remove nsHTMLCSSUtils::GetComputedStyle(nsIDOMElement*); r=ehsan (6c51103bc) - Bug 1147412 part 8 - Clean up nsHTMLCSSUtils::IsCSSInvertible; r=ehsan (01e60c446) - Bug 1147412 part 9 - Convert some nsHTMLEditor members to Element; r=ehsan (e7efb1ac4) - Bug 1147412 part 10 - Clean up nsHTMLCSSUtils::Get*Property, GetCSSInlinePropertyBase; r=ehsan (54154143d) - Bug 1149163 part 9 - Clean up nsHTMLEditRules::GetNodesFromSelection; r=froydnj (5186308b9) - Bug 1154701 part 1 - Clean up nsHTMLEditor::CreateListOfNodesToPaste; r=ehsan (ea95238d5) - Bug 1153629 part 1 - Clean up nsHTMLEditRules::GetListActionNodes; r=ehsan (51f3b3e95) - Bug 1153629 part 2 - Clean up nsHTMLEditRules::GetParagraphFormatNodes; r=ehsan (a27bd7751) - Bug 1153629 part 3 - Clean up nsHTMLEditRules::GetNodesFromPoint; r=ehsan (edc7e4561) - Bug 1153629 part 4 - Clean up nsHTMLEditRules::ListIsEmptyLine; r=ehsan (ce3289bc7) - Bug 1153629 part 5 - Clean up nsHTMLEditRules::GetChildNodesForOperation; r=ehsan (b3a509dbf) - Bug 1153629 part 6 - Clean up nsHTMLEditRules::MakeBlockquote; r=ehsan (cb3808182) - Bug 1153629 part 7 - Clean up nsHTMLEditRules::RemoveBlockStyle, RemovePartOfBlock; r=ehsan (660b9f76e) - Bug 1153629 part 8 - Clean up nsHTMLEditRules::ApplyBlockStyle; r=ehsan (f54f9538c) - Bug 1153629 part 9 - Clean up nsHTMLEditRules::MakeTransitionList; r=ehsan (fb63cf6d8) - Bug 1153629 part 10 - Clean up nsHTMLEditRules::AlignInnerBlocks; r=ehsan (752d2df7a) - Bug 1153629 part 11 - Clean up nsHTMLEditRules::AdjustSpecialBreaks; r=ehsan (16ef0416b) - Bug 1153629 part 12 - Clean up nsHTMLEditRules::RemoveEmptyNodes; r=ehsan (d528e70e6) - Bug 1154701 part 2 - Use more OwningNonNull in editor; r=ehsan (85b1929e6) - Bug 1154701 part 3 - Clean up nsHTMLEditor::GetListAndTableParents, DiscoverPartialListsAndTables, ScanForListAndTableStructure, ReplaceOrphanedStructure; r=ehsan (7fe31f058) - Bug 1154701 part 4 - Switch nsHTMLEditor::mContentFilters to nsTArray; r=ehsan (64e6dd160) - Bug 1154701 part 5 - Switch nsHTMLEditor::objectResizeEventListeners to nsTArray; r=ehsan (036bc65fe) - Bug 1154701 part 6 - Clean up nsHTMLEditor::SetInlinePropertyOnNodeImpl; r=ehsan (2d619ca16) - Bug 1154701 part 7 - Clean up nsHTMLEditor::SetInlineProperty; r=ehsan (7a367d31b) - Bug 1154701 part 8 - Clean up nsHTMLEditor::SetInlinePropertyOnNode; r=ehsan (707c07d93) - Bug 1154701 part 9 - Clean up nsHTMLEditor::RelativeFontChange; r=ehsan (273ae9c64) - Bug 1154701 part 10 - Switch nsEditor::mActionListeners to nsTArray; r=ehsan (d2b5732fe) - Bug 1154701 part 11 - Switch nsEditor::mEditorObservers to nsTArray; r=ehsan (25a5af12e) - Bug 1154701 part 12 - Switch nsEditor::mDocStateListeners to nsTArray; r=ehsan (665af0792) - Bug 1154701 part 13 - Clean up nsHTMLEditor::SetCSSBackgroundColor; r=ehsan (ba424ade8) - Bug 1154701 part 14 - Remove unused nsCOMArray cruft; r=ehsan (3a8679a67) - Bug 1101651 - Part 1: xpcomrt version of dom media library need for standalone webrtcs. r=jesup (ae37b5464) - Bug 1137447 - New app update telemetry for patch type (complete or partial), extended error codes, and general cleanup. r=bbondy (c736ae502)
923 lines
29 KiB
C++
923 lines
29 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/FetchDriver.h"
|
|
|
|
#include "nsIDocument.h"
|
|
#include "nsIInputStream.h"
|
|
#include "nsIOutputStream.h"
|
|
#include "nsIHttpChannel.h"
|
|
#include "nsIHttpChannelInternal.h"
|
|
#include "nsIHttpHeaderVisitor.h"
|
|
#include "nsIScriptSecurityManager.h"
|
|
#include "nsIThreadRetargetableRequest.h"
|
|
#include "nsIUploadChannel2.h"
|
|
|
|
#include "nsContentPolicyUtils.h"
|
|
#include "nsCORSListenerProxy.h"
|
|
#include "nsDataHandler.h"
|
|
#include "nsHostObjectProtocolHandler.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsPrintfCString.h"
|
|
#include "nsStreamUtils.h"
|
|
#include "nsStringStream.h"
|
|
|
|
#include "mozilla/dom/File.h"
|
|
#include "mozilla/dom/workers/Workers.h"
|
|
|
|
#include "Fetch.h"
|
|
#include "InternalRequest.h"
|
|
#include "InternalResponse.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_ISUPPORTS(FetchDriver,
|
|
nsIStreamListener, nsIChannelEventSink, nsIInterfaceRequestor,
|
|
nsIAsyncVerifyRedirectCallback)
|
|
|
|
FetchDriver::FetchDriver(InternalRequest* aRequest, nsIPrincipal* aPrincipal,
|
|
nsILoadGroup* aLoadGroup)
|
|
: mPrincipal(aPrincipal)
|
|
, mLoadGroup(aLoadGroup)
|
|
, mRequest(aRequest)
|
|
, mFetchRecursionCount(0)
|
|
, mResponseAvailableCalled(false)
|
|
{
|
|
}
|
|
|
|
FetchDriver::~FetchDriver()
|
|
{
|
|
// We assert this since even on failures, we should call
|
|
// FailWithNetworkError().
|
|
MOZ_ASSERT(mResponseAvailableCalled);
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::Fetch(FetchDriverObserver* aObserver)
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
mObserver = aObserver;
|
|
|
|
return Fetch(false /* CORS flag */);
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::Fetch(bool aCORSFlag)
|
|
{
|
|
// We do not currently implement parts of the spec that lead to recursion.
|
|
MOZ_ASSERT(mFetchRecursionCount == 0);
|
|
mFetchRecursionCount++;
|
|
|
|
// FIXME(nsm): Deal with HSTS.
|
|
|
|
if (!mRequest->IsSynchronous() && mFetchRecursionCount <= 1) {
|
|
nsCOMPtr<nsIRunnable> r =
|
|
NS_NewRunnableMethodWithArg<bool>(this, &FetchDriver::ContinueFetch, aCORSFlag);
|
|
nsresult rv = NS_DispatchToCurrentThread(r);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
MOZ_CRASH("Synchronous fetch not supported");
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::ContinueFetch(bool aCORSFlag)
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
|
|
nsAutoCString url;
|
|
mRequest->GetURL(url);
|
|
nsCOMPtr<nsIURI> requestURI;
|
|
nsresult rv = NS_NewURI(getter_AddRefs(requestURI), url,
|
|
nullptr, nullptr);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
// CSP/mixed content checks.
|
|
int16_t shouldLoad;
|
|
rv = NS_CheckContentLoadPolicy(mRequest->ContentPolicyType(),
|
|
requestURI,
|
|
mPrincipal,
|
|
mDocument,
|
|
// FIXME(nsm): Should MIME be extracted from
|
|
// Content-Type header?
|
|
EmptyCString(), /* mime guess */
|
|
nullptr, /* extra */
|
|
&shouldLoad,
|
|
nsContentUtils::GetContentPolicy(),
|
|
nsContentUtils::GetSecurityManager());
|
|
if (NS_WARN_IF(NS_FAILED(rv) || NS_CP_REJECTED(shouldLoad))) {
|
|
// Disallowed by content policy.
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
// Begin Step 4 of the Fetch algorithm
|
|
// https://fetch.spec.whatwg.org/#fetching
|
|
|
|
nsAutoCString scheme;
|
|
rv = requestURI->GetScheme(scheme);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
rv = mPrincipal->CheckMayLoad(requestURI, false /* report */, false /* allowIfInheritsPrincipal */);
|
|
if ((!aCORSFlag && NS_SUCCEEDED(rv)) ||
|
|
(scheme.EqualsLiteral("data") && mRequest->SameOriginDataURL()) ||
|
|
scheme.EqualsLiteral("about")) {
|
|
return BasicFetch();
|
|
}
|
|
|
|
if (mRequest->Mode() == RequestMode::Same_origin) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
if (mRequest->Mode() == RequestMode::No_cors) {
|
|
mRequest->SetResponseTainting(InternalRequest::RESPONSETAINT_OPAQUE);
|
|
return BasicFetch();
|
|
}
|
|
|
|
if (!scheme.EqualsLiteral("http") && !scheme.EqualsLiteral("https")) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
bool corsPreflight = false;
|
|
if (mRequest->Mode() == RequestMode::Cors_with_forced_preflight ||
|
|
(mRequest->UnsafeRequest() && (!mRequest->HasSimpleMethod() || !mRequest->Headers()->HasOnlySimpleHeaders()))) {
|
|
corsPreflight = true;
|
|
}
|
|
|
|
mRequest->SetResponseTainting(InternalRequest::RESPONSETAINT_CORS);
|
|
return HttpFetch(true /* aCORSFlag */, corsPreflight);
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::BasicFetch()
|
|
{
|
|
nsAutoCString url;
|
|
mRequest->GetURL(url);
|
|
nsCOMPtr<nsIURI> uri;
|
|
nsresult rv = NS_NewURI(getter_AddRefs(uri),
|
|
url,
|
|
nullptr,
|
|
nullptr);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
nsAutoCString scheme;
|
|
rv = uri->GetScheme(scheme);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
if (scheme.LowerCaseEqualsLiteral("about")) {
|
|
if (url.EqualsLiteral("about:blank")) {
|
|
nsRefPtr<InternalResponse> response =
|
|
new InternalResponse(200, NS_LITERAL_CSTRING("OK"));
|
|
ErrorResult result;
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("content-type"),
|
|
NS_LITERAL_CSTRING("text/html;charset=utf-8"),
|
|
result);
|
|
MOZ_ASSERT(!result.Failed());
|
|
nsCOMPtr<nsIInputStream> body;
|
|
rv = NS_NewCStringInputStream(getter_AddRefs(body), EmptyCString());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
response->SetBody(body);
|
|
BeginResponse(response);
|
|
return SucceedWithResponse();
|
|
}
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
if (scheme.LowerCaseEqualsLiteral("blob")) {
|
|
nsRefPtr<FileImpl> blobImpl;
|
|
rv = NS_GetBlobForBlobURI(uri, getter_AddRefs(blobImpl));
|
|
FileImpl* blob = static_cast<FileImpl*>(blobImpl.get());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
nsRefPtr<InternalResponse> response = new InternalResponse(200, NS_LITERAL_CSTRING("OK"));
|
|
{
|
|
ErrorResult result;
|
|
uint64_t size = blob->GetSize(result);
|
|
if (NS_WARN_IF(result.Failed())) {
|
|
FailWithNetworkError();
|
|
return result.StealNSResult();
|
|
}
|
|
|
|
nsAutoString sizeStr;
|
|
sizeStr.AppendInt(size);
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("Content-Length"), NS_ConvertUTF16toUTF8(sizeStr), result);
|
|
if (NS_WARN_IF(result.Failed())) {
|
|
FailWithNetworkError();
|
|
return result.StealNSResult();
|
|
}
|
|
|
|
nsAutoString type;
|
|
blob->GetType(type);
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("Content-Type"), NS_ConvertUTF16toUTF8(type), result);
|
|
if (NS_WARN_IF(result.Failed())) {
|
|
FailWithNetworkError();
|
|
return result.StealNSResult();
|
|
}
|
|
}
|
|
|
|
nsCOMPtr<nsIInputStream> stream;
|
|
rv = blob->GetInternalStream(getter_AddRefs(stream));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
response->SetBody(stream);
|
|
BeginResponse(response);
|
|
return SucceedWithResponse();
|
|
}
|
|
|
|
if (scheme.LowerCaseEqualsLiteral("data")) {
|
|
nsAutoCString method;
|
|
mRequest->GetMethod(method);
|
|
if (method.LowerCaseEqualsASCII("get")) {
|
|
// Use nsDataHandler directly so that we can extract the content type.
|
|
// XXX(nsm): Is there a way to acquire the charset without such tight
|
|
// coupling with the DataHandler? nsIProtocolHandler does not provide
|
|
// anything similar.
|
|
nsAutoCString contentType, contentCharset, dataBuffer, hashRef;
|
|
bool isBase64;
|
|
rv = nsDataHandler::ParseURI(url,
|
|
contentType,
|
|
contentCharset,
|
|
isBase64,
|
|
dataBuffer,
|
|
hashRef);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
ErrorResult result;
|
|
nsRefPtr<InternalResponse> response = new InternalResponse(200, NS_LITERAL_CSTRING("OK"));
|
|
if (!contentCharset.IsEmpty()) {
|
|
contentType.Append(";charset=");
|
|
contentType.Append(contentCharset);
|
|
}
|
|
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("Content-Type"), contentType, result);
|
|
if (!result.Failed()) {
|
|
nsCOMPtr<nsIInputStream> stream;
|
|
rv = NS_NewCStringInputStream(getter_AddRefs(stream), dataBuffer);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
response->SetBody(stream);
|
|
BeginResponse(response);
|
|
return SucceedWithResponse();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
if (scheme.LowerCaseEqualsLiteral("file")) {
|
|
} else if (scheme.LowerCaseEqualsLiteral("http") ||
|
|
scheme.LowerCaseEqualsLiteral("https")) {
|
|
return HttpFetch();
|
|
}
|
|
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
// This function implements the "HTTP Fetch" algorithm from the Fetch spec.
|
|
// Functionality is often split between here, the CORS listener proxy and the
|
|
// Necko HTTP implementation.
|
|
nsresult
|
|
FetchDriver::HttpFetch(bool aCORSFlag, bool aCORSPreflightFlag, bool aAuthenticationFlag)
|
|
{
|
|
// Step 1. "Let response be null."
|
|
mResponse = nullptr;
|
|
nsresult rv;
|
|
|
|
nsCOMPtr<nsIIOService> ios = do_GetIOService(&rv);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
nsAutoCString url;
|
|
mRequest->GetURL(url);
|
|
nsCOMPtr<nsIURI> uri;
|
|
rv = NS_NewURI(getter_AddRefs(uri),
|
|
url,
|
|
nullptr,
|
|
nullptr,
|
|
ios);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
// Step 2 deals with letting ServiceWorkers intercept requests. This is
|
|
// handled by Necko after the channel is opened.
|
|
// FIXME(nsm): Bug 1119026: The channel's skip service worker flag should be
|
|
// set based on the Request's flag.
|
|
|
|
// Step 3.1 "If the CORS preflight flag is set and one of these conditions is
|
|
// true..." is handled by the CORS proxy.
|
|
//
|
|
// Step 3.2 "Set request's skip service worker flag." This isn't required
|
|
// since Necko will fall back to the network if the ServiceWorker does not
|
|
// respond with a valid Response.
|
|
//
|
|
// NS_StartCORSPreflight() will automatically kick off the original request
|
|
// if it succeeds, so we need to have everything setup for the original
|
|
// request too.
|
|
|
|
// Step 3.3 "Let credentials flag be set if either request's credentials mode
|
|
// is include, or request's credentials mode is same-origin and the CORS flag
|
|
// is unset, and unset otherwise."
|
|
bool useCredentials = false;
|
|
if (mRequest->GetCredentialsMode() == RequestCredentials::Include ||
|
|
(mRequest->GetCredentialsMode() == RequestCredentials::Same_origin && !aCORSFlag)) {
|
|
useCredentials = true;
|
|
}
|
|
|
|
// This is effectivetly the opposite of the use credentials flag in "HTTP
|
|
// network or cache fetch" in the spec and decides whether to transmit
|
|
// cookies and other identifying information. LOAD_ANONYMOUS also prevents
|
|
// new cookies sent by the server from being stored.
|
|
const nsLoadFlags credentialsFlag = useCredentials ? 0 : nsIRequest::LOAD_ANONYMOUS;
|
|
|
|
// From here on we create a channel and set its properties with the
|
|
// information from the InternalRequest. This is an implementation detail.
|
|
MOZ_ASSERT(mLoadGroup);
|
|
nsCOMPtr<nsIChannel> chan;
|
|
rv = NS_NewChannel(getter_AddRefs(chan),
|
|
uri,
|
|
mPrincipal,
|
|
nsILoadInfo::SEC_NORMAL,
|
|
mRequest->ContentPolicyType(),
|
|
mLoadGroup,
|
|
nullptr, /* aCallbacks */
|
|
nsIRequest::LOAD_NORMAL | credentialsFlag,
|
|
ios);
|
|
mLoadGroup = nullptr;
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
// Insert ourselves into the notification callbacks chain so we can handle
|
|
// cross-origin redirects.
|
|
chan->GetNotificationCallbacks(getter_AddRefs(mNotificationCallbacks));
|
|
chan->SetNotificationCallbacks(this);
|
|
|
|
// FIXME(nsm): Bug 1120715.
|
|
// Step 3.4 "If request's cache mode is default and request's header list
|
|
// contains a header named `If-Modified-Since`, `If-None-Match`,
|
|
// `If-Unmodified-Since`, `If-Match`, or `If-Range`, set request's cache mode
|
|
// to no-store."
|
|
|
|
// Step 3.5 begins "HTTP network or cache fetch".
|
|
// HTTP network or cache fetch
|
|
// ---------------------------
|
|
// Step 1 "Let HTTPRequest..." The channel is the HTTPRequest.
|
|
nsCOMPtr<nsIHttpChannel> httpChan = do_QueryInterface(chan);
|
|
if (httpChan) {
|
|
// Copy the method.
|
|
nsAutoCString method;
|
|
mRequest->GetMethod(method);
|
|
rv = httpChan->SetRequestMethod(method);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
// Set the same headers.
|
|
nsAutoTArray<InternalHeaders::Entry, 5> headers;
|
|
mRequest->Headers()->GetEntries(headers);
|
|
for (uint32_t i = 0; i < headers.Length(); ++i) {
|
|
httpChan->SetRequestHeader(headers[i].mName, headers[i].mValue, false /* merge */);
|
|
}
|
|
|
|
// Step 2. Set the referrer.
|
|
nsAutoString referrer;
|
|
mRequest->GetReferrer(referrer);
|
|
// The referrer should have already been resolved to a URL by the caller.
|
|
MOZ_ASSERT(!referrer.EqualsLiteral(kFETCH_CLIENT_REFERRER_STR));
|
|
if (!referrer.IsEmpty()) {
|
|
nsCOMPtr<nsIURI> refURI;
|
|
rv = NS_NewURI(getter_AddRefs(refURI), referrer, nullptr, nullptr);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
net::ReferrerPolicy referrerPolicy = net::RP_Default;
|
|
if (mDocument) {
|
|
referrerPolicy = mDocument->GetReferrerPolicy();
|
|
}
|
|
|
|
rv = httpChan->SetReferrerWithPolicy(refURI, referrerPolicy);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
}
|
|
|
|
// Step 3 "If HTTPRequest's force Origin header flag is set..."
|
|
if (mRequest->ForceOriginHeader()) {
|
|
nsAutoString origin;
|
|
rv = nsContentUtils::GetUTFOrigin(mPrincipal, origin);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
httpChan->SetRequestHeader(NS_LITERAL_CSTRING("origin"),
|
|
NS_ConvertUTF16toUTF8(origin),
|
|
false /* merge */);
|
|
}
|
|
// Bug 1120722 - Authorization will be handled later.
|
|
// Auth may require prompting, we don't support it yet.
|
|
// The next patch in this same bug prevents this from aborting the request.
|
|
// Credentials checks for CORS are handled by nsCORSListenerProxy,
|
|
}
|
|
|
|
// Step 5. Proxy authentication will be handled by Necko.
|
|
// FIXME(nsm): Bug 1120715.
|
|
// Step 7-10. "If request's cache mode is neither no-store nor reload..."
|
|
|
|
// Continue setting up 'HTTPRequest'. Content-Type and body data.
|
|
nsCOMPtr<nsIUploadChannel2> uploadChan = do_QueryInterface(chan);
|
|
if (uploadChan) {
|
|
nsAutoCString contentType;
|
|
ErrorResult result;
|
|
mRequest->Headers()->Get(NS_LITERAL_CSTRING("content-type"), contentType, result);
|
|
// This is an error because the Request constructor explicitly extracts and
|
|
// sets a content-type per spec.
|
|
if (result.Failed()) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
nsCOMPtr<nsIInputStream> bodyStream;
|
|
mRequest->GetBody(getter_AddRefs(bodyStream));
|
|
if (bodyStream) {
|
|
nsAutoCString method;
|
|
mRequest->GetMethod(method);
|
|
rv = uploadChan->ExplicitSetUploadStream(bodyStream, contentType, -1, method, false /* aStreamHasHeaders */);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set skip serviceworker flag.
|
|
// While the spec also gates on the client being a ServiceWorker, we can't
|
|
// infer that here. Instead we rely on callers to set the flag correctly.
|
|
if (mRequest->SkipServiceWorker()) {
|
|
nsCOMPtr<nsIHttpChannelInternal> internalChan = do_QueryInterface(httpChan);
|
|
internalChan->ForceNoIntercept();
|
|
}
|
|
|
|
nsCOMPtr<nsIStreamListener> listener = this;
|
|
|
|
// Unless the cors mode is explicitly no-cors, we set up a cors proxy even in
|
|
// the same-origin case, since the proxy does not enforce cors header checks
|
|
// in the same-origin case.
|
|
if (mRequest->Mode() != RequestMode::No_cors) {
|
|
// Set up a CORS proxy that will handle the various requirements of the CORS
|
|
// protocol. It handles the preflight cache and CORS response headers.
|
|
// If the request is allowed, it will start our original request
|
|
// and our observer will be notified. On failure, our observer is notified
|
|
// directly.
|
|
nsRefPtr<nsCORSListenerProxy> corsListener =
|
|
new nsCORSListenerProxy(this, mPrincipal, useCredentials);
|
|
rv = corsListener->Init(chan, DataURIHandling::Allow);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
listener = corsListener.forget();
|
|
}
|
|
|
|
// If preflight is required, start a "CORS preflight fetch"
|
|
// https://fetch.spec.whatwg.org/#cors-preflight-fetch-0. All the
|
|
// implementation is handled by NS_StartCORSPreflight, we just set up the
|
|
// unsafeHeaders so they can be verified against the response's
|
|
// "Access-Control-Allow-Headers" header.
|
|
if (aCORSPreflightFlag) {
|
|
nsCOMPtr<nsIChannel> preflightChannel;
|
|
nsAutoTArray<nsCString, 5> unsafeHeaders;
|
|
mRequest->Headers()->GetUnsafeHeaders(unsafeHeaders);
|
|
|
|
rv = NS_StartCORSPreflight(chan, listener, mPrincipal,
|
|
useCredentials,
|
|
unsafeHeaders,
|
|
getter_AddRefs(preflightChannel));
|
|
} else {
|
|
rv = chan->AsyncOpen(listener, nullptr);
|
|
}
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return FailWithNetworkError();
|
|
}
|
|
|
|
// Step 4 onwards of "HTTP Fetch" is handled internally by Necko.
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::ContinueHttpFetchAfterNetworkFetch()
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
MOZ_ASSERT(mResponse);
|
|
MOZ_ASSERT(!mResponse->IsError());
|
|
|
|
return SucceedWithResponse();
|
|
}
|
|
|
|
already_AddRefed<InternalResponse>
|
|
FetchDriver::BeginAndGetFilteredResponse(InternalResponse* aResponse)
|
|
{
|
|
MOZ_ASSERT(aResponse);
|
|
if (!aResponse->FinalURL()) {
|
|
nsAutoCString reqURL;
|
|
mRequest->GetURL(reqURL);
|
|
aResponse->SetUrl(reqURL);
|
|
}
|
|
|
|
// FIXME(nsm): Handle mixed content check, step 7 of fetch.
|
|
|
|
nsRefPtr<InternalResponse> filteredResponse;
|
|
switch (mRequest->GetResponseTainting()) {
|
|
case InternalRequest::RESPONSETAINT_BASIC:
|
|
filteredResponse = aResponse->BasicResponse();
|
|
break;
|
|
case InternalRequest::RESPONSETAINT_CORS:
|
|
filteredResponse = aResponse->CORSResponse();
|
|
break;
|
|
case InternalRequest::RESPONSETAINT_OPAQUE:
|
|
filteredResponse = aResponse->OpaqueResponse();
|
|
break;
|
|
default:
|
|
MOZ_CRASH("Unexpected case");
|
|
}
|
|
|
|
MOZ_ASSERT(filteredResponse);
|
|
MOZ_ASSERT(mObserver);
|
|
mObserver->OnResponseAvailable(filteredResponse);
|
|
mResponseAvailableCalled = true;
|
|
return filteredResponse.forget();
|
|
}
|
|
|
|
void
|
|
FetchDriver::BeginResponse(InternalResponse* aResponse)
|
|
{
|
|
nsRefPtr<InternalResponse> r = BeginAndGetFilteredResponse(aResponse);
|
|
// Release the ref.
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::SucceedWithResponse()
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
if (mObserver) {
|
|
mObserver->OnResponseEnd();
|
|
mObserver = nullptr;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::FailWithNetworkError()
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
nsRefPtr<InternalResponse> error = InternalResponse::NetworkError();
|
|
if (mObserver) {
|
|
mObserver->OnResponseAvailable(error);
|
|
mResponseAvailableCalled = true;
|
|
mObserver->OnResponseEnd();
|
|
mObserver = nullptr;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
namespace {
|
|
class FillResponseHeaders final : public nsIHttpHeaderVisitor {
|
|
InternalResponse* mResponse;
|
|
|
|
~FillResponseHeaders()
|
|
{ }
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit FillResponseHeaders(InternalResponse* aResponse)
|
|
: mResponse(aResponse)
|
|
{
|
|
}
|
|
|
|
NS_IMETHOD
|
|
VisitHeader(const nsACString & aHeader, const nsACString & aValue) override
|
|
{
|
|
ErrorResult result;
|
|
mResponse->Headers()->Append(aHeader, aValue, result);
|
|
if (result.Failed()) {
|
|
NS_WARNING(nsPrintfCString("Fetch ignoring illegal header - '%s': '%s'",
|
|
PromiseFlatCString(aHeader).get(),
|
|
PromiseFlatCString(aValue).get()).get());
|
|
result.SuppressException();
|
|
}
|
|
return NS_OK;
|
|
}
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(FillResponseHeaders, nsIHttpHeaderVisitor)
|
|
} // namespace
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::OnStartRequest(nsIRequest* aRequest,
|
|
nsISupports* aContext)
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
MOZ_ASSERT(!mPipeOutputStream);
|
|
MOZ_ASSERT(mObserver);
|
|
nsresult rv;
|
|
aRequest->GetStatus(&rv);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
nsCOMPtr<nsIHttpChannel> channel = do_QueryInterface(aRequest);
|
|
// For now we only support HTTP.
|
|
MOZ_ASSERT(channel);
|
|
|
|
aRequest->GetStatus(&rv);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
uint32_t responseStatus;
|
|
channel->GetResponseStatus(&responseStatus);
|
|
|
|
nsAutoCString statusText;
|
|
channel->GetResponseStatusText(statusText);
|
|
|
|
nsRefPtr<InternalResponse> response = new InternalResponse(responseStatus, statusText);
|
|
|
|
nsRefPtr<FillResponseHeaders> visitor = new FillResponseHeaders(response);
|
|
rv = channel->VisitResponseHeaders(visitor);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
NS_WARNING("Failed to visit all headers.");
|
|
}
|
|
|
|
// We open a pipe so that we can immediately set the pipe's read end as the
|
|
// response's body. Setting the segment size to UINT32_MAX means that the
|
|
// pipe has infinite space. The nsIChannel will continue to buffer data in
|
|
// xpcom events even if we block on a fixed size pipe. It might be possible
|
|
// to suspend the channel and then resume when there is space available, but
|
|
// for now use an infinite pipe to avoid blocking.
|
|
nsCOMPtr<nsIInputStream> pipeInputStream;
|
|
rv = NS_NewPipe(getter_AddRefs(pipeInputStream),
|
|
getter_AddRefs(mPipeOutputStream),
|
|
0, /* default segment size */
|
|
UINT32_MAX /* infinite pipe */);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
// Cancel request.
|
|
return rv;
|
|
}
|
|
response->SetBody(pipeInputStream);
|
|
|
|
nsCOMPtr<nsISupports> securityInfo;
|
|
rv = channel->GetSecurityInfo(getter_AddRefs(securityInfo));
|
|
if (securityInfo) {
|
|
response->SetSecurityInfo(securityInfo);
|
|
}
|
|
|
|
// Resolves fetch() promise which may trigger code running in a worker. Make
|
|
// sure the Response is fully initialized before calling this.
|
|
mResponse = BeginAndGetFilteredResponse(response);
|
|
|
|
nsCOMPtr<nsIEventTarget> sts = do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &rv);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
// Cancel request.
|
|
return rv;
|
|
}
|
|
|
|
// Try to retarget off main thread.
|
|
nsCOMPtr<nsIThreadRetargetableRequest> rr = do_QueryInterface(aRequest);
|
|
if (rr) {
|
|
rr->RetargetDeliveryTo(sts);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::OnDataAvailable(nsIRequest* aRequest,
|
|
nsISupports* aContext,
|
|
nsIInputStream* aInputStream,
|
|
uint64_t aOffset,
|
|
uint32_t aCount)
|
|
{
|
|
uint32_t aRead;
|
|
MOZ_ASSERT(mResponse);
|
|
MOZ_ASSERT(mPipeOutputStream);
|
|
|
|
nsresult rv = aInputStream->ReadSegments(NS_CopySegmentToStream,
|
|
mPipeOutputStream,
|
|
aCount, &aRead);
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::OnStopRequest(nsIRequest* aRequest,
|
|
nsISupports* aContext,
|
|
nsresult aStatusCode)
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
if (mPipeOutputStream) {
|
|
mPipeOutputStream->Close();
|
|
}
|
|
|
|
if (NS_FAILED(aStatusCode)) {
|
|
FailWithNetworkError();
|
|
return aStatusCode;
|
|
}
|
|
|
|
ContinueHttpFetchAfterNetworkFetch();
|
|
return NS_OK;
|
|
}
|
|
|
|
// This is called when the channel is redirected.
|
|
NS_IMETHODIMP
|
|
FetchDriver::AsyncOnChannelRedirect(nsIChannel* aOldChannel,
|
|
nsIChannel* aNewChannel,
|
|
uint32_t aFlags,
|
|
nsIAsyncVerifyRedirectCallback *aCallback)
|
|
{
|
|
NS_PRECONDITION(aNewChannel, "Redirect without a channel?");
|
|
|
|
nsresult rv;
|
|
|
|
// Section 4.2, Step 4.6-4.7, enforcing a redirect count is done by Necko.
|
|
// The pref used is "network.http.redirection-limit" which is set to 20 by
|
|
// default.
|
|
//
|
|
// Step 4.8. We only unset this for spec compatibility. Any actions we take
|
|
// on mRequest here do not affect what the channel does.
|
|
mRequest->UnsetSameOriginDataURL();
|
|
|
|
//
|
|
// Requests that require preflight are not permitted to redirect.
|
|
// Fetch spec section 4.2 "HTTP Fetch", step 4.9 just uses the manual
|
|
// redirect flag to decide whether to execute step 4.10 or not. We do not
|
|
// represent it in our implementation.
|
|
// The only thing we do is to check if the request requires a preflight (part
|
|
// of step 4.9), in which case we abort. This part cannot be done by
|
|
// nsCORSListenerProxy since it does not have access to mRequest.
|
|
// which case. Step 4.10.3 is handled by OnRedirectVerifyCallback(), and all
|
|
// the other steps are handled by nsCORSListenerProxy.
|
|
if (!NS_IsInternalSameURIRedirect(aOldChannel, aNewChannel, aFlags)) {
|
|
rv = DoesNotRequirePreflight(aNewChannel);
|
|
if (NS_FAILED(rv)) {
|
|
NS_WARNING("FetchDriver::OnChannelRedirect: "
|
|
"DoesNotRequirePreflight returned failure");
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
mRedirectCallback = aCallback;
|
|
mOldRedirectChannel = aOldChannel;
|
|
mNewRedirectChannel = aNewChannel;
|
|
|
|
nsCOMPtr<nsIChannelEventSink> outer =
|
|
do_GetInterface(mNotificationCallbacks);
|
|
if (outer) {
|
|
// The callee is supposed to call OnRedirectVerifyCallback() on success,
|
|
// and nobody has to call it on failure, so we can just return after this
|
|
// block.
|
|
rv = outer->AsyncOnChannelRedirect(aOldChannel, aNewChannel, aFlags, this);
|
|
if (NS_FAILED(rv)) {
|
|
aOldChannel->Cancel(rv);
|
|
mRedirectCallback = nullptr;
|
|
mOldRedirectChannel = nullptr;
|
|
mNewRedirectChannel = nullptr;
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
(void) OnRedirectVerifyCallback(NS_OK);
|
|
return NS_OK;
|
|
}
|
|
|
|
// Returns NS_OK if no preflight is required, error otherwise.
|
|
nsresult
|
|
FetchDriver::DoesNotRequirePreflight(nsIChannel* aChannel)
|
|
{
|
|
// If this is a same-origin request or the channel's URI inherits
|
|
// its principal, it's allowed.
|
|
if (nsContentUtils::CheckMayLoad(mPrincipal, aChannel, true)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Check if we need to do a preflight request.
|
|
nsCOMPtr<nsIHttpChannel> httpChannel = do_QueryInterface(aChannel);
|
|
NS_ENSURE_TRUE(httpChannel, NS_ERROR_DOM_BAD_URI);
|
|
|
|
nsAutoCString method;
|
|
httpChannel->GetRequestMethod(method);
|
|
if (mRequest->Mode() == RequestMode::Cors_with_forced_preflight ||
|
|
!mRequest->Headers()->HasOnlySimpleHeaders() ||
|
|
(!method.LowerCaseEqualsLiteral("get") &&
|
|
!method.LowerCaseEqualsLiteral("post") &&
|
|
!method.LowerCaseEqualsLiteral("head"))) {
|
|
return NS_ERROR_DOM_BAD_URI;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::GetInterface(const nsIID& aIID, void **aResult)
|
|
{
|
|
if (aIID.Equals(NS_GET_IID(nsIChannelEventSink))) {
|
|
*aResult = static_cast<nsIChannelEventSink*>(this);
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult rv;
|
|
|
|
if (mNotificationCallbacks) {
|
|
rv = mNotificationCallbacks->GetInterface(aIID, aResult);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
NS_ASSERTION(*aResult, "Lying nsIInterfaceRequestor implementation!");
|
|
return rv;
|
|
}
|
|
}
|
|
else if (aIID.Equals(NS_GET_IID(nsIStreamListener))) {
|
|
*aResult = static_cast<nsIStreamListener*>(this);
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
else if (aIID.Equals(NS_GET_IID(nsIRequestObserver))) {
|
|
*aResult = static_cast<nsIRequestObserver*>(this);
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
|
|
return QueryInterface(aIID, aResult);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::OnRedirectVerifyCallback(nsresult aResult)
|
|
{
|
|
// On a successful redirect we perform the following substeps of Section 4.2,
|
|
// step 4.10.
|
|
if (NS_SUCCEEDED(aResult)) {
|
|
// Step 4.10.3 "Set request's url to locationURL." so that when we set the
|
|
// Response's URL from the Request's URL in Section 4, step 6, we get the
|
|
// final value.
|
|
nsCOMPtr<nsIURI> newURI;
|
|
nsresult rv = NS_GetFinalChannelURI(mNewRedirectChannel, getter_AddRefs(newURI));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
aResult = rv;
|
|
} else {
|
|
// We need to update our request's URL.
|
|
nsAutoCString newUrl;
|
|
newURI->GetSpec(newUrl);
|
|
mRequest->SetURL(newUrl);
|
|
}
|
|
}
|
|
|
|
if (NS_FAILED(aResult)) {
|
|
mOldRedirectChannel->Cancel(aResult);
|
|
}
|
|
|
|
mOldRedirectChannel = nullptr;
|
|
mNewRedirectChannel = nullptr;
|
|
mRedirectCallback->OnRedirectVerifyCallback(aResult);
|
|
mRedirectCallback = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
FetchDriver::SetDocument(nsIDocument* aDocument)
|
|
{
|
|
// Cannot set document after Fetch() has been called.
|
|
MOZ_ASSERT(mFetchRecursionCount == 0);
|
|
mDocument = aDocument;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|