mirror of
https://gitee.com/g1879/DrissionPage.git
synced 2024-12-10 04:00:23 +08:00
Tab默认单例;增加TargetNotFoundError、Settings.singleton_tab_obj
This commit is contained in:
parent
1eaa56efdb
commit
b19e62bac5
@ -63,12 +63,13 @@ class Browser(object):
|
||||
self._driver.set_callback('Target.targetDestroyed', self._onTargetDestroyed)
|
||||
self._driver.set_callback('Target.targetCreated', self._onTargetCreated)
|
||||
|
||||
def _get_driver(self, tab_id):
|
||||
def _get_driver(self, tab_id, owner=None):
|
||||
"""获取对应tab id的Driver
|
||||
:param tab_id: 标签页id
|
||||
:param owner: 使用该驱动的对象
|
||||
:return: Driver对象
|
||||
"""
|
||||
return self._drivers.pop(tab_id, Driver(tab_id, 'page', self.address))
|
||||
return self._drivers.pop(tab_id, Driver(tab_id, 'page', self.address, owner))
|
||||
|
||||
def _onTargetCreated(self, **kwargs):
|
||||
"""标签页创建时执行"""
|
||||
@ -201,8 +202,8 @@ class Browser(object):
|
||||
except TypeError:
|
||||
pass
|
||||
|
||||
def _on_quit(self):
|
||||
self.page._on_quit()
|
||||
def _on_disconnect(self):
|
||||
self.page._on_disconnect()
|
||||
Browser.BROWSERS.pop(self.id, None)
|
||||
if self.page._chromium_options.is_auto_port and self.page._chromium_options.user_data_path:
|
||||
path = Path(self.page._chromium_options.user_data_path)
|
||||
|
@ -28,7 +28,7 @@ class Browser(object):
|
||||
|
||||
def __init__(self, address: str, browser_id: str, page: ChromiumPage): ...
|
||||
|
||||
def _get_driver(self, tab_id: str) -> Driver: ...
|
||||
def _get_driver(self, tab_id: str, owner=None) -> Driver: ...
|
||||
|
||||
def run_cdp(self, cmd, **cmd_args) -> dict: ...
|
||||
|
||||
@ -61,4 +61,4 @@ class Browser(object):
|
||||
|
||||
def quit(self, timeout: float = 5, force: bool = False) -> None: ...
|
||||
|
||||
def _on_quit(self) -> None: ...
|
||||
def _on_disconnect(self) -> None: ...
|
||||
|
@ -12,21 +12,23 @@ from time import perf_counter, sleep
|
||||
|
||||
from requests import get
|
||||
from websocket import (WebSocketTimeoutException, WebSocketConnectionClosedException, create_connection,
|
||||
WebSocketException)
|
||||
WebSocketException, WebSocketBadStatusException)
|
||||
|
||||
from ..errors import PageDisconnectedError
|
||||
from ..errors import PageDisconnectedError, TargetNotFoundError
|
||||
|
||||
|
||||
class Driver(object):
|
||||
def __init__(self, tab_id, tab_type, address):
|
||||
def __init__(self, tab_id, tab_type, address, owner=None):
|
||||
"""
|
||||
:param tab_id: 标签页id
|
||||
:param tab_type: 标签页类型
|
||||
:param address: 浏览器连接地址
|
||||
:param owner: 创建这个驱动的对象
|
||||
"""
|
||||
self.id = tab_id
|
||||
self.address = address
|
||||
self.type = tab_type
|
||||
self.owner = owner
|
||||
self._debug = False
|
||||
self.alert_flag = False # 标记alert出现,跳过一条请求后复原
|
||||
|
||||
@ -195,7 +197,10 @@ class Driver(object):
|
||||
def start(self):
|
||||
"""启动连接"""
|
||||
self._stopped.clear()
|
||||
self._ws = create_connection(self._websocket_url, enable_multithread=True, suppress_origin=True)
|
||||
try:
|
||||
self._ws = create_connection(self._websocket_url, enable_multithread=True, suppress_origin=True)
|
||||
except WebSocketBadStatusException as e:
|
||||
raise TargetNotFoundError(f'找不到页面:{self.id}。') if 'No such target id' in str(e) else e
|
||||
self._recv_th.start()
|
||||
self._handle_event_th.start()
|
||||
return True
|
||||
@ -230,6 +235,9 @@ class Driver(object):
|
||||
self.method_results.clear()
|
||||
self.event_queue.queue.clear()
|
||||
|
||||
if hasattr(self.owner, '_on_disconnect'):
|
||||
self.owner._on_disconnect()
|
||||
|
||||
def set_callback(self, event, callback, immediate=False):
|
||||
"""绑定cdp event和回调方法
|
||||
:param event: cdp event
|
||||
@ -247,18 +255,17 @@ class Driver(object):
|
||||
class BrowserDriver(Driver):
|
||||
BROWSERS = {}
|
||||
|
||||
def __new__(cls, tab_id, tab_type, address, browser):
|
||||
def __new__(cls, tab_id, tab_type, address, owner):
|
||||
if tab_id in cls.BROWSERS:
|
||||
return cls.BROWSERS[tab_id]
|
||||
return object.__new__(cls)
|
||||
|
||||
def __init__(self, tab_id, tab_type, address, browser):
|
||||
def __init__(self, tab_id, tab_type, address, owner):
|
||||
if hasattr(self, '_created'):
|
||||
return
|
||||
self._created = True
|
||||
BrowserDriver.BROWSERS[tab_id] = self
|
||||
super().__init__(tab_id, tab_type, address)
|
||||
self.browser = browser
|
||||
super().__init__(tab_id, tab_type, address, owner)
|
||||
|
||||
def __repr__(self):
|
||||
return f'<BrowserDriver {self.id}>'
|
||||
@ -267,7 +274,3 @@ class BrowserDriver(Driver):
|
||||
r = get(url, headers={'Connection': 'close'})
|
||||
r.close()
|
||||
return r
|
||||
|
||||
def _stop(self):
|
||||
super()._stop()
|
||||
self.browser._on_quit()
|
||||
|
@ -27,7 +27,7 @@ class Driver(object):
|
||||
id: str
|
||||
address: str
|
||||
type: str
|
||||
# _debug: bool
|
||||
owner = ...
|
||||
alert_flag: bool
|
||||
_websocket_url: str
|
||||
_cur_id: int
|
||||
@ -42,7 +42,7 @@ class Driver(object):
|
||||
event_queue: Queue
|
||||
immediate_event_queue: Queue
|
||||
|
||||
def __init__(self, tab_id: str, tab_type: str, address: str): ...
|
||||
def __init__(self, tab_id: str, tab_type: str, address: str, owner=None): ...
|
||||
|
||||
def _send(self, message: dict, timeout: float = None) -> dict: ...
|
||||
|
||||
@ -67,10 +67,10 @@ class Driver(object):
|
||||
|
||||
class BrowserDriver(Driver):
|
||||
BROWSERS: Dict[str, Driver] = ...
|
||||
browser: Browser = ...
|
||||
owner: Browser = ...
|
||||
|
||||
def __new__(cls, tab_id: str, tab_type: str, address: str, browser: Browser): ...
|
||||
def __new__(cls, tab_id: str, tab_type: str, address: str, owner: Browser): ...
|
||||
|
||||
def __init__(self, tab_id: str, tab_type: str, address: str, browser: Browser): ...
|
||||
def __init__(self, tab_id: str, tab_type: str, address: str, owner: Browser): ...
|
||||
|
||||
def get(self, url) -> Response: ...
|
||||
|
@ -11,3 +11,4 @@ class Settings(object):
|
||||
raise_when_ele_not_found = False
|
||||
raise_when_click_failed = False
|
||||
raise_when_wait_failed = False
|
||||
singleton_tab_obj = True
|
||||
|
@ -124,7 +124,7 @@ class ChromiumBase(BasePage):
|
||||
:return: None
|
||||
"""
|
||||
self._is_loading = True
|
||||
self._driver = self.browser._get_driver(tab_id)
|
||||
self._driver = self.browser._get_driver(tab_id, self)
|
||||
|
||||
self._alert = Alert()
|
||||
self._driver.set_callback('Page.javascriptDialogOpening', self._on_alert_open, immediate=True)
|
||||
|
@ -245,7 +245,7 @@ class ChromiumPage(ChromiumBase):
|
||||
"""
|
||||
self.browser.quit(timeout, force)
|
||||
|
||||
def _on_quit(self):
|
||||
def _on_disconnect(self):
|
||||
"""浏览器退出时执行"""
|
||||
ChromiumPage.PAGES.pop(self._browser_id, None)
|
||||
|
||||
|
@ -20,6 +20,11 @@ from .._units.waiter import PageWaiter
|
||||
class ChromiumPage(ChromiumBase):
|
||||
PAGES: dict = ...
|
||||
|
||||
def __new__(cls,
|
||||
addr_or_opts: Union[str, int, ChromiumOptions] = None,
|
||||
tab_id: str = None,
|
||||
timeout: float = None): ...
|
||||
|
||||
def __init__(self,
|
||||
addr_or_opts: Union[str, int, ChromiumOptions] = None,
|
||||
tab_id: str = None,
|
||||
@ -28,7 +33,7 @@ class ChromiumPage(ChromiumBase):
|
||||
self._browser: Browser = ...
|
||||
self._browser_id: str = ...
|
||||
self._rect: Optional[TabRect] = ...
|
||||
self._is_exist:bool = ...
|
||||
self._is_exist: bool = ...
|
||||
|
||||
def _handle_options(self, addr_or_opts: Union[str, ChromiumOptions]) -> str: ...
|
||||
|
||||
@ -98,7 +103,7 @@ class ChromiumPage(ChromiumBase):
|
||||
|
||||
def quit(self, timeout: float = 5, force: bool = True) -> None: ...
|
||||
|
||||
def _on_quit(self) -> None: ...
|
||||
def _on_disconnect(self) -> None: ...
|
||||
|
||||
|
||||
def get_rename(original: str, rename: str) -> str: ...
|
||||
|
@ -9,6 +9,7 @@ from copy import copy
|
||||
|
||||
from .._base.base import BasePage
|
||||
from .._configs.session_options import SessionOptions
|
||||
from .._functions.settings import Settings
|
||||
from .._functions.web import set_session_cookies, set_browser_cookies
|
||||
from .._pages.chromium_base import ChromiumBase, get_mhtml, get_pdf
|
||||
from .._pages.session_page import SessionPage
|
||||
@ -18,12 +19,28 @@ from .._units.waiter import TabWaiter
|
||||
|
||||
class ChromiumTab(ChromiumBase):
|
||||
"""实现浏览器标签页的类"""
|
||||
TABS = {}
|
||||
|
||||
def __init__(self, page, tab_id=None):
|
||||
def __new__(cls, page, tab_id):
|
||||
"""
|
||||
:param page: ChromiumPage对象
|
||||
:param tab_id: 要控制的标签页id,不指定默认为激活的
|
||||
:param tab_id: 要控制的标签页id
|
||||
"""
|
||||
if Settings.singleton_tab_obj and tab_id in cls.TABS:
|
||||
return cls.TABS[tab_id]
|
||||
r = object.__new__(cls)
|
||||
cls.TABS[tab_id] = r
|
||||
return r
|
||||
|
||||
def __init__(self, page, tab_id):
|
||||
"""
|
||||
:param page: ChromiumPage对象
|
||||
:param tab_id: 要控制的标签页id
|
||||
"""
|
||||
if Settings.singleton_tab_obj and hasattr(self, '_created'):
|
||||
return
|
||||
self._created = True
|
||||
|
||||
self._page = page
|
||||
self._browser = page.browser
|
||||
super().__init__(page.address, tab_id, page.timeout)
|
||||
@ -73,6 +90,9 @@ class ChromiumTab(ChromiumBase):
|
||||
def __repr__(self):
|
||||
return f'<ChromiumTab browser_id={self.browser.id} tab_id={self.tab_id}>'
|
||||
|
||||
def _on_disconnect(self):
|
||||
ChromiumTab.TABS.pop(self.tab_id, None)
|
||||
|
||||
|
||||
class WebPageTab(SessionPage, ChromiumTab, BasePage):
|
||||
def __init__(self, page, tab_id):
|
||||
|
@ -25,8 +25,11 @@ from .._units.waiter import TabWaiter
|
||||
|
||||
|
||||
class ChromiumTab(ChromiumBase):
|
||||
TABS: dict = ...
|
||||
|
||||
def __init__(self, page: ChromiumPage, tab_id: str = None):
|
||||
def __new__(cls, page: ChromiumPage, tab_id: str): ...
|
||||
|
||||
def __init__(self, page: ChromiumPage, tab_id: str):
|
||||
self._page: ChromiumPage = ...
|
||||
self._browser: Browser = ...
|
||||
self._rect: Optional[TabRect] = ...
|
||||
|
@ -89,3 +89,7 @@ class StorageError(BaseError):
|
||||
|
||||
class CookieFormatError(BaseError):
|
||||
_info = 'cookie格式不正确。'
|
||||
|
||||
|
||||
class TargetNotFoundError(BaseError):
|
||||
_info = '找不到指定页面。'
|
||||
|
Loading…
x
Reference in New Issue
Block a user