/* -*- 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/. */ #ifndef mozilla_dom_InternalResponse_h #define mozilla_dom_InternalResponse_h #include "nsIInputStream.h" #include "nsISupportsImpl.h" #include "mozilla/dom/InternalHeaders.h" #include "mozilla/dom/ResponseBinding.h" #include "mozilla/dom/RequestBinding.h" #include "mozilla/dom/ChannelInfo.h" #include "mozilla/UniquePtr.h" namespace mozilla { namespace ipc { class PrincipalInfo; } // namespace ipc namespace dom { class InternalHeaders; class InternalResponse final { friend class FetchDriver; public: NS_INLINE_DECL_THREADSAFE_REFCOUNTING(InternalResponse) InternalResponse(uint16_t aStatus, const nsACString& aStatusText, RequestCredentials aCredentialsMode = RequestCredentials::Omit); already_AddRefed Clone(); static already_AddRefed NetworkError() { RefPtr response = new InternalResponse(0, EmptyCString()); ErrorResult result; response->Headers()->SetGuard(HeadersGuardEnum::Immutable, result); MOZ_ASSERT(!result.Failed()); response->mType = ResponseType::Error; return response.forget(); } already_AddRefed OpaqueResponse(); already_AddRefed OpaqueRedirectResponse(); already_AddRefed BasicResponse(); already_AddRefed CORSResponse(); ResponseType Type() const { MOZ_ASSERT_IF(mType == ResponseType::Error, !mWrappedResponse); MOZ_ASSERT_IF(mType == ResponseType::Default, !mWrappedResponse); MOZ_ASSERT_IF(mType == ResponseType::Basic, mWrappedResponse); MOZ_ASSERT_IF(mType == ResponseType::Cors, mWrappedResponse); MOZ_ASSERT_IF(mType == ResponseType::Opaque, mWrappedResponse); MOZ_ASSERT_IF(mType == ResponseType::Opaqueredirect, mWrappedResponse); return mType; } bool IsError() const { return Type() == ResponseType::Error; } // FIXME(nsm): Return with exclude fragment. void GetUrl(nsCString& aURL) const { aURL.Assign(mURL); } void GetUnfilteredUrl(nsCString& aURL) const { if (mWrappedResponse) { return mWrappedResponse->GetUrl(aURL); } return GetUrl(aURL); } // SetUrl should only be called when the fragment has alredy been stripped void SetUrl(const nsACString& aURL) { mURL.Assign(aURL); } uint16_t GetStatus() const { return mStatus; } uint16_t GetUnfilteredStatus() const { if (mWrappedResponse) { return mWrappedResponse->GetStatus(); } return GetStatus(); } const nsCString& GetStatusText() const { return mStatusText; } const nsCString& GetUnfilteredStatusText() const { if (mWrappedResponse) { return mWrappedResponse->GetStatusText(); } return GetStatusText(); } InternalHeaders* Headers() { return mHeaders; } InternalHeaders* UnfilteredHeaders() { if (mWrappedResponse) { return mWrappedResponse->Headers(); }; return Headers(); } void GetUnfilteredBody(nsIInputStream** aStream) { if (mWrappedResponse) { MOZ_ASSERT(!mBody); return mWrappedResponse->GetBody(aStream); } nsCOMPtr stream = mBody; stream.forget(aStream); } void GetBody(nsIInputStream** aStream) { if (Type() == ResponseType::Opaque || Type() == ResponseType::Opaqueredirect) { *aStream = nullptr; return; } return GetUnfilteredBody(aStream); } void SetBody(nsIInputStream* aBody) { if (mWrappedResponse) { return mWrappedResponse->SetBody(aBody); } // A request's body may not be reset once set. MOZ_ASSERT(!mBody); mBody = aBody; } void InitChannelInfo(nsIChannel* aChannel) { mChannelInfo.InitFromChannel(aChannel); } void InitChannelInfo(const mozilla::ipc::IPCChannelInfo& aChannelInfo) { mChannelInfo.InitFromIPCChannelInfo(aChannelInfo); } void InitChannelInfo(const ChannelInfo& aChannelInfo) { mChannelInfo = aChannelInfo; } const ChannelInfo& GetChannelInfo() const { return mChannelInfo; } const UniquePtr& GetPrincipalInfo() const { return mPrincipalInfo; } // Takes ownership of the principal info. void SetPrincipalInfo(UniquePtr aPrincipalInfo); nsresult StripFragmentAndSetUrl(const nsACString& aUrl); LoadTainting GetTainting() const; private: ~InternalResponse(); explicit InternalResponse(const InternalResponse& aOther) = delete; InternalResponse& operator=(const InternalResponse&) = delete; // Returns an instance of InternalResponse which is a copy of this // InternalResponse, except headers, body and wrapped response (if any) which // are left uninitialized. Used for cloning and filtering. already_AddRefed CreateIncompleteCopy(); ResponseType mType; nsCString mTerminationReason; nsCString mURL; const uint16_t mStatus; const nsCString mStatusText; RefPtr mHeaders; nsCOMPtr mBody; ChannelInfo mChannelInfo; UniquePtr mPrincipalInfo; RequestCredentials mCredentialsMode; // For filtered responses. // Cache, and SW interception should always serialize/access the underlying // unfiltered headers and when deserializing, create an InternalResponse // with the unfiltered headers followed by wrapping it. RefPtr mWrappedResponse; }; } // namespace dom } // namespace mozilla #endif // mozilla_dom_InternalResponse_h