Как включить селеновый веб-драйвер для работы с удаленными объектами Python (Pyro) в документе bokeh? - PullRequest
0 голосов
/ 16 мая 2018

У меня есть документ Python, работающий на сервере Bokeh с bokeh serve view_server_so.py --show.Веб-страница сразу открывается на вкладке браузера.

В целях тестирования я использую Pyro, чтобы разрешить удаленный доступ к некоторым объектам / методам, работающим на сервере bokeh.

Мой тестовый скрипт сервера создаеттолько одна кнопка и один метод (доступны для дистанционного управления), bokeh отображает веб-страницу на http://localhost:5006/view_server_so.

. Кнопка представляет собой кнопку переключения, при каждом нажатии кнопки ее активное состояние меняется от True доFalse, и его цвет меняется.

# view_server_so.py
# This script shall be started from the command line with
# bokeh serve view_server_so.py --show

from bokeh.layouts import layout, row
bokeh.models import Toggle
from bokeh.io import curdoc
import os
import Pyro4
import socket
import subprocess
import threading

# The layout used in the html-page
class View():
    def __init__(self):
        self.document = curdoc()
        self.btn_task = Toggle(label="Press me",
                               button_type="success",
                               css_classes=["btn_task"])        
        self.layout = layout(row(self.btn_task))

    @Pyro4.expose
    def get_btn_task_active_state(self):
        # This method is enabled for remote control
        # The btn_task changes its active state each time it is pressed (bool)
        print('active state: {}'.format(self.btn_task.active))
        return self.btn_task.active

# As the script is started with 'bokeh serve', 
# bokeh has control of the main event loop.
# To enable Pyro to listen for client requests for the object view,
# and its exposed method,
# the Pyro event loop needs to run in its own thread.
class PyroDaemon(threading.Thread):
    def __init__(self, obj):
        threading.Thread.__init__(self)
        self.obj=obj
        self.started=threading.Event()
        print('PyroDaemon.__init__()')
    def run(self):
        daemon=Pyro4.Daemon()
        obj = self.obj
        self.uri=daemon.register(obj,"test")
        self.started.set()
        print('PyroDaemon.run()')
        daemon.requestLoop()

view = View()
print('view created')

# starting a Pyro name server
hostname = socket.gethostname()
print('hostname: {}'.format(hostname))
try:
    print('try Pyro4.locateNS()')
    name_server=Pyro4.locateNS()
except Pyro4.errors.NamingError:
    print('except Pyro4.errorsNamingError')
    args = ['python', '-m', 'Pyro4.naming', '-n', hostname]
    p = subprocess.Popen(args) 
    name_server=Pyro4.locateNS()
print('Nameserver is started')

# create a pyro daemon with object view, running in its own worker thread
pyro_thread=PyroDaemon(view)
pyro_thread.setDaemon(True)
pyro_thread.start()
pyro_thread.started.wait()
print('Daemon is started')

# The object view is registered at the Pyro name server
name_server.register('view', pyro_thread.uri, metadata = {'View'})
print('view is registered')

view.document.add_root(view.layout)

Теперь у меня есть второй тестовый сценарий.После некоторой настройки удаленного объекта он просто несколько раз вызывает метод get_btn_task_active_state() и печатает результат.Когда между вызовами нажимается кнопка в браузере, активное состояние переключается и правильно печатается.

import Pyro4
import time

# looking for the Pyro name server created in view_server_so.py
nameserver=Pyro4.locateNS(broadcast = True)
# get the URI (universal resource identifier)
uris = nameserver.list(metadata_all = {'View'})
# create a proxy object that interacts (remote control)
# with the remote object 'view'
view_rc = Pyro4.Proxy(uris['view'])

for i in range(3):
    time.sleep(3)
    # change the state manually: click on the button
    print('state: {}'.format(view_rc.get_btn_task_active_state()))
    # prints
    # state: False
    # state: True
    # state: False

Поскольку ручное тестирование становится утомительным, я хочу автоматизировать ручные нажатия на кнопку.Поэтому я добавляю поддержку веб-драйвера, ищу кнопку на веб-странице и получаю несколько автоматических кликов и звонков на get_btn_task_active_state(), как и раньше.

import Pyro4
from selenium import webdriver
import socket
import time

# looking for the Pyro name server created in view_server_so.py
nameserver=Pyro4.locateNS(broadcast = True)
# get the URI (universal resource identifier)
uris = nameserver.list(metadata_all = {'View'})
# create a proxy object that interacts (remote control)
# with the remote object 'view'
view_rc = Pyro4.Proxy(uris['view'])

# use the Chrome webdriver
driver = webdriver.Chrome()
# open a Chrome Browser and the given webpage
driver.get("http://localhost:5006/view_server_so")
time.sleep(1)
# Find the button
btn_task = driver.find_element_by_class_name("btn_task")

for i in range(3):
    time.sleep(1)
    print('state: {}'.format(view_rc.get_btn_task_active_state()))
    btn_task.click()
    # prints
    # state: False
    # state: False
    # state: False
    #
    # but should alternate between False and True

Откроется браузер, управляемый веб-драйвером, я вижу цветкнопки меняются визуально, так как они автоматически нажимаются, но активное состояние кнопки, кажется, больше не меняется.

Какие изменения потребуются, чтобы автоматический скрипт выдал те же результаты, что и руководствотестирование?

...