зеркало из https://github.com/mozilla/gecko-dev.git
399 строки
12 KiB
C++
399 строки
12 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 "FetchStreamReader.h"
|
|
#include "InternalResponse.h"
|
|
#include "js/Stream.h"
|
|
#include "mozilla/ConsoleReportCollector.h"
|
|
#include "mozilla/ErrorResult.h"
|
|
#include "mozilla/dom/AutoEntryScript.h"
|
|
#include "mozilla/dom/Promise.h"
|
|
#include "mozilla/dom/PromiseBinding.h"
|
|
#include "mozilla/dom/ReadableStream.h"
|
|
#include "mozilla/dom/ReadableStreamDefaultController.h"
|
|
#include "mozilla/dom/ReadableStreamDefaultReader.h"
|
|
#include "mozilla/dom/WorkerPrivate.h"
|
|
#include "mozilla/dom/WorkerRef.h"
|
|
#include "mozilla/HoldDropJSObjects.h"
|
|
#include "mozilla/TaskCategory.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsDebug.h"
|
|
#include "nsIAsyncInputStream.h"
|
|
#include "nsIPipe.h"
|
|
#include "nsIScriptError.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "jsapi.h"
|
|
|
|
namespace mozilla::dom {
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(FetchStreamReader)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(FetchStreamReader)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(FetchStreamReader)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(FetchStreamReader)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mGlobal)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mReader)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(FetchStreamReader)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mGlobal)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mReader)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(FetchStreamReader)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(FetchStreamReader)
|
|
NS_INTERFACE_MAP_ENTRY(nsIOutputStreamCallback)
|
|
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIOutputStreamCallback)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
/* static */
|
|
nsresult FetchStreamReader::Create(JSContext* aCx, nsIGlobalObject* aGlobal,
|
|
FetchStreamReader** aStreamReader,
|
|
nsIInputStream** aInputStream) {
|
|
MOZ_ASSERT(aCx);
|
|
MOZ_ASSERT(aGlobal);
|
|
MOZ_ASSERT(aStreamReader);
|
|
MOZ_ASSERT(aInputStream);
|
|
|
|
RefPtr<FetchStreamReader> streamReader = new FetchStreamReader(aGlobal);
|
|
|
|
nsCOMPtr<nsIAsyncInputStream> pipeIn;
|
|
|
|
nsresult rv =
|
|
NS_NewPipe2(getter_AddRefs(pipeIn),
|
|
getter_AddRefs(streamReader->mPipeOut), true, true, 0, 0);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (!NS_IsMainThread()) {
|
|
WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(aCx);
|
|
MOZ_ASSERT(workerPrivate);
|
|
|
|
RefPtr<WeakWorkerRef> workerRef =
|
|
WeakWorkerRef::Create(workerPrivate, [streamReader]() {
|
|
MOZ_ASSERT(streamReader);
|
|
MOZ_ASSERT(streamReader->mWorkerRef);
|
|
|
|
WorkerPrivate* workerPrivate = streamReader->mWorkerRef->GetPrivate();
|
|
MOZ_ASSERT(workerPrivate);
|
|
|
|
streamReader->CloseAndRelease(workerPrivate->GetJSContext(),
|
|
NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
});
|
|
|
|
if (NS_WARN_IF(!workerRef)) {
|
|
streamReader->mPipeOut->CloseWithStatus(NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return NS_ERROR_DOM_INVALID_STATE_ERR;
|
|
}
|
|
|
|
// These 2 objects create a ref-cycle here that is broken when the stream is
|
|
// closed or the worker shutsdown.
|
|
streamReader->mWorkerRef = std::move(workerRef);
|
|
}
|
|
|
|
pipeIn.forget(aInputStream);
|
|
streamReader.forget(aStreamReader);
|
|
return NS_OK;
|
|
}
|
|
|
|
FetchStreamReader::FetchStreamReader(nsIGlobalObject* aGlobal)
|
|
: mGlobal(aGlobal),
|
|
mOwningEventTarget(mGlobal->EventTargetFor(TaskCategory::Other)),
|
|
mBufferRemaining(0),
|
|
mBufferOffset(0),
|
|
mStreamClosed(false) {
|
|
MOZ_ASSERT(aGlobal);
|
|
|
|
mozilla::HoldJSObjects(this);
|
|
}
|
|
|
|
FetchStreamReader::~FetchStreamReader() {
|
|
CloseAndRelease(nullptr, NS_BASE_STREAM_CLOSED);
|
|
|
|
mozilla::DropJSObjects(this);
|
|
}
|
|
|
|
// If a context is provided, an attempt will be made to cancel the reader. The
|
|
// only situation where we don't expect to have a context is when closure is
|
|
// being triggered from the destructor or the WorkerRef is notifying. If
|
|
// we're at the destructor, it's far too late to cancel anything. And if the
|
|
// WorkerRef is being notified, the global is going away, so there's also
|
|
// no need to do further JS work.
|
|
void FetchStreamReader::CloseAndRelease(JSContext* aCx, nsresult aStatus) {
|
|
NS_ASSERT_OWNINGTHREAD(FetchStreamReader);
|
|
|
|
if (mStreamClosed) {
|
|
// Already closed.
|
|
return;
|
|
}
|
|
|
|
RefPtr<FetchStreamReader> kungFuDeathGrip = this;
|
|
if (aCx && mReader) {
|
|
ErrorResult rv;
|
|
if (aStatus == NS_ERROR_DOM_WRONG_TYPE_ERR) {
|
|
rv.ThrowTypeError<MSG_FETCH_BODY_WRONG_TYPE>();
|
|
} else {
|
|
rv = aStatus;
|
|
}
|
|
JS::Rooted<JS::Value> errorValue(aCx);
|
|
if (ToJSValue(aCx, std::move(rv), &errorValue)) {
|
|
IgnoredErrorResult ignoredError;
|
|
// It's currently safe to cancel an already closed reader because, per the
|
|
// comments in ReadableStream::cancel() conveying the spec, step 2 of
|
|
// 3.4.3 that specified ReadableStreamCancel is: If stream.[[state]] is
|
|
// "closed", return a new promise resolved with undefined.
|
|
RefPtr<Promise> ignoredResultPromise =
|
|
MOZ_KnownLive(mReader)->Cancel(aCx, errorValue, ignoredError);
|
|
NS_WARNING_ASSERTION(!ignoredError.Failed(),
|
|
"Failed to cancel stream during close and release");
|
|
}
|
|
|
|
// We don't want to propagate exceptions during the cleanup.
|
|
JS_ClearPendingException(aCx);
|
|
}
|
|
|
|
mStreamClosed = true;
|
|
|
|
mGlobal = nullptr;
|
|
|
|
if (mPipeOut) {
|
|
mPipeOut->CloseWithStatus(aStatus);
|
|
}
|
|
mPipeOut = nullptr;
|
|
|
|
mWorkerRef = nullptr;
|
|
|
|
mReader = nullptr;
|
|
mBuffer.Clear();
|
|
}
|
|
|
|
void FetchStreamReader::StartConsuming(JSContext* aCx, ReadableStream* aStream,
|
|
ReadableStreamDefaultReader** aReader,
|
|
ErrorResult& aRv) {
|
|
MOZ_DIAGNOSTIC_ASSERT(!mReader);
|
|
MOZ_DIAGNOSTIC_ASSERT(aStream);
|
|
|
|
RefPtr<ReadableStreamDefaultReader> reader =
|
|
AcquireReadableStreamDefaultReader(aStream, aRv);
|
|
if (aRv.Failed()) {
|
|
CloseAndRelease(aCx, NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return;
|
|
}
|
|
|
|
mReader = reader;
|
|
reader.forget(aReader);
|
|
|
|
aRv = mPipeOut->AsyncWait(this, 0, 0, mOwningEventTarget);
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
struct FetchReadRequest : public ReadRequest {
|
|
public:
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(FetchReadRequest, ReadRequest)
|
|
|
|
explicit FetchReadRequest(FetchStreamReader* aReader)
|
|
: mFetchStreamReader(aReader) {}
|
|
|
|
void ChunkSteps(JSContext* aCx, JS::Handle<JS::Value> aChunk,
|
|
ErrorResult& aRv) override {
|
|
mFetchStreamReader->ChunkSteps(aCx, aChunk, aRv);
|
|
}
|
|
|
|
void CloseSteps(JSContext* aCx, ErrorResult& aRv) override {
|
|
mFetchStreamReader->CloseAndRelease(aCx, NS_BASE_STREAM_CLOSED);
|
|
}
|
|
|
|
void ErrorSteps(JSContext* aCx, JS::Handle<JS::Value> aError,
|
|
ErrorResult& aRv) override {
|
|
mFetchStreamReader->ErrorSteps(aCx, aError, aRv);
|
|
}
|
|
|
|
protected:
|
|
virtual ~FetchReadRequest() = default;
|
|
|
|
RefPtr<FetchStreamReader> mFetchStreamReader;
|
|
};
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(FetchReadRequest, ReadRequest,
|
|
mFetchStreamReader)
|
|
NS_IMPL_ADDREF_INHERITED(FetchReadRequest, ReadRequest)
|
|
NS_IMPL_RELEASE_INHERITED(FetchReadRequest, ReadRequest)
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(FetchReadRequest)
|
|
NS_INTERFACE_MAP_END_INHERITING(ReadRequest)
|
|
|
|
// nsIOutputStreamCallback interface
|
|
MOZ_CAN_RUN_SCRIPT_BOUNDARY
|
|
NS_IMETHODIMP
|
|
FetchStreamReader::OnOutputStreamReady(nsIAsyncOutputStream* aStream) {
|
|
NS_ASSERT_OWNINGTHREAD(FetchStreamReader);
|
|
if (mStreamClosed) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Only assert if we know the stream is not closed yet.
|
|
MOZ_ASSERT(aStream == mPipeOut);
|
|
MOZ_ASSERT(mReader);
|
|
|
|
if (!mBuffer.IsEmpty()) {
|
|
return WriteBuffer();
|
|
}
|
|
|
|
// Here we can retrieve data from the reader using any global we want because
|
|
// it is not observable. We want to use the reader's global, which is also the
|
|
// Response's one.
|
|
AutoEntryScript aes(mGlobal, "ReadableStreamReader.read", !mWorkerRef);
|
|
|
|
IgnoredErrorResult rv;
|
|
|
|
// https://fetch.spec.whatwg.org/#incrementally-read-loop
|
|
// The below very loosely tries to implement the incrementally-read-loop from
|
|
// the fetch spec.
|
|
RefPtr<ReadRequest> readRequest = new FetchReadRequest(this);
|
|
ReadableStreamDefaultReaderRead(aes.cx(), MOZ_KnownLive(mReader), readRequest,
|
|
rv);
|
|
|
|
if (NS_WARN_IF(rv.Failed())) {
|
|
// Let's close the stream.
|
|
CloseAndRelease(aes.cx(), NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void FetchStreamReader::ChunkSteps(JSContext* aCx, JS::Handle<JS::Value> aChunk,
|
|
ErrorResult& aRv) {
|
|
// This roughly implements the chunk steps from
|
|
// https://fetch.spec.whatwg.org/#incrementally-read-loop.
|
|
|
|
// Step 2. If chunk is not a Uint8Array object, then set continueAlgorithm to
|
|
// this step: run processBodyError given a TypeError.
|
|
RootedSpiderMonkeyInterface<Uint8Array> chunk(aCx);
|
|
if (!aChunk.isObject() || !chunk.Init(&aChunk.toObject())) {
|
|
CloseAndRelease(aCx, NS_ERROR_DOM_WRONG_TYPE_ERR);
|
|
return;
|
|
}
|
|
chunk.ComputeState();
|
|
|
|
uint32_t len = chunk.Length();
|
|
if (len == 0) {
|
|
// If there is nothing to read, let's do another reading.
|
|
OnOutputStreamReady(mPipeOut);
|
|
return;
|
|
}
|
|
|
|
MOZ_DIAGNOSTIC_ASSERT(mBuffer.IsEmpty());
|
|
|
|
// Let's take a copy of the data.
|
|
if (!mBuffer.AppendElements(chunk.Data(), len, fallible)) {
|
|
CloseAndRelease(aCx, NS_ERROR_OUT_OF_MEMORY);
|
|
return;
|
|
}
|
|
|
|
mBufferOffset = 0;
|
|
mBufferRemaining = len;
|
|
|
|
nsresult rv = WriteBuffer();
|
|
if (NS_FAILED(rv)) {
|
|
// Normalize to a generic DOM exception.
|
|
CloseAndRelease(aCx, NS_ERROR_DOM_ABORT_ERR);
|
|
}
|
|
}
|
|
|
|
void FetchStreamReader::ErrorSteps(JSContext* aCx, JS::Handle<JS::Value> aError,
|
|
ErrorResult& aRv) {
|
|
ReportErrorToConsole(aCx, aError);
|
|
CloseAndRelease(aCx, NS_ERROR_FAILURE);
|
|
}
|
|
|
|
nsresult FetchStreamReader::WriteBuffer() {
|
|
MOZ_ASSERT(!mBuffer.IsEmpty());
|
|
|
|
char* data = reinterpret_cast<char*>(mBuffer.Elements());
|
|
|
|
while (1) {
|
|
uint32_t written = 0;
|
|
nsresult rv =
|
|
mPipeOut->Write(data + mBufferOffset, mBufferRemaining, &written);
|
|
|
|
if (rv == NS_BASE_STREAM_WOULD_BLOCK) {
|
|
break;
|
|
}
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
MOZ_ASSERT(written <= mBufferRemaining);
|
|
mBufferRemaining -= written;
|
|
mBufferOffset += written;
|
|
|
|
if (mBufferRemaining == 0) {
|
|
mBuffer.Clear();
|
|
break;
|
|
}
|
|
}
|
|
|
|
nsresult rv = mPipeOut->AsyncWait(this, 0, 0, mOwningEventTarget);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void FetchStreamReader::ReportErrorToConsole(JSContext* aCx,
|
|
JS::Handle<JS::Value> aValue) {
|
|
nsCString sourceSpec;
|
|
uint32_t line = 0;
|
|
uint32_t column = 0;
|
|
nsString valueString;
|
|
|
|
nsContentUtils::ExtractErrorValues(aCx, aValue, sourceSpec, &line, &column,
|
|
valueString);
|
|
|
|
nsTArray<nsString> params;
|
|
params.AppendElement(valueString);
|
|
|
|
RefPtr<ConsoleReportCollector> reporter = new ConsoleReportCollector();
|
|
reporter->AddConsoleReport(nsIScriptError::errorFlag,
|
|
"ReadableStreamReader.read"_ns,
|
|
nsContentUtils::eDOM_PROPERTIES, sourceSpec, line,
|
|
column, "ReadableStreamReadingFailed"_ns, params);
|
|
|
|
uint64_t innerWindowId = 0;
|
|
|
|
if (NS_IsMainThread()) {
|
|
nsCOMPtr<nsPIDOMWindowInner> window = do_QueryInterface(mGlobal);
|
|
if (window) {
|
|
innerWindowId = window->WindowID();
|
|
}
|
|
reporter->FlushReportsToConsole(innerWindowId);
|
|
return;
|
|
}
|
|
|
|
WorkerPrivate* workerPrivate = GetWorkerPrivateFromContext(aCx);
|
|
if (workerPrivate) {
|
|
innerWindowId = workerPrivate->WindowID();
|
|
}
|
|
|
|
RefPtr<Runnable> r = NS_NewRunnableFunction(
|
|
"FetchStreamReader::ReportErrorToConsole", [reporter, innerWindowId]() {
|
|
reporter->FlushReportsToConsole(innerWindowId);
|
|
});
|
|
|
|
workerPrivate->DispatchToMainThread(r.forget());
|
|
}
|
|
|
|
} // namespace mozilla::dom
|