Skip to content

Commit 3df1290

Browse files
authored
Merge pull request #653 from plotly/jupyter-test-docs
Add docs for Jupyter NB test suite
2 parents 8fdd412 + 57a4005 commit 3df1290

File tree

2 files changed

+113
-0
lines changed

2 files changed

+113
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,106 @@
1+
# plotly.py Jupyter Notebook JS test suite
2+
3+
To make sure that plotly.py works properly inside Jupyter Notebooks in both
4+
`plotly.plotly` and `plotly.offline`, we here run JavaScript tests in a browser.
5+
6+
See PR [#545](https://github.com/plotly/plotly.py/pull/545) for example of what
7+
can go wrong in Jupyter Notebooks.
8+
9+
10+
## Running the tests
11+
12+
Install JavaScript dependencies:
13+
14+
```bash
15+
(plotly.py/plotly/tests/test_optional/test_jupyter) $ npm install
16+
```
17+
18+
Run the tests:
19+
20+
```bash
21+
(plotly.py/plotly/tests/test_optional/test_jupyter) $ nosetests
22+
23+
# or from the repo root
24+
(plotly.py) $ nosetests plotly/tests/test_optional/test_jupyter
25+
```
26+
27+
## Add a test case
28+
29+
- Open up `jupyter notebook`.
30+
31+
- Add a few code cells testing what you wish to test. No need to execute them!
32+
33+
- Save the resulting `.ipynb` file in `fixtures/` (e.g. `my_test.ipynb`)
34+
35+
- Add a JavaScript test file in `js_tests/` by first requiring the in-house
36+
[`tape`](https://github.com/substack/tape) test wrapper found in `lib/tape-wrapper.js`.
37+
38+
For example,
39+
40+
```js
41+
var test = require('../lib/tape-wrapper');
42+
43+
test('should have one plotly.js graph', function(t) {
44+
t.plan(1);
45+
46+
var nodes = document.querySelectorAll('.js-plotly-plot');
47+
t.equal(nodes.length, 1);
48+
});
49+
```
50+
51+
asserts that one plotly graph is present in the notebook.
52+
53+
At the moment, it is highly recommended that the `js` test file has the same name
54+
(minus the extension) as the `.ipynb` fixture (i.e. `my_test.js` in this
55+
example).
56+
57+
- Add a test case in `test_jupyter.py`. If both the fixture and `js` test file
58+
have the same name, simply add:
59+
60+
```py
61+
class MyTest(Common):
62+
__test__ = True
63+
name = 'my_test'
64+
```
65+
66+
to `test_jupyter.py` and you're done :beers:
67+
68+
69+
## How does this work?
70+
71+
The `Common` test class in `test_jupyter.py`:
72+
73+
- Loads up a given `.ipynb` fixture
74+
75+
- Executes all code cells and converts them to HTML using the
76+
[`nbconvert`](https://nbconvert.readthedocs.io/en/latest/) and
77+
[`ipykernel`](http://ipython.readthedocs.io/en/stable/install/kernel_install.html)
78+
modules.
79+
80+
- Saves the resulting HTML file is saved in `fixtures/` but is git-ignored.
81+
82+
Then, `Common` runs an `npm` command in the shell:
83+
84+
```bash
85+
npm test -- <path-to-html-fixture> <path-to-js-test-file>
86+
```
87+
88+
which runs a minimal server using `lib/server.js`.
89+
90+
In details, `lib/server.js`:
91+
92+
- bundles up the `js` test code into a single bundle using
93+
[`browserify`](https://github.com/substack/node-browserify)
94+
95+
- stubs in a `<script>` tag to include the `js` test bundle in the fixture HTML
96+
97+
- starts a simple http server
98+
99+
- launches Chrome at the HTML fixture URL
100+
101+
- once the page is loaded, the `js` tests are run and results are logged in the
102+
terminal
103+
104+
105+
See PR [#549](https://github.com/plotly/plotly.py/pull/549) for the details on
106+
how this suite was first implemented.

plotly/tests/test_optional/test_jupyter/test_jupyter.py

+7
Original file line numberDiff line numberDiff line change
@@ -12,10 +12,17 @@
1212
import subprocess
1313

1414
PATH_ROOT = path.dirname(__file__)
15+
PATH_NODE_MODULES = path.join(PATH_ROOT, 'node_modules')
1516
PATH_FIXTURES = path.join(PATH_ROOT, 'fixtures')
1617
PATH_JS_TESTS = path.join(PATH_ROOT, 'js_tests')
1718

1819

20+
class PlotlyJupyterTestDeps(TestCase):
21+
22+
def test_node_modules(self):
23+
self.assertTrue(path.isdir(PATH_NODE_MODULES))
24+
25+
1926
class Common(TestCase):
2027
__test__ = False
2128
name = None

0 commit comments

Comments
 (0)