forked from mirrors/gecko-dev
		
	 32ab8293ff
			
		
	
	
		32ab8293ff
		
	
	
	
	
		
			
			ReferrerPolicy gets tossed back and forth as a uint32_t and ReferrerPolicy enum in header file. Expose ReferrerPolicyValues from webidl file and use consistently in native code. Differential Revision: https://phabricator.services.mozilla.com/D41954 --HG-- extra : moz-landing-system : lando
		
			
				
	
	
		
			643 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			643 lines
		
	
	
	
		
			20 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 "Request.h"
 | |
| 
 | |
| #include "nsIURI.h"
 | |
| #include "nsPIDOMWindow.h"
 | |
| 
 | |
| #include "mozilla/ErrorResult.h"
 | |
| #include "mozilla/dom/Headers.h"
 | |
| #include "mozilla/dom/Fetch.h"
 | |
| #include "mozilla/dom/FetchUtil.h"
 | |
| #include "mozilla/dom/Promise.h"
 | |
| #include "mozilla/dom/URL.h"
 | |
| #include "mozilla/dom/WorkerPrivate.h"
 | |
| #include "mozilla/Unused.h"
 | |
| 
 | |
| namespace mozilla {
 | |
| namespace dom {
 | |
| 
 | |
| NS_IMPL_ADDREF_INHERITED(Request, FetchBody<Request>)
 | |
| NS_IMPL_RELEASE_INHERITED(Request, FetchBody<Request>)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_CLASS(Request)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(Request, FetchBody<Request>)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mOwner)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mHeaders)
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK(mSignal)
 | |
|   tmp->Unfollow();
 | |
|   NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_END
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(Request, FetchBody<Request>)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mOwner)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mHeaders)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSignal)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mFollowingSignal)
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(Request, FetchBody<Request>)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mReadableStreamBody)
 | |
|   MOZ_DIAGNOSTIC_ASSERT(!tmp->mReadableStreamReader);
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mReadableStreamReader)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER
 | |
| NS_IMPL_CYCLE_COLLECTION_TRACE_END
 | |
| 
 | |
| NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(Request)
 | |
|   NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
 | |
| NS_INTERFACE_MAP_END_INHERITING(FetchBody<Request>)
 | |
| 
 | |
| Request::Request(nsIGlobalObject* aOwner, InternalRequest* aRequest,
 | |
|                  AbortSignal* aSignal)
 | |
|     : FetchBody<Request>(aOwner), mRequest(aRequest) {
 | |
|   MOZ_ASSERT(aRequest->Headers()->Guard() == HeadersGuardEnum::Immutable ||
 | |
|              aRequest->Headers()->Guard() == HeadersGuardEnum::Request ||
 | |
|              aRequest->Headers()->Guard() == HeadersGuardEnum::Request_no_cors);
 | |
|   SetMimeType();
 | |
| 
 | |
|   if (aSignal) {
 | |
|     // If we don't have a signal as argument, we will create it when required by
 | |
|     // content, otherwise the Request's signal must follow what has been passed.
 | |
|     mSignal = new AbortSignal(aOwner, aSignal->Aborted());
 | |
|     if (!mSignal->Aborted()) {
 | |
|       mSignal->Follow(aSignal);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| Request::~Request() {}
 | |
| 
 | |
| already_AddRefed<InternalRequest> Request::GetInternalRequest() {
 | |
|   RefPtr<InternalRequest> r = mRequest;
 | |
|   return r.forget();
 | |
| }
 | |
| 
 | |
| namespace {
 | |
| already_AddRefed<nsIURI> ParseURLFromDocument(Document* aDocument,
 | |
|                                               const nsAString& aInput,
 | |
|                                               ErrorResult& aRv) {
 | |
|   MOZ_ASSERT(aDocument);
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
| 
 | |
|   nsCOMPtr<nsIURI> resolvedURI;
 | |
|   aRv = NS_NewURI(getter_AddRefs(resolvedURI), aInput, nullptr,
 | |
|                   aDocument->GetBaseURI());
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     aRv.ThrowTypeError<MSG_INVALID_URL>(aInput);
 | |
|   }
 | |
|   return resolvedURI.forget();
 | |
| }
 | |
| void GetRequestURLFromDocument(Document* aDocument, const nsAString& aInput,
 | |
|                                nsAString& aRequestURL, nsACString& aURLfragment,
 | |
|                                ErrorResult& aRv) {
 | |
|   nsCOMPtr<nsIURI> resolvedURI = ParseURLFromDocument(aDocument, aInput, aRv);
 | |
|   if (aRv.Failed()) {
 | |
|     return;
 | |
|   }
 | |
|   // This fails with URIs with weird protocols, even when they are valid,
 | |
|   // so we ignore the failure
 | |
|   nsAutoCString credentials;
 | |
|   Unused << resolvedURI->GetUserPass(credentials);
 | |
|   if (!credentials.IsEmpty()) {
 | |
|     aRv.ThrowTypeError<MSG_URL_HAS_CREDENTIALS>(aInput);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIURI> resolvedURIClone;
 | |
|   aRv = NS_GetURIWithoutRef(resolvedURI, getter_AddRefs(resolvedURIClone));
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
|   nsAutoCString spec;
 | |
|   aRv = resolvedURIClone->GetSpec(spec);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
|   CopyUTF8toUTF16(spec, aRequestURL);
 | |
| 
 | |
|   // Get the fragment from nsIURI.
 | |
|   aRv = resolvedURI->GetRef(aURLfragment);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| already_AddRefed<nsIURI> ParseURLFromChrome(const nsAString& aInput,
 | |
|                                             ErrorResult& aRv) {
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
|   nsCOMPtr<nsIURI> uri;
 | |
|   aRv = NS_NewURI(getter_AddRefs(uri), aInput);
 | |
|   if (aRv.Failed()) {
 | |
|     aRv.ThrowTypeError<MSG_INVALID_URL>(aInput);
 | |
|   }
 | |
|   return uri.forget();
 | |
| }
 | |
| void GetRequestURLFromChrome(const nsAString& aInput, nsAString& aRequestURL,
 | |
|                              nsACString& aURLfragment, ErrorResult& aRv) {
 | |
|   nsCOMPtr<nsIURI> uri = ParseURLFromChrome(aInput, aRv);
 | |
|   if (aRv.Failed()) {
 | |
|     return;
 | |
|   }
 | |
|   // This fails with URIs with weird protocols, even when they are valid,
 | |
|   // so we ignore the failure
 | |
|   nsAutoCString credentials;
 | |
|   Unused << uri->GetUserPass(credentials);
 | |
|   if (!credentials.IsEmpty()) {
 | |
|     aRv.ThrowTypeError<MSG_URL_HAS_CREDENTIALS>(aInput);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsCOMPtr<nsIURI> uriClone;
 | |
|   aRv = NS_GetURIWithoutRef(uri, getter_AddRefs(uriClone));
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
|   nsAutoCString spec;
 | |
|   aRv = uriClone->GetSpec(spec);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
|   CopyUTF8toUTF16(spec, aRequestURL);
 | |
| 
 | |
|   // Get the fragment from nsIURI.
 | |
|   aRv = uri->GetRef(aURLfragment);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| already_AddRefed<URL> ParseURLFromWorker(const GlobalObject& aGlobal,
 | |
|                                          const nsAString& aInput,
 | |
|                                          ErrorResult& aRv) {
 | |
|   WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
 | |
|   MOZ_ASSERT(worker);
 | |
|   worker->AssertIsOnWorkerThread();
 | |
| 
 | |
|   NS_ConvertUTF8toUTF16 baseURL(worker->GetLocationInfo().mHref);
 | |
|   RefPtr<URL> url =
 | |
|       URL::Constructor(aGlobal.GetAsSupports(), aInput, baseURL, aRv);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     aRv.ThrowTypeError<MSG_INVALID_URL>(aInput);
 | |
|   }
 | |
|   return url.forget();
 | |
| }
 | |
| void GetRequestURLFromWorker(const GlobalObject& aGlobal,
 | |
|                              const nsAString& aInput, nsAString& aRequestURL,
 | |
|                              nsACString& aURLfragment, ErrorResult& aRv) {
 | |
|   RefPtr<URL> url = ParseURLFromWorker(aGlobal, aInput, aRv);
 | |
|   if (aRv.Failed()) {
 | |
|     return;
 | |
|   }
 | |
|   nsString username;
 | |
|   url->GetUsername(username);
 | |
| 
 | |
|   nsString password;
 | |
|   url->GetPassword(password);
 | |
| 
 | |
|   if (!username.IsEmpty() || !password.IsEmpty()) {
 | |
|     aRv.ThrowTypeError<MSG_URL_HAS_CREDENTIALS>(aInput);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Get the fragment from URL.
 | |
|   nsAutoString fragment;
 | |
|   url->GetHash(fragment);
 | |
| 
 | |
|   // Note: URL::GetHash() includes the "#" and we want the fragment with out
 | |
|   // the hash symbol.
 | |
|   if (!fragment.IsEmpty()) {
 | |
|     CopyUTF16toUTF8(Substring(fragment, 1), aURLfragment);
 | |
|   }
 | |
| 
 | |
|   url->SetHash(EmptyString());
 | |
|   url->Stringify(aRequestURL);
 | |
| }
 | |
| 
 | |
| class ReferrerSameOriginChecker final : public WorkerMainThreadRunnable {
 | |
|  public:
 | |
|   ReferrerSameOriginChecker(WorkerPrivate* aWorkerPrivate,
 | |
|                             const nsAString& aReferrerURL, nsresult& aResult)
 | |
|       : WorkerMainThreadRunnable(
 | |
|             aWorkerPrivate,
 | |
|             NS_LITERAL_CSTRING("Fetch :: Referrer same origin check")),
 | |
|         mReferrerURL(aReferrerURL),
 | |
|         mResult(aResult) {
 | |
|     mWorkerPrivate->AssertIsOnWorkerThread();
 | |
|   }
 | |
| 
 | |
|   bool MainThreadRun() override {
 | |
|     nsCOMPtr<nsIURI> uri;
 | |
|     if (NS_SUCCEEDED(NS_NewURI(getter_AddRefs(uri), mReferrerURL))) {
 | |
|       nsCOMPtr<nsIPrincipal> principal = mWorkerPrivate->GetPrincipal();
 | |
|       if (principal) {
 | |
|         mResult = principal->CheckMayLoad(uri, /* report */ false,
 | |
|                                           /* allowIfInheritsPrincipal */ false);
 | |
|       }
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   const nsString mReferrerURL;
 | |
|   nsresult& mResult;
 | |
| };
 | |
| 
 | |
| }  // namespace
 | |
| 
 | |
| /*static*/
 | |
| already_AddRefed<Request> Request::Constructor(const GlobalObject& aGlobal,
 | |
|                                                const RequestOrUSVString& aInput,
 | |
|                                                const RequestInit& aInit,
 | |
|                                                ErrorResult& aRv) {
 | |
|   bool hasCopiedBody = false;
 | |
|   RefPtr<InternalRequest> request;
 | |
| 
 | |
|   nsCOMPtr<nsIGlobalObject> global = do_QueryInterface(aGlobal.GetAsSupports());
 | |
| 
 | |
|   RefPtr<AbortSignal> signal;
 | |
| 
 | |
|   if (aInput.IsRequest()) {
 | |
|     RefPtr<Request> inputReq = &aInput.GetAsRequest();
 | |
|     nsCOMPtr<nsIInputStream> body;
 | |
|     inputReq->GetBody(getter_AddRefs(body));
 | |
|     bool used = inputReq->GetBodyUsed(aRv);
 | |
|     if (NS_WARN_IF(aRv.Failed())) {
 | |
|       return nullptr;
 | |
|     }
 | |
|     if (used) {
 | |
|       aRv.ThrowTypeError<MSG_FETCH_BODY_CONSUMED_ERROR>();
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     // The body will be copied when GetRequestConstructorCopy() is executed.
 | |
|     if (body) {
 | |
|       hasCopiedBody = true;
 | |
|     }
 | |
| 
 | |
|     request = inputReq->GetInternalRequest();
 | |
|     signal = inputReq->GetOrCreateSignal();
 | |
|   } else {
 | |
|     // aInput is USVString.
 | |
|     // We need to get url before we create a InternalRequest.
 | |
|     nsAutoString input;
 | |
|     input.Assign(aInput.GetAsUSVString());
 | |
|     nsAutoString requestURL;
 | |
|     nsCString fragment;
 | |
|     if (NS_IsMainThread()) {
 | |
|       nsCOMPtr<nsPIDOMWindowInner> inner(do_QueryInterface(global));
 | |
|       Document* doc = inner ? inner->GetExtantDoc() : nullptr;
 | |
|       if (doc) {
 | |
|         GetRequestURLFromDocument(doc, input, requestURL, fragment, aRv);
 | |
|       } else {
 | |
|         // If we don't have a document, we must assume that this is a full URL.
 | |
|         GetRequestURLFromChrome(input, requestURL, fragment, aRv);
 | |
|       }
 | |
|     } else {
 | |
|       GetRequestURLFromWorker(aGlobal, input, requestURL, fragment, aRv);
 | |
|     }
 | |
|     if (aRv.Failed()) {
 | |
|       return nullptr;
 | |
|     }
 | |
|     request = new InternalRequest(NS_ConvertUTF16toUTF8(requestURL), fragment);
 | |
|   }
 | |
|   request = request->GetRequestConstructorCopy(global, aRv);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   RequestMode fallbackMode = RequestMode::EndGuard_;
 | |
|   RequestCredentials fallbackCredentials = RequestCredentials::EndGuard_;
 | |
|   RequestCache fallbackCache = RequestCache::EndGuard_;
 | |
|   if (aInput.IsUSVString()) {
 | |
|     fallbackMode = RequestMode::Cors;
 | |
|     fallbackCredentials = RequestCredentials::Same_origin;
 | |
|     fallbackCache = RequestCache::Default;
 | |
|   }
 | |
| 
 | |
|   RequestMode mode =
 | |
|       aInit.mMode.WasPassed() ? aInit.mMode.Value() : fallbackMode;
 | |
|   RequestCredentials credentials = aInit.mCredentials.WasPassed()
 | |
|                                        ? aInit.mCredentials.Value()
 | |
|                                        : fallbackCredentials;
 | |
| 
 | |
|   if (mode == RequestMode::Navigate) {
 | |
|     aRv.ThrowTypeError<MSG_INVALID_REQUEST_MODE>(NS_LITERAL_STRING("navigate"));
 | |
|     return nullptr;
 | |
|   }
 | |
|   if (aInit.IsAnyMemberPresent() && request->Mode() == RequestMode::Navigate) {
 | |
|     mode = RequestMode::Same_origin;
 | |
|   }
 | |
| 
 | |
|   if (aInit.IsAnyMemberPresent()) {
 | |
|     request->SetReferrer(NS_LITERAL_STRING(kFETCH_CLIENT_REFERRER_STR));
 | |
|     request->SetReferrerPolicy(ReferrerPolicy::_empty);
 | |
|   }
 | |
|   if (aInit.mReferrer.WasPassed()) {
 | |
|     const nsString& referrer = aInit.mReferrer.Value();
 | |
|     if (referrer.IsEmpty()) {
 | |
|       request->SetReferrer(NS_LITERAL_STRING(""));
 | |
|     } else {
 | |
|       nsAutoString referrerURL;
 | |
|       if (NS_IsMainThread()) {
 | |
|         nsCOMPtr<nsPIDOMWindowInner> inner(do_QueryInterface(global));
 | |
|         Document* doc = inner ? inner->GetExtantDoc() : nullptr;
 | |
|         nsCOMPtr<nsIURI> uri;
 | |
|         if (doc) {
 | |
|           uri = ParseURLFromDocument(doc, referrer, aRv);
 | |
|         } else {
 | |
|           // If we don't have a document, we must assume that this is a full
 | |
|           // URL.
 | |
|           uri = ParseURLFromChrome(referrer, aRv);
 | |
|         }
 | |
|         if (NS_WARN_IF(aRv.Failed())) {
 | |
|           aRv.ThrowTypeError<MSG_INVALID_REFERRER_URL>(referrer);
 | |
|           return nullptr;
 | |
|         }
 | |
|         nsAutoCString spec;
 | |
|         uri->GetSpec(spec);
 | |
|         CopyUTF8toUTF16(spec, referrerURL);
 | |
|         if (!referrerURL.EqualsLiteral(kFETCH_CLIENT_REFERRER_STR)) {
 | |
|           nsCOMPtr<nsIPrincipal> principal = global->PrincipalOrNull();
 | |
|           if (principal) {
 | |
|             nsresult rv =
 | |
|                 principal->CheckMayLoad(uri, /* report */ false,
 | |
|                                         /* allowIfInheritsPrincipal */ false);
 | |
|             if (NS_FAILED(rv)) {
 | |
|               referrerURL.AssignLiteral(kFETCH_CLIENT_REFERRER_STR);
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         RefPtr<URL> url = ParseURLFromWorker(aGlobal, referrer, aRv);
 | |
|         if (NS_WARN_IF(aRv.Failed())) {
 | |
|           aRv.ThrowTypeError<MSG_INVALID_REFERRER_URL>(referrer);
 | |
|           return nullptr;
 | |
|         }
 | |
|         url->Stringify(referrerURL);
 | |
|         if (!referrerURL.EqualsLiteral(kFETCH_CLIENT_REFERRER_STR)) {
 | |
|           WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
 | |
|           nsresult rv = NS_OK;
 | |
|           // ReferrerSameOriginChecker uses a sync loop to get the main thread
 | |
|           // to perform the same-origin check.  Overall, on Workers this method
 | |
|           // can create 3 sync loops (two for constructing URLs and one here) so
 | |
|           // in the future we may want to optimize it all by off-loading all of
 | |
|           // this work in a single sync loop.
 | |
|           RefPtr<ReferrerSameOriginChecker> checker =
 | |
|               new ReferrerSameOriginChecker(worker, referrerURL, rv);
 | |
|           IgnoredErrorResult error;
 | |
|           checker->Dispatch(Canceling, error);
 | |
|           if (error.Failed() || NS_FAILED(rv)) {
 | |
|             referrerURL.AssignLiteral(kFETCH_CLIENT_REFERRER_STR);
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       request->SetReferrer(referrerURL);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (aInit.mReferrerPolicy.WasPassed()) {
 | |
|     request->SetReferrerPolicy(aInit.mReferrerPolicy.Value());
 | |
|   }
 | |
| 
 | |
|   if (aInit.mSignal.WasPassed()) {
 | |
|     signal = aInit.mSignal.Value();
 | |
|   }
 | |
| 
 | |
|   UniquePtr<mozilla::ipc::PrincipalInfo> principalInfo;
 | |
| 
 | |
|   if (NS_IsMainThread()) {
 | |
|     nsCOMPtr<nsPIDOMWindowInner> window = do_QueryInterface(global);
 | |
|     if (window) {
 | |
|       nsCOMPtr<Document> doc;
 | |
|       doc = window->GetExtantDoc();
 | |
|       if (doc) {
 | |
|         request->SetEnvironmentReferrerPolicy(doc->GetReferrerPolicy());
 | |
| 
 | |
|         principalInfo.reset(new mozilla::ipc::PrincipalInfo());
 | |
|         nsresult rv =
 | |
|             PrincipalToPrincipalInfo(doc->NodePrincipal(), principalInfo.get());
 | |
|         if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|           aRv.ThrowTypeError<MSG_FETCH_BODY_CONSUMED_ERROR>();
 | |
|           return nullptr;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     WorkerPrivate* worker = GetCurrentThreadWorkerPrivate();
 | |
|     if (worker) {
 | |
|       worker->AssertIsOnWorkerThread();
 | |
|       request->SetEnvironmentReferrerPolicy(worker->GetReferrerPolicy());
 | |
|       principalInfo =
 | |
|           MakeUnique<mozilla::ipc::PrincipalInfo>(worker->GetPrincipalInfo());
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   request->SetPrincipalInfo(std::move(principalInfo));
 | |
| 
 | |
|   if (mode != RequestMode::EndGuard_) {
 | |
|     request->ClearCreatedByFetchEvent();
 | |
|     request->SetMode(mode);
 | |
|   }
 | |
| 
 | |
|   if (credentials != RequestCredentials::EndGuard_) {
 | |
|     request->ClearCreatedByFetchEvent();
 | |
|     request->SetCredentialsMode(credentials);
 | |
|   }
 | |
| 
 | |
|   RequestCache cache =
 | |
|       aInit.mCache.WasPassed() ? aInit.mCache.Value() : fallbackCache;
 | |
|   if (cache != RequestCache::EndGuard_) {
 | |
|     if (cache == RequestCache::Only_if_cached &&
 | |
|         request->Mode() != RequestMode::Same_origin) {
 | |
|       uint32_t t = static_cast<uint32_t>(request->Mode());
 | |
|       NS_ConvertASCIItoUTF16 modeString(RequestModeValues::strings[t].value,
 | |
|                                         RequestModeValues::strings[t].length);
 | |
|       aRv.ThrowTypeError<MSG_ONLY_IF_CACHED_WITHOUT_SAME_ORIGIN>(modeString);
 | |
|       return nullptr;
 | |
|     }
 | |
|     request->ClearCreatedByFetchEvent();
 | |
|     request->SetCacheMode(cache);
 | |
|   }
 | |
| 
 | |
|   if (aInit.mRedirect.WasPassed()) {
 | |
|     request->SetRedirectMode(aInit.mRedirect.Value());
 | |
|   }
 | |
| 
 | |
|   if (aInit.mIntegrity.WasPassed()) {
 | |
|     request->SetIntegrity(aInit.mIntegrity.Value());
 | |
|   }
 | |
| 
 | |
|   if (aInit.mMozErrors.WasPassed() && aInit.mMozErrors.Value()) {
 | |
|     request->SetMozErrors();
 | |
|   }
 | |
| 
 | |
|   // Request constructor step 14.
 | |
|   if (aInit.mMethod.WasPassed()) {
 | |
|     nsAutoCString method(aInit.mMethod.Value());
 | |
| 
 | |
|     // Step 14.1. Disallow forbidden methods, and anything that is not a HTTP
 | |
|     // token, since HTTP states that Method may be any of the defined values or
 | |
|     // a token (extension method).
 | |
|     nsAutoCString outMethod;
 | |
|     nsresult rv = FetchUtil::GetValidRequestMethod(method, outMethod);
 | |
|     if (NS_FAILED(rv)) {
 | |
|       NS_ConvertUTF8toUTF16 label(method);
 | |
|       aRv.ThrowTypeError<MSG_INVALID_REQUEST_METHOD>(label);
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     // Step 14.2
 | |
|     request->ClearCreatedByFetchEvent();
 | |
|     request->SetMethod(outMethod);
 | |
|   }
 | |
| 
 | |
|   RefPtr<InternalHeaders> requestHeaders = request->Headers();
 | |
| 
 | |
|   RefPtr<InternalHeaders> headers;
 | |
|   if (aInit.mHeaders.WasPassed()) {
 | |
|     RefPtr<Headers> h =
 | |
|         Headers::Constructor(aGlobal, aInit.mHeaders.Value(), aRv);
 | |
|     if (aRv.Failed()) {
 | |
|       return nullptr;
 | |
|     }
 | |
|     request->ClearCreatedByFetchEvent();
 | |
|     headers = h->GetInternalHeaders();
 | |
|   } else {
 | |
|     headers = new InternalHeaders(*requestHeaders);
 | |
|   }
 | |
| 
 | |
|   requestHeaders->Clear();
 | |
|   // From "Let r be a new Request object associated with request and a new
 | |
|   // Headers object whose guard is "request"."
 | |
|   requestHeaders->SetGuard(HeadersGuardEnum::Request, aRv);
 | |
|   MOZ_ASSERT(!aRv.Failed());
 | |
| 
 | |
|   if (request->Mode() == RequestMode::No_cors) {
 | |
|     if (!request->HasSimpleMethod()) {
 | |
|       nsAutoCString method;
 | |
|       request->GetMethod(method);
 | |
|       NS_ConvertUTF8toUTF16 label(method);
 | |
|       aRv.ThrowTypeError<MSG_INVALID_REQUEST_METHOD>(label);
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     requestHeaders->SetGuard(HeadersGuardEnum::Request_no_cors, aRv);
 | |
|     if (aRv.Failed()) {
 | |
|       return nullptr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   requestHeaders->Fill(*headers, aRv);
 | |
|   if (aRv.Failed()) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   if ((aInit.mBody.WasPassed() && !aInit.mBody.Value().IsNull()) ||
 | |
|       hasCopiedBody) {
 | |
|     // HEAD and GET are not allowed to have a body.
 | |
|     nsAutoCString method;
 | |
|     request->GetMethod(method);
 | |
|     // method is guaranteed to be uppercase due to step 14.2 above.
 | |
|     if (method.EqualsLiteral("HEAD") || method.EqualsLiteral("GET")) {
 | |
|       aRv.ThrowTypeError<MSG_NO_BODY_ALLOWED_FOR_GET_AND_HEAD>();
 | |
|       return nullptr;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (aInit.mBody.WasPassed()) {
 | |
|     const Nullable<fetch::OwningBodyInit>& bodyInitNullable =
 | |
|         aInit.mBody.Value();
 | |
|     if (!bodyInitNullable.IsNull()) {
 | |
|       const fetch::OwningBodyInit& bodyInit = bodyInitNullable.Value();
 | |
|       nsCOMPtr<nsIInputStream> stream;
 | |
|       nsAutoCString contentTypeWithCharset;
 | |
|       uint64_t contentLength = 0;
 | |
|       aRv = ExtractByteStreamFromBody(bodyInit, getter_AddRefs(stream),
 | |
|                                       contentTypeWithCharset, contentLength);
 | |
|       if (NS_WARN_IF(aRv.Failed())) {
 | |
|         return nullptr;
 | |
|       }
 | |
| 
 | |
|       nsCOMPtr<nsIInputStream> temporaryBody = stream;
 | |
| 
 | |
|       if (!contentTypeWithCharset.IsVoid() &&
 | |
|           !requestHeaders->Has(NS_LITERAL_CSTRING("Content-Type"), aRv)) {
 | |
|         requestHeaders->Append(NS_LITERAL_CSTRING("Content-Type"),
 | |
|                                contentTypeWithCharset, aRv);
 | |
|       }
 | |
| 
 | |
|       if (NS_WARN_IF(aRv.Failed())) {
 | |
|         return nullptr;
 | |
|       }
 | |
| 
 | |
|       request->ClearCreatedByFetchEvent();
 | |
| 
 | |
|       if (hasCopiedBody) {
 | |
|         request->SetBody(nullptr, 0);
 | |
|       }
 | |
| 
 | |
|       request->SetBody(temporaryBody, contentLength);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   RefPtr<Request> domRequest = new Request(global, request, signal);
 | |
|   domRequest->SetMimeType();
 | |
| 
 | |
|   if (aInput.IsRequest()) {
 | |
|     RefPtr<Request> inputReq = &aInput.GetAsRequest();
 | |
|     nsCOMPtr<nsIInputStream> body;
 | |
|     inputReq->GetBody(getter_AddRefs(body));
 | |
|     if (body) {
 | |
|       inputReq->SetBody(nullptr, 0);
 | |
|       inputReq->SetBodyUsed(aGlobal.Context(), aRv);
 | |
|       if (NS_WARN_IF(aRv.Failed())) {
 | |
|         return nullptr;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return domRequest.forget();
 | |
| }
 | |
| 
 | |
| already_AddRefed<Request> Request::Clone(ErrorResult& aRv) {
 | |
|   bool used = GetBodyUsed(aRv);
 | |
|   if (NS_WARN_IF(aRv.Failed())) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   if (used) {
 | |
|     aRv.ThrowTypeError<MSG_FETCH_BODY_CONSUMED_ERROR>();
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   RefPtr<InternalRequest> ir = mRequest->Clone();
 | |
|   if (!ir) {
 | |
|     aRv.Throw(NS_ERROR_FAILURE);
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   RefPtr<Request> request = new Request(mOwner, ir, GetOrCreateSignal());
 | |
| 
 | |
|   return request.forget();
 | |
| }
 | |
| 
 | |
| Headers* Request::Headers_() {
 | |
|   if (!mHeaders) {
 | |
|     mHeaders = new Headers(mOwner, mRequest->Headers());
 | |
|   }
 | |
| 
 | |
|   return mHeaders;
 | |
| }
 | |
| 
 | |
| AbortSignal* Request::GetOrCreateSignal() {
 | |
|   if (!mSignal) {
 | |
|     mSignal = new AbortSignal(mOwner, false);
 | |
|   }
 | |
| 
 | |
|   return mSignal;
 | |
| }
 | |
| 
 | |
| AbortSignalImpl* Request::GetSignalImpl() const { return mSignal; }
 | |
| 
 | |
| }  // namespace dom
 | |
| }  // namespace mozilla
 |