Skip to content

Commit eaa9394

Browse files
committed
Enable bind mounting a subfolder of the home directory
- Resolve #1
1 parent ce04d23 commit eaa9394

File tree

5 files changed

+65
-25
lines changed

5 files changed

+65
-25
lines changed

CUDA.md

+3-6
Original file line numberDiff line numberDiff line change
@@ -106,15 +106,12 @@ docker run --rm \
106106
```
107107

108108
It will be *bind mounted* as the JupyterLab user's home directory and
109-
automatically populated on first run.
109+
automatically populated.
110+
:exclamation: *Bind mounting* a subfolder of the home directory is only possible
111+
for images with Python version ≥ 3.12.2.
110112

111113
### Run container
112114

113-
| :exclamation: Always mount the user's **entire** home directory.<br>Mounting a subfolder prevents the container from starting.[^1] |
114-
|:-----------------------------------------------------------------------------------------------------------------------------------|
115-
116-
[^1]: The only exception is the use case described at [Jupyter Docker Stacks > Quick Start > Example 2](https://github.com/jupyter/docker-stacks#quick-start).
117-
118115
self built:
119116

120117
```bash

README.md

+5-8
Original file line numberDiff line numberDiff line change
@@ -154,15 +154,12 @@ docker run --rm \
154154
```
155155

156156
It will be *bind mounted* as the JupyterLab user's home directory and
157-
automatically populated on first run.
157+
automatically populated.
158+
:exlamation: *Bind mounting* a subfolder of the home directory is only possible
159+
for images with Python version ≥ 3.12.2.
158160

159161
### Run container
160162

161-
| :exclamation: Always mount the user's **entire** home directory.<br>Mounting a subfolder prevents the container from starting.[^1] |
162-
|:-----------------------------------------------------------------------------------------------------------------------------------|
163-
164-
[^1]: The only exception is the use case described at [Jupyter Docker Stacks > Quick Start > Example 2](https://github.com/jupyter/docker-stacks#quick-start).
165-
166163
self built:
167164

168165
```bash
@@ -255,10 +252,10 @@ docker run -it --rm \
255252
256253
| Extension | Environment variable |
257254
|:--------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------|
258-
| GitHub Pull Requests and Issues | `GITHUB_TOKEN`: Personal access token with scopes `repo` and `user`.[^2] |
255+
| GitHub Pull Requests and Issues | `GITHUB_TOKEN`: Personal access token with scopes `repo` and `user`.[^1] |
259256
| GitLab Workflow | `GITLAB_WORKFLOW_INSTANCE_URL`: GitLab instance URL (e.g. https://gitlab.com).<br>`GITLAB_WORKFLOW_TOKEN`: Personal access token with scopes `api` and `read_user`. |
260257
261-
[^2]: *Device activation* may require a one-time login from the extension's sidebar.
258+
[^1]: *Device activation* may require a one-time login from the extension's sidebar.
262259
263260
## Similar projects
264261

base/latest.Dockerfile

+7
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,11 @@ RUN dpkgArch="$(dpkg --print-architecture)" \
177177
fi \
178178
## Add user
179179
&& useradd -l -m -s $(which zsh) -N -u ${NB_UID} ${NB_USER} \
180+
## Mark home directory as populated
181+
&& touch /home/${NB_USER}/.populated \
182+
&& chown ${NB_UID}:${NB_GID} /home/${NB_USER}/.populated \
183+
&& chmod go+w /home/${NB_USER}/.populated \
184+
## Create backup directory for home directory
180185
&& mkdir -p /var/backups/skel \
181186
&& chown ${NB_UID}:${NB_GID} /var/backups/skel \
182187
## Install Tini
@@ -277,6 +282,8 @@ RUN sh -c "$(curl -fsSL https://raw.githubusercontent.com/ohmyzsh/ohmyzsh/master
277282
&& echo "[[ ! -f ~/.p10k.zsh ]] || source ~/.p10k.zsh" >> ${HOME}/.zshrc \
278283
## Create user's private bin
279284
&& mkdir -p ${HOME}/.local/bin \
285+
## Record population timestamp
286+
&& date -uIseconds > ${HOME}/.populated \
280287
## Create backup of home directory
281288
&& cp -a ${HOME}/. /var/backups/skel
282289

base/scripts/usr/local/bin/start-notebook.d/10-populate.sh

+16-4
Original file line numberDiff line numberDiff line change
@@ -4,10 +4,22 @@
44

55
set -e
66

7-
if [ "$(ls -A "/home/jovyan" 2> /dev/null)" == "" ]; then
8-
_log "Populating home dir /home/jovyan..."
9-
if cp -a /var/backups/skel/. /home/jovyan; then
10-
_log "Success!"
7+
if [ ! -f /home/jovyan/.populated ]; then
8+
# Create list of missing files (top level only)
9+
fd="$(comm -13 <(cd /home/jovyan; ls -A) <(cd /var/backups/skel; ls -A) \
10+
| paste -sd ',' -)"
11+
# Handle case when only marker is missing
12+
if [[ "${fd}" == ".populated" ]]; then
13+
sf="${fd}"
14+
else
15+
sf="{${fd}}"
16+
fi
17+
_log "Populating home dir: /home/jovyan"
18+
_log "Copying files/directories (recursively):"
19+
_log "- ${fd}"
20+
if eval "cp -a /var/backups/skel/${sf} /home/jovyan"; then
21+
date -uIseconds > /home/jovyan/.populated
22+
_log "Done populating home dir"
1123
else
1224
_log "ERROR: Failed to copy data from /var/backups/skel to /home/jovyan!"
1325
exit 1

base/scripts/usr/local/bin/start.sh

+34-7
Original file line numberDiff line numberDiff line change
@@ -139,12 +139,23 @@ if [ "$(id -u)" == 0 ] ; then
139139
exit 1
140140
fi
141141
fi
142-
# The home directory could be bind mounted. Populate it if it is empty
143-
elif [[ "$(ls -A "/home/${NB_USER}${DOMAIN:+@$DOMAIN}" 2> /dev/null)" == "" ]]; then
144-
_log "Populating home dir /home/${NB_USER}${DOMAIN:+@$DOMAIN}..."
145-
# shellcheck disable=SC2086
146-
if cp ${CP_OPTS:--a} /home/jovyan/. "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/"; then
147-
_log "Success!"
142+
# The home directory could be bind mounted. Populate it.
143+
elif [[ ! -f "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/.populated" ]]; then
144+
# Create list of missing files (top level only)
145+
fd="$(comm -13 <(cd "/home/${NB_USER}${DOMAIN:+@$DOMAIN}"; ls -A) <(cd /home/jovyan; ls -A) \
146+
| paste -sd ',' -)"
147+
# Handle case when only marker is missing
148+
if [[ "${fd}" == ".populated" ]]; then
149+
sf="${fd}"
150+
else
151+
sf="{${fd}}"
152+
fi
153+
_log "Populating home dir: /home/${NB_USER}${DOMAIN:+@$DOMAIN}"
154+
_log "Copying files/directories (recursively):"
155+
_log "- ${fd}"
156+
if eval "cp ${CP_OPTS:--a} /home/jovyan/${sf} /home/${NB_USER}${DOMAIN:+@$DOMAIN}"; then
157+
date -uIseconds > "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/.populated"
158+
_log "Done populating home dir"
148159
else
149160
_log "ERROR: Failed to copy data from /home/jovyan to /home/${NB_USER}${DOMAIN:+@$DOMAIN}!"
150161
exit 1
@@ -159,10 +170,18 @@ if [ "$(id -u)" == 0 ] ; then
159170
fi
160171
fi
161172

173+
# Trigger changing ownership of the desired user's home folder
174+
if [[ "${CP_OPTS}" == "" && "$(stat -c '%u:%g' "/home/${NB_USER}${DOMAIN:+@$DOMAIN}")" != "$(id -u "${NB_USER}"):$(id -g "${NB_USER}")" ]]; then
175+
if [[ "${CHOWN_HOME}" != "1" && "${CHOWN_HOME}" != "yes" ]]; then
176+
export CHOWN_HOME="1"
177+
trg_msg="(Automatic) "
178+
fi
179+
fi
180+
162181
# Optionally ensure the desired user get filesystem ownership of it's home
163182
# folder and/or additional folders
164183
if [[ "${CHOWN_HOME}" == "1" || "${CHOWN_HOME}" == "yes" ]]; then
165-
_log "Ensuring /home/${NB_USER}${DOMAIN:+@$DOMAIN} is owned by ${NB_UID}:${NB_GID} ${CHOWN_HOME_OPTS:+(chown options: ${CHOWN_HOME_OPTS})}"
184+
_log "${trg_msg}Ensuring /home/${NB_USER}${DOMAIN:+@$DOMAIN} is owned by ${NB_UID}:${NB_GID} ${CHOWN_HOME_OPTS:+(chown options: ${CHOWN_HOME_OPTS})}"
166185
if [[ ! -L "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/mnt" || ("$(stat -c '%u:%g' "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/mnt")" != "$(id -u "${NB_USER}"):$(id -g "${NB_USER}")") ]]; then
167186
# shellcheck disable=SC2086
168187
chown ${CHOWN_HOME_OPTS} "${NB_UID}:${NB_GID}" "/home/${NB_USER}${DOMAIN:+@$DOMAIN}"
@@ -178,6 +197,14 @@ if [ "$(id -u)" == 0 ] ; then
178197
fi
179198
fi
180199
fi
200+
if [[ "${CP_OPTS}" == "" && "${sf}" != "" ]]; then
201+
if [[ "$(stat -c '%u:%g' "/home/${NB_USER}${DOMAIN:+@$DOMAIN}/.populated")" != "$(id -u "${NB_USER}"):$(id -g "${NB_USER}")" ]]; then
202+
_log "(Automatic) Ensuring the following files/directories in /home/${NB_USER}${DOMAIN:+@$DOMAIN} are owned by ${NB_UID}:${NB_GID} (chown options: -R):"
203+
_log "- ${fd}"
204+
# shellcheck disable=SC2086
205+
eval "chown -R ${NB_UID}:${NB_GID} /home/${NB_USER}${DOMAIN:+@$DOMAIN}/${sf}"
206+
fi
207+
fi
181208
if [ -n "${CHOWN_EXTRA}" ]; then
182209
for extra_dir in $(echo "${CHOWN_EXTRA}" | tr ',' ' '); do
183210
_log "Ensuring ${extra_dir} is owned by ${NB_UID}:${NB_GID} ${CHOWN_EXTRA_OPTS:+(chown options: ${CHOWN_EXTRA_OPTS})}"

0 commit comments

Comments
 (0)