Skip to content
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

Merged
merged 40 commits into from
Mar 28, 2018
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
9d4cf94
add utils module and wait_for_selector
mpacer Mar 20, 2018
7c659f5
add quick_selenium script for quickly starting up a selenium interact…
mpacer Mar 20, 2018
0994db2
Introduce Notebook class and give usage example in quick_selenium
mpacer Mar 20, 2018
f9dd7c1
add basic cell functionality to Notebook class
mpacer Mar 20, 2018
c9a8504
add body property
mpacer Mar 20, 2018
3d3086e
add edit_cell and execute_cell methods; remove focus_cell edit parameter
mpacer Mar 20, 2018
4b6a0a7
Add CellTypeError and convert_cell_type and wait_for_stale_cell methods
mpacer Mar 20, 2018
2b7f549
Add add_cell and add_markdown_cell methods
mpacer Mar 20, 2018
e2243d6
add method that allows you to close the notebook without confirming
mpacer Mar 22, 2018
86ae162
add first markdown test and a notebook fixture for new notebook
mpacer Mar 22, 2018
bf39dec
move sauce driver logic into isolated function, simplify selenium_driver
mpacer Mar 22, 2018
e0ed2c4
make authenticated browser module scope fixture for permission reasons
mpacer Mar 22, 2018
c220215
use wait_for_selector from utils module in test_dashboard_nav
mpacer Mar 22, 2018
3092800
add wait_for_selector to the cells property and to_command_mode
mpacer Mar 22, 2018
3615d4a
enhance wait_for_selector to handle returning single elements
mpacer Mar 23, 2018
d634f1d
add new_window contextmanager for creating, switching, & waiting on n…
mpacer Mar 23, 2018
a112ab6
add select_kernel function for clicking "new" & selecting a kernel
mpacer Mar 23, 2018
ebef7ba
create new_notebook classmethod creating/switching to new Notebook page
mpacer Mar 23, 2018
5d19785
small changes to naming things
mpacer Mar 23, 2018
3571f16
introduce cell __iter__, & __setitem__/__getitem__ for cells & indices
mpacer Mar 23, 2018
e183fc1
enrich signature for execute_cell to accept both index and cell directly
mpacer Mar 23, 2018
c16dac2
use new rich container __*__ methods to make adding & executing nicer
mpacer Mar 23, 2018
e9971a9
simplify __getitem__ __setitem__ to handle only ints/slices and ints
mpacer Mar 23, 2018
bf4868d
remove wait_for_selector call inside self.cells property
mpacer Mar 23, 2018
02e0ac3
fix add_cell logic, add content param, edit_cell default not render
mpacer Mar 23, 2018
98c09f8
use index method to get cell index in execute_cell
mpacer Mar 23, 2018
5e43458
add append, extend & coerce_to_cell methods
mpacer Mar 23, 2018
d9dd5d8
add run_all method
mpacer Mar 23, 2018
33ca649
add get_rendered_contents helper function
mpacer Mar 23, 2018
0999798
use new utilities, enable more markdown cells to be added easily
mpacer Mar 23, 2018
7808a89
make append actually append to the end of cell list
mpacer Mar 23, 2018
c081af5
add other markdown string conversion examples for test
mpacer Mar 23, 2018
d598ef5
switch fstring to format string
mpacer Mar 23, 2018
3c4596b
Take into account lab as a potential endpoint for the driver
mpacer Mar 23, 2018
79603b4
add quick_notebook utility and give docstring reminder to quit browsers
mpacer Mar 23, 2018
74af79c
Improve docstrings & comments; Remove unused code; Relocate script
mpacer Mar 23, 2018
515f8e2
nicer error and use append directly do not iterate over cells
mpacer Mar 27, 2018
c23ba2a
Docstrings and naming clarity
takluyver Mar 28, 2018
a6f604a
No need for copy
takluyver Mar 28, 2018
7266fd5
Remove unused variant of append for now
takluyver Mar 28, 2018
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
56 changes: 36 additions & 20 deletions notebook/tests/selenium/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@

pjoin = os.path.join


def _wait_for_server(proc, info_file_path):
"""Wait 30 seconds for the notebook server to start"""
for i in range(300):
Expand All @@ -28,6 +29,7 @@ def _wait_for_server(proc, info_file_path):
time.sleep(0.1)
raise RuntimeError("Didn't find %s in 30 seconds", info_file_path)


@pytest.fixture(scope='session')
def notebook_server():
info = {}
Expand All @@ -50,10 +52,11 @@ def notebook_server():
# run with a base URL that would be escaped,
# to test that we don't double-escape URLs
'--NotebookApp.base_url=/a@b/',
]
]
print("command=", command)
proc = info['popen'] = Popen(command, cwd=nbdir, env=env)
info_file_path = pjoin(td, 'jupyter_runtime', 'nbserver-%i.json' % proc.pid)
info_file_path = pjoin(td, 'jupyter_runtime',
'nbserver-%i.json' % proc.pid)
info.update(_wait_for_server(proc, info_file_path))

print("Notebook server info:", info)
Expand All @@ -63,26 +66,38 @@ def notebook_server():
requests.post(urljoin(info['url'], 'api/shutdown'),
headers={'Authorization': 'token '+info['token']})


def make_sauce_driver():
"""This function helps travis create a driver on Sauce Labs.

This function will err if used without specifying the variables expected
in that context.
"""

username = os.environ["SAUCE_USERNAME"]
access_key = os.environ["SAUCE_ACCESS_KEY"]
capabilities = {
"tunnel-identifier": os.environ["TRAVIS_JOB_NUMBER"],
"build": os.environ["TRAVIS_BUILD_NUMBER"],
"tags": [os.environ['TRAVIS_PYTHON_VERSION'], 'CI'],
"platform": "Windows 10",
"browserName": os.environ['JUPYTER_TEST_BROWSER'],
"version": "latest",
}
if capabilities['browserName'] == 'firefox':
# Attempt to work around issue where browser loses authentication
capabilities['version'] = '57.0'
hub_url = "%s:%s@localhost:4445" % (username, access_key)
print("Connecting remote driver on Sauce Labs")
driver = Remote(desired_capabilities=capabilities,
command_executor="http://%s/wd/hub" % hub_url)
return driver


@pytest.fixture(scope='session')
def selenium_driver():
if os.environ.get('SAUCE_USERNAME'):
username = os.environ["SAUCE_USERNAME"]
access_key = os.environ["SAUCE_ACCESS_KEY"]
capabilities = {
"tunnel-identifier": os.environ["TRAVIS_JOB_NUMBER"],
"build": os.environ["TRAVIS_BUILD_NUMBER"],
"tags": [os.environ['TRAVIS_PYTHON_VERSION'], 'CI'],
"platform": "Windows 10",
"browserName": os.environ['JUPYTER_TEST_BROWSER'],
"version": "latest",
}
if capabilities['browserName'] == 'firefox':
# Attempt to work around issue where browser loses authentication
capabilities['version'] = '57.0'
hub_url = "%s:%s@localhost:4445" % (username, access_key)
print("Connecting remote driver on Sauce Labs")
driver = Remote(desired_capabilities=capabilities,
command_executor="http://%s/wd/hub" % hub_url)
driver = make_sauce_driver()
elif os.environ.get('JUPYTER_TEST_BROWSER') == 'chrome':
driver = Chrome()
else:
Expand All @@ -93,7 +108,8 @@ def selenium_driver():
# Teardown
driver.quit()

@pytest.fixture

@pytest.fixture(scope='module')
def authenticated_browser(selenium_driver, notebook_server):
selenium_driver.jupyter_server_info = notebook_server
selenium_driver.get("{url}?token={token}".format(**notebook_server))
Expand Down
22 changes: 22 additions & 0 deletions notebook/tests/selenium/quick_selenium.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
from selenium.webdriver import Firefox
Copy link
Member

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.

Copy link
Member Author

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.

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"]}'
Copy link
Member

Choose a reason for hiding this comment

The 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 .format() method for now.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

driver.get(auth_url)
return driver
8 changes: 1 addition & 7 deletions notebook/tests/selenium/test_dashboard_nav.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
from selenium.webdriver.support import expected_conditions as EC

from notebook.utils import url_path_join
from notebook.tests.selenium.utils import wait_for_selector
pjoin = os.path.join


Expand Down Expand Up @@ -40,7 +41,6 @@ def get_list_items(browser):
'element': a,
} for a in browser.find_elements_by_class_name('item_link')]


def only_dir_links(browser):
"""Return only links that point at other directories in the tree

Expand All @@ -49,12 +49,6 @@ def only_dir_links(browser):
return [i for i in items
if url_in_tree(browser, i['link']) and i['label'] != '..']


def wait_for_selector(browser, selector, timeout=10):
wait = WebDriverWait(browser, timeout)
return wait.until(EC.visibility_of_element_located((By.CSS_SELECTOR, selector)))


def test_items(authenticated_browser):
visited_dict = {}
# Going down the tree to collect links
Expand Down
26 changes: 26 additions & 0 deletions notebook/tests/selenium/test_markdown.py
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
163 changes: 163 additions & 0 deletions notebook/tests/selenium/utils.py
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
Copy link
Member

Choose a reason for hiding this comment

The 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.

Copy link
Member Author

Choose a reason for hiding this comment

The 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")
Copy link
Member

Choose a reason for hiding this comment

The 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.

Copy link
Member Author

Choose a reason for hiding this comment

The 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):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

'safety' is rather vague. remove_unsaved_check, maybe? Or base it on the hook name: remove_onbeforeunload.

Copy link
Member

Choose a reason for hiding this comment

The 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'):
Copy link
Member

Choose a reason for hiding this comment

The 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)")
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it OK that this is in new_notebook() but remove_safety_check() is called from __init__? I'd expect they'd go together.

Copy link
Member Author

Choose a reason for hiding this comment

The 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.
Copy link
Member

Choose a reason for hiding this comment

The 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.

Copy link
Member Author

Choose a reason for hiding this comment

The 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)