2d3dbc15e06bdddd62921ba85de23ae934cc9f06
[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)
56 {
57     CallData callData;
58     CallType callType = getCallData(jsFunction, callData);
59     return call(&exec, jsFunction, callType, callData, thisValue, arguments);
60 }
61
62 JSValue ReadableJSStream::invoke(ExecState& exec, const char* propertyName)
63 {
64     JSValue function = getPropertyFromObject(exec, m_source.get(), propertyName);
65     if (exec.hadException())
66         return jsUndefined();
67
68     if (!function.isFunction()) {
69         if (!function.isUndefined())
70             throwVMError(&exec, createTypeError(&exec, ASCIILiteral("ReadableStream trying to call a property that is not callable")));
71         return jsUndefined();
72     }
73
74     MarkedArgumentBuffer arguments;
75     arguments.append(jsController(exec, globalObject()));
76     return callFunction(exec, function, m_source.get(), arguments);
77 }
78
79 JSDOMGlobalObject* ReadableJSStream::globalObject()
80 {
81     return jsDynamicCast<JSDOMGlobalObject*>(m_source->globalObject());
82 }
83
84 static void startReadableStreamAsync(ReadableStream& readableStream)
85 {
86     RefPtr<ReadableStream> stream = &readableStream;
87     stream->scriptExecutionContext()->postTask([stream](ScriptExecutionContext&) {
88         stream->start();
89     });
90 }
91
92 void ReadableJSStream::doStart(ExecState& exec)
93 {
94     JSLockHolder lock(&exec);
95
96     invoke(exec, "start");
97
98     if (exec.hadException())
99         return;
100
101     // FIXME: Implement handling promise as result of calling start function.
102     startReadableStreamAsync(*this);
103 }
104
105 RefPtr<ReadableJSStream> ReadableJSStream::create(ExecState& exec, ScriptExecutionContext& scriptExecutionContext)
106 {
107     JSObject* jsSource;
108     JSValue value = exec.argument(0);
109     if (value.isObject())
110         jsSource = value.getObject();
111     else if (!value.isUndefined()) {
112         throwVMError(&exec, createTypeError(&exec, ASCIILiteral("ReadableStream constructor first argument, if any, should be an object")));
113         return nullptr;
114     } else
115         jsSource = JSFinalObject::create(exec.vm(), JSFinalObject::createStructure(exec.vm(), exec.callee()->globalObject(), jsNull(), 1));
116
117     RefPtr<ReadableJSStream> readableStream = adoptRef(*new ReadableJSStream(scriptExecutionContext, exec, jsSource));
118     readableStream->doStart(exec);
119
120     if (exec.hadException())
121         return nullptr;
122
123     return readableStream;
124 }
125
126 ReadableJSStream::ReadableJSStream(ScriptExecutionContext& scriptExecutionContext, ExecState& exec, JSObject* source)
127     : ReadableStream(scriptExecutionContext)
128 {
129     m_source.set(exec.vm(), source);
130 }
131
132 JSValue ReadableJSStream::jsController(ExecState& exec, JSDOMGlobalObject* globalObject)
133 {
134     if (!m_controller)
135         m_controller = std::make_unique<ReadableStreamController>(*this);
136     return toJS(&exec, globalObject, m_controller.get());
137 }
138
139 void ReadableJSStream::storeError(JSC::ExecState& exec)
140 {
141     if (m_error)
142         return;
143     JSValue error = exec.argumentCount() ? exec.argument(0) : createError(&exec, ASCIILiteral("Error function called."));
144     m_error.set(exec.vm(), error);
145
146     changeStateToErrored();
147 }
148
149 bool ReadableJSStream::hasValue() const
150 {
151     return m_chunkQueue.size();
152 }
153
154 JSValue ReadableJSStream::read()
155 {
156     ASSERT(hasValue());
157
158     return m_chunkQueue.takeFirst().get();
159 }
160
161 void ReadableJSStream::enqueue(ExecState& exec)
162 {
163     ASSERT(!isCloseRequested());
164
165     if (!isReadable())
166         return;
167
168     JSValue chunk = exec.argumentCount() ? exec.argument(0) : jsUndefined();
169     if (resolveReadCallback(chunk))
170         return;
171
172     m_chunkQueue.append(JSC::Strong<JSC::Unknown>(exec.vm(), chunk));
173     // FIXME: Compute chunk size.
174     // FIXME: Add pulling of data here and also when data is passed to resolve callback.
175 }
176
177 } // namespace WebCore
178
179 #endif