• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012 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 
33 
34 #include "core/inspector/InjectedScript.h"
35 
36 #include "bindings/v8/ScriptFunctionCall.h"
37 #include "core/inspector/InjectedScriptHost.h"
38 #include "platform/JSONValues.h"
39 #include "wtf/text/WTFString.h"
40 
41 using WebCore::TypeBuilder::Array;
42 using WebCore::TypeBuilder::Debugger::CallFrame;
43 using WebCore::TypeBuilder::Runtime::PropertyDescriptor;
44 using WebCore::TypeBuilder::Runtime::InternalPropertyDescriptor;
45 using WebCore::TypeBuilder::Debugger::FunctionDetails;
46 using WebCore::TypeBuilder::Runtime::RemoteObject;
47 
48 namespace WebCore {
49 
InjectedScript()50 InjectedScript::InjectedScript()
51     : InjectedScriptBase("InjectedScript")
52 {
53 }
54 
InjectedScript(ScriptObject injectedScriptObject,InspectedStateAccessCheck accessCheck)55 InjectedScript::InjectedScript(ScriptObject injectedScriptObject, InspectedStateAccessCheck accessCheck)
56     : InjectedScriptBase("InjectedScript", injectedScriptObject, accessCheck)
57 {
58 }
59 
evaluate(ErrorString * errorString,const String & expression,const String & objectGroup,bool includeCommandLineAPI,bool returnByValue,bool generatePreview,RefPtr<TypeBuilder::Runtime::RemoteObject> * result,TypeBuilder::OptOutput<bool> * wasThrown)60 void InjectedScript::evaluate(ErrorString* errorString, const String& expression, const String& objectGroup, bool includeCommandLineAPI, bool returnByValue, bool generatePreview, RefPtr<TypeBuilder::Runtime::RemoteObject>* result, TypeBuilder::OptOutput<bool>* wasThrown)
61 {
62     ScriptFunctionCall function(injectedScriptObject(), "evaluate");
63     function.appendArgument(expression);
64     function.appendArgument(objectGroup);
65     function.appendArgument(includeCommandLineAPI);
66     function.appendArgument(returnByValue);
67     function.appendArgument(generatePreview);
68     makeEvalCall(errorString, function, result, wasThrown);
69 }
70 
callFunctionOn(ErrorString * errorString,const String & objectId,const String & expression,const String & arguments,bool returnByValue,bool generatePreview,RefPtr<TypeBuilder::Runtime::RemoteObject> * result,TypeBuilder::OptOutput<bool> * wasThrown)71 void InjectedScript::callFunctionOn(ErrorString* errorString, const String& objectId, const String& expression, const String& arguments, bool returnByValue, bool generatePreview, RefPtr<TypeBuilder::Runtime::RemoteObject>* result, TypeBuilder::OptOutput<bool>* wasThrown)
72 {
73     ScriptFunctionCall function(injectedScriptObject(), "callFunctionOn");
74     function.appendArgument(objectId);
75     function.appendArgument(expression);
76     function.appendArgument(arguments);
77     function.appendArgument(returnByValue);
78     function.appendArgument(generatePreview);
79     makeEvalCall(errorString, function, result, wasThrown);
80 }
81 
evaluateOnCallFrame(ErrorString * errorString,const ScriptValue & callFrames,const String & callFrameId,const String & expression,const String & objectGroup,bool includeCommandLineAPI,bool returnByValue,bool generatePreview,RefPtr<RemoteObject> * result,TypeBuilder::OptOutput<bool> * wasThrown)82 void InjectedScript::evaluateOnCallFrame(ErrorString* errorString, const ScriptValue& callFrames, const String& callFrameId, const String& expression, const String& objectGroup, bool includeCommandLineAPI, bool returnByValue, bool generatePreview, RefPtr<RemoteObject>* result, TypeBuilder::OptOutput<bool>* wasThrown)
83 {
84     ScriptFunctionCall function(injectedScriptObject(), "evaluateOnCallFrame");
85     function.appendArgument(callFrames);
86     function.appendArgument(callFrameId);
87     function.appendArgument(expression);
88     function.appendArgument(objectGroup);
89     function.appendArgument(includeCommandLineAPI);
90     function.appendArgument(returnByValue);
91     function.appendArgument(generatePreview);
92     makeEvalCall(errorString, function, result, wasThrown);
93 }
94 
restartFrame(ErrorString * errorString,const ScriptValue & callFrames,const String & callFrameId,RefPtr<JSONObject> * result)95 void InjectedScript::restartFrame(ErrorString* errorString, const ScriptValue& callFrames, const String& callFrameId, RefPtr<JSONObject>* result)
96 {
97     ScriptFunctionCall function(injectedScriptObject(), "restartFrame");
98     function.appendArgument(callFrames);
99     function.appendArgument(callFrameId);
100     RefPtr<JSONValue> resultValue;
101     makeCall(function, &resultValue);
102     if (resultValue) {
103         if (resultValue->type() == JSONValue::TypeString) {
104             resultValue->asString(errorString);
105             return;
106         }
107         if (resultValue->type() == JSONValue::TypeObject) {
108             *result = resultValue->asObject();
109             return;
110         }
111     }
112     *errorString = "Internal error";
113 }
114 
getStepInPositions(ErrorString * errorString,const ScriptValue & callFrames,const String & callFrameId,RefPtr<Array<TypeBuilder::Debugger::Location>> & positions)115 void InjectedScript::getStepInPositions(ErrorString* errorString, const ScriptValue& callFrames, const String& callFrameId, RefPtr<Array<TypeBuilder::Debugger::Location> >& positions)
116 {
117     ScriptFunctionCall function(injectedScriptObject(), "getStepInPositions");
118     function.appendArgument(callFrames);
119     function.appendArgument(callFrameId);
120     RefPtr<JSONValue> resultValue;
121     makeCall(function, &resultValue);
122     if (resultValue) {
123         if (resultValue->type() == JSONValue::TypeString) {
124             resultValue->asString(errorString);
125             return;
126         }
127         if (resultValue->type() == JSONValue::TypeArray) {
128             positions = Array<TypeBuilder::Debugger::Location>::runtimeCast(resultValue);
129             return;
130         }
131     }
132     *errorString = "Internal error";
133 }
134 
setVariableValue(ErrorString * errorString,const ScriptValue & callFrames,const String * callFrameIdOpt,const String * functionObjectIdOpt,int scopeNumber,const String & variableName,const String & newValueStr)135 void InjectedScript::setVariableValue(ErrorString* errorString, const ScriptValue& callFrames, const String* callFrameIdOpt, const String* functionObjectIdOpt, int scopeNumber, const String& variableName, const String& newValueStr)
136 {
137     ScriptFunctionCall function(injectedScriptObject(), "setVariableValue");
138     if (callFrameIdOpt) {
139         function.appendArgument(callFrames);
140         function.appendArgument(*callFrameIdOpt);
141     } else {
142         function.appendArgument(false);
143         function.appendArgument(false);
144     }
145     if (functionObjectIdOpt)
146         function.appendArgument(*functionObjectIdOpt);
147     else
148         function.appendArgument(false);
149     function.appendArgument(scopeNumber);
150     function.appendArgument(variableName);
151     function.appendArgument(newValueStr);
152     RefPtr<JSONValue> resultValue;
153     makeCall(function, &resultValue);
154     if (!resultValue) {
155         *errorString = "Internal error";
156         return;
157     }
158     if (resultValue->type() == JSONValue::TypeString) {
159         resultValue->asString(errorString);
160         return;
161     }
162     // Normal return.
163 }
164 
getFunctionDetails(ErrorString * errorString,const String & functionId,RefPtr<FunctionDetails> * result)165 void InjectedScript::getFunctionDetails(ErrorString* errorString, const String& functionId, RefPtr<FunctionDetails>* result)
166 {
167     ScriptFunctionCall function(injectedScriptObject(), "getFunctionDetails");
168     function.appendArgument(functionId);
169     RefPtr<JSONValue> resultValue;
170     makeCall(function, &resultValue);
171     if (!resultValue || resultValue->type() != JSONValue::TypeObject) {
172         if (!resultValue->asString(errorString))
173             *errorString = "Internal error";
174         return;
175     }
176     *result = FunctionDetails::runtimeCast(resultValue);
177 }
178 
getProperties(ErrorString * errorString,const String & objectId,bool ownProperties,bool accessorPropertiesOnly,RefPtr<Array<PropertyDescriptor>> * properties)179 void InjectedScript::getProperties(ErrorString* errorString, const String& objectId, bool ownProperties, bool accessorPropertiesOnly, RefPtr<Array<PropertyDescriptor> >* properties)
180 {
181     ScriptFunctionCall function(injectedScriptObject(), "getProperties");
182     function.appendArgument(objectId);
183     function.appendArgument(ownProperties);
184     function.appendArgument(accessorPropertiesOnly);
185 
186     RefPtr<JSONValue> result;
187     makeCall(function, &result);
188     if (!result || result->type() != JSONValue::TypeArray) {
189         *errorString = "Internal error";
190         return;
191     }
192     *properties = Array<PropertyDescriptor>::runtimeCast(result);
193 }
194 
getInternalProperties(ErrorString * errorString,const String & objectId,RefPtr<Array<InternalPropertyDescriptor>> * properties)195 void InjectedScript::getInternalProperties(ErrorString* errorString, const String& objectId, RefPtr<Array<InternalPropertyDescriptor> >* properties)
196 {
197     ScriptFunctionCall function(injectedScriptObject(), "getInternalProperties");
198     function.appendArgument(objectId);
199 
200     RefPtr<JSONValue> result;
201     makeCall(function, &result);
202     if (!result || result->type() != JSONValue::TypeArray) {
203         *errorString = "Internal error";
204         return;
205     }
206     RefPtr<Array<InternalPropertyDescriptor> > array = Array<InternalPropertyDescriptor>::runtimeCast(result);
207     if (array->length() > 0)
208         *properties = array;
209 }
210 
nodeForObjectId(const String & objectId)211 Node* InjectedScript::nodeForObjectId(const String& objectId)
212 {
213     if (hasNoValue() || !canAccessInspectedWindow())
214         return 0;
215 
216     ScriptFunctionCall function(injectedScriptObject(), "nodeForObjectId");
217     function.appendArgument(objectId);
218 
219     bool hadException = false;
220     ScriptValue resultValue = callFunctionWithEvalEnabled(function, hadException);
221     ASSERT(!hadException);
222 
223     return InjectedScriptHost::scriptValueAsNode(resultValue);
224 }
225 
releaseObject(const String & objectId)226 void InjectedScript::releaseObject(const String& objectId)
227 {
228     ScriptFunctionCall function(injectedScriptObject(), "releaseObject");
229     function.appendArgument(objectId);
230     RefPtr<JSONValue> result;
231     makeCall(function, &result);
232 }
233 
wrapCallFrames(const ScriptValue & callFrames)234 PassRefPtr<Array<CallFrame> > InjectedScript::wrapCallFrames(const ScriptValue& callFrames)
235 {
236     ASSERT(!hasNoValue());
237     ScriptFunctionCall function(injectedScriptObject(), "wrapCallFrames");
238     function.appendArgument(callFrames);
239     bool hadException = false;
240     ScriptValue callFramesValue = callFunctionWithEvalEnabled(function, hadException);
241     ASSERT(!hadException);
242     RefPtr<JSONValue> result = callFramesValue.toJSONValue(scriptState());
243     if (result->type() == JSONValue::TypeArray)
244         return Array<CallFrame>::runtimeCast(result);
245     return Array<CallFrame>::create();
246 }
247 
wrapObject(const ScriptValue & value,const String & groupName,bool generatePreview) const248 PassRefPtr<TypeBuilder::Runtime::RemoteObject> InjectedScript::wrapObject(const ScriptValue& value, const String& groupName, bool generatePreview) const
249 {
250     ASSERT(!hasNoValue());
251     ScriptFunctionCall wrapFunction(injectedScriptObject(), "wrapObject");
252     wrapFunction.appendArgument(value);
253     wrapFunction.appendArgument(groupName);
254     wrapFunction.appendArgument(canAccessInspectedWindow());
255     wrapFunction.appendArgument(generatePreview);
256     bool hadException = false;
257     ScriptValue r = callFunctionWithEvalEnabled(wrapFunction, hadException);
258     if (hadException)
259         return 0;
260     RefPtr<JSONObject> rawResult = r.toJSONValue(scriptState())->asObject();
261     return TypeBuilder::Runtime::RemoteObject::runtimeCast(rawResult);
262 }
263 
wrapTable(const ScriptValue & table,const ScriptValue & columns) const264 PassRefPtr<TypeBuilder::Runtime::RemoteObject> InjectedScript::wrapTable(const ScriptValue& table, const ScriptValue& columns) const
265 {
266     ASSERT(!hasNoValue());
267     ScriptFunctionCall wrapFunction(injectedScriptObject(), "wrapTable");
268     wrapFunction.appendArgument(canAccessInspectedWindow());
269     wrapFunction.appendArgument(table);
270     if (columns.hasNoValue())
271         wrapFunction.appendArgument(false);
272     else
273         wrapFunction.appendArgument(columns);
274     bool hadException = false;
275     ScriptValue r = callFunctionWithEvalEnabled(wrapFunction, hadException);
276     if (hadException)
277         return 0;
278     RefPtr<JSONObject> rawResult = r.toJSONValue(scriptState())->asObject();
279     return TypeBuilder::Runtime::RemoteObject::runtimeCast(rawResult);
280 }
281 
wrapNode(Node * node,const String & groupName)282 PassRefPtr<TypeBuilder::Runtime::RemoteObject> InjectedScript::wrapNode(Node* node, const String& groupName)
283 {
284     return wrapObject(nodeAsScriptValue(node), groupName);
285 }
286 
findObjectById(const String & objectId) const287 ScriptValue InjectedScript::findObjectById(const String& objectId) const
288 {
289     ASSERT(!hasNoValue());
290     ScriptFunctionCall function(injectedScriptObject(), "findObjectById");
291     function.appendArgument(objectId);
292 
293     bool hadException = false;
294     ScriptValue resultValue = callFunctionWithEvalEnabled(function, hadException);
295     ASSERT(!hadException);
296     return resultValue;
297 }
298 
findCallFrameById(ErrorString * errorString,const ScriptValue & topCallFrame,const String & callFrameId)299 ScriptValue InjectedScript::findCallFrameById(ErrorString* errorString, const ScriptValue& topCallFrame, const String& callFrameId)
300 {
301     ScriptFunctionCall function(injectedScriptObject(), "callFrameForId");
302     function.appendArgument(topCallFrame);
303     function.appendArgument(callFrameId);
304     bool hadException = false;
305     ScriptValue resultValue = callFunctionWithEvalEnabled(function, hadException);
306     ASSERT(!hadException);
307     if (hadException || resultValue.hasNoValue() || !resultValue.isObject()) {
308         *errorString = "Internal error";
309         return ScriptValue();
310     }
311     return resultValue;
312 }
313 
inspectNode(Node * node)314 void InjectedScript::inspectNode(Node* node)
315 {
316     ASSERT(!hasNoValue());
317     ScriptFunctionCall function(injectedScriptObject(), "inspectNode");
318     function.appendArgument(nodeAsScriptValue(node));
319     RefPtr<JSONValue> result;
320     makeCall(function, &result);
321 }
322 
releaseObjectGroup(const String & objectGroup)323 void InjectedScript::releaseObjectGroup(const String& objectGroup)
324 {
325     ASSERT(!hasNoValue());
326     ScriptFunctionCall releaseFunction(injectedScriptObject(), "releaseObjectGroup");
327     releaseFunction.appendArgument(objectGroup);
328     bool hadException = false;
329     callFunctionWithEvalEnabled(releaseFunction, hadException);
330     ASSERT(!hadException);
331 }
332 
nodeAsScriptValue(Node * node)333 ScriptValue InjectedScript::nodeAsScriptValue(Node* node)
334 {
335     return InjectedScriptHost::nodeAsScriptValue(scriptState(), node);
336 }
337 
338 } // namespace WebCore
339 
340