diff --git a/LICENSE b/LICENSE index a7133a96b68..8805a8182c6 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ The MIT License (MIT) -Copyright (c) 2014-2024 Michael Mintz +Copyright (c) 2014-2025 Michael Mintz Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/examples/cdp_mode/ReadMe.md b/examples/cdp_mode/ReadMe.md index 95e3aa6a6ea..60d204a0eb0 100644 --- a/examples/cdp_mode/ReadMe.md +++ b/examples/cdp_mode/ReadMe.md @@ -6,11 +6,16 @@ -------- - +

(Watch the CDP Mode tutorial on YouTube! ▶️)

-------- + +

(Watch "Hacking websites with CDP" on YouTube! ▶️)

+ +-------- + 👤 UC Mode avoids bot-detection by first disconnecting WebDriver from the browser at strategic times, calling special PyAutoGUI methods to bypass CAPTCHAs (as needed), and finally reconnecting the driver afterwards so that WebDriver actions can be performed again. Although this approach works for bypassing simple CAPTCHAs, more flexibility is needed for bypassing bot-detection on websites with advanced protection. (That's where CDP Mode comes in.) 🐙 CDP Mode is based on python-cdp, trio-cdp, and nodriver. trio-cdp is an early implementation of python-cdp, and nodriver is a modern implementation of python-cdp. (Refactored Python-CDP code is imported from MyCDP.) diff --git a/examples/presenter/hacking_with_cdp.py b/examples/presenter/hacking_with_cdp.py new file mode 100644 index 00000000000..0e5341f11fa --- /dev/null +++ b/examples/presenter/hacking_with_cdp.py @@ -0,0 +1,146 @@ +# https://www.youtube.com/watch?v=vt2zsdiNh3U +from seleniumbase import BaseCase +BaseCase.main(__name__, __file__) + + +class UCPresentationClass(BaseCase): + def test_hacking_with_cdp(self): + self.open("data:,") + self.set_window_position(4, 40) + self._output_file_saves = False + self.create_presentation(theme="serif", transition="none") + self.add_slide("

Press SPACE to begin!

\n") + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "
", + ) + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "

", + ) + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "

", + ) + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "

", + ) + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "

", + ) + self.add_slide( + "

Coming up on the Hacker Show...

\n" + "

", + ) + self.add_slide( + "

Get ready for some
serious hacking!

" + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '' + ) + self.add_slide( + '

The remote-debugging-port

' + '' + ) + self.add_slide( + "

Let's get to the fun part...

" + '' + ) + self.begin_presentation(filename="uc_presentation.html") diff --git a/examples/raw_skype_mobile.py b/examples/raw_skype_mobile.py new file mode 100644 index 00000000000..2f4ef2c6a6c --- /dev/null +++ b/examples/raw_skype_mobile.py @@ -0,0 +1,15 @@ +"""Mobile emulation test for Skype.""" +from seleniumbase import SB + +with SB(mobile=True, test=True) as sb: + sb.open("https://www.skype.com/en/get-skype/") + sb.assert_element('[aria-label="Microsoft"]') + sb.assert_text("Download Skype", "h1") + sb.highlight("div.appBannerContent") + sb.highlight("h1") + sb.assert_text("Skype for Mobile", "h2") + sb.highlight("h2") + sb.highlight("#get-skype-0") + sb.highlight_click("span[data-dropdown-icon]") + sb.highlight("#get-skype-0_android-download") + sb.highlight('[data-bi-id*="ios"]') diff --git a/mkdocs_build/requirements.txt b/mkdocs_build/requirements.txt index 4608be4cd1e..4379c934aae 100644 --- a/mkdocs_build/requirements.txt +++ b/mkdocs_build/requirements.txt @@ -2,7 +2,7 @@ # Minimum Python version: 3.9 (for generating docs only) regex>=2024.11.6 -pymdown-extensions>=10.13 +pymdown-extensions>=10.14 pipdeptree>=2.24.0 python-dateutil>=2.8.2 Markdown==3.7 diff --git a/requirements.txt b/requirements.txt index 806752a8c0d..b0b6de70735 100755 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,7 @@ pip>=24.3.1 packaging>=24.2 setuptools~=70.2;python_version<"3.10" -setuptools>=75.6.0;python_version>="3.10" +setuptools>=75.8.0;python_version>="3.10" wheel>=0.45.1 attrs>=24.3.0 certifi>=2024.12.14 @@ -23,7 +23,7 @@ parse>=1.20.2 parse-type>=0.6.4 colorama>=0.4.6 pyyaml>=6.0.2 -pygments>=2.18.0 +pygments>=2.19.1 pyreadline3>=3.5.3;platform_system=="Windows" tabcompleter>=1.4.0 pdbp>=1.6.1 @@ -36,7 +36,8 @@ requests==2.32.3 sniffio==1.3.1 h11==0.14.0 outcome==1.3.0.post0 -trio==0.27.0 +trio==0.27.0;python_version<"3.9" +trio==0.28.0;python_version>="3.9" trio-websocket==0.11.1 wsproto==1.2.0 websocket-client==1.8.0 @@ -67,7 +68,7 @@ rich==13.9.4 # ("pip install -r requirements.txt" also installs this, but "pip install -e ." won't.) coverage>=7.6.1;python_version<"3.9" -coverage>=7.6.9;python_version>="3.9" +coverage>=7.6.10;python_version>="3.9" pytest-cov>=5.0.0;python_version<"3.9" pytest-cov>=6.0.0;python_version>="3.9" flake8==5.0.4;python_version<"3.9" diff --git a/seleniumbase/__version__.py b/seleniumbase/__version__.py index 5a2eda19a6a..634fe5c2052 100755 --- a/seleniumbase/__version__.py +++ b/seleniumbase/__version__.py @@ -1,2 +1,2 @@ # seleniumbase package -__version__ = "4.33.12" +__version__ = "4.33.13" diff --git a/seleniumbase/core/browser_launcher.py b/seleniumbase/core/browser_launcher.py index eb50a898daf..56439970876 100644 --- a/seleniumbase/core/browser_launcher.py +++ b/seleniumbase/core/browser_launcher.py @@ -533,10 +533,26 @@ def uc_open_with_cdp_mode(driver, url=None): if url_protocol not in ["about", "data", "chrome"]: safe_url = False + headless = False + headed = None + xvfb = None + if hasattr(sb_config, "headless"): + headless = sb_config.headless + if hasattr(sb_config, "headed"): + headed = sb_config.headed + if hasattr(sb_config, "xvfb"): + xvfb = sb_config.xvfb + loop = asyncio.new_event_loop() asyncio.set_event_loop(loop) driver.cdp_base = loop.run_until_complete( - cdp_util.start(host=cdp_host, port=cdp_port) + cdp_util.start( + host=cdp_host, + port=cdp_port, + headless=headless, + headed=headed, + xvfb=xvfb, + ) ) loop.run_until_complete(driver.cdp_base.wait(0)) @@ -863,13 +879,15 @@ def __install_pyautogui_if_missing(): xvfb_height = 768 sb_config._xvfb_height = xvfb_height with suppress(Exception): - xvfb_display = Display( + _xvfb_display = Display( visible=True, size=(xvfb_width, xvfb_height), backend="xvfb", use_xauth=True, ) - xvfb_display.start() + _xvfb_display.start() + sb_config._virtual_display = _xvfb_display + sb_config.headless_active = True def install_pyautogui_if_missing(driver): diff --git a/seleniumbase/fixtures/base_case.py b/seleniumbase/fixtures/base_case.py index e8092ba0583..b34e3c7d5bd 100644 --- a/seleniumbase/fixtures/base_case.py +++ b/seleniumbase/fixtures/base_case.py @@ -13795,7 +13795,8 @@ def __switch_to_newest_window_if_not_blank(self): if self.get_current_url() == "about:blank": self.switch_to_window(current_window) except Exception: - self.switch_to_window(current_window) + with suppress(Exception): + self.switch_to_window(current_window) def __needs_minimum_wait(self): if ( @@ -14004,9 +14005,10 @@ def __activate_standard_virtual_display(self): visible=0, size=(width, height) ) self._xvfb_display.start() - sb_config._virtual_display = self._xvfb_display self.headless_active = True - sb_config.headless_active = True + if not self.undetectable: + sb_config._virtual_display = self._xvfb_display + sb_config.headless_active = True def __activate_virtual_display(self): if self.undetectable and not (self.headless or self.headless2): @@ -14029,6 +14031,8 @@ def __activate_virtual_display(self): "\nX11 display failed! Will use regular xvfb!" ) self.__activate_standard_virtual_display() + else: + self.headless_active = True except Exception as e: if hasattr(e, "msg"): print("\n" + str(e.msg)) @@ -16601,6 +16605,7 @@ def tearDown(self): self.__quit_all_drivers() # Resume tearDown() for all test runners, (Pytest / Pynose / Behave) if hasattr(self, "_xvfb_display") and self._xvfb_display: + # Stop the Xvfb virtual display launched from BaseCase try: if hasattr(self._xvfb_display, "stop"): self._xvfb_display.stop() @@ -16610,6 +16615,20 @@ def tearDown(self): pass except Exception: pass + if ( + hasattr(sb_config, "_virtual_display") + and sb_config._virtual_display + and hasattr(sb_config._virtual_display, "stop") + ): + # CDP Mode may launch a 2nd Xvfb virtual display + try: + sb_config._virtual_display.stop() + sb_config._virtual_display = None + sb_config.headless_active = False + except AttributeError: + pass + except Exception: + pass if self.__visual_baseline_copies: sb_config._visual_baseline_copies = True if has_exception: diff --git a/seleniumbase/plugins/sb_manager.py b/seleniumbase/plugins/sb_manager.py index 3c39eca7c15..96f916cf2e0 100644 --- a/seleniumbase/plugins/sb_manager.py +++ b/seleniumbase/plugins/sb_manager.py @@ -1256,6 +1256,19 @@ def SB( print(traceback.format_exc().strip()) if test and not test_passed: print("********** ERROR: The test AND the tearDown() FAILED!") + if ( + hasattr(sb_config, "_virtual_display") + and sb_config._virtual_display + and hasattr(sb_config._virtual_display, "stop") + ): + try: + sb_config._virtual_display.stop() + sb_config._virtual_display = None + sb_config.headless_active = False + except AttributeError: + pass + except Exception: + pass end_time = time.time() run_time = end_time - start_time sb_config = sb_config_backup diff --git a/seleniumbase/undetected/cdp_driver/cdp_util.py b/seleniumbase/undetected/cdp_driver/cdp_util.py index 7899436f2eb..237ba3b28b3 100644 --- a/seleniumbase/undetected/cdp_driver/cdp_util.py +++ b/seleniumbase/undetected/cdp_driver/cdp_util.py @@ -84,6 +84,9 @@ def __activate_virtual_display_as_needed( "\nX11 display failed! Will use regular xvfb!" ) __activate_standard_virtual_display() + else: + sb_config._virtual_display = _xvfb_display + sb_config.headless_active = True except Exception as e: if hasattr(e, "msg"): print("\n" + str(e.msg)) diff --git a/setup.py b/setup.py index 01615ed533e..c9308093c1e 100755 --- a/setup.py +++ b/setup.py @@ -150,7 +150,7 @@ 'pip>=24.3.1', 'packaging>=24.2', 'setuptools~=70.2;python_version<"3.10"', # Newer ones had issues - 'setuptools>=75.6.0;python_version>="3.10"', + 'setuptools>=75.8.0;python_version>="3.10"', 'wheel>=0.45.1', 'attrs>=24.3.0', "certifi>=2024.12.14", @@ -172,7 +172,7 @@ 'parse-type>=0.6.4', 'colorama>=0.4.6', 'pyyaml>=6.0.2', - 'pygments>=2.18.0', + 'pygments>=2.19.1', 'pyreadline3>=3.5.3;platform_system=="Windows"', "tabcompleter>=1.4.0", "pdbp>=1.6.1", @@ -185,7 +185,8 @@ 'sniffio==1.3.1', 'h11==0.14.0', 'outcome==1.3.0.post0', - 'trio==0.27.0', + 'trio==0.27.0;python_version<"3.9"', + 'trio==0.28.0;python_version>="3.9"', 'trio-websocket==0.11.1', 'wsproto==1.2.0', 'websocket-client==1.8.0', @@ -225,7 +226,7 @@ # Usage: coverage run -m pytest; coverage html; coverage report "coverage": [ 'coverage>=7.6.1;python_version<"3.9"', - 'coverage>=7.6.9;python_version>="3.9"', + 'coverage>=7.6.10;python_version>="3.9"', 'pytest-cov>=5.0.0;python_version<"3.9"', 'pytest-cov>=6.0.0;python_version>="3.9"', ], @@ -264,7 +265,7 @@ # (An optional library for image-processing.) "pillow": [ 'Pillow>=10.4.0;python_version<"3.9"', - 'Pillow>=11.0.0;python_version>="3.9"', + 'Pillow>=11.1.0;python_version>="3.9"', ], # pip install -e .[pip-system-certs] # (If you see [SSL: CERTIFICATE_VERIFY_FAILED], then get this.)