[Streams API] Collecting a ReadableStreamReader should not unlock its stream
[WebKit-https.git] / Source / WebCore / bindings / js / JSReadableStreamReaderCustom.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
32 #if ENABLE(STREAMS_API)
33 #include "JSReadableStreamReader.h"
34
35 #include "ExceptionCode.h"
36 #include "JSDOMPromise.h"
37 #include "JSReadableStream.h"
38 #include "ReadableStreamJSSource.h"
39 #include <runtime/Error.h>
40 #include <wtf/NeverDestroyed.h>
41
42 using namespace JSC;
43
44 namespace WebCore {
45
46 JSValue JSReadableStreamReader::read(ExecState* exec)
47 {
48     JSValue error = createError(exec, ASCIILiteral("read is not implemented"));
49     return exec->vm().throwException(exec, error);
50 }
51
52 static JSPromiseDeferred* getOrCreatePromiseDeferredFromObject(ExecState* exec, JSValue thisObject, JSGlobalObject* globalObject, PrivateName &name)
53 {
54     JSValue slot = getInternalSlotFromObject(exec, thisObject, name);
55     JSPromiseDeferred* promiseDeferred = slot ? jsDynamicCast<JSPromiseDeferred*>(slot) : nullptr;
56
57     if (!promiseDeferred) {
58         promiseDeferred = JSPromiseDeferred::create(exec, globalObject);
59         setInternalSlotToObject(exec, thisObject, name, promiseDeferred);
60     }
61     return promiseDeferred;
62 }
63
64 static JSC::PrivateName& closedPromiseSlotName()
65 {
66     static NeverDestroyed<JSC::PrivateName> closedPromiseSlotName("closedPromise");
67     return closedPromiseSlotName;
68 }
69
70 JSValue JSReadableStreamReader::closed(ExecState* exec) const
71 {
72     JSPromiseDeferred* promiseDeferred = getOrCreatePromiseDeferredFromObject(exec, this, globalObject(), closedPromiseSlotName());
73     DeferredWrapper wrapper(exec, globalObject(), promiseDeferred);
74     auto successCallback = [this, wrapper]() mutable {
75         // FIXME: return jsUndefined().
76         wrapper.resolve(&impl());
77     };
78     auto failureCallback = [this, wrapper]() mutable {
79         // FIXME: return stored error.
80         wrapper.reject(&impl());
81     };
82
83     impl().closed(WTF::move(successCallback), WTF::move(failureCallback));
84
85     return wrapper.promise();
86 }
87
88 JSValue JSReadableStreamReader::cancel(ExecState* exec)
89 {
90     JSValue error = createError(exec, ASCIILiteral("cancel is not implemented"));
91     return exec->vm().throwException(exec, error);
92 }
93
94 JSValue JSReadableStreamReader::releaseLock(ExecState* exec)
95 {
96     JSValue error = createError(exec, ASCIILiteral("releaseLock is not implemented"));
97     return exec->vm().throwException(exec, error);
98 }
99
100 EncodedJSValue JSC_HOST_CALL constructJSReadableStreamReader(ExecState* exec)
101 {
102     if (!exec->argumentCount())
103         return throwVMError(exec, createTypeError(exec, ASCIILiteral("ReadableStreamReader constructor takes a ReadableStream as parameter")));
104
105     JSReadableStream* stream = jsDynamicCast<JSReadableStream*>(exec->argument(0));
106     if (!stream)
107         return throwVMError(exec, createTypeError(exec, ASCIILiteral("ReadableStreamReader constructor parameter is not a ReadableStream")));
108
109     if (stream->impl().isLocked())
110         return throwVMError(exec, createTypeError(exec, ASCIILiteral("ReadableStreamReader constructor parameter is a locked ReadableStream")));
111
112     return JSValue::encode(toJS(exec, stream->globalObject(), stream->impl().createReader()));
113 }
114
115 } // namespace WebCore
116
117 #endif