mirror of
https://gitee.com/g1879/DrissionPage.git
synced 2024-12-10 04:00:23 +08:00
251 lines
8.9 KiB
Python
251 lines
8.9 KiB
Python
# -*- coding:utf-8 -*-
|
||
"""
|
||
@Author : g1879
|
||
@Contact : g1879@qq.com
|
||
@Copyright: (c) 2024 by g1879, Inc. All Rights Reserved.
|
||
@License : BSD 3-Clause.
|
||
"""
|
||
from pathlib import Path
|
||
from platform import system
|
||
from shutil import rmtree
|
||
from tempfile import gettempdir
|
||
from threading import Lock
|
||
from time import perf_counter, sleep
|
||
|
||
from .._configs.options_manage import OptionsManager
|
||
from ..errors import (ContextLostError, ElementLostError, CDPError, PageDisconnectedError, NoRectError,
|
||
AlertExistsError, WrongURLError, StorageError, CookieFormatError, JavaScriptError)
|
||
|
||
|
||
class PortFinder(object):
|
||
used_port = set()
|
||
prev_time = 0
|
||
lock = Lock()
|
||
checked_paths = set()
|
||
|
||
def __init__(self, path=None):
|
||
"""
|
||
:param path: 临时文件保存路径,为None时使用系统临时文件夹
|
||
"""
|
||
tmp = Path(path) if path else Path(gettempdir()) / 'DrissionPage'
|
||
self.tmp_dir = tmp / 'autoPortData'
|
||
self.tmp_dir.mkdir(parents=True, exist_ok=True)
|
||
if str(self.tmp_dir.absolute()) not in PortFinder.checked_paths:
|
||
for i in self.tmp_dir.iterdir():
|
||
if i.is_dir() and not port_is_using('127.0.0.1', i.name):
|
||
rmtree(i, ignore_errors=True)
|
||
PortFinder.checked_paths.add(str(self.tmp_dir.absolute()))
|
||
|
||
def get_port(self, scope=None):
|
||
"""查找一个可用端口
|
||
:param scope: 指定端口范围,不含最后的数字,为None则使用[9600-59600)
|
||
:return: 可以使用的端口和用户文件夹路径组成的元组
|
||
"""
|
||
from random import randint
|
||
with PortFinder.lock:
|
||
if PortFinder.prev_time and perf_counter() - PortFinder.prev_time > 60:
|
||
PortFinder.used_port.clear()
|
||
if scope in (True, None):
|
||
scope = (9600, 59600)
|
||
max_times = scope[1] - scope[0]
|
||
times = 0
|
||
while times < max_times:
|
||
times += 1
|
||
port = randint(*scope)
|
||
if port in PortFinder.used_port or port_is_using('127.0.0.1', port):
|
||
continue
|
||
path = self.tmp_dir / str(port)
|
||
if path.exists():
|
||
try:
|
||
rmtree(path)
|
||
except:
|
||
continue
|
||
PortFinder.used_port.add(port)
|
||
PortFinder.prev_time = perf_counter()
|
||
return port, str(path)
|
||
raise OSError('未找到可用端口。')
|
||
|
||
|
||
def port_is_using(ip, port):
|
||
"""检查端口是否被占用
|
||
:param ip: 浏览器地址
|
||
:param port: 浏览器端口
|
||
:return: bool
|
||
"""
|
||
from socket import socket, AF_INET, SOCK_STREAM
|
||
s = socket(AF_INET, SOCK_STREAM)
|
||
s.settimeout(.1)
|
||
result = s.connect_ex((ip, int(port)))
|
||
s.close()
|
||
return result == 0
|
||
|
||
|
||
def clean_folder(folder_path, ignore=None):
|
||
"""清空一个文件夹,除了ignore里的文件和文件夹
|
||
:param folder_path: 要清空的文件夹路径
|
||
:param ignore: 忽略列表
|
||
:return: None
|
||
"""
|
||
ignore = [] if not ignore else ignore
|
||
p = Path(folder_path)
|
||
|
||
for f in p.iterdir():
|
||
if f.name not in ignore:
|
||
if f.is_file():
|
||
f.unlink()
|
||
elif f.is_dir():
|
||
rmtree(f, True)
|
||
|
||
|
||
def show_or_hide_browser(page, hide=True):
|
||
"""执行显示或隐藏浏览器窗口
|
||
:param page: ChromePage对象
|
||
:param hide: 是否隐藏
|
||
:return: None
|
||
"""
|
||
if not page.browser.address.startswith(('127.0.0.1', 'localhost')):
|
||
return
|
||
|
||
if system().lower() != 'windows':
|
||
raise OSError('该方法只能在Windows系统使用。')
|
||
|
||
try:
|
||
from win32gui import ShowWindow
|
||
from win32con import SW_HIDE, SW_SHOW
|
||
except ImportError:
|
||
raise ImportError('请先安装:pip install pypiwin32')
|
||
|
||
pid = page._page.process_id
|
||
if not pid:
|
||
return None
|
||
hds = get_hwnds_from_pid(pid, page.title)
|
||
sw = SW_HIDE if hide else SW_SHOW
|
||
for hd in hds:
|
||
ShowWindow(hd, sw)
|
||
|
||
|
||
def get_browser_progress_id(progress, address):
|
||
"""获取浏览器进程id
|
||
:param progress: 已知的进程对象,没有时传入None
|
||
:param address: 浏览器管理地址,含端口
|
||
:return: 进程id或None
|
||
"""
|
||
if progress:
|
||
return progress.pid
|
||
|
||
from os import popen
|
||
port = address.split(':')[-1]
|
||
txt = ''
|
||
progresses = popen(f'netstat -nao | findstr :{port}').read().split('\n')
|
||
for progress in progresses:
|
||
if 'LISTENING' in progress:
|
||
txt = progress
|
||
break
|
||
if not txt:
|
||
return None
|
||
|
||
return txt.split(' ')[-1]
|
||
|
||
|
||
def get_hwnds_from_pid(pid, title):
|
||
"""通过PID查询句柄ID
|
||
:param pid: 进程id
|
||
:param title: 窗口标题
|
||
:return: 进程句柄组成的列表
|
||
"""
|
||
try:
|
||
from win32gui import IsWindow, GetWindowText, EnumWindows
|
||
from win32process import GetWindowThreadProcessId
|
||
except ImportError:
|
||
raise ImportError('请先安装win32gui,pip install pypiwin32')
|
||
|
||
def callback(hwnd, hds):
|
||
if IsWindow(hwnd) and title in GetWindowText(hwnd):
|
||
_, found_pid = GetWindowThreadProcessId(hwnd)
|
||
if str(found_pid) == str(pid):
|
||
hds.append(hwnd)
|
||
return True
|
||
|
||
hwnds = []
|
||
EnumWindows(callback, hwnds)
|
||
return hwnds
|
||
|
||
|
||
def wait_until(function, kwargs=None, timeout=10):
|
||
"""等待传入的方法返回值不为假
|
||
:param function: 要执行的方法
|
||
:param kwargs: 方法参数
|
||
:param timeout: 超时时间(秒)
|
||
:return: 执行结果,超时抛出TimeoutError
|
||
"""
|
||
if kwargs is None:
|
||
kwargs = {}
|
||
end_time = perf_counter() + timeout
|
||
while perf_counter() < end_time:
|
||
value = function(**kwargs)
|
||
if value:
|
||
return value
|
||
sleep(.01)
|
||
raise TimeoutError
|
||
|
||
|
||
def configs_to_here(save_name=None):
|
||
"""把默认ini文件复制到当前目录
|
||
:param save_name: 指定文件名,为None则命名为'dp_configs.ini'
|
||
:return: None
|
||
"""
|
||
om = OptionsManager('default')
|
||
save_name = f'{save_name}.ini' if save_name is not None else 'dp_configs.ini'
|
||
om.save(save_name)
|
||
|
||
|
||
def raise_error(result, ignore=None, user=False):
|
||
"""抛出error对应报错
|
||
:param result: 包含error的dict
|
||
:param ignore: 要忽略的错误
|
||
:param user: 是否用户调用的
|
||
:return: None
|
||
"""
|
||
error = result['error']
|
||
if error in ('Cannot find context with specified id', 'Inspected target navigated or closed',
|
||
'No frame with given id found'):
|
||
r = ContextLostError()
|
||
elif error in ('Could not find node with given id', 'Could not find object with given id',
|
||
'No node with given id found', 'Node with given id does not belong to the document',
|
||
'No node found for given backend id'):
|
||
r = ElementLostError()
|
||
elif error in ('connection disconnected', 'No target with given id found'):
|
||
r = PageDisconnectedError()
|
||
elif error == 'alert exists.':
|
||
r = AlertExistsError()
|
||
elif error in ('Node does not have a layout object', 'Could not compute box model.'):
|
||
r = NoRectError()
|
||
elif error == 'Cannot navigate to invalid URL':
|
||
r = WrongURLError(f'无效的url:{result["args"]["url"]}。也许要加上"http://"?')
|
||
elif error == 'Frame corresponds to an opaque origin and its storage key cannot be serialized':
|
||
r = StorageError()
|
||
elif error == 'Sanitizing cookie failed':
|
||
r = CookieFormatError(f'cookie格式不正确:{result["args"]}')
|
||
elif error == 'Invalid header name':
|
||
r = ValueError(f'header名不正确。\n参数:{result["args"]["headers"]}')
|
||
elif error == 'Given expression does not evaluate to a function':
|
||
r = JavaScriptError(f'传入的js无法解析成函数:\n{result["args"]["functionDeclaration"]}')
|
||
elif error.endswith("' wasn't found"):
|
||
r = RuntimeError(f'没有找到对应功能,方法错误或你的浏览器太旧。\n方法:{result["method"]}\n参数:{result["args"]}')
|
||
elif result['type'] == 'timeout':
|
||
from DrissionPage import __version__
|
||
txt = f'\n错误:{result["error"]}\n方法:{result["method"]}\n参数:{result["args"]}\n' \
|
||
f'版本:{__version__}\n超时,可能是浏览器卡了。'
|
||
r = TimeoutError(txt)
|
||
elif result['type'] == 'call_method_error' and not user:
|
||
from DrissionPage import __version__
|
||
txt = f'\n错误:{result["error"]}\n方法:{result["method"]}\n参数:{result["args"]}\n' \
|
||
f'版本:{__version__}\n出现这个错误可能意味着程序有bug,请把错误信息和重现方法' \
|
||
'告知作者,谢谢。\n报告网站:https://gitee.com/g1879/DrissionPage/issues'
|
||
r = CDPError(txt)
|
||
else:
|
||
r = RuntimeError(result)
|
||
|
||
if not ignore or not isinstance(r, ignore):
|
||
raise r
|