/* -*- 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 "InternalRequest.h" #include "nsIContentPolicy.h" #include "nsIDocument.h" #include "nsStreamUtils.h" #include "mozilla/ErrorResult.h" #include "mozilla/dom/ScriptSettings.h" #include "mozilla/dom/workers/Workers.h" #include "WorkerPrivate.h" namespace mozilla { namespace dom { // The global is used to extract the principal. already_AddRefed InternalRequest::GetRequestConstructorCopy(nsIGlobalObject* aGlobal, ErrorResult& aRv) const { nsRefPtr copy = new InternalRequest(); copy->mURL.Assign(mURL); copy->SetMethod(mMethod); copy->mHeaders = new InternalHeaders(*mHeaders); copy->SetUnsafeRequest(); copy->mBodyStream = mBodyStream; copy->mForceOriginHeader = true; // The "client" is not stored in our implementation. Fetch API users should // use the appropriate window/document/principal and other Gecko security // mechanisms as appropriate. copy->mSameOriginDataURL = true; copy->mPreserveContentCodings = true; // The default referrer is already about:client. copy->mContentPolicyType = nsIContentPolicy::TYPE_FETCH; copy->mMode = mMode; copy->mCredentialsMode = mCredentialsMode; copy->mCacheMode = mCacheMode; copy->mCreatedByFetchEvent = mCreatedByFetchEvent; return copy.forget(); } already_AddRefed InternalRequest::Clone() { nsRefPtr clone = new InternalRequest(*this); if (!mBodyStream) { return clone.forget(); } nsCOMPtr clonedBody; nsCOMPtr replacementBody; nsresult rv = NS_CloneInputStream(mBodyStream, getter_AddRefs(clonedBody), getter_AddRefs(replacementBody)); if (NS_WARN_IF(NS_FAILED(rv))) { return nullptr; } clone->mBodyStream.swap(clonedBody); if (replacementBody) { mBodyStream.swap(replacementBody); } return clone.forget(); } InternalRequest::InternalRequest(const InternalRequest& aOther) : mMethod(aOther.mMethod) , mURL(aOther.mURL) , mHeaders(new InternalHeaders(*aOther.mHeaders)) , mContentPolicyType(aOther.mContentPolicyType) , mReferrer(aOther.mReferrer) , mMode(aOther.mMode) , mCredentialsMode(aOther.mCredentialsMode) , mResponseTainting(aOther.mResponseTainting) , mCacheMode(aOther.mCacheMode) , mAuthenticationFlag(aOther.mAuthenticationFlag) , mForceOriginHeader(aOther.mForceOriginHeader) , mPreserveContentCodings(aOther.mPreserveContentCodings) , mSameOriginDataURL(aOther.mSameOriginDataURL) , mSandboxedStorageAreaURLs(aOther.mSandboxedStorageAreaURLs) , mSkipServiceWorker(aOther.mSkipServiceWorker) , mSynchronous(aOther.mSynchronous) , mUnsafeRequest(aOther.mUnsafeRequest) , mUseURLCredentials(aOther.mUseURLCredentials) , mCreatedByFetchEvent(aOther.mCreatedByFetchEvent) { // NOTE: does not copy body stream... use the fallible Clone() for that } InternalRequest::~InternalRequest() { } void InternalRequest::SetContentPolicyType(nsContentPolicyType aContentPolicyType) { mContentPolicyType = aContentPolicyType; } /* static */ RequestContext InternalRequest::MapContentPolicyTypeToRequestContext(nsContentPolicyType aContentPolicyType) { RequestContext context = RequestContext::Internal; switch (aContentPolicyType) { case nsIContentPolicy::TYPE_OTHER: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_INTERNAL_SCRIPT: context = RequestContext::Script; break; case nsIContentPolicy::TYPE_INTERNAL_WORKER: context = RequestContext::Worker; break; case nsIContentPolicy::TYPE_INTERNAL_SHARED_WORKER: context = RequestContext::Sharedworker; break; case nsIContentPolicy::TYPE_IMAGE: context = RequestContext::Image; break; case nsIContentPolicy::TYPE_STYLESHEET: context = RequestContext::Style; break; case nsIContentPolicy::TYPE_INTERNAL_OBJECT: context = RequestContext::Object; break; case nsIContentPolicy::TYPE_INTERNAL_EMBED: context = RequestContext::Embed; break; case nsIContentPolicy::TYPE_DOCUMENT: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_INTERNAL_IFRAME: context = RequestContext::Iframe; break; case nsIContentPolicy::TYPE_INTERNAL_FRAME: context = RequestContext::Frame; break; case nsIContentPolicy::TYPE_REFRESH: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_XBL: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_PING: context = RequestContext::Ping; break; case nsIContentPolicy::TYPE_XMLHTTPREQUEST: context = RequestContext::Xmlhttprequest; break; case nsIContentPolicy::TYPE_OBJECT_SUBREQUEST: context = RequestContext::Plugin; break; case nsIContentPolicy::TYPE_DTD: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_FONT: context = RequestContext::Font; break; case nsIContentPolicy::TYPE_INTERNAL_AUDIO: context = RequestContext::Audio; break; case nsIContentPolicy::TYPE_INTERNAL_VIDEO: context = RequestContext::Video; break; case nsIContentPolicy::TYPE_INTERNAL_TRACK: context = RequestContext::Track; break; case nsIContentPolicy::TYPE_WEBSOCKET: context = RequestContext::Internal; break; case nsIContentPolicy::TYPE_CSP_REPORT: context = RequestContext::Cspreport; break; case nsIContentPolicy::TYPE_XSLT: context = RequestContext::Xslt; break; case nsIContentPolicy::TYPE_BEACON: context = RequestContext::Beacon; break; case nsIContentPolicy::TYPE_FETCH: context = RequestContext::Fetch; break; case nsIContentPolicy::TYPE_IMAGESET: context = RequestContext::Imageset; break; case nsIContentPolicy::TYPE_WEB_MANIFEST: context = RequestContext::Manifest; break; default: MOZ_ASSERT(false, "Unhandled nsContentPolicyType value"); break; } return context; } bool InternalRequest::IsNavigationRequest() const { // https://fetch.spec.whatwg.org/#navigation-request-context // // A navigation request context is one of "form", "frame", "hyperlink", // "iframe", "internal" (as long as context frame type is not "none"), // "location", "metarefresh", and "prerender". // // TODO: include equivalent check for "form" context // TODO: include equivalent check for "prerender" context return mContentPolicyType == nsIContentPolicy::TYPE_DOCUMENT || mContentPolicyType == nsIContentPolicy::TYPE_SUBDOCUMENT || mContentPolicyType == nsIContentPolicy::TYPE_INTERNAL_FRAME || mContentPolicyType == nsIContentPolicy::TYPE_INTERNAL_IFRAME || mContentPolicyType == nsIContentPolicy::TYPE_REFRESH; } bool InternalRequest::IsWorkerRequest() const { // https://fetch.spec.whatwg.org/#worker-request-context // // A worker request context is one of "serviceworker", "sharedworker", and // "worker". // // Note, service workers are not included here because currently there is // no way to generate a Request with a "serviceworker" RequestContext. // ServiceWorker scripts cannot be intercepted. return mContentPolicyType == nsIContentPolicy::TYPE_INTERNAL_WORKER || mContentPolicyType == nsIContentPolicy::TYPE_INTERNAL_SHARED_WORKER; } bool InternalRequest::IsClientRequest() const { return IsNavigationRequest() || IsWorkerRequest(); } } // namespace dom } // namespace mozilla