1 # Copyright 2013 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file.
5 import command_executor
6 from command_executor
import Command
7 from webelement
import WebElement
10 class ChromeDriverException(Exception):
12 class NoSuchElement(ChromeDriverException
):
14 class NoSuchFrame(ChromeDriverException
):
16 class UnknownCommand(ChromeDriverException
):
18 class StaleElementReference(ChromeDriverException
):
20 class UnknownError(ChromeDriverException
):
22 class JavaScriptError(ChromeDriverException
):
24 class XPathLookupError(ChromeDriverException
):
26 class NoSuchWindow(ChromeDriverException
):
28 class InvalidCookieDomain(ChromeDriverException
):
30 class ScriptTimeout(ChromeDriverException
):
32 class InvalidSelector(ChromeDriverException
):
34 class SessionNotCreatedException(ChromeDriverException
):
36 class NoSuchSession(ChromeDriverException
):
39 def _ExceptionForResponse(response
):
40 exception_class_map
= {
45 10: StaleElementReference
,
50 24: InvalidCookieDomain
,
53 33: SessionNotCreatedException
55 status
= response
['status']
56 msg
= response
['value']['message']
57 return exception_class_map
.get(status
, ChromeDriverException
)(msg
)
60 class ChromeDriver(object):
61 """Starts and controls a single Chrome instance on this machine."""
63 def __init__(self
, server_url
, chrome_binary
=None, android_package
=None,
64 android_activity
=None, android_process
=None,
65 android_use_running_app
=None, chrome_switches
=None,
66 chrome_extensions
=None, chrome_log_path
=None,
67 debugger_address
=None, browser_log_level
=None,
68 performance_log_level
=None, mobile_emulation
=None,
69 experimental_options
=None):
70 self
._executor
= command_executor
.CommandExecutor(server_url
)
74 if experimental_options
:
75 assert isinstance(experimental_options
, dict)
76 options
= experimental_options
.copy()
79 options
['androidPackage'] = android_package
81 options
['androidActivity'] = android_activity
83 options
['androidProcess'] = android_process
84 if android_use_running_app
:
85 options
['androidUseRunningApp'] = android_use_running_app
87 options
['binary'] = chrome_binary
90 assert type(chrome_switches
) is list
91 options
['args'] = chrome_switches
94 assert type(mobile_emulation
) is dict
95 options
['mobileEmulation'] = mobile_emulation
98 assert type(chrome_extensions
) is list
99 options
['extensions'] = chrome_extensions
102 assert type(chrome_log_path
) is str
103 options
['logPath'] = chrome_log_path
106 assert type(debugger_address
) is str
107 options
['debuggerAddress'] = debugger_address
110 log_levels
= ['ALL', 'DEBUG', 'INFO', 'WARNING', 'SEVERE', 'OFF']
111 if browser_log_level
:
112 assert browser_log_level
in log_levels
113 logging_prefs
['browser'] = browser_log_level
114 if performance_log_level
:
115 assert performance_log_level
in log_levels
116 logging_prefs
['performance'] = performance_log_level
119 'desiredCapabilities': {
120 'chromeOptions': options
,
121 'loggingPrefs': logging_prefs
125 response
= self
._ExecuteCommand
(Command
.NEW_SESSION
, params
)
126 self
._session
_id
= response
['sessionId']
127 self
.capabilities
= self
._UnwrapValue
(response
['value'])
129 def _WrapValue(self
, value
):
130 """Wrap value from client side for chromedriver side."""
131 if isinstance(value
, dict):
133 for key
, val
in value
.items():
134 converted
[key
] = self
._WrapValue
(val
)
136 elif isinstance(value
, WebElement
):
137 return {'ELEMENT': value
._id
}
138 elif isinstance(value
, list):
139 return list(self
._WrapValue
(item
) for item
in value
)
143 def _UnwrapValue(self
, value
):
144 """Unwrap value from chromedriver side for client side."""
145 if isinstance(value
, dict):
146 if (len(value
) == 1 and 'ELEMENT' in value
147 and isinstance(value
['ELEMENT'], basestring
)):
148 return WebElement(self
, value
['ELEMENT'])
151 for key
, val
in value
.items():
152 unwraped
[key
] = self
._UnwrapValue
(val
)
154 elif isinstance(value
, list):
155 return list(self
._UnwrapValue
(item
) for item
in value
)
159 def _ExecuteCommand(self
, command
, params
={}):
160 params
= self
._WrapValue
(params
)
161 response
= self
._executor
.Execute(command
, params
)
162 if response
['status'] != 0:
163 raise _ExceptionForResponse(response
)
166 def ExecuteCommand(self
, command
, params
={}):
167 params
['sessionId'] = self
._session
_id
168 response
= self
._ExecuteCommand
(command
, params
)
169 return self
._UnwrapValue
(response
['value'])
171 def GetWindowHandles(self
):
172 return self
.ExecuteCommand(Command
.GET_WINDOW_HANDLES
)
174 def SwitchToWindow(self
, handle_or_name
):
175 self
.ExecuteCommand(Command
.SWITCH_TO_WINDOW
, {'name': handle_or_name
})
177 def GetCurrentWindowHandle(self
):
178 return self
.ExecuteCommand(Command
.GET_CURRENT_WINDOW_HANDLE
)
180 def CloseWindow(self
):
181 self
.ExecuteCommand(Command
.CLOSE
)
184 self
.ExecuteCommand(Command
.GET
, {'url': url
})
186 def LaunchApp(self
, app_id
):
187 self
.ExecuteCommand(Command
.LAUNCH_APP
, {'id': app_id
})
189 def ExecuteScript(self
, script
, *args
):
190 converted_args
= list(args
)
191 return self
.ExecuteCommand(
192 Command
.EXECUTE_SCRIPT
, {'script': script
, 'args': converted_args
})
194 def ExecuteAsyncScript(self
, script
, *args
):
195 converted_args
= list(args
)
196 return self
.ExecuteCommand(
197 Command
.EXECUTE_ASYNC_SCRIPT
,
198 {'script': script
, 'args': converted_args
})
200 def SwitchToFrame(self
, id_or_name
):
201 self
.ExecuteCommand(Command
.SWITCH_TO_FRAME
, {'id': id_or_name
})
203 def SwitchToFrameByIndex(self
, index
):
204 self
.SwitchToFrame(index
)
206 def SwitchToMainFrame(self
):
207 self
.SwitchToFrame(None)
209 def SwitchToParentFrame(self
):
210 self
.ExecuteCommand(Command
.SWITCH_TO_PARENT_FRAME
)
213 return self
.ExecuteCommand(Command
.GET_TITLE
)
215 def GetPageSource(self
):
216 return self
.ExecuteCommand(Command
.GET_PAGE_SOURCE
)
218 def FindElement(self
, strategy
, target
):
219 return self
.ExecuteCommand(
220 Command
.FIND_ELEMENT
, {'using': strategy
, 'value': target
})
222 def FindElements(self
, strategy
, target
):
223 return self
.ExecuteCommand(
224 Command
.FIND_ELEMENTS
, {'using': strategy
, 'value': target
})
226 def SetTimeout(self
, type, timeout
):
227 return self
.ExecuteCommand(
228 Command
.SET_TIMEOUT
, {'type' : type, 'ms': timeout
})
230 def GetCurrentUrl(self
):
231 return self
.ExecuteCommand(Command
.GET_CURRENT_URL
)
234 return self
.ExecuteCommand(Command
.GO_BACK
)
237 return self
.ExecuteCommand(Command
.GO_FORWARD
)
240 return self
.ExecuteCommand(Command
.REFRESH
)
242 def MouseMoveTo(self
, element
=None, x_offset
=None, y_offset
=None):
244 if element
is not None:
245 params
['element'] = element
._id
246 if x_offset
is not None:
247 params
['xoffset'] = x_offset
248 if y_offset
is not None:
249 params
['yoffset'] = y_offset
250 self
.ExecuteCommand(Command
.MOUSE_MOVE_TO
, params
)
252 def MouseClick(self
, button
=0):
253 self
.ExecuteCommand(Command
.MOUSE_CLICK
, {'button': button
})
255 def MouseButtonDown(self
, button
=0):
256 self
.ExecuteCommand(Command
.MOUSE_BUTTON_DOWN
, {'button': button
})
258 def MouseButtonUp(self
, button
=0):
259 self
.ExecuteCommand(Command
.MOUSE_BUTTON_UP
, {'button': button
})
261 def MouseDoubleClick(self
, button
=0):
262 self
.ExecuteCommand(Command
.MOUSE_DOUBLE_CLICK
, {'button': button
})
264 def TouchDown(self
, x
, y
):
265 self
.ExecuteCommand(Command
.TOUCH_DOWN
, {'x': x
, 'y': y
})
267 def TouchUp(self
, x
, y
):
268 self
.ExecuteCommand(Command
.TOUCH_UP
, {'x': x
, 'y': y
})
270 def TouchMove(self
, x
, y
):
271 self
.ExecuteCommand(Command
.TOUCH_MOVE
, {'x': x
, 'y': y
})
273 def TouchFlick(self
, element
, xoffset
, yoffset
, speed
):
275 'element': element
._id
,
280 self
.ExecuteCommand(Command
.TOUCH_FLICK
, params
)
282 def GetCookies(self
):
283 return self
.ExecuteCommand(Command
.GET_COOKIES
)
285 def AddCookie(self
, cookie
):
286 self
.ExecuteCommand(Command
.ADD_COOKIE
, {'cookie': cookie
})
288 def DeleteCookie(self
, name
):
289 self
.ExecuteCommand(Command
.DELETE_COOKIE
, {'name': name
})
291 def DeleteAllCookies(self
):
292 self
.ExecuteCommand(Command
.DELETE_ALL_COOKIES
)
294 def IsAlertOpen(self
):
295 return self
.ExecuteCommand(Command
.GET_ALERT
)
297 def GetAlertMessage(self
):
298 return self
.ExecuteCommand(Command
.GET_ALERT_TEXT
)
300 def HandleAlert(self
, accept
, prompt_text
=''):
302 self
.ExecuteCommand(Command
.SET_ALERT_VALUE
, {'text': prompt_text
})
304 cmd
= Command
.ACCEPT_ALERT
306 cmd
= Command
.DISMISS_ALERT
307 self
.ExecuteCommand(cmd
)
310 return self
.ExecuteCommand(Command
.IS_LOADING
)
312 def GetWindowPosition(self
):
313 position
= self
.ExecuteCommand(Command
.GET_WINDOW_POSITION
,
314 {'windowHandle': 'current'})
315 return [position
['x'], position
['y']]
317 def SetWindowPosition(self
, x
, y
):
318 self
.ExecuteCommand(Command
.SET_WINDOW_POSITION
,
319 {'windowHandle': 'current', 'x': x
, 'y': y
})
321 def GetWindowSize(self
):
322 size
= self
.ExecuteCommand(Command
.GET_WINDOW_SIZE
,
323 {'windowHandle': 'current'})
324 return [size
['width'], size
['height']]
326 def SetWindowSize(self
, width
, height
):
328 Command
.SET_WINDOW_SIZE
,
329 {'windowHandle': 'current', 'width': width
, 'height': height
})
331 def MaximizeWindow(self
):
332 self
.ExecuteCommand(Command
.MAXIMIZE_WINDOW
, {'windowHandle': 'current'})
335 """Quits the browser and ends the session."""
336 self
.ExecuteCommand(Command
.QUIT
)
338 def GetLog(self
, type):
339 return self
.ExecuteCommand(Command
.GET_LOG
, {'type': type})
341 def GetAvailableLogTypes(self
):
342 return self
.ExecuteCommand(Command
.GET_AVAILABLE_LOG_TYPES
)
344 def IsAutoReporting(self
):
345 return self
.ExecuteCommand(Command
.IS_AUTO_REPORTING
)
347 def SetAutoReporting(self
, enabled
):
348 self
.ExecuteCommand(Command
.SET_AUTO_REPORTING
, {'enabled': enabled
})