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 void ReadableStream::start()
121 void ReadableStream::pull()
123 if (!m_isStarted || m_state == State::Closed || m_state == State::Errored || m_closeRequested)
125 // FIXME: Implement queueSize check.
126 if (m_readRequests.isEmpty() && hasValue())
128 // FIXME: Implement async pull check.
132 ReadableStreamReader& ReadableStream::getReader()
136 std::unique_ptr<ReadableStreamReader> newReader = std::make_unique<ReadableStreamReader>(*this);
137 ReadableStreamReader& reader = *newReader.get();
139 if (m_state == State::Readable) {
140 m_reader = WTF::move(newReader);
144 m_releasedReaders.append(WTF::move(newReader));
148 void ReadableStream::closed(ClosedSuccessCallback&& successCallback, FailureCallback&& failureCallback)
150 if (m_state == State::Closed) {
154 if (m_state == State::Errored) {
155 failureCallback(error());
158 m_closedSuccessCallback = WTF::move(successCallback);
159 m_closedFailureCallback = WTF::move(failureCallback);
162 void ReadableStream::read(ReadSuccessCallback&& successCallback, ReadEndCallback&& endCallback, FailureCallback&& failureCallback)
164 if (m_state == State::Closed) {
168 if (m_state == State::Errored) {
169 failureCallback(error());
173 successCallback(read());
174 if (!m_closeRequested)
176 else if (!hasValue())
180 m_readRequests.append({ WTF::move(successCallback), WTF::move(endCallback), WTF::move(failureCallback) });
184 bool ReadableStream::resolveReadCallback(JSC::JSValue value)
186 if (m_readRequests.isEmpty())
189 m_readRequests.takeFirst().successCallback(value);
193 const char* ReadableStream::activeDOMObjectName() const
195 return "ReadableStream";
198 bool ReadableStream::canSuspendForPageCache() const
200 // FIXME: We should try and do better here.