2009-04-09 Jian Li <jianli@chromium.org>
[WebKit-https.git] / WebCore / bindings / v8 / custom / V8XMLHttpRequestCustom.cpp
1 /*
2  * Copyright (C) 2008, 2009 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "XMLHttpRequest.h"
33
34 #include "Frame.h"
35 #include "V8Binding.h"
36 #include "V8Document.h"
37 #include "V8CustomBinding.h"
38 #include "V8HTMLDocument.h"
39 #include "V8ObjectEventListener.h"
40 #include "V8Proxy.h"
41 #include "V8XMLHttpRequestUtilities.h"
42 #include "WorkerContext.h"
43 #include "WorkerContextExecutionProxy.h"
44
45 namespace WebCore {
46
47 PassRefPtr<EventListener> getEventListener(XMLHttpRequest* xmlHttpRequest, v8::Local<v8::Value> value, bool findOnly)
48 {
49 #if ENABLE(WORKERS)
50     WorkerContextExecutionProxy* workerContextProxy = WorkerContextExecutionProxy::retrieve();
51     if (workerContextProxy)
52         return workerContextProxy->findOrCreateObjectEventListener(value, false, findOnly);
53 #endif
54
55     V8Proxy* proxy = V8Proxy::retrieve(xmlHttpRequest->scriptExecutionContext());
56     if (proxy)
57         return findOnly ? proxy->FindObjectEventListener(value, false) : proxy->FindOrCreateObjectEventListener(value, false);
58
59     return PassRefPtr<EventListener>();
60 }
61
62 ACCESSOR_GETTER(XMLHttpRequestOnabort)
63 {
64     INC_STATS("DOM.XMLHttpRequest.onabort._get");
65     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
66     if (xmlHttpRequest->onabort()) {
67         V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onabort());
68         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
69         return v8Listener;
70     }
71     return v8::Undefined();
72 }
73
74 ACCESSOR_SETTER(XMLHttpRequestOnabort)
75 {
76     INC_STATS("DOM.XMLHttpRequest.onabort._set");
77     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
78     if (value->IsNull()) {
79         if (xmlHttpRequest->onabort()) {
80             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onabort());
81             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
82             removeHiddenXHRDependency(info.Holder(), v8Listener);
83         }
84
85         // Clear the listener.
86         xmlHttpRequest->setOnabort(0);
87     } else {
88         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
89         if (listener) {
90             xmlHttpRequest->setOnabort(listener);
91             createHiddenXHRDependency(info.Holder(), value);
92         }
93     }
94 }
95
96 ACCESSOR_GETTER(XMLHttpRequestOnerror)
97 {
98     INC_STATS("DOM.XMLHttpRequest.onerror._get");
99     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
100     if (xmlHttpRequest->onerror()) {
101         RefPtr<V8ObjectEventListener> listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onerror());
102         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
103         return v8Listener;
104     }
105     return v8::Undefined();
106 }
107
108 ACCESSOR_SETTER(XMLHttpRequestOnerror)
109 {
110     INC_STATS("DOM.XMLHttpRequest.onerror._set");
111     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
112     if (value->IsNull()) {
113         if (xmlHttpRequest->onerror()) {
114             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onerror());
115             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
116             removeHiddenXHRDependency(info.Holder(), v8Listener);
117         }
118
119         // Clear the listener.
120         xmlHttpRequest->setOnerror(0);
121     } else {
122         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
123         if (listener) {
124             xmlHttpRequest->setOnerror(listener);
125             createHiddenXHRDependency(info.Holder(), value);
126         }
127     }
128 }
129
130 ACCESSOR_GETTER(XMLHttpRequestOnload)
131 {
132     INC_STATS("DOM.XMLHttpRequest.onload._get");
133     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
134     if (xmlHttpRequest->onload()) {
135         V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onload());
136         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
137         return v8Listener;
138     }
139     return v8::Undefined();
140 }
141
142 ACCESSOR_SETTER(XMLHttpRequestOnload)
143 {
144     INC_STATS("DOM.XMLHttpRequest.onload._set");
145     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
146     if (value->IsNull()) {
147         if (xmlHttpRequest->onload()) {
148             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onload());
149             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
150             removeHiddenXHRDependency(info.Holder(), v8Listener);
151         }
152
153         xmlHttpRequest->setOnload(0);
154
155     } else {
156         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
157         if (listener) {
158             xmlHttpRequest->setOnload(listener.get());
159             createHiddenXHRDependency(info.Holder(), value);
160         }
161     }
162 }
163
164 ACCESSOR_GETTER(XMLHttpRequestOnloadstart)
165 {
166     INC_STATS("DOM.XMLHttpRequest.onloadstart._get");
167     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
168     if (xmlHttpRequest->onloadstart()) {
169         V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onloadstart());
170         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
171         return v8Listener;
172     }
173     return v8::Undefined();
174 }
175
176 ACCESSOR_SETTER(XMLHttpRequestOnloadstart)
177 {
178     INC_STATS("DOM.XMLHttpRequest.onloadstart._set");
179     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
180     if (value->IsNull()) {
181         if (xmlHttpRequest->onloadstart()) {
182             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onloadstart());
183             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
184             removeHiddenXHRDependency(info.Holder(), v8Listener);
185         }
186
187         // Clear the listener.
188         xmlHttpRequest->setOnloadstart(0);
189     } else {
190         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
191         if (listener) {
192             xmlHttpRequest->setOnloadstart(listener);
193             createHiddenXHRDependency(info.Holder(), value);
194         }
195     }
196 }
197
198 ACCESSOR_GETTER(XMLHttpRequestOnprogress)
199 {
200     INC_STATS("DOM.XMLHttpRequest.onprogress._get");
201     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
202     if (xmlHttpRequest->onprogress()) {
203         V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onprogress());
204         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
205         return v8Listener;
206     }
207     return v8::Undefined();
208 }
209
210 ACCESSOR_SETTER(XMLHttpRequestOnprogress)
211 {
212     INC_STATS("DOM.XMLHttpRequest.onprogress._set");
213     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
214     if (value->IsNull()) {
215         if (xmlHttpRequest->onprogress()) {
216             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onprogress());
217             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
218             removeHiddenXHRDependency(info.Holder(), v8Listener);
219         }
220
221         // Clear the listener.
222         xmlHttpRequest->setOnprogress(0);
223     } else {
224         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
225         if (listener) {
226             xmlHttpRequest->setOnprogress(listener);
227             createHiddenXHRDependency(info.Holder(), value);
228         }
229     }
230 }
231
232 ACCESSOR_GETTER(XMLHttpRequestOnreadystatechange)
233 {
234     INC_STATS("DOM.XMLHttpRequest.onreadystatechange._get");
235     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
236     if (xmlHttpRequest->onreadystatechange()) {
237         V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onreadystatechange());
238         v8::Local<v8::Object> v8Listener = listener->getListenerObject();
239         return v8Listener;
240     }
241     return v8::Undefined();
242 }
243
244 ACCESSOR_SETTER(XMLHttpRequestOnreadystatechange)
245 {
246     INC_STATS("DOM.XMLHttpRequest.onreadystatechange._set");
247     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
248     if (value->IsNull()) {
249         if (xmlHttpRequest->onreadystatechange()) {
250             V8ObjectEventListener* listener = static_cast<V8ObjectEventListener*>(xmlHttpRequest->onreadystatechange());
251             v8::Local<v8::Object> v8Listener = listener->getListenerObject();
252             removeHiddenXHRDependency(info.Holder(), v8Listener);
253         }
254
255         // Clear the listener.
256         xmlHttpRequest->setOnreadystatechange(0);
257     } else {
258         RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, value, false);
259         if (listener) {
260             xmlHttpRequest->setOnreadystatechange(listener.get());
261             createHiddenXHRDependency(info.Holder(), value);
262         }
263     }
264 }
265
266 ACCESSOR_GETTER(XMLHttpRequestResponseText)
267 {
268     // FIXME: This is only needed because webkit set this getter as custom.
269     // So we need a custom method to avoid forking the IDL file.
270     INC_STATS("DOM.XMLHttpRequest.responsetext._get");
271     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, info.Holder());
272     return v8StringOrNull(xmlHttpRequest->responseText());
273 }
274
275 CALLBACK_FUNC_DECL(XMLHttpRequestAddEventListener)
276 {
277     INC_STATS("DOM.XMLHttpRequest.addEventListener()");
278     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
279
280     RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, args[1], false);
281     if (listener) {
282         String type = toWebCoreString(args[0]);
283         bool useCapture = args[2]->BooleanValue();
284         xmlHttpRequest->addEventListener(type, listener, useCapture);
285
286         createHiddenXHRDependency(args.Holder(), args[1]);
287     }
288     return v8::Undefined();
289 }
290
291 CALLBACK_FUNC_DECL(XMLHttpRequestRemoveEventListener)
292 {
293     INC_STATS("DOM.XMLHttpRequest.removeEventListener()");
294     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
295
296     RefPtr<EventListener> listener = getEventListener(xmlHttpRequest, args[1], true);
297     if (listener) {
298         String type = toWebCoreString(args[0]);
299         bool useCapture = args[2]->BooleanValue();
300         xmlHttpRequest->removeEventListener(type, listener.get(), useCapture);
301
302         removeHiddenXHRDependency(args.Holder(), args[1]);
303     }
304
305     return v8::Undefined();
306 }
307
308 CALLBACK_FUNC_DECL(XMLHttpRequestOpen)
309 {
310     INC_STATS("DOM.XMLHttpRequest.open()");
311     // Four cases:
312     // open(method, url)
313     // open(method, url, async)
314     // open(method, url, async, user)
315     // open(method, url, async, user, passwd)
316
317     if (args.Length() < 2)
318         return throwError("Not enough arguments", V8Proxy::SYNTAX_ERROR);
319
320     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
321
322     String method = toWebCoreString(args[0]);
323     String urlstring = toWebCoreString(args[1]);
324     ScriptExecutionContext* context = 0;
325 #if ENABLE(WORKERS)
326     WorkerContextExecutionProxy* proxy = WorkerContextExecutionProxy::retrieve();
327     if (proxy)
328         context = proxy->workerContext();
329     else
330 #endif
331         context = V8Proxy::retrieve()->frame()->document();
332     KURL url = context->completeURL(urlstring);
333
334     bool async = (args.Length() < 3) ? true : args[2]->BooleanValue();
335
336     ExceptionCode ec = 0;
337     String user, passwd;
338     if (args.Length() >= 4 && !args[3]->IsUndefined()) {
339         user = valueToStringWithNullCheck(args[3]);
340
341         if (args.Length() >= 5 && !args[4]->IsUndefined()) {
342             passwd = valueToStringWithNullCheck(args[4]);
343             xmlHttpRequest->open(method, url, async, user, passwd, ec);
344         } else
345             xmlHttpRequest->open(method, url, async, user, ec);
346     } else
347         xmlHttpRequest->open(method, url, async, ec);
348
349     if (ec)
350         return throwError(ec);
351
352     return v8::Undefined();
353 }
354
355 static bool IsDocumentType(v8::Handle<v8::Value> value)
356 {
357     // FIXME: add other document types.
358     return V8Document::HasInstance(value) || V8HTMLDocument::HasInstance(value);
359 }
360
361 CALLBACK_FUNC_DECL(XMLHttpRequestSend)
362 {
363     INC_STATS("DOM.XMLHttpRequest.send()");
364     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
365
366     ExceptionCode ec = 0;
367     if (args.Length() < 1)
368         xmlHttpRequest->send(ec);
369     else {
370         v8::Handle<v8::Value> arg = args[0];
371         // FIXME: upstream handles "File" objects too.
372         if (IsDocumentType(arg)) {
373             v8::Handle<v8::Object> object = v8::Handle<v8::Object>::Cast(arg);
374             Document* document = V8Proxy::DOMWrapperToNode<Document>(object);
375             ASSERT(document);
376             xmlHttpRequest->send(document, ec);
377         } else
378             xmlHttpRequest->send(valueToStringWithNullCheck(arg), ec);
379     }
380
381     if (ec)
382         return throwError(ec);
383
384     return v8::Undefined();
385 }
386
387 CALLBACK_FUNC_DECL(XMLHttpRequestSetRequestHeader) {
388     INC_STATS("DOM.XMLHttpRequest.setRequestHeader()");
389     if (args.Length() < 2)
390         return throwError("Not enough arguments", V8Proxy::SYNTAX_ERROR);
391
392     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
393     ExceptionCode ec = 0;
394     String header = toWebCoreString(args[0]);
395     String value = toWebCoreString(args[1]);
396     xmlHttpRequest->setRequestHeader(header, value, ec);
397     if (ec)
398         return throwError(ec);
399     return v8::Undefined();
400 }
401
402 CALLBACK_FUNC_DECL(XMLHttpRequestGetResponseHeader)
403 {
404     INC_STATS("DOM.XMLHttpRequest.getResponseHeader()");
405     if (args.Length() < 1)
406         return throwError("Not enough arguments", V8Proxy::SYNTAX_ERROR);
407
408     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
409     ExceptionCode ec = 0;
410     String header = toWebCoreString(args[0]);
411     String result = xmlHttpRequest->getResponseHeader(header, ec);
412     if (ec)
413         return throwError(ec);
414     return v8StringOrNull(result);
415 }
416
417 CALLBACK_FUNC_DECL(XMLHttpRequestOverrideMimeType)
418 {
419     INC_STATS("DOM.XMLHttpRequest.overrideMimeType()");
420     if (args.Length() < 1)
421         return throwError("Not enough arguments", V8Proxy::SYNTAX_ERROR);
422
423     XMLHttpRequest* xmlHttpRequest = V8Proxy::ToNativeObject<XMLHttpRequest>(V8ClassIndex::XMLHTTPREQUEST, args.Holder());
424     String value = toWebCoreString(args[0]);
425     xmlHttpRequest->overrideMimeType(value);
426     return v8::Undefined();
427 }
428
429 CALLBACK_FUNC_DECL(XMLHttpRequestDispatchEvent)
430 {
431     INC_STATS("DOM.XMLHttpRequest.dispatchEvent()");
432     return v8::Undefined();
433 }
434
435 } // namespace WebCore