2 * Copyright (C) 2015 Canon Inc.
3 * Copyright (C) 2015 Igalia S.L.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted, provided that the following conditions
7 * are required to be met:
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. Neither the name of Canon Inc. nor the names of
15 * its contributors may be used to endorse or promote products derived
16 * from this software without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY CANON INC. AND ITS CONTRIBUTORS "AS IS" AND ANY
19 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL CANON INC. AND ITS CONTRIBUTORS BE LIABLE FOR
22 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
24 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
25 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
26 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #include "ReadableStream.h"
33 #if ENABLE(STREAMS_API)
35 #include "NotImplemented.h"
36 #include "ReadableStreamReader.h"
37 #include <runtime/JSCJSValueInlines.h>
38 #include <wtf/RefCountedLeakCounter.h>
42 DEFINE_DEBUG_ONLY_GLOBAL(WTF::RefCountedLeakCounter, readableStreamCounter, ("ReadableStream"));
44 ReadableStream::ReadableStream(ScriptExecutionContext& scriptExecutionContext)
45 : ActiveDOMObject(&scriptExecutionContext)
48 readableStreamCounter.increment();
53 ReadableStream::~ReadableStream()
56 readableStreamCounter.decrement();
60 void ReadableStream::clearCallbacks()
62 m_closedSuccessCallback = nullptr;
63 m_closedFailureCallback = nullptr;
65 m_readRequests.clear();
68 void ReadableStream::changeStateToClosed()
70 ASSERT(!m_closeRequested);
71 ASSERT(m_state != State::Errored);
73 m_closeRequested = true;
75 if (m_state != State::Readable || hasValue())
80 void ReadableStream::close()
82 m_state = State::Closed;
85 m_releasedReaders.append(WTF::move(m_reader));
87 if (m_closedSuccessCallback)
88 m_closedSuccessCallback();
90 for (auto& request : m_readRequests)
91 request.endCallback();
96 void ReadableStream::changeStateToErrored()
98 if (m_state != State::Readable)
100 m_state = State::Errored;
103 m_releasedReaders.append(WTF::move(m_reader));
105 JSC::JSValue error = this->error();
106 if (m_closedFailureCallback)
107 m_closedFailureCallback(error);
109 for (auto& request : m_readRequests)
110 request.failureCallback(error);
115 ReadableStreamReader& ReadableStream::getReader()
119 std::unique_ptr<ReadableStreamReader> newReader = std::make_unique<ReadableStreamReader>(*this);
120 ReadableStreamReader& reader = *newReader.get();
122 if (m_state == State::Readable) {
123 m_reader = WTF::move(newReader);
127 m_releasedReaders.append(WTF::move(newReader));
131 void ReadableStream::closed(ClosedSuccessCallback&& successCallback, FailureCallback&& failureCallback)
133 if (m_state == State::Closed) {
137 if (m_state == State::Errored) {
138 failureCallback(error());
141 m_closedSuccessCallback = WTF::move(successCallback);
142 m_closedFailureCallback = WTF::move(failureCallback);
145 void ReadableStream::read(ReadSuccessCallback&& successCallback, ReadEndCallback&& endCallback, FailureCallback&& failureCallback)
147 if (m_state == State::Closed) {
151 if (m_state == State::Errored) {
152 failureCallback(error());
156 successCallback(read());
157 if (m_closeRequested && !hasValue())
161 m_readRequests.append({ WTF::move(successCallback), WTF::move(endCallback), WTF::move(failureCallback) });
162 // FIXME: We should try to pull.
165 bool ReadableStream::resolveReadCallback(JSC::JSValue value)
167 if (m_readRequests.isEmpty())
170 m_readRequests.takeFirst().successCallback(value);
174 void ReadableStream::start()
179 const char* ReadableStream::activeDOMObjectName() const
181 return "ReadableStream";
184 bool ReadableStream::canSuspendForPageCache() const
186 // FIXME: We should try and do better here.