/* -*- 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_FetchStreamReader_h #define mozilla_dom_FetchStreamReader_h #include "js/RootingAPI.h" #include "js/TypeDecls.h" #include "mozilla/Attributes.h" #include "mozilla/WeakPtr.h" #include "mozilla/dom/FetchBinding.h" #include "mozilla/dom/PromiseNativeHandler.h" #include "nsIAsyncOutputStream.h" #include "nsIGlobalObject.h" namespace mozilla::dom { class ReadableStream; class ReadableStreamDefaultReader; class StrongWorkerRef; class FetchStreamReader; class OutputStreamHolder final : public nsIOutputStreamCallback { public: NS_DECL_ISUPPORTS NS_DECL_NSIOUTPUTSTREAMCALLBACK OutputStreamHolder(FetchStreamReader* aReader, nsIAsyncOutputStream* aOutput); nsresult Init(JSContext* aCx); void Shutdown(); // These just proxy the calls to the nsIAsyncOutputStream nsresult AsyncWait(uint32_t aFlags, uint32_t aRequestedCount, nsIEventTarget* aEventTarget); nsresult Write(char* aBuffer, uint32_t aLength, uint32_t* aWritten) { return mOutput->Write(aBuffer, aLength, aWritten); } nsresult CloseWithStatus(nsresult aStatus) { return mOutput->CloseWithStatus(aStatus); } nsresult StreamStatus() { return mOutput->StreamStatus(); } nsIAsyncOutputStream* GetOutputStream() { return mOutput; } private: ~OutputStreamHolder(); RefPtr mAsyncWaitReader; // WeakPtr to avoid cycles WeakPtr mReader; // To ensure the worker sticks around RefPtr mAsyncWaitWorkerRef; RefPtr mWorkerRef; nsCOMPtr mOutput; }; class FetchStreamReader final : public nsISupports, public SupportsWeakPtr { public: NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_CLASS(FetchStreamReader) // This creates a nsIInputStream able to retrieve data from the ReadableStream // object. The reading starts when StartConsuming() is called. static nsresult Create(JSContext* aCx, nsIGlobalObject* aGlobal, FetchStreamReader** aStreamReader, nsIInputStream** aInputStream); bool OnOutputStreamReady(); MOZ_CAN_RUN_SCRIPT void ChunkSteps(JSContext* aCx, JS::Handle aChunk, ErrorResult& aRv); MOZ_CAN_RUN_SCRIPT void CloseSteps(JSContext* aCx, ErrorResult& aRv); MOZ_CAN_RUN_SCRIPT void ErrorSteps(JSContext* aCx, JS::Handle aError, ErrorResult& aRv); // Idempotently close the output stream and null out all state. If aCx is // provided, the reader will also be canceled. aStatus must be a DOM error // as understood by DOMException because it will be provided as the // cancellation reason. // // This is a script boundary minimize annotation changes required while // we figure out how to handle some more tricky annotation cases (for // example, the destructor of this class. Tracking under Bug 1750656) MOZ_CAN_RUN_SCRIPT_BOUNDARY void CloseAndRelease(JSContext* aCx, nsresult aStatus); void StartConsuming(JSContext* aCx, ReadableStream* aStream, ErrorResult& aRv); private: explicit FetchStreamReader(nsIGlobalObject* aGlobal); ~FetchStreamReader(); nsresult WriteBuffer(); // Attempt to copy data from mBuffer into mPipeOut. Returns `true` if data was // written, and AsyncWait callbacks or FetchReadRequest calls have been set up // to write more data in the future, and `false` otherwise. MOZ_CAN_RUN_SCRIPT bool Process(JSContext* aCx); void ReportErrorToConsole(JSContext* aCx, JS::Handle aValue); nsCOMPtr mGlobal; nsCOMPtr mOwningEventTarget; RefPtr mOutput; RefPtr mReader; nsTArray mBuffer; uint32_t mBufferRemaining = 0; uint32_t mBufferOffset = 0; bool mHasOutstandingReadRequest = false; bool mStreamClosed = false; }; } // namespace mozilla::dom #endif // mozilla_dom_FetchStreamReader_h