finally blocks should not set the exception stack trace when re-throwing the exception.
[WebKit-https.git] / Source / WebCore / bindings / js / ReadableJSStream.cpp
1 /*
2  * Copyright (C) 2015 Canon Inc.
3  * Copyright (C) 2015 Igalia S.L.
4  *
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:
8  *
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.
17  *
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.
28  */
29
30 #include "config.h"
31 #include "ReadableJSStream.h"
32
33 #if ENABLE(STREAMS_API)
34
35 #include "DOMWrapperWorld.h"
36 #include "JSDOMPromise.h"
37 #include "JSReadableStream.h"
38 #include "JSReadableStreamController.h"
39 #include "NotImplemented.h"
40 #include "ScriptExecutionContext.h"
41 #include <runtime/Error.h>
42 #include <runtime/JSCJSValueInlines.h>
43 #include <runtime/JSString.h>
44 #include <runtime/StructureInlines.h>
45
46 using namespace JSC;
47
48 namespace WebCore {
49
50 static inline JSValue getPropertyFromObject(ExecState* exec, JSObject* object, const char* identifier)
51 {
52     return object->get(exec, Identifier::fromString(exec, identifier));
53 }
54
55 static inline JSValue callFunction(ExecState* exec, JSValue jsFunction, JSValue thisValue, const ArgList& arguments, Exception*& exception)
56 {
57     CallData callData;
58     CallType callType = getCallData(jsFunction, callData);
59     return call(exec, jsFunction, callType, callData, thisValue, arguments, exception);
60 }
61
62 JSDOMGlobalObject* ReadableJSStream::globalObject()
63 {
64     return jsDynamicCast<JSDOMGlobalObject*>(m_source->globalObject());
65 }
66
67 static void startReadableStreamAsync(ReadableStream& readableStream)
68 {
69     RefPtr<ReadableStream> stream = &readableStream;
70     stream->scriptExecutionContext()->postTask([stream](ScriptExecutionContext&) {
71         stream->start();
72     });
73 }
74
75 void ReadableJSStream::doStart(ExecState& exec)
76 {
77     JSLockHolder lock(&exec);
78
79     JSValue startFunction = getPropertyFromObject(&exec, m_source.get(), "start");
80     if (!startFunction.isFunction()) {
81         if (!startFunction.isUndefined())
82             throwVMError(&exec, createTypeError(&exec, ASCIILiteral("ReadableStream constructor object start property should be a function.")));
83         else
84             startReadableStreamAsync(*this);
85         return;
86     }
87
88     MarkedArgumentBuffer arguments;
89     arguments.append(jsController(exec, globalObject()));
90
91     Exception* exception;
92     callFunction(&exec, startFunction, m_source.get(), arguments, exception);
93
94     if (exception) {
95         throwVMError(&exec, exception);
96         return;
97     }
98
99     // FIXME: Implement handling promise as result of calling start function.
100     startReadableStreamAsync(*this);
101 }
102
103 Ref<ReadableJSStream> ReadableJSStream::create(ExecState& exec, ScriptExecutionContext& scriptExecutionContext)
104 {
105     ASSERT_WITH_MESSAGE(!exec.argumentCount() || exec.argument(0).isObject(), "Caller of ReadableJSStream constructor should ensure that passed argument if any is an object.");
106     JSObject* source =  exec.argumentCount() ? exec.argument(0).getObject() : JSFinalObject::create(exec.vm(), JSFinalObject::createStructure(exec.vm(), exec.callee()->globalObject(), jsNull(), 1));
107
108     Ref<ReadableJSStream> readableStream = adoptRef(*new ReadableJSStream(scriptExecutionContext, exec, source));
109     readableStream->doStart(exec);
110     return readableStream;
111 }
112
113 ReadableJSStream::ReadableJSStream(ScriptExecutionContext& scriptExecutionContext, ExecState& exec, JSObject* source)
114     : ReadableStream(scriptExecutionContext)
115 {
116     m_source.set(exec.vm(), source);
117 }
118
119 JSValue ReadableJSStream::jsController(ExecState& exec, JSDOMGlobalObject* globalObject)
120 {
121     if (!m_controller)
122         m_controller = std::make_unique<ReadableStreamController>(*this);
123     return toJS(&exec, globalObject, m_controller.get());
124 }
125
126 void ReadableJSStream::storeError(JSC::ExecState& exec)
127 {
128     if (m_error)
129         return;
130     JSValue error = exec.argumentCount() ? exec.argument(0) : createError(&exec, ASCIILiteral("Error function called."));
131     m_error.set(exec.vm(), error);
132
133     changeStateToErrored();
134 }
135
136 bool ReadableJSStream::hasValue() const
137 {
138     return m_chunkQueue.size();
139 }
140
141 JSValue ReadableJSStream::read()
142 {
143     ASSERT(hasValue());
144
145     return m_chunkQueue.takeFirst().get();
146 }
147
148 void ReadableJSStream::enqueue(ExecState& exec)
149 {
150     ASSERT(!isCloseRequested());
151
152     if (!isReadable())
153         return;
154
155     JSValue chunk = exec.argumentCount() ? exec.argument(0) : jsUndefined();
156     if (resolveReadCallback(chunk))
157         return;
158
159     m_chunkQueue.append(JSC::Strong<JSC::Unknown>(exec.vm(), chunk));
160     // FIXME: Compute chunk size.
161     // FIXME: Add pulling of data here and also when data is passed to resolve callback.
162 }
163
164 } // namespace WebCore
165
166 #endif