• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
3  * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
4  * Copyright (C) 2012 Google Inc. All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  *
10  * 1.  Redistributions of source code must retain the above copyright
11  *     notice, this list of conditions and the following disclaimer.
12  * 2.  Redistributions in binary form must reproduce the above copyright
13  *     notice, this list of conditions and the following disclaimer in the
14  *     documentation and/or other materials provided with the distribution.
15  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
16  *     its contributors may be used to endorse or promote products derived
17  *     from this software without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
20  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
23  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
26  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30 
31 #include "config.h"
32 #include "core/inspector/InjectedScriptManager.h"
33 
34 #include "InjectedScriptSource.h"
35 #include "bindings/v8/ScriptObject.h"
36 #include "core/inspector/InjectedScript.h"
37 #include "core/inspector/InjectedScriptHost.h"
38 #include "core/inspector/JSONParser.h"
39 #include "platform/JSONValues.h"
40 #include "wtf/PassOwnPtr.h"
41 
42 namespace WebCore {
43 
createForPage()44 PassOwnPtr<InjectedScriptManager> InjectedScriptManager::createForPage()
45 {
46     return adoptPtr(new InjectedScriptManager(&InjectedScriptManager::canAccessInspectedWindow));
47 }
48 
createForWorker()49 PassOwnPtr<InjectedScriptManager> InjectedScriptManager::createForWorker()
50 {
51     return adoptPtr(new InjectedScriptManager(&InjectedScriptManager::canAccessInspectedWorkerGlobalScope));
52 }
53 
InjectedScriptManager(InspectedStateAccessCheck accessCheck)54 InjectedScriptManager::InjectedScriptManager(InspectedStateAccessCheck accessCheck)
55     : m_nextInjectedScriptId(1)
56     , m_injectedScriptHost(InjectedScriptHost::create())
57     , m_inspectedStateAccessCheck(accessCheck)
58 {
59 }
60 
~InjectedScriptManager()61 InjectedScriptManager::~InjectedScriptManager()
62 {
63 }
64 
disconnect()65 void InjectedScriptManager::disconnect()
66 {
67     m_injectedScriptHost->disconnect();
68     m_injectedScriptHost.clear();
69 }
70 
injectedScriptHost()71 InjectedScriptHost* InjectedScriptManager::injectedScriptHost()
72 {
73     return m_injectedScriptHost.get();
74 }
75 
injectedScriptForId(int id)76 InjectedScript InjectedScriptManager::injectedScriptForId(int id)
77 {
78     IdToInjectedScriptMap::iterator it = m_idToInjectedScript.find(id);
79     if (it != m_idToInjectedScript.end())
80         return it->value;
81     for (ScriptStateToId::iterator it = m_scriptStateToId.begin(); it != m_scriptStateToId.end(); ++it) {
82         if (it->value == id)
83             return injectedScriptFor(it->key);
84     }
85     return InjectedScript();
86 }
87 
injectedScriptIdFor(ScriptState * scriptState)88 int InjectedScriptManager::injectedScriptIdFor(ScriptState* scriptState)
89 {
90     ScriptStateToId::iterator it = m_scriptStateToId.find(scriptState);
91     if (it != m_scriptStateToId.end())
92         return it->value;
93     int id = m_nextInjectedScriptId++;
94     m_scriptStateToId.set(scriptState, id);
95     return id;
96 }
97 
injectedScriptForObjectId(const String & objectId)98 InjectedScript InjectedScriptManager::injectedScriptForObjectId(const String& objectId)
99 {
100     RefPtr<JSONValue> parsedObjectId = parseJSON(objectId);
101     if (parsedObjectId && parsedObjectId->type() == JSONValue::TypeObject) {
102         long injectedScriptId = 0;
103         bool success = parsedObjectId->asObject()->getNumber("injectedScriptId", &injectedScriptId);
104         if (success)
105             return m_idToInjectedScript.get(injectedScriptId);
106     }
107     return InjectedScript();
108 }
109 
discardInjectedScripts()110 void InjectedScriptManager::discardInjectedScripts()
111 {
112     m_idToInjectedScript.clear();
113     m_scriptStateToId.clear();
114 }
115 
discardInjectedScriptsFor(DOMWindow * window)116 void InjectedScriptManager::discardInjectedScriptsFor(DOMWindow* window)
117 {
118     if (m_scriptStateToId.isEmpty())
119         return;
120 
121     Vector<long> idsToRemove;
122     IdToInjectedScriptMap::iterator end = m_idToInjectedScript.end();
123     for (IdToInjectedScriptMap::iterator it = m_idToInjectedScript.begin(); it != end; ++it) {
124         ScriptState* scriptState = it->value.scriptState();
125         if (window != scriptState->domWindow())
126             continue;
127         m_scriptStateToId.remove(scriptState);
128         idsToRemove.append(it->key);
129     }
130 
131     for (size_t i = 0; i < idsToRemove.size(); i++)
132         m_idToInjectedScript.remove(idsToRemove[i]);
133 
134     // Now remove script states that have id but no injected script.
135     Vector<ScriptState*> scriptStatesToRemove;
136     for (ScriptStateToId::iterator it = m_scriptStateToId.begin(); it != m_scriptStateToId.end(); ++it) {
137         ScriptState* scriptState = it->key;
138         if (window == scriptState->domWindow())
139             scriptStatesToRemove.append(scriptState);
140     }
141     for (size_t i = 0; i < scriptStatesToRemove.size(); i++)
142         m_scriptStateToId.remove(scriptStatesToRemove[i]);
143 }
144 
canAccessInspectedWorkerGlobalScope(ScriptState *)145 bool InjectedScriptManager::canAccessInspectedWorkerGlobalScope(ScriptState*)
146 {
147     return true;
148 }
149 
releaseObjectGroup(const String & objectGroup)150 void InjectedScriptManager::releaseObjectGroup(const String& objectGroup)
151 {
152     Vector<int> keys;
153     keys.appendRange(m_idToInjectedScript.keys().begin(), m_idToInjectedScript.keys().end());
154     for (Vector<int>::iterator k = keys.begin(); k != keys.end(); ++k) {
155         IdToInjectedScriptMap::iterator s = m_idToInjectedScript.find(*k);
156         if (s != m_idToInjectedScript.end())
157             s->value.releaseObjectGroup(objectGroup); // m_idToInjectedScript may change here.
158     }
159 }
160 
injectedScriptSource()161 String InjectedScriptManager::injectedScriptSource()
162 {
163     return String(reinterpret_cast<const char*>(InjectedScriptSource_js), sizeof(InjectedScriptSource_js));
164 }
165 
injectedScriptFor(ScriptState * inspectedScriptState)166 InjectedScript InjectedScriptManager::injectedScriptFor(ScriptState* inspectedScriptState)
167 {
168     ScriptStateToId::iterator it = m_scriptStateToId.find(inspectedScriptState);
169     if (it != m_scriptStateToId.end()) {
170         IdToInjectedScriptMap::iterator it1 = m_idToInjectedScript.find(it->value);
171         if (it1 != m_idToInjectedScript.end())
172             return it1->value;
173     }
174 
175     if (!m_inspectedStateAccessCheck(inspectedScriptState))
176         return InjectedScript();
177 
178     int id = injectedScriptIdFor(inspectedScriptState);
179     ScriptObject injectedScriptObject = createInjectedScript(injectedScriptSource(), inspectedScriptState, id);
180     InjectedScript result(injectedScriptObject, m_inspectedStateAccessCheck);
181     m_idToInjectedScript.set(id, result);
182     return result;
183 }
184 
185 } // namespace WebCore
186 
187