mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
7f8ba9c1d7
- Bug 1236786 - [WebGL2] pass getVertexAttrib in gl-object-get-calls.html, r=jgilbert (60a2c91a38)
- Bug 1233046 - Fix OES_texture_float on OSX. - r=jrmuizel (4bc0059f5f)
- Bug 1233557 - Allow RGB8 to be renderable again for web-compat. - r=jrmuizel (4c13bfd8e8)
- Bug 1233549. Disallow ES3 compressed texture formats. r=jgilbert (1073033161)
- Bug 1241702 - Allow unsized DEPTH_STENCIL for RBs in WebGL 2. - r=kamidphish (87d17d2cf9)
- Bug 1239126. Handle gl_InstanceID attribute with no location. r=jgilbert (4894997e98)
- Bug 1236782 - [WebGL2] pass getProgramParameter in gl-object-get-calls.html; r=jgilbert (2136fcce48)
- Bug 1232462. Only ask for a higher version of GLSL when using WebGL2. r=jgilbert (0317be4eb4)
- Bug 1242330 - "Four extensions were promoted to core in WebGL 2 and should no longer be available as extensions." r=jgilbert r=jmuizelaar (6df020b8d4)
- Bug 1233626 - Default MaxDrawingBuffers to 1 unless ext/webgl2. - r=jrmuizel (a7580d661c)
- Bug 1231657. Don't allow linking different versions shaders. r=jgilbert (e610f98066)
- Bug 1241777 - TexCompareFunc should be stored in ascending order. r=jgilbert (b6151a0076)
- Bug 1228885 - Implement WebGLTexture::MemoryUsage. - r=kamidphish (ea06815414)
- Bug 1239259 - Fix WebGL2 generateMipmap checking. r=jgilbert (39f587c421)
- Bug 1242347 - Allow unsized internal format when generate mipmap. r=jgilbert (b203a8898c)
- Bug 1232502. Use the correct internalFormat when calling CopyTexImage2D. r=jgilbert (eeaef3215e)
- Bug 1243663 - Max uniform and attribute location lengths in WebGL2 should be 1024. r=jgilbert (c4ec6de507)
- Bug 1239488 - Add int/uint to vertex attrib data type. r=jgilbert (11b4968025)
- Bug 1184242 - Remove aTabParent != sActiveTabParent warning from IMEStateManager::SetInputContextForChildProcess. r=masayuki (0fcda10e15)
- Bug 1178652 - Send NOTIFY_IME_OF_COMPOSITION_UPDATE to parent process correctly. r=masayuki (bce28e2c91)
- Bug 1107782 - Only accept certain mouse, gamepad events as user-active. r=smaug (00542c80b9)
- Bug 1247850 - Shrink NameTableKey in nsStaticCaseInsensitiveNameTable. r=froydnj,erahm. (ce3cb3edfb)
- Bug 1247359 - micro-optimize the common case of String{Begins,End}With; r=erahm (333e042b31)
- Bug 1239125. Add operator!=(char_type*) to nsTSubstring. r=froydnj (0cc047a9a1)
- Bug 1213862 - Align nsString whitespace handling with web specs; r=froydnj (db5b11ca52)
- Bug 1141884 - Trigger compositor smooth scrolling to snap points when APZ is enabled. r=mstange,kip (593af59f2a)
- Bug 1244582: Add back in a null check that was accidentally removed. r=smaug (76bff1b01f)
- Bug 1234176 - Introduce and use the WriteSysFile() helper function. r=dhylands (22a46fbe8b)
- missing bit of Bug 1198124 - Enable -Wshadow (f84535a7a2)
- Bug 1249171 - Simplify nsCOMArray::SizeOfExcludingThis(). r=erahm. (57efdce1c6)
- Bug 1156416 - Validate camera parameters supplied by the application. r=mikeh (f8b4b84ccf)
- Bug 1186808 - Replace nsBaseHashtable::EnumerateRead() calls in dom/camera/ with iterators. r=mikeh. (7b1db5f6a1)
- Bug 1158378 - Fix how a failed set configuration call would try to shutdown the camera after release. (9d5e323bca)
- Bug 1171374 - Permit software video codecs with the emulated camera. r=sotaro (c1ae26ea0d)
- Bug 1234458 P1 Allow the CacheChild to be "locked" into memory so it will delay destruction. r=ehsan a=ritu (9e46185779)
- Bug 1234458 P2 Lock the CacheChild actor while Cache DOM methods are running. r=ehsan a=ritu (038342a6e2)
- Bug 1244764 P1 Make Cache .add()/.addAll() fail if a Response.ok() is false. r=ehsan (ae26ca9ef1)
- Bug 1172562 - Clear QuotaManager storage when uninstalling an app. Test. r=bkelly (b07311a3b7)
- Bug 1172629 - Use the caches global property from an iframe loaded after setting the pref in order to make the tests pass with the pref disabled; r=bkelly a=RyanVM (e7c05d8b79)
- Bug 1244764 P2 Make dom/cache mochitests pass with new add()/addAll() behavior. r=ehsan (e1f667c1b4)
- Bug 1244764 P3 Make service worker tests pass with new Cache add()/addAll() behavior. r=ehsan (1518ae5225)
- Bug 1003860 - Simplify storage setup tasks in storage inspector tests. r=mratcliffe (249a8bdb2b)
- Bug 1003860 - Service worker cache for storage actor. r=mratcliffe (5c3d1ecd0c)
- Bug 1244764 P5 Fix devtools test to work with new Cache add()/addAll() behavior. r=ehsan (bf85405de8)
- Bug 1232901 - Use channel.asyncOpen2 within dom/browser-element/BrowserElementParent.js (r=sicking,aus) (2a228ed551)
- Bug 1180330 - http auth prompt shown when opening browser if prompt canceled/dismissed earlier. r=fabrice (ba3666f4bd)
- Bug 1234118 - Delete code for supporting 'do-command' and 'copypaste-docommand'. r=mtseng, r=smaug (b1b575d3c5)
- Bug 1238883 - [TV Browser] It shows "The page cannot be displayed" when user browse some webpages. r=roc (e6d7739dd6)
- Bug 1238440 - FileReader should throw an error when the blob changed size when reading, r=khuey (b006adba10)
- Bug 1230422 - FileReader should handle nested ReadAs*() calls. r=khuey (5a3ff84a31)
- Bug 1225202, part 3 - Create files in test_fileapi_slice.html using SpecialPowers.createFiles. r=baku (1137975548)
- Bug 1241171 - FormData should not force 'blob' as filename, r=smaug (748055f751)
- Bug 1246375 - Restore the previous spec version of FormData, r=smaug (3586af2b88)
- Bug 1237183 - Modify implementation of reading preference. r=seanlin (a132bc7246)
- Bug 801545 - Remove DocumentType.internalSubset, r=bz (ea30c9b5ee)
- Bug 1226440 - Expose a method to get a node's immediate dominator; r=bz,sfink (f77ae44037)
- Bug 825318 - Implement adoptDownload for mozDownloadManager, r=aus, r=sicking (e98cb05210)
- Bug 1237370 - Always log the reason for remote AppRep lookup failures. r=gcp (2c804e68fc)
- Bug 1167493 - Application Reputation: disable remote lookup of zip files on Mac/Linux, r=gcp (517459e064)
- Bug 1195519 - Use channel->ascynOpen2 toolkit/components/downloads/ApplicationReputation.cpp (r=sicking) (2856e5213a)
- Bug 1237856 - Add prefs to honor/ignore Application Reputation verdicts. r=gcp (54ee06264f)
- Bug 1243643 - Deprecate unsafe CPOW usage in contentAreaUtils' saveImage. r=jld (6ae790f1ef)
- Bug 1229224: Add an eslint plugin for importing all browser.js globals for browser-chrome tests. r=miker (9df52a7f3b)
- Bug 1245916: Add additional browser window scripts to eslint globals. r=felipe (92d316ca5e)
- Bug 1246244 - Allow non-CPOW documents to pass through saveImageURL properly. r=jaws,Margaret (c8d4ca241d)
- some missing bits after world fix (c0439eebb0)
- add some missing stuff (ddbd47dc03)
- bissing bit of 1229519 (4e255c3dae)
- Bug 1199662 - Crash ping environment block is broken when any string field contains a quotation mark. Unescape INI fields properly using the library that already exists for the purpose. r=ted (874a999edc)
- Bug 1216150 - Turn on the experimental Intl.DateTimeFormat.prototype.formatToParts in b2g certified apps. r=fabrice (40eeb1a4d4)
- Bug 1216150 - Mini-bustage fix for something I think I unintentionally qref'd into the final patch. r=bustage in a CLOSED TREE (36d9b21a67)
- Bug 1141311 - Add async mode support to GonkNativeWindow on Lollipop Gonk r=pchang (39d9d56326)
- Bug 1146671 - Ensure camera not already released when performing operations. r=dhylands (71b59caa1f)
- Bug 1248737. Improve documentation for WorkerRunnable and associated classes. r=khuey (4ff57790c5)
- Bug 1235629 - Remove dead code in WorkerFeature.h, r=smaug (75a51fcf03)
- Bug 1212333 - WorkerDebuggerManager should live on the main thread;r=khuey (11fdfbbae6)
- Bug 1226443 P3 Re-enable service worker update wpt tests. r=ehsan (605dac5f9e)
- Bug 1226443 P4 Cleanup ServiceWorkerScriptCache objects when initialization fails. r=ehsan (43de3429a2)
- Bug 1234127: Change |BluetoothAdapter.pairingReqs| as a nullable object; r=btian, r=mrbkap (45d2038f6a)
- Bug 1188487 - BrowserElement webidl changes for muting and setting volume. r=ehsan (21bea70a07)
- Bug 1238210 - Correct the Promise return types on two Clients methods; r=baku (fa41b25df0)
- Bug 1246784 - Expose Console to the WorkerDebuggerGlobalScope - part 2, r=khuey (0da9ce8ff6)
- Bug 1228702. Don't expose the 'location' property of Exception/DOMException on workers. r=bholley (0fe86ea586)
- Bug 1223825 - Change Directory.path to include the directory's name. r=baku (0cdae4c2f0)
- Bug 1238225 - Mark ExtendableMessageEvent.ports as SameObject; r=baku (45b9a9746f)
- Bug 1236933 - Return null from FetchEvent.clientId for non-subresource network requests; r=bkelly (4a9c4b40cb)
- Bug 1238213 - Make FetchEvent.request non-nullable; r=baku (751082c8ba)
- Bug 1193125 - Avoid corrupting image data in test_fetch_event.html. r=bkelly (9f6bff232f)
- Bug 1201664 - Avoid using Request's constructor when creating FetchEvent.request; r=bkelly (7a3401e345)
- Bug 1175944 - Packaged app's (app://) JS files are not loaded and do not trigger "onfetch" handler. r=jdm (62df139153)
- Bug 1233644 - use pattern matching when listening clear-origin-data. r=baku (ea2594f50e)
- Bug 1237363 - Part 1: Unregister all service workers registered in mochitests at the end of the test; r=jdm (5be97e5bb0)
- Bug 1237363 - Part 2: Fail mochitests which register a service worker without unregistering it; r=jdm (c4160ffd5f)
- Bug 1237363 - Part 3: Add a test for a mochitest finishing without unregistering its service worker; r=jdm (911d37291b)
- Bug 1174078 - Calling "fetch" inside Service Worker's "onfetch" handler in b2g causes "onfetch" again that leads to an infinite loop. Test. r=nsm (208451f346)
- Bug 1197379 - Remove support for intercepting app:// URIs using service workers; r=jdm (3cbdd725f1)
- Bug 1179399 - Part 1: Relax the ShouldIntercept checks when overriding JAR channel info; r=jdm (850bb2bdb8)
- Bug 1238213 follow-up: Mark the FetchEventInit dictionary argument to FetchEvent's constructor optional too; r=bzbarsky (356cbe6db7)
- Bug 1232732 - modify NS_WARNING in MOZ_WIN_MEM_TRY_CATCH; r=aklotz (e2be4d6919)
- Bug 1247658 - Expose a method to JS for find the shortest retaining paths of some nodes in a heap snapshot; r=bz r=jimb (2c82198808)
- Bug 1188115: Expose IDBCursorWithValue in workers. r=baku (e1c40aeb6e)
- Bug 1162680 - Notify Keyboard.jsm to send blur event when the message manager is closed first. r=timdream (53727ab300)
- Bug 1192986 Also mark Cache/CacheStorage as release interfaces on workers. r=ehsan a=bustage (25cf83c154)
- Bug 1159742. Get rid of the pref annotation from test_interfaces, since it basically corresponds to disabling the test. r=jst (c229e3f881)
- Bug 1203160 - Part 2: Fix the interfaces tests to allow SW interfaces for non-release Fennec; r=baku (072840db1f)
- Bug 1197700 - Correct mistakes in InputMethod.webidl. r=kanru, r=janjongboom, sr=smaug (4edb6f201f)
- Bug 1206970 - Stop expecting AnimationPlaybackEvent to be exposed on release branches, where it's disabled by pref, r=smaug (30ae2b13db)
- Bug 1177276 - Pref on canvas.captureStream by default. r=smaug,mt (0cfe0f72f2)
- Bug 1215147 - Enable VR API's on FF for Android by default. r=snorp, r=vlad, r=bz (5ff3725318)
- Bug 1218482 - Enable WebVR By Default,r=bz (f26111ed82)
- Bug 1159755. Stop forcing the media.eme.apiVisible preference to be true in our test harness. r=cpearce (09f7887917)
- Bug 1149312 - Obtain test coverage for the file-backed case of MediaRecorder. r=roc (bd2e7e40f0)
- Bug 1154559 - Remove flaky timeouts from manifest.js and register SimpleTest.registerCleanupFunction() to report unfinished tests. r=cpearce. (eb68db0fb2)
- Bug 1154564 - Add the ability to notify timeouts to MediaTestManager and remove flaky timeouts from test_playback.html. r=cpearce. (c89b4e58d9)
- Bug 1135170 - Fix up racey test_seek-1.html. rpending=mattwoodrow (b3a7d0dcd6)
- Bug 902686 - Change manifest.js to use SpecialPowers.pushPrefEnv. r=edwin (636b0edc1a)
- Bug 1183502 - give androidVersion a correct value in manifest.js. r=sotaro. (933e9ea712)
- Bug 1235588 - add null check to SimpleTest. r=bechen. (958ede68de)
- misspatch (c8922447ff)
- Bug 1151740 - pass the callback object as-is to SpecialPowers.exactGC(). r=edwin (99ca873bce)
- Bug 1197682 - InputMethodManager#setSupportsSwitchingTypes, r=janjongboom, sr=smaug (e7eb54e491)
- Bug 1201407 - Add input-manage-only events for InputMethod API. r=janjongboom, sr=smaug (776d064bd1)
- Bug 1234459 - Expose full text in the input box to InputMethod API, r=masayuki, sr=smaug (4fa0554356)
- Bug 1198163 - Workaround Mochitest app and assign frame proper permissions, r=kanru (c3bcf8ecc1)
- Bug 990250 - Fold nsIStyleSheet into CSSStyleSheet. r=dbaron (23579cb300)
697 lines
23 KiB
C++
697 lines
23 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* 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/DebugOnly.h"
|
|
#include "mozilla/dom/FetchDriver.h"
|
|
|
|
#include "nsIAsyncVerifyRedirectCallback.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 "nsIInterfaceRequestorUtils.h"
|
|
#include "nsIPipe.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 "nsHttpChannel.h"
|
|
|
|
#include "mozilla/dom/File.h"
|
|
#include "mozilla/dom/workers/Workers.h"
|
|
#include "mozilla/unused.h"
|
|
|
|
#include "Fetch.h"
|
|
#include "InternalRequest.h"
|
|
#include "InternalResponse.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_ISUPPORTS(FetchDriver,
|
|
nsIStreamListener, nsIInterfaceRequestor,
|
|
nsIThreadRetargetableStreamListener)
|
|
|
|
FetchDriver::FetchDriver(InternalRequest* aRequest, nsIPrincipal* aPrincipal,
|
|
nsILoadGroup* aLoadGroup)
|
|
: mPrincipal(aPrincipal)
|
|
, mLoadGroup(aLoadGroup)
|
|
, mRequest(aRequest)
|
|
, mResponseAvailableCalled(false)
|
|
, mFetchCalled(false)
|
|
{
|
|
}
|
|
|
|
FetchDriver::~FetchDriver()
|
|
{
|
|
// We assert this since even on failures, we should call
|
|
// FailWithNetworkError().
|
|
MOZ_ASSERT(mResponseAvailableCalled);
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::Fetch(FetchDriverObserver* aObserver)
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
MOZ_ASSERT(!mFetchCalled);
|
|
mFetchCalled = true;
|
|
|
|
mObserver = aObserver;
|
|
|
|
Telemetry::Accumulate(Telemetry::SERVICE_WORKER_REQUEST_PASSTHROUGH,
|
|
mRequest->WasCreatedByFetchEvent());
|
|
|
|
// FIXME(nsm): Deal with HSTS.
|
|
|
|
MOZ_RELEASE_ASSERT(!mRequest->IsSynchronous(),
|
|
"Synchronous fetch not supported");
|
|
|
|
nsCOMPtr<nsIRunnable> r =
|
|
NS_NewRunnableMethod(this, &FetchDriver::ContinueFetch);
|
|
return NS_DispatchToCurrentThread(r);
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::ContinueFetch()
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
|
|
nsresult rv = HttpFetch();
|
|
if (NS_FAILED(rv)) {
|
|
FailWithNetworkError();
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
// 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()
|
|
{
|
|
// Step 1. "Let response be null."
|
|
mResponse = nullptr;
|
|
nsresult rv;
|
|
|
|
nsCOMPtr<nsIIOService> ios = do_GetIOService(&rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
nsAutoCString url;
|
|
mRequest->GetURL(url);
|
|
nsCOMPtr<nsIURI> uri;
|
|
rv = NS_NewURI(getter_AddRefs(uri),
|
|
url,
|
|
nullptr,
|
|
nullptr,
|
|
ios);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Unsafe requests aren't allowed with when using no-core mode.
|
|
if (mRequest->Mode() == RequestMode::No_cors &&
|
|
mRequest->UnsafeRequest() &&
|
|
(!mRequest->HasSimpleMethod() ||
|
|
!mRequest->Headers()->HasOnlySimpleHeaders())) {
|
|
MOZ_ASSERT(false, "The API should have caught this");
|
|
return NS_ERROR_DOM_BAD_URI;
|
|
}
|
|
|
|
// 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 one of
|
|
// - request's credentials mode is "include"
|
|
// - request's credentials mode is "same-origin" and either the CORS flag
|
|
// is unset or response tainting is "opaque"
|
|
// is true, and unset otherwise."
|
|
|
|
// 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.
|
|
const nsLoadFlags bypassFlag = mRequest->SkipServiceWorker() ?
|
|
nsIChannel::LOAD_BYPASS_SERVICE_WORKER : 0;
|
|
|
|
nsSecurityFlags secFlags = nsILoadInfo::SEC_ABOUT_BLANK_INHERITS;
|
|
if (mRequest->Mode() == RequestMode::Cors) {
|
|
secFlags |= nsILoadInfo::SEC_REQUIRE_CORS_DATA_INHERITS;
|
|
} else if (mRequest->Mode() == RequestMode::Same_origin ||
|
|
mRequest->Mode() == RequestMode::Navigate) {
|
|
secFlags |= nsILoadInfo::SEC_REQUIRE_SAME_ORIGIN_DATA_INHERITS;
|
|
} else if (mRequest->Mode() == RequestMode::No_cors) {
|
|
secFlags |= nsILoadInfo::SEC_ALLOW_CROSS_ORIGIN_DATA_INHERITS;
|
|
} else {
|
|
MOZ_ASSERT_UNREACHABLE("Unexpected request mode!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
if (mRequest->GetRedirectMode() != RequestRedirect::Follow) {
|
|
secFlags |= nsILoadInfo::SEC_DONT_FOLLOW_REDIRECTS;
|
|
}
|
|
|
|
// This is handles the use credentials flag in "HTTP
|
|
// network or cache fetch" in the spec and decides whether to transmit
|
|
// cookies and other identifying information.
|
|
if (mRequest->GetCredentialsMode() == RequestCredentials::Include) {
|
|
secFlags |= nsILoadInfo::SEC_COOKIES_INCLUDE;
|
|
} else if (mRequest->GetCredentialsMode() == RequestCredentials::Omit) {
|
|
secFlags |= nsILoadInfo::SEC_COOKIES_OMIT;
|
|
} else if (mRequest->GetCredentialsMode() == RequestCredentials::Same_origin) {
|
|
secFlags |= nsILoadInfo::SEC_COOKIES_SAME_ORIGIN;
|
|
} else {
|
|
MOZ_ASSERT_UNREACHABLE("Unexpected credentials mode!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
// 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;
|
|
|
|
nsLoadFlags loadFlags = nsIRequest::LOAD_NORMAL |
|
|
bypassFlag | nsIChannel::LOAD_CLASSIFY_URI;
|
|
if (mDocument) {
|
|
MOZ_ASSERT(mDocument->NodePrincipal() == mPrincipal);
|
|
rv = NS_NewChannel(getter_AddRefs(chan),
|
|
uri,
|
|
mDocument,
|
|
secFlags,
|
|
mRequest->ContentPolicyType(),
|
|
mLoadGroup,
|
|
nullptr, /* aCallbacks */
|
|
loadFlags,
|
|
ios);
|
|
} else {
|
|
rv = NS_NewChannel(getter_AddRefs(chan),
|
|
uri,
|
|
mPrincipal,
|
|
secFlags,
|
|
mRequest->ContentPolicyType(),
|
|
mLoadGroup,
|
|
nullptr, /* aCallbacks */
|
|
loadFlags,
|
|
ios);
|
|
}
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mLoadGroup = nullptr;
|
|
|
|
// 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);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Set the same headers.
|
|
AutoTArray<InternalHeaders::Entry, 5> headers;
|
|
mRequest->Headers()->GetEntries(headers);
|
|
bool hasAccept = false;
|
|
for (uint32_t i = 0; i < headers.Length(); ++i) {
|
|
if (!hasAccept && headers[i].mName.EqualsLiteral("accept")) {
|
|
hasAccept = true;
|
|
}
|
|
if (headers[i].mValue.IsEmpty()) {
|
|
httpChan->SetEmptyRequestHeader(headers[i].mName);
|
|
} else {
|
|
httpChan->SetRequestHeader(headers[i].mName, headers[i].mValue, false /* merge */);
|
|
}
|
|
}
|
|
|
|
if (!hasAccept) {
|
|
httpChan->SetRequestHeader(NS_LITERAL_CSTRING("accept"),
|
|
NS_LITERAL_CSTRING("*/*"),
|
|
false /* merge */);
|
|
}
|
|
|
|
// Step 2. Set the referrer.
|
|
nsAutoString referrer;
|
|
mRequest->GetReferrer(referrer);
|
|
if (referrer.EqualsLiteral(kFETCH_CLIENT_REFERRER_STR)) {
|
|
rv = nsContentUtils::SetFetchReferrerURIWithPolicy(mPrincipal,
|
|
mDocument,
|
|
httpChan);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
} else if (referrer.IsEmpty()) {
|
|
rv = httpChan->SetReferrerWithPolicy(nullptr, net::RP_No_Referrer);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
} else {
|
|
// From "Determine request's Referrer" step 3
|
|
// "If request's referrer is a URL, let referrerSource be request's
|
|
// referrer."
|
|
//
|
|
// XXXnsm - We never actually hit this from a fetch() call since both
|
|
// fetch and Request() create a new internal request whose referrer is
|
|
// always set to about:client. Should we just crash here instead until
|
|
// someone tries to use FetchDriver for non-fetch() APIs?
|
|
nsCOMPtr<nsIURI> referrerURI;
|
|
rv = NS_NewURI(getter_AddRefs(referrerURI), referrer, nullptr, nullptr);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
rv =
|
|
httpChan->SetReferrerWithPolicy(referrerURI,
|
|
mDocument ? mDocument->GetReferrerPolicy() :
|
|
net::RP_Default);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
// Step 3 "If HTTPRequest's force Origin header flag is set..."
|
|
if (mRequest->ForceOriginHeader()) {
|
|
nsAutoString origin;
|
|
rv = nsContentUtils::GetUTFOrigin(mPrincipal, origin);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
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,
|
|
|
|
nsCOMPtr<nsIHttpChannelInternal> internalChan = do_QueryInterface(httpChan);
|
|
|
|
// Conversion between enumerations is safe due to static asserts in
|
|
// dom/workers/ServiceWorkerManager.cpp
|
|
internalChan->SetCorsMode(static_cast<uint32_t>(mRequest->Mode()));
|
|
internalChan->SetRedirectMode(static_cast<uint32_t>(mRequest->GetRedirectMode()));
|
|
}
|
|
|
|
// 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 result.StealNSResult();
|
|
}
|
|
|
|
nsCOMPtr<nsIInputStream> bodyStream;
|
|
mRequest->GetBody(getter_AddRefs(bodyStream));
|
|
if (bodyStream) {
|
|
nsAutoCString method;
|
|
mRequest->GetMethod(method);
|
|
rv = uploadChan->ExplicitSetUploadStream(bodyStream, contentType, -1, method, false /* aStreamHasHeaders */);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
}
|
|
|
|
// If preflight is required, start a "CORS preflight fetch"
|
|
// https://fetch.spec.whatwg.org/#cors-preflight-fetch-0. All the
|
|
// implementation is handled by the http channel calling into
|
|
// nsCORSListenerProxy. We just inform it which unsafe headers are included
|
|
// in the request.
|
|
if (mRequest->Mode() == RequestMode::Cors) {
|
|
AutoTArray<nsCString, 5> unsafeHeaders;
|
|
mRequest->Headers()->GetUnsafeHeaders(unsafeHeaders);
|
|
nsCOMPtr<nsILoadInfo> loadInfo = chan->GetLoadInfo();
|
|
loadInfo->SetCorsPreflightInfo(unsafeHeaders, false);
|
|
}
|
|
|
|
rv = chan->AsyncOpen2(this);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Step 4 onwards of "HTTP Fetch" is handled internally by Necko.
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<InternalResponse>
|
|
FetchDriver::BeginAndGetFilteredResponse(InternalResponse* aResponse,
|
|
nsIURI* aFinalURI,
|
|
bool aFoundOpaqueRedirect)
|
|
{
|
|
MOZ_ASSERT(aResponse);
|
|
nsAutoCString reqURL;
|
|
if (aFinalURI) {
|
|
aFinalURI->GetSpec(reqURL);
|
|
} else {
|
|
mRequest->GetURL(reqURL);
|
|
}
|
|
DebugOnly<nsresult> rv = aResponse->StripFragmentAndSetUrl(reqURL);
|
|
MOZ_ASSERT(NS_SUCCEEDED(rv));
|
|
|
|
RefPtr<InternalResponse> filteredResponse;
|
|
if (aFoundOpaqueRedirect) {
|
|
filteredResponse = aResponse->OpaqueRedirectResponse();
|
|
} else {
|
|
switch (mRequest->GetResponseTainting()) {
|
|
case LoadTainting::Basic:
|
|
filteredResponse = aResponse->BasicResponse();
|
|
break;
|
|
case LoadTainting::CORS:
|
|
filteredResponse = aResponse->CORSResponse();
|
|
break;
|
|
case LoadTainting::Opaque:
|
|
filteredResponse = aResponse->OpaqueResponse();
|
|
break;
|
|
default:
|
|
MOZ_CRASH("Unexpected case");
|
|
}
|
|
}
|
|
|
|
MOZ_ASSERT(filteredResponse);
|
|
MOZ_ASSERT(mObserver);
|
|
mObserver->OnResponseAvailable(filteredResponse);
|
|
mResponseAvailableCalled = true;
|
|
return filteredResponse.forget();
|
|
}
|
|
|
|
nsresult
|
|
FetchDriver::FailWithNetworkError()
|
|
{
|
|
workers::AssertIsOnMainThread();
|
|
RefPtr<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();
|
|
|
|
// Note, this can be called multiple times if we are doing an opaqueredirect.
|
|
// In that case we will get a simulated OnStartRequest() and then the real
|
|
// channel will call in with an errored OnStartRequest().
|
|
|
|
nsresult rv;
|
|
aRequest->GetStatus(&rv);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
// We should only get to the following code once.
|
|
MOZ_ASSERT(!mPipeOutputStream);
|
|
MOZ_ASSERT(mObserver);
|
|
|
|
RefPtr<InternalResponse> response;
|
|
nsCOMPtr<nsIChannel> channel = do_QueryInterface(aRequest);
|
|
nsCOMPtr<nsIHttpChannel> httpChannel = do_QueryInterface(aRequest);
|
|
|
|
// On a successful redirect we perform the following substeps of HTTP Fetch,
|
|
// step 5, "redirect status", step 11.
|
|
|
|
// Step 11.5 "Append locationURL to request's url list." so that when we set the
|
|
// Response's URL from the Request's URL in Main Fetch, step 15, we get the
|
|
// final value. Note, we still use a single URL value instead of a list.
|
|
// Because of that we only need to do this after the request finishes.
|
|
nsCOMPtr<nsIURI> newURI;
|
|
rv = NS_GetFinalChannelURI(channel, getter_AddRefs(newURI));
|
|
if (NS_FAILED(rv)) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
nsAutoCString newUrl;
|
|
newURI->GetSpec(newUrl);
|
|
mRequest->SetURL(newUrl);
|
|
|
|
bool foundOpaqueRedirect = false;
|
|
|
|
if (httpChannel) {
|
|
uint32_t responseStatus;
|
|
httpChannel->GetResponseStatus(&responseStatus);
|
|
|
|
if (mozilla::net::nsHttpChannel::IsRedirectStatus(responseStatus)) {
|
|
if (mRequest->GetRedirectMode() == RequestRedirect::Error) {
|
|
FailWithNetworkError();
|
|
return NS_BINDING_FAILED;
|
|
}
|
|
if (mRequest->GetRedirectMode() == RequestRedirect::Manual) {
|
|
foundOpaqueRedirect = true;
|
|
}
|
|
}
|
|
|
|
nsAutoCString statusText;
|
|
httpChannel->GetResponseStatusText(statusText);
|
|
|
|
response = new InternalResponse(responseStatus, statusText);
|
|
|
|
RefPtr<FillResponseHeaders> visitor = new FillResponseHeaders(response);
|
|
rv = httpChannel->VisitResponseHeaders(visitor);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
NS_WARNING("Failed to visit all headers.");
|
|
}
|
|
} else {
|
|
response = new InternalResponse(200, NS_LITERAL_CSTRING("OK"));
|
|
|
|
ErrorResult result;
|
|
nsAutoCString contentType;
|
|
rv = channel->GetContentType(contentType);
|
|
if (NS_SUCCEEDED(rv) && !contentType.IsEmpty()) {
|
|
nsAutoCString contentCharset;
|
|
channel->GetContentCharset(contentCharset);
|
|
if (NS_SUCCEEDED(rv) && !contentCharset.IsEmpty()) {
|
|
contentType += NS_LITERAL_CSTRING(";charset=") + contentCharset;
|
|
}
|
|
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("Content-Type"),
|
|
contentType,
|
|
result);
|
|
MOZ_ASSERT(!result.Failed());
|
|
}
|
|
|
|
int64_t contentLength;
|
|
rv = channel->GetContentLength(&contentLength);
|
|
if (NS_SUCCEEDED(rv) && contentLength) {
|
|
nsAutoCString contentLenStr;
|
|
contentLenStr.AppendInt(contentLength);
|
|
response->Headers()->Append(NS_LITERAL_CSTRING("Content-Length"),
|
|
contentLenStr,
|
|
result);
|
|
MOZ_ASSERT(!result.Failed());
|
|
}
|
|
}
|
|
|
|
// 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 */,
|
|
true /* non-blocking input, otherwise you deadlock */,
|
|
false /* blocking output, since the pipe is 'in'finite */ );
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
// Cancel request.
|
|
return rv;
|
|
}
|
|
response->SetBody(pipeInputStream);
|
|
|
|
response->InitChannelInfo(channel);
|
|
|
|
nsCOMPtr<nsIURI> channelURI;
|
|
rv = channel->GetURI(getter_AddRefs(channelURI));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
// Cancel request.
|
|
return rv;
|
|
}
|
|
|
|
nsCOMPtr<nsILoadInfo> loadInfo;
|
|
rv = channel->GetLoadInfo(getter_AddRefs(loadInfo));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
FailWithNetworkError();
|
|
return rv;
|
|
}
|
|
|
|
// Propagate any tainting from the channel back to our response here. This
|
|
// step is not reflected in the spec because the spec is written such that
|
|
// FetchEvent.respondWith() just passes the already-tainted Response back to
|
|
// the outer fetch(). In gecko, however, we serialize the Response through
|
|
// the channel and must regenerate the tainting from the channel in the
|
|
// interception case.
|
|
mRequest->MaybeIncreaseResponseTainting(loadInfo->GetTainting());
|
|
|
|
// Resolves fetch() promise which may trigger code running in a worker. Make
|
|
// sure the Response is fully initialized before calling this.
|
|
mResponse = BeginAndGetFilteredResponse(response, channelURI,
|
|
foundOpaqueRedirect);
|
|
|
|
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.
|
|
if (nsCOMPtr<nsIThreadRetargetableRequest> rr = do_QueryInterface(aRequest)) {
|
|
NS_WARN_IF(NS_FAILED(rr->RetargetDeliveryTo(sts)));
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::OnDataAvailable(nsIRequest* aRequest,
|
|
nsISupports* aContext,
|
|
nsIInputStream* aInputStream,
|
|
uint64_t aOffset,
|
|
uint32_t aCount)
|
|
{
|
|
// NB: This can be called on any thread! But we're guaranteed that it is
|
|
// called between OnStartRequest and OnStopRequest, so we don't need to worry
|
|
// about races.
|
|
|
|
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 (NS_FAILED(aStatusCode)) {
|
|
nsCOMPtr<nsIAsyncOutputStream> outputStream = do_QueryInterface(mPipeOutputStream);
|
|
if (outputStream) {
|
|
outputStream->CloseWithStatus(NS_BINDING_FAILED);
|
|
}
|
|
|
|
// We proceed as usual here, since we've already created a successful response
|
|
// from OnStartRequest.
|
|
} else {
|
|
MOZ_ASSERT(mResponse);
|
|
MOZ_ASSERT(!mResponse->IsError());
|
|
|
|
if (mPipeOutputStream) {
|
|
mPipeOutputStream->Close();
|
|
}
|
|
}
|
|
|
|
if (mObserver) {
|
|
mObserver->OnResponseEnd();
|
|
mObserver = nullptr;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::CheckListenerChain()
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
FetchDriver::GetInterface(const nsIID& aIID, void **aResult)
|
|
{
|
|
if (aIID.Equals(NS_GET_IID(nsIStreamListener))) {
|
|
*aResult = static_cast<nsIStreamListener*>(this);
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
if (aIID.Equals(NS_GET_IID(nsIRequestObserver))) {
|
|
*aResult = static_cast<nsIRequestObserver*>(this);
|
|
NS_ADDREF_THIS();
|
|
return NS_OK;
|
|
}
|
|
|
|
return QueryInterface(aIID, aResult);
|
|
}
|
|
|
|
void
|
|
FetchDriver::SetDocument(nsIDocument* aDocument)
|
|
{
|
|
// Cannot set document after Fetch() has been called.
|
|
MOZ_ASSERT(!mFetchCalled);
|
|
mDocument = aDocument;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|