123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- import _thread
- import logging
- import multiprocessing as mp
- import queue
- import socket
- import sys
- import tempfile
- import time
- import warnings
- from threading import Event, Thread
- from typing import Any, Callable, Dict, List, Tuple
- from . import globals, helpers, native
- try:
- with warnings.catch_warnings():
- # webview depends on bottle which uses the deprecated CGI function (https://github.com/bottlepy/bottle/issues/1403)
- warnings.filterwarnings('ignore', category=DeprecationWarning)
- import webview
- except ModuleNotFoundError:
- class webview:
- class Window:
- pass
- pass
- def open_window(
- host: str, port: int, title: str, width: int, height: int, fullscreen: bool,
- method_queue: mp.Queue, response_queue: mp.Queue,
- ) -> None:
- while not helpers.is_port_open(host, port):
- time.sleep(0.1)
- window_kwargs = dict(url=f'http://{host}:{port}', title=title, width=width, height=height, fullscreen=fullscreen)
- window_kwargs.update(globals.app.native.window_args)
- try:
- window = webview.create_window(**window_kwargs)
- closing = Event()
- window.events.closing += closing.set
- start_window_method_executor(window, method_queue, response_queue, closing)
- webview.start(storage_path=tempfile.mkdtemp(), **globals.app.native.start_args)
- except NameError:
- logging.error('''Native mode is not supported in this configuration.
- Please run "pip install nicegui[native] to use it.''')
- sys.exit(1)
- def start_window_method_executor(
- window: webview.Window, method_queue: mp.Queue, response_queue: mp.Queue, closing: Event
- ) -> None:
- def execute(method: Callable, args: Tuple[Any, ...], kwargs: Dict[str, Any]) -> None:
- try:
- response = method(*args, **kwargs)
- if response is not None or 'dialog' in method.__name__:
- response_queue.put(response)
- except Exception:
- logging.exception(f'error in window.{method.__name__}')
- def window_method_executor() -> None:
- pending_executions: List[Thread] = []
- while not closing.is_set():
- try:
- method_name, args, kwargs = method_queue.get(block=False)
- if method_name == 'signal_server_shutdown':
- if pending_executions:
- logging.warning('shutdown is possibly blocked by opened dialogs like a file picker')
- while pending_executions:
- pending_executions.pop().join()
- elif method_name == 'get_always_on_top':
- response_queue.put(window.on_top)
- elif method_name == 'set_always_on_top':
- window.on_top = args[0]
- elif method_name == 'get_position':
- response_queue.put((int(window.x), int(window.y)))
- elif method_name == 'get_size':
- response_queue.put((int(window.width), int(window.height)))
- else:
- method = getattr(window, method_name)
- if callable(method):
- pending_executions.append(Thread(target=execute, args=(method, args, kwargs)))
- pending_executions[-1].start()
- else:
- logging.error(f'window.{method_name} is not callable')
- except queue.Empty:
- time.sleep(0.01)
- except Exception:
- logging.exception(f'error in window.{method_name}')
- Thread(target=window_method_executor).start()
- def activate(host: str, port: int, title: str, width: int, height: int, fullscreen: bool) -> None:
- def check_shutdown() -> None:
- while process.is_alive():
- time.sleep(0.1)
- globals.server.should_exit = True
- while globals.state != globals.State.STOPPED:
- time.sleep(0.1)
- _thread.interrupt_main()
- mp.freeze_support()
- args = host, port, title, width, height, fullscreen, native.method_queue, native.response_queue
- process = mp.Process(target=open_window, args=args, daemon=False)
- process.start()
- Thread(target=check_shutdown, daemon=True).start()
- def find_open_port(start_port: int = 8000, end_port: int = 8999) -> int:
- """Reliably find an open port in a given range.
- This function will actually try to open the port to ensure no firewall blocks it.
- This is better than, e.g., passing port=0 to uvicorn.
- """
- for port in range(start_port, end_port + 1):
- try:
- with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
- s.bind(('localhost', port))
- return port
- except OSError:
- pass
- raise OSError('No open port found')
|