-
Notifications
You must be signed in to change notification settings - Fork 5.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Selenium utils + markdown rendering tests #3458
Changes from 18 commits
9d4cf94
7c659f5
0994db2
f9dd7c1
c9a8504
3d3086e
4b6a0a7
2b7f549
e2243d6
86ae162
bf39dec
e0ed2c4
c220215
3092800
3615d4a
d634f1d
a112ab6
ebef7ba
5d19785
3571f16
e183fc1
c16dac2
e9971a9
bf4868d
02e0ac3
98c09f8
5e43458
d9dd5d8
33ca649
0999798
7808a89
c081af5
d598ef5
3c4596b
79603b4
74af79c
515f8e2
c23ba2a
a6f604a
7266fd5
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
from selenium.webdriver import Firefox | ||
from notebook.notebookapp import list_running_servers | ||
|
||
|
||
def quick_driver(): | ||
"""Quickly create a selenium driver pointing at an active noteboook server. | ||
|
||
Usage example | ||
|
||
from inside the selenium test directory: | ||
|
||
import quick_selenium, test_markdown, utils | ||
nb = utils.Notebook(test_markdown.notebook(quick_selenium.quick_driver())) | ||
""" | ||
try: | ||
server = list(list_running_servers())[0] | ||
except IndexError as e: | ||
e.message = 'You need a server running before you can run this command' | ||
driver = Firefox() | ||
auth_url = f'{server["url"]}?token={server["token"]}' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. f strings are great, but we're still supporting Python 3.4 and 3.5 for now, so we'll need to stick with the explicit There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ✓ |
||
driver.get(auth_url) | ||
return driver |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
import os | ||
|
||
import pytest | ||
from selenium.webdriver.common.keys import Keys | ||
|
||
from .utils import wait_for_selector, Notebook | ||
|
||
pjoin = os.path.join | ||
|
||
|
||
@pytest.fixture(scope='module') | ||
def notebook(authenticated_browser): | ||
return Notebook.new_notebook(authenticated_browser) | ||
|
||
|
||
def test_markdown_cell(notebook): | ||
nb = notebook | ||
cell = nb.cells[0] | ||
nb.convert_cell_type(cell_type="markdown") | ||
assert nb.current_cell != cell | ||
nb.edit_cell(index=0, content="# Foo") | ||
nb.wait_for_stale_cell(cell) | ||
rendered_cells = nb.browser.find_elements_by_class_name('text_cell_render') | ||
outputs = [x.get_attribute('innerHTML') for x in rendered_cells] | ||
expected = '<h1 id="Foo">Foo<a class="anchor-link" href="#Foo">¶</a></h1>' | ||
assert outputs[0].strip() == expected |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,163 @@ | ||
import os | ||
|
||
from selenium.webdriver.common.by import By | ||
from selenium.webdriver.common.keys import Keys | ||
from selenium.webdriver.support.ui import WebDriverWait | ||
from selenium.webdriver.support import expected_conditions as EC | ||
|
||
from contextlib import contextmanager | ||
|
||
pjoin = os.path.join | ||
|
||
|
||
def wait_for_selector(browser, selector, timeout=10, visible=False, single=False): | ||
wait = WebDriverWait(browser, timeout) | ||
if single: | ||
if visible: | ||
conditional = EC.visibility_of_element_located | ||
else: | ||
conditional = EC.presence_of_element_located | ||
else: | ||
if visible: | ||
conditional = EC.visibility_of_all_elements_located | ||
else: | ||
conditional = EC.presence_of_all_elements_located | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is steadily heading towards replicating the complexity of the underlying Selenium API. It's fine for now, but we should keep in mind that it may be better to use the underlying APIs directly in some cases. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Agreed — however, I think this covers all the cases where we wanted it and it hides a tonne of the selenium boilerplate so I think it's still ok. |
||
return wait.until(conditional((By.CSS_SELECTOR, selector))) | ||
|
||
|
||
class CellTypeError(ValueError): | ||
|
||
def __init__(self, message=""): | ||
self.message = message | ||
|
||
class Notebook: | ||
|
||
def __init__(self, browser): | ||
self.browser = browser | ||
self.remove_safety_check() | ||
|
||
@property | ||
def body(self): | ||
return self.browser.find_element_by_tag_name("body") | ||
|
||
@property | ||
def cells(self): | ||
"""Gets all cells once they are visible. | ||
|
||
""" | ||
wait_for_selector(self.browser, ".cell") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't like having a wait inside a property - the rule of thumb for properties is that anything they need to do should be near-instant. If it needs to do something that's not, let's make it a regular method - the call gives people reading the code a hint that there's something going on underneath. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was wrong in my belief that it was needed here. One of the reasons I really didn't like it was that it couldn't handle notebooks with no cells; so this always seemed wrong. So it's gone! |
||
return self.browser.find_elements_by_class_name("cell") | ||
|
||
|
||
@property | ||
def current_cell_index(self): | ||
return self.cells.index(self.current_cell) | ||
|
||
def remove_safety_check(self): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 'safety' is rather vague. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Let's find a more representative name for this. |
||
"""Disable request to save before closing window. | ||
|
||
This is most easily done by using js directly. | ||
""" | ||
self.browser.execute_script("window.onbeforeunload = null;") | ||
|
||
def to_command_mode(self): | ||
"""Changes us into command mode on currently focused cell | ||
|
||
""" | ||
self.cells[0].send_keys(Keys.ESCAPE) | ||
self.browser.execute_script("return Jupyter.notebook.handle_command_mode(" | ||
"Jupyter.notebook.get_cell(" | ||
"Jupyter.notebook.get_edit_index()))") | ||
|
||
def focus_cell(self, index=0): | ||
cell = self.cells[index] | ||
cell.click() | ||
self.to_command_mode() | ||
self.current_cell = cell | ||
|
||
def convert_cell_type(self, index=0, cell_type="code"): | ||
# TODO add check to see if it is already present | ||
self.focus_cell(index) | ||
cell = self.cells[index] | ||
if cell_type == "markdown": | ||
self.current_cell.send_keys("m") | ||
elif cell_type == "raw": | ||
self.current_cell.send_keys("r") | ||
elif cell_type == "code": | ||
self.current_cell.send_keys("y") | ||
else: | ||
raise CellTypeError(("{} is not a valid cell type," | ||
"use 'code', 'markdown', or 'raw'").format(cell_type)) | ||
|
||
self.wait_for_stale_cell(cell) | ||
self.focus_cell(index) | ||
return self.current_cell | ||
|
||
def wait_for_stale_cell(self, cell): | ||
""" This is needed to switch a cell's mode and refocus it, or to render it. | ||
|
||
Warning: there is currently no way to do this when changing between | ||
markdown and raw cells. | ||
""" | ||
wait = WebDriverWait(self.browser, 10) | ||
element = wait.until(EC.staleness_of(cell)) | ||
|
||
def edit_cell(self, cell=None, index=0, content="", render=True): | ||
if cell is None: | ||
cell = self.cells[index] | ||
else: | ||
index = self.cells.index(cell) | ||
self.focus_cell(index) | ||
|
||
for line_no, line in enumerate(content.splitlines()): | ||
if line_no != 0: | ||
self.current_cell.send_keys(Keys.ENTER, "\n") | ||
self.current_cell.send_keys(Keys.ENTER, line) | ||
if render: | ||
self.execute_cell(self.current_cell_index) | ||
|
||
def execute_cell(self, index=0): | ||
self.focus_cell(index) | ||
self.current_cell.send_keys(Keys.CONTROL, Keys.ENTER) | ||
|
||
def add_cell(self, index=-1, cell_type="code"): | ||
self.focus_cell(index) | ||
self.current_cell.send_keys("a") | ||
new_index = index + 1 if index >= 0 else index | ||
self.convert_cell_type(index=new_index, cell_type=cell_type) | ||
|
||
def add_markdown_cell(self, index=-1, content="", render=True): | ||
self.add_cell(index, cell_type="markdown") | ||
self.edit_cell(index=index, content=content, render=render) | ||
|
||
|
||
@classmethod | ||
def new_notebook(cls, browser, kernel_name='kernel-python3'): | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Docstring! It should mention that this expects to be called with the browser on the dashboard page. You could use your other utilities to verify that and throw an error if it's not. |
||
# initial_window_handles = browser.window_handles | ||
with new_window(browser, selector=".cell"): | ||
select_kernel(browser, kernel_name=kernel_name) | ||
browser.execute_script("Jupyter.notebook.set_autosave_interval(0)") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is it OK that this is in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yea… I think you're right on that account. |
||
return cls(browser) | ||
|
||
|
||
def select_kernel(browser, kernel_name='kernel-python3'): | ||
"""Clicks the "new" button and selects a kernel from the options. | ||
""" | ||
new_button = wait_for_selector(browser, "#new-buttons", single=True) | ||
new_button.click() | ||
kernel_selector = '#{} a'.format(kernel_name) | ||
kernel = wait_for_selector(browser, kernel_selector, single=True) | ||
kernel.click() | ||
|
||
@contextmanager | ||
def new_window(browser, selector=None): | ||
"""Creates new window, switches you to that window, waits for selector if set. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Neat! The docstring needs some work, though - this doesn't create a new window, rather it expects that code called in its context will create a new window, and then it switches to that on leaving the context. This is worth explaining clearly, because most context managers do the interesting stuff when you create/enter them, and exiting just cleans something up. Here the interesting stuff is on exit. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think I took care of this one. |
||
|
||
""" | ||
initial_window_handles = browser.window_handles | ||
yield | ||
new_window_handle = next(window for window in browser.window_handles | ||
if window not in initial_window_handles) | ||
browser.switch_to_window(new_window_handle) | ||
if selector is not None: | ||
wait_for_selector(browser, selector) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's have a module level docstring to explain that these utilities are not used by the tests, but they're intended for interactive exploration while writing tests.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I had intended to add this.