Skip to content

Commit bceaead

Browse files
yuvipandamathbunnyru
andauthoredOct 17, 2023
Migrate start-notebook & start-singleuser to python (#2006)
* Migrate start-notebook.sh to bash Based on > Stop using bash, haha 👍 from #1532. If there's more apetite for this, I'll try to migrate `start.sh` and `start-singleuser.sh` as well - I think they should all be merged together. We can remove the `.sh` suffixes for accuracy, and keep symlinks in so old config still works. Since the shebang is what is used to launch the correct interpreter, the `.sh` doesn't matter. Will help fix #1532, as I believe all those things are going to be easier to do from python than bash * Rename start-notebook.sh to start-notebook * Cleanup start-notebook a little * Fix typo * Migrate start-singleuser as well * Remove unused import * Run symlink commands as root * Combine repetitive RUN commands * Remove multiple args to env -u can not be set by shebang, we must set the env var instead * Fix conditional inversion Co-authored-by: Ayaz Salikhov <mathbunnyru@users.noreply.github.com> * Fix how start-singleuser is exec'd * Actually call jupyterhub-singleuser in start-singleuser * Pass through any additional args we get * Put .py suffix on the start-* scripts * Add .sh shims for the start-* scripts * Document start-notebook.sh and start-singleuser.sh * Partially test start-notebook.sh * Reflow warning docs Co-authored-by: Ayaz Salikhov <mathbunnyru@users.noreply.github.com> --------- Co-authored-by: Ayaz Salikhov <mathbunnyru@users.noreply.github.com>
1 parent 4c0c0aa commit bceaead

20 files changed

+117
-70
lines changed
 

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ system when the container exits, but any changes made to the `~/work` directory
6565
By default, [jupyter's root_dir](https://jupyter-server.readthedocs.io/en/latest/other/full-config.html) is `/home/jovyan`.
6666
So, new notebooks will be saved there, unless you change the directory in the file browser.
6767
68-
To change the default directory, you will need to specify `ServerApp.root_dir` by adding this line to previous command: `start-notebook.sh --ServerApp.root_dir=/home/jovyan/work`.
68+
To change the default directory, you will need to specify `ServerApp.root_dir` by adding this line to previous command: `start-notebook.py --ServerApp.root_dir=/home/jovyan/work`.
6969
```
7070

7171
## Contributing

‎docs/using/common.md

+9-9
Original file line numberDiff line numberDiff line change
@@ -1,34 +1,34 @@
11
# Common Features
22

33
Except for `jupyter/docker-stacks-foundation`, a container launched from any Jupyter Docker Stacks image runs a Jupyter Server with the JupyterLab frontend.
4-
The container does so by executing a `start-notebook.sh` script.
4+
The container does so by executing a `start-notebook.py` script.
55
This script configures the internal container environment and then runs `jupyter lab`, passing any command-line arguments received.
66

77
This page describes the options supported by the startup script and how to bypass it to run alternative commands.
88

99
## Jupyter Server Options
1010

11-
You can pass [Jupyter Server options](https://jupyter-server.readthedocs.io/en/latest/operators/public-server.html) to the `start-notebook.sh` script when launching the container.
11+
You can pass [Jupyter Server options](https://jupyter-server.readthedocs.io/en/latest/operators/public-server.html) to the `start-notebook.py` script when launching the container.
1212

1313
1. For example, to secure the Jupyter Server with a [custom password](https://jupyter-server.readthedocs.io/en/latest/operators/public-server.html#preparing-a-hashed-password)
1414
hashed using `jupyter_server.auth.passwd()` instead of the default token,
1515
you can run the following (this hash was generated for the `my-password` password):
1616

1717
```bash
1818
docker run -it --rm -p 8888:8888 jupyter/base-notebook \
19-
start-notebook.sh --PasswordIdentityProvider.hashed_password='argon2:$argon2id$v=19$m=10240,t=10,p=8$JdAN3fe9J45NvK/EPuGCvA$O/tbxglbwRpOFuBNTYrymAEH6370Q2z+eS1eF4GM6Do'
19+
start-notebook.py --PasswordIdentityProvider.hashed_password='argon2:$argon2id$v=19$m=10240,t=10,p=8$JdAN3fe9J45NvK/EPuGCvA$O/tbxglbwRpOFuBNTYrymAEH6370Q2z+eS1eF4GM6Do'
2020
```
2121

2222
2. To set the [base URL](https://jupyter-server.readthedocs.io/en/latest/operators/public-server.html#running-the-notebook-with-a-customized-url-prefix) of the Jupyter Server, you can run the following:
2323

2424
```bash
2525
docker run -it --rm -p 8888:8888 jupyter/base-notebook \
26-
start-notebook.sh --ServerApp.base_url=/customized/url/prefix/
26+
start-notebook.py --ServerApp.base_url=/customized/url/prefix/
2727
```
2828

2929
## Docker Options
3030

31-
You may instruct the `start-notebook.sh` script to customize the container environment before launching the Server.
31+
You may instruct the `start-notebook.py` script to customize the container environment before launching the Server.
3232
You do so by passing arguments to the `docker run` command.
3333

3434
### User-related configurations
@@ -104,7 +104,7 @@ You do so by passing arguments to the `docker run` command.
104104
You do **not** need this option to allow the user to `conda` or `pip` install additional packages.
105105
This option is helpful for cases when you wish to give `${NB_USER}` the ability to install OS packages with `apt` or modify other root-owned files in the container.
106106
You **must** run the container with `--user root` for this option to take effect.
107-
(The `start-notebook.sh` script will `su ${NB_USER}` after adding `${NB_USER}` to sudoers.)
107+
(The `start-notebook.py` script will `su ${NB_USER}` after adding `${NB_USER}` to sudoers.)
108108
**You should only enable `sudo` if you trust the user or if the container runs on an isolated host.**
109109

110110
### Additional runtime configurations
@@ -147,7 +147,7 @@ For example, to mount a host folder containing a `notebook.key` and `notebook.cr
147147
docker run -it --rm -p 8888:8888 \
148148
-v /some/host/folder:/etc/ssl/notebook \
149149
jupyter/base-notebook \
150-
start-notebook.sh \
150+
start-notebook.py \
151151
--ServerApp.keyfile=/etc/ssl/notebook/notebook.key \
152152
--ServerApp.certfile=/etc/ssl/notebook/notebook.crt
153153
```
@@ -159,7 +159,7 @@ For example:
159159
docker run -it --rm -p 8888:8888 \
160160
-v /some/host/folder/notebook.pem:/etc/ssl/notebook.pem \
161161
jupyter/base-notebook \
162-
start-notebook.sh \
162+
start-notebook.py \
163163
--ServerApp.certfile=/etc/ssl/notebook.pem
164164
```
165165

@@ -220,7 +220,7 @@ docker run -it --rm \
220220

221221
### `start.sh`
222222

223-
The `start-notebook.sh` script inherits most of its option handling capability from a more generic `start.sh` script.
223+
The `start-notebook.py` script inherits most of its option handling capability from a more generic `start.sh` script.
224224
The `start.sh` script supports all the features described above but allows you to specify an arbitrary command to execute.
225225
For example, to run the text-based `ipython` console in a container, do the following:
226226

‎docs/using/recipes.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -375,14 +375,14 @@ Credit: [britishbadger](https://github.com/britishbadger) from [docker-stacks/is
375375
The default security is very good.
376376
There are use cases, encouraged by containers, where the jupyter container and the system it runs within lie inside the security boundary.
377377
It is convenient to launch the server without a password or token in these use cases.
378-
In this case, you should use the `start-notebook.sh` script to launch the server with no token:
378+
In this case, you should use the `start-notebook.py` script to launch the server with no token:
379379

380380
For JupyterLab:
381381

382382
```bash
383383
docker run -it --rm \
384384
jupyter/base-notebook \
385-
start-notebook.sh --IdentityProvider.token=''
385+
start-notebook.py --IdentityProvider.token=''
386386
```
387387

388388
For Jupyter Notebook:
@@ -391,7 +391,7 @@ For Jupyter Notebook:
391391
docker run -it --rm \
392392
-e DOCKER_STACKS_JUPYTER_CMD=notebook \
393393
jupyter/base-notebook \
394-
start-notebook.sh --IdentityProvider.token=''
394+
start-notebook.py --IdentityProvider.token=''
395395
```
396396

397397
## Enable nbclassic-extension spellchecker for markdown (or any other nbclassic-extension)

‎docs/using/running.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ Any other changes made in the container will be lost.
6969
By default, [jupyter's root_dir](https://jupyter-server.readthedocs.io/en/latest/other/full-config.html) is `/home/jovyan`.
7070
So, new notebooks will be saved there, unless you change the directory in the file browser.
7171
72-
To change the default directory, you will need to specify `ServerApp.root_dir` by adding this line to previous command: `start-notebook.sh --ServerApp.root_dir=/home/jovyan/work`.
72+
To change the default directory, you will need to specify `ServerApp.root_dir` by adding this line to previous command: `start-notebook.py --ServerApp.root_dir=/home/jovyan/work`.
7373
```
7474

7575
### Example 3

‎docs/using/selecting.md

+9-2
Original file line numberDiff line numberDiff line change
@@ -56,10 +56,17 @@ It contains:
5656
- Everything in `jupyter/docker-stacks-foundation`
5757
- Minimally functional Server (e.g., no LaTeX support for saving notebooks as PDFs)
5858
- `notebook`, `jupyterhub` and `jupyterlab` packages
59-
- A `start-notebook.sh` script as the default command
60-
- A `start-singleuser.sh` script useful for launching containers in JupyterHub
59+
- A `start-notebook.py` script as the default command
60+
- A `start-singleuser.py` script useful for launching containers in JupyterHub
6161
- Options for a self-signed HTTPS certificate
6262

63+
```{warning}
64+
`jupyter/base-notebook` also contains `start-notebook.sh` and `start-singleuser.sh` files to maintain backwards compatibility.
65+
External config that explicitly refers to those files should instead
66+
update to refer to `start-notebook.py` and `start-singleuser.py`.
67+
The shim `.sh` files will be removed at some future date.
68+
```
69+
6370
### jupyter/minimal-notebook
6471

6572
[Source on GitHub](https://github.com/jupyter/docker-stacks/tree/main/images/minimal-notebook) |

‎examples/docker-compose/notebook/letsencrypt-notebook.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ services:
1818
USE_HTTPS: "yes"
1919
PASSWORD: ${PASSWORD}
2020
command: >
21-
start-notebook.sh
21+
start-notebook.py
2222
--ServerApp.certfile=/etc/letsencrypt/fullchain.pem
2323
--ServerApp.keyfile=/etc/letsencrypt/privkey.pem
2424

‎examples/make-deploy/Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ define RUN_NOTEBOOK
1313
--name $(NAME) \
1414
-v $(WORK_VOLUME):/home/jovyan/work \
1515
$(DOCKER_ARGS) \
16-
$(IMAGE) bash -c "$(PRE_CMD) chown jovyan /home/jovyan/work && start-notebook.sh $(ARGS)" > /dev/null
16+
$(IMAGE) bash -c "$(PRE_CMD) chown jovyan /home/jovyan/work && start-notebook.py $(ARGS)" > /dev/null
1717
@echo "DONE: Notebook '$(NAME)' listening on $$(docker-machine ip $$(docker-machine active)):$(PORT)"
1818
endef
1919

‎examples/openshift/templates.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@
8080
"name": "jupyter-notebook",
8181
"image": "${NOTEBOOK_IMAGE}",
8282
"command": [
83-
"start-notebook.sh",
83+
"start-notebook.py",
8484
"--config=/etc/jupyter/openshift/jupyter_server_config.py",
8585
"--no-browser",
8686
"--ip=0.0.0.0"

‎examples/source-to-image/README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -117,7 +117,7 @@ with the extra system packages, and then use that image with the S2I build to co
117117
The `run` script in this directory is very simple and just runs the notebook application.
118118

119119
```bash
120-
exec start-notebook.sh "$@"
120+
exec start-notebook.py "$@"
121121
```
122122

123123
## Integration with OpenShift

‎examples/source-to-image/run

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,4 @@
22

33
# Start up the notebook instance.
44

5-
exec start-notebook.sh "$@"
5+
exec start-notebook.py "$@"

‎examples/source-to-image/templates.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -274,7 +274,7 @@
274274
"name": "jupyter-notebook",
275275
"image": "${APPLICATION_NAME}:latest",
276276
"command": [
277-
"start-notebook.sh",
277+
"start-notebook.py",
278278
"--config=/etc/jupyter/openshift/jupyter_server_config.py",
279279
"--no-browser",
280280
"--ip=0.0.0.0"

‎images/base-notebook/Dockerfile

+2-2
Original file line numberDiff line numberDiff line change
@@ -52,10 +52,10 @@ ENV JUPYTER_PORT=8888
5252
EXPOSE $JUPYTER_PORT
5353

5454
# Configure container startup
55-
CMD ["start-notebook.sh"]
55+
CMD ["start-notebook.py"]
5656

5757
# Copy local files as late as possible to avoid cache busting
58-
COPY start-notebook.sh start-singleuser.sh /usr/local/bin/
58+
COPY start-notebook.py start-notebook.sh start-singleuser.py start-singleuser.sh /usr/local/bin/
5959
COPY jupyter_server_config.py docker_healthcheck.py /etc/jupyter/
6060

6161
# Fix permissions on /etc/jupyter as root
+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
#!/usr/bin/env python
2+
# Copyright (c) Jupyter Development Team.
3+
# Distributed under the terms of the Modified BSD License.
4+
import os
5+
import shlex
6+
import sys
7+
8+
# If we are in a JupyterHub, we pass on to `start-singleuser.py` instead so it does the right thing
9+
if "JUPYTERHUB_API_TOKEN" in os.environ:
10+
print(
11+
"WARNING: using start-singleuser.py instead of start-notebook.py to start a server associated with JupyterHub."
12+
)
13+
command = ["/usr/local/bin/start-singleuser.py"] + sys.argv[1:]
14+
os.execvp(command[0], command)
15+
16+
17+
# Wrap everything in start.sh, no matter what
18+
command = ["/usr/local/bin/start.sh"]
19+
20+
# If we want to survive restarts, tell that to start.sh
21+
if os.environ.get("RESTARTABLE") == "yes":
22+
command.append("run-one-constantly")
23+
24+
# We always launch a jupyter subcommand from this script
25+
command.append("jupyter")
26+
27+
# Launch the configured subcommand. Note that this should be a single string, so we don't split it
28+
# We default to lab
29+
jupyter_command = os.environ.get("DOCKER_STACKS_JUPYTER_CMD", "lab")
30+
command.append(jupyter_command)
31+
32+
# Append any optional NOTEBOOK_ARGS we were passed in. This is supposed to be multiple args passed
33+
# on to the notebook command, so we split it correctly with shlex
34+
if "NOTEBOOK_ARGS" in os.environ:
35+
command += shlex.split(os.environ["NOTEBOOK_ARGS"])
36+
37+
# Pass through any other args we were passed on the commandline
38+
command += sys.argv[1:]
39+
40+
# Execute the command!
41+
os.execvp(command[0], command)
+3-20
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,5 @@
11
#!/bin/bash
2-
# Copyright (c) Jupyter Development Team.
3-
# Distributed under the terms of the Modified BSD License.
2+
# Shim to emit warning and call start-notebook.py
3+
echo "WARNING: Use start-notebook.py instead"
44

5-
set -e
6-
7-
# The Jupyter command to launch
8-
# JupyterLab by default
9-
DOCKER_STACKS_JUPYTER_CMD="${DOCKER_STACKS_JUPYTER_CMD:=lab}"
10-
11-
if [[ -n "${JUPYTERHUB_API_TOKEN}" ]]; then
12-
echo "WARNING: using start-singleuser.sh instead of start-notebook.sh to start a server associated with JupyterHub."
13-
exec /usr/local/bin/start-singleuser.sh "$@"
14-
fi
15-
16-
wrapper=""
17-
if [[ "${RESTARTABLE}" == "yes" ]]; then
18-
wrapper="run-one-constantly"
19-
fi
20-
21-
# shellcheck disable=SC1091,SC2086
22-
exec /usr/local/bin/start.sh ${wrapper} jupyter ${DOCKER_STACKS_JUPYTER_CMD} ${NOTEBOOK_ARGS} "$@"
5+
exec /usr/local/bin/start-notebook.py "$@"
+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
#!/usr/bin/env python
2+
# Copyright (c) Jupyter Development Team.
3+
# Distributed under the terms of the Modified BSD License.
4+
import os
5+
import shlex
6+
import sys
7+
8+
command = ["/usr/local/bin/start.sh", "jupyterhub-singleuser"]
9+
10+
# set default ip to 0.0.0.0
11+
if "--ip=" not in os.environ.get("NOTEBOOK_ARGS", ""):
12+
command.append("--ip=0.0.0.0")
13+
14+
# Append any optional NOTEBOOK_ARGS we were passed in. This is supposed to be multiple args passed
15+
# on to the notebook command, so we split it correctly with shlex
16+
if "NOTEBOOK_ARGS" in os.environ:
17+
command += shlex.split(os.environ["NOTEBOOK_ARGS"])
18+
19+
# Pass any other args we have been passed through
20+
command += sys.argv[1:]
21+
22+
# Execute the command!
23+
os.execvp(command[0], command)
+3-11
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,5 @@
11
#!/bin/bash
2-
# Copyright (c) Jupyter Development Team.
3-
# Distributed under the terms of the Modified BSD License.
2+
# Shim to emit warning and call start-singleuser.py
3+
echo "WARNING: Use start-singleuser.py instead"
44

5-
set -e
6-
7-
# set default ip to 0.0.0.0
8-
if [[ "${NOTEBOOK_ARGS} $*" != *"--ip="* ]]; then
9-
NOTEBOOK_ARGS="--ip=0.0.0.0 ${NOTEBOOK_ARGS}"
10-
fi
11-
12-
# shellcheck disable=SC1091,SC2086
13-
. /usr/local/bin/start.sh jupyterhub-singleuser ${NOTEBOOK_ARGS} "$@"
5+
exec /usr/local/bin/start-singleuser.py "$@"

‎tests/base-notebook/test_container_options.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ def test_cli_args(container: TrackedContainer, http_client: requests.Session) ->
1515
"""Image should respect command line args (e.g., disabling token security)"""
1616
host_port = find_free_port()
1717
running_container = container.run_detached(
18-
command=["start-notebook.sh", "--IdentityProvider.token=''"],
18+
command=["start-notebook.py", "--IdentityProvider.token=''"],
1919
ports={"8888/tcp": host_port},
2020
)
2121
resp = http_client.get(f"http://localhost:{host_port}")
@@ -102,7 +102,7 @@ def test_custom_internal_port(
102102
host_port = find_free_port()
103103
internal_port = env.get("JUPYTER_PORT", 8888)
104104
running_container = container.run_detached(
105-
command=["start-notebook.sh", "--IdentityProvider.token=''"],
105+
command=["start-notebook.py", "--IdentityProvider.token=''"],
106106
environment=env,
107107
ports={internal_port: host_port},
108108
)

‎tests/base-notebook/test_healthcheck.py

+10-9
Original file line numberDiff line numberDiff line change
@@ -22,23 +22,24 @@
2222
(["RESTARTABLE=yes"], None, None),
2323
(["JUPYTER_PORT=8171"], None, None),
2424
(["JUPYTER_PORT=8117", "DOCKER_STACKS_JUPYTER_CMD=notebook"], None, None),
25-
(None, ["start-notebook.sh", "--ServerApp.base_url=/test"], None),
26-
(None, ["start-notebook.sh", "--ServerApp.base_url=/test/"], None),
27-
(["GEN_CERT=1"], ["start-notebook.sh", "--ServerApp.base_url=/test"], None),
25+
(None, ["start-notebook.sh"], None),
26+
(None, ["start-notebook.py", "--ServerApp.base_url=/test"], None),
27+
(None, ["start-notebook.py", "--ServerApp.base_url=/test/"], None),
28+
(["GEN_CERT=1"], ["start-notebook.py", "--ServerApp.base_url=/test"], None),
2829
(
2930
["GEN_CERT=1", "JUPYTER_PORT=7891"],
30-
["start-notebook.sh", "--ServerApp.base_url=/test"],
31+
["start-notebook.py", "--ServerApp.base_url=/test"],
3132
None,
3233
),
3334
(["NB_USER=testuser", "CHOWN_HOME=1"], None, "root"),
3435
(
3536
["NB_USER=testuser", "CHOWN_HOME=1"],
36-
["start-notebook.sh", "--ServerApp.base_url=/test"],
37+
["start-notebook.py", "--ServerApp.base_url=/test"],
3738
"root",
3839
),
3940
(
4041
["NB_USER=testuser", "CHOWN_HOME=1", "JUPYTER_PORT=8123"],
41-
["start-notebook.sh", "--ServerApp.base_url=/test"],
42+
["start-notebook.py", "--ServerApp.base_url=/test"],
4243
"root",
4344
),
4445
],
@@ -85,7 +86,7 @@ def test_health(
8586
"HTTPS_PROXY=host.docker.internal",
8687
"HTTP_PROXY=host.docker.internal",
8788
],
88-
["start-notebook.sh", "--ServerApp.base_url=/test"],
89+
["start-notebook.py", "--ServerApp.base_url=/test"],
8990
"root",
9091
),
9192
],
@@ -122,12 +123,12 @@ def test_health_proxy(
122123
(["NB_USER=testuser", "CHOWN_HOME=1"], None, None),
123124
(
124125
["NB_USER=testuser", "CHOWN_HOME=1"],
125-
["start-notebook.sh", "--ServerApp.base_url=/test"],
126+
["start-notebook.py", "--ServerApp.base_url=/test"],
126127
None,
127128
),
128129
(
129130
["NB_USER=testuser", "CHOWN_HOME=1", "JUPYTER_PORT=8123"],
130-
["start-notebook.sh", "--ServerApp.base_url=/test"],
131+
["start-notebook.py", "--ServerApp.base_url=/test"],
131132
None,
132133
),
133134
],

‎tests/base-notebook/test_start_container.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525
["JUPYTERHUB_API_TOKEN=my_token"],
2626
"jupyterhub-singleuser",
2727
False,
28-
["WARNING: using start-singleuser.sh"],
28+
["WARNING: using start-singleuser.py"],
2929
),
3030
],
3131
)
@@ -37,9 +37,9 @@ def test_start_notebook(
3737
expected_start: bool,
3838
expected_warnings: list[str],
3939
) -> None:
40-
"""Test the notebook start-notebook script"""
40+
"""Test the notebook start-notebook.py script"""
4141
LOGGER.info(
42-
f"Test that the start-notebook launches the {expected_command} server from the env {env} ..."
42+
f"Test that the start-notebook.py launches the {expected_command} server from the env {env} ..."
4343
)
4444
host_port = find_free_port()
4545
running_container = container.run_detached(

‎tests/pluto_check.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ def check_pluto_proxy(
1818
token = secrets.token_hex()
1919
container.run_detached(
2020
command=[
21-
"start-notebook.sh",
21+
"start-notebook.py",
2222
f"--IdentityProvider.token={token}",
2323
],
2424
ports={"8888/tcp": host_port},

0 commit comments

Comments
 (0)
Please sign in to comment.