增加wait.alert_closed()、wait.has_rect();run_js()无视其中产生的弹窗

This commit is contained in:
g1879 2023-12-22 17:49:49 +08:00
parent 98b992ac90
commit da5a8a9e42
7 changed files with 63 additions and 27 deletions

View File

@ -5,6 +5,7 @@
"""
from time import sleep, perf_counter
from errors import PageClosedError
from .driver import BrowserDriver, Driver
from .._functions.tools import stop_process_on_port, raise_error
from .._units.downloader import DownloadManager
@ -65,9 +66,10 @@ class Browser(object):
def _onTargetCreated(self, **kwargs):
"""标签页创建时执行"""
if kwargs['targetInfo']['type'] == 'page' and not kwargs['targetInfo']['url'].startswith('devtools://'):
self._drivers[kwargs['targetInfo']['targetId']] = Driver(kwargs['targetInfo']['targetId'], 'page',
self.address)
if (kwargs['targetInfo']['type'] in ('page', 'webview')
and not kwargs['targetInfo']['url'].startswith('devtools://')):
self._drivers[kwargs['targetInfo']['targetId']] = Driver(kwargs['targetInfo']['targetId'],
'page', self.address)
def _onTargetDestroyed(self, **kwargs):
"""标签页关闭时执行"""
@ -101,13 +103,13 @@ class Browser(object):
def tabs_count(self):
"""返回标签页数量"""
j = self.run_cdp('Target.getTargets')['targetInfos'] # 不要改用get避免卡死
return len([i for i in j if i['type'] == 'page' and not i['url'].startswith('devtools://')])
return len([i for i in j if i['type'] in ('page', 'webview') and not i['url'].startswith('devtools://')])
@property
def tabs(self):
"""返回所有标签页id组成的列表"""
j = self._driver.get(f'http://{self.address}/json').json() # 不要改用cdp因为顺序不对
return [i['id'] for i in j if i['type'] == 'page' and not i['url'].startswith('devtools://')]
return [i['id'] for i in j if i['type'] in ('page', 'webview') and not i['url'].startswith('devtools://')]
@property
def process_id(self):
@ -140,7 +142,7 @@ class Browser(object):
:param tab_id: 标签页id
:return: None
"""
self.run_cdp('Target.closeTarget', targetId=tab_id)
self.run_cdp('Target.closeTarget', targetId=tab_id, _ignore=PageClosedError)
def activate_tab(self, tab_id):
"""使标签页变为活动状态
@ -162,8 +164,12 @@ class Browser(object):
:param force: 是否立刻强制终止进程
:return: None
"""
self.run_cdp('Browser.close')
self.driver.stop()
try:
self.run_cdp('Browser.close')
self.driver.stop()
except PageClosedError:
self.driver.stop()
return
if force:
ip, port = self.address.split(':')

View File

@ -83,8 +83,8 @@ class Driver(object):
return result
except Empty:
if self.alert_flag and message['method'].startswith('Input.'):
return {'result': {'message': 'alert exists.'}}
if self.alert_flag and message['method'].startswith(('Input.', 'Runtime.')):
return {'error': {'message': 'alert exists.'}}
if timeout is not None and perf_counter() > end_time:
self.method_results.pop(ws_id, None)
@ -156,7 +156,7 @@ class Driver(object):
if self._stopped.is_set():
return {'error': 'tab closed', 'type': 'tab_closed'}
timeout = kwargs.pop('_timeout', 10)
timeout = kwargs.pop('_timeout', 15)
result = self._send({'method': _method, 'params': kwargs}, timeout=timeout)
if result is None:
return {'error': 'tab closed', 'type': 'tab_closed'}

View File

@ -603,6 +603,7 @@ class ChromiumElement(DrissionElement):
if clear and vals not in ('\n', '\ue007'):
self.clear(by_js=False)
else:
self.wait.has_rect()
self._input_focus()
input_text_or_keys(self.page, vals)
@ -614,10 +615,10 @@ class ChromiumElement(DrissionElement):
"""
if by_js:
self.run_js("this.value='';")
return
else:
self._input_focus()
self.input(('\ue009', 'a', '\ue017'), clear=False)
self._input_focus()
self.input(('\ue009', 'a', '\ue017'), clear=False)
def _input_focus(self):
"""输入前使元素获取焦点"""
@ -1248,8 +1249,10 @@ def make_chromium_eles(page, node_ids=None, obj_ids=None, single=True, ele_only=
if ele_only:
continue
else:
# todo: Node()
pass
if single:
return node['node']['nodeValue']
else:
nodes.append(node['node']['nodeValue'])
obj_id = page.run_cdp('DOM.resolveNode', nodeId=node_id)['object']['objectId']
ele = ChromiumElement(page, obj_id=obj_id, node_id=node_id, backend_id=node['node']['backendNodeId'])
@ -1269,8 +1272,10 @@ def make_chromium_eles(page, node_ids=None, obj_ids=None, single=True, ele_only=
if ele_only:
continue
else:
# todo: Node
pass
if single:
return node['node']['nodeValue']
else:
nodes.append(node['node']['nodeValue'])
ele = ChromiumElement(page, obj_id=obj_id, node_id=node['node']['nodeId'],
backend_id=node['node']['backendNodeId'])
@ -1356,7 +1361,7 @@ def run_js(page_or_ele, script, as_expr=False, timeout=None, args=None):
try:
if as_expr:
res = page.run_cdp('Runtime.evaluate', expression=script, returnByValue=False,
awaitPromise=True, userGesture=True, _timeout=timeout)
awaitPromise=True, userGesture=True, _timeout=timeout, _ignore=AlertExistsError)
else:
args = args or ()
@ -1364,7 +1369,7 @@ def run_js(page_or_ele, script, as_expr=False, timeout=None, args=None):
script = f'function(){{{script}}}'
res = page.run_cdp('Runtime.callFunctionOn', functionDeclaration=script, objectId=obj_id,
arguments=[convert_argument(arg) for arg in args], returnByValue=False,
awaitPromise=True, userGesture=True, _timeout=timeout)
awaitPromise=True, userGesture=True, _timeout=timeout, _ignore=AlertExistsError)
except ContextLostError:
if is_page:

View File

@ -203,7 +203,7 @@ def test_connect(ip, port, timeout=30):
tabs = requests_get(f'http://{ip}:{port}/json', timeout=10, headers={'Connection': 'close'},
proxies={'http': None, 'https': None}).json()
for tab in tabs:
if tab['type'] == 'page':
if tab['type'] in ('page', 'webview'):
return
except Exception:
sleep(.2)

View File

@ -89,7 +89,8 @@ class ChromiumBase(BasePage):
if not tab_id:
tabs = self.browser.driver.get(f'http://{self.address}/json').json()
tabs = [(i['id'], i['url']) for i in tabs if i['type'] == 'page' and not i['url'].startswith('devtools://')]
tabs = [(i['id'], i['url']) for i in tabs
if i['type'] in ('page', 'webview') and not i['url'].startswith('devtools://')]
dialog = None
if len(tabs) > 1:
for k, t in enumerate(tabs):
@ -588,7 +589,8 @@ class ChromiumBase(BasePage):
search_ids = []
try:
search_result = self.run_cdp_loaded('DOM.performSearch', query=loc, includeUserAgentShadowDOM=True)
search_result = self.run_cdp_loaded('DOM.performSearch', query=loc, _timeout=timeout,
includeUserAgentShadowDOM=True)
count = search_result['resultCount']
search_ids.append(search_result['searchId'])
except ContextLostError:
@ -615,7 +617,11 @@ class ChromiumBase(BasePage):
ok = False
try:
search_result = self.run_cdp_loaded('DOM.performSearch', query=loc, includeUserAgentShadowDOM=True)
timeout = end_time - perf_counter()
if timeout <= 0:
timeout = .5
search_result = self.run_cdp_loaded('DOM.performSearch', query=loc, _timeout=timeout,
includeUserAgentShadowDOM=True)
count = search_result['resultCount']
search_ids.append(search_result['searchId'])
except ContextLostError:

View File

@ -238,6 +238,13 @@ class TabWaiter(BaseWaiter):
else:
return True
def alert_closed(self):
"""等待弹出框关闭"""
while not self._driver.states.has_alert:
sleep(.2)
while self._driver.states.has_alert:
sleep(.2)
class PageWaiter(TabWaiter):
def __init__(self, page):
@ -415,6 +422,14 @@ class ElementWaiter(object):
else:
return False
def has_rect(self, timeout=None, raise_err=None):
"""等待当前元素有大小及位置属性
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
"""
return self._wait_state('has_rect', True, timeout, raise_err, err_text='等待元素拥有大小及位置属性失败。')
def _wait_state(self, attr, mode=False, timeout=None, raise_err=None, err_text=None):
"""等待元素某个元素状态到达指定状态
:param attr: 状态名称

View File

@ -24,9 +24,9 @@ class BaseWaiter(object):
raise_err: bool = None) -> bool: ...
def ele_displayed(self,
loc_or_ele: Union[str, tuple, ChromiumElement],
timeout: float = None,
raise_err: bool = None) -> bool: ...
loc_or_ele: Union[str, tuple, ChromiumElement],
timeout: float = None,
raise_err: bool = None) -> bool: ...
def ele_hidden(self, loc_or_ele: Union[str, tuple, ChromiumElement], timeout: float = None,
raise_err: bool = None) -> bool: ...
@ -60,6 +60,8 @@ class TabWaiter(BaseWaiter):
def downloads_done(self, timeout: float = None, cancel_if_timeout: bool = True) -> bool: ...
def alert_closed(self) -> None: ...
class PageWaiter(TabWaiter):
_driver: ChromiumPage = ...
@ -90,6 +92,8 @@ class ElementWaiter(object):
def disabled(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def has_rect(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def disabled_or_deleted(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def stop_moving(self, gap: float = .1, timeout: float = None, raise_err: bool = None) -> bool: ...