JupyterHub URL scheme#
This document describes how JupyterHub routes requests.
This does not include the REST API URLs.
In general, all URLs can be prefixed with
run the whole JupyterHub application on a prefix.
All authenticated handlers redirect to
/hub/login to log-in users
before being redirected back to the originating page.
The returned request should preserve all query parameters.
The top-level request is always a simple redirect to
to be handled by the default JupyterHub handler.
In general, all requests to
/anything that do not start with
but are routed to the Hub, will be redirected to
/hub/anything before being handled by the Hub.
This is an authenticated URL.
This handler redirects users to the default URL of the application,
which defaults to the user’s default server.
That is, the handler redirects to
/hub/spawn if the user’s server is not running,
or to the server itself (
/user/:name) if the server is running.
This default URL behavior can be customized in two ways:
First, to redirect users to the JupyterHub home page (
instead of spawning their server,
redirect_to_server to False:
c.JupyterHub.redirect_to_server = False
This might be useful if you have a Hub where you expect users to be managing multiple server configurations but automatic spawning is not desirable.
Second, you can customise the landing page to any page you like, such as a custom service you have deployed e.g. with course information:
c.JupyterHub.default_url = '/services/my-landing-service'
By default, the Hub home page has just one or two buttons for starting and stopping the user’s server.
If named servers are enabled, there will be some additional tools for management of the named servers.
Version added: 1.0 named server UI is new in 1.0.
This is the JupyterHub login page. If you have a form-based username+password login, such as the default PAMAuthenticator, this page will render the login form.
If login is handled by an external service, e.g. with OAuth, this page will have a button, declaring “Log in with …” which users can click to log in with the chosen service.
If you want to skip the user interaction and initiate login via the button, you can set:
c.Authenticator.auto_login = True
This can be useful when the user is “already logged in” via some mechanism.
However, a handshake via
redirects is necessary to complete the authentication with JupyterHub.
/hub/logout clears cookies from the current browser.
Note that logging out does not stop a user’s server(s) by default.
If you would like to shut down user servers on logout, you can enable this behavior with:
c.JupyterHub.shutdown_on_logout = True
Be careful with this setting because logging out one browser does not mean the user is no longer actively using their server from another machine.
If a user’s server is running, this URL is handled by the user’s given server, not by the Hub. The username is the first part, and if using named servers, the server name is the second part.
If the user’s server is not running, this will be redirected to
This URL indicates a request for a user server that is not running
/user/... would have been handled by the notebook server
if the specified server were running).
Handling this URL depends on two conditions: whether a requested user is found as a match and the state of the requested user’s notebook server, for example:
the server is not active a. user matches b. user doesn’t match
the server is ready
the server is pending, but not ready
If the server is pending spawn,
the browser will be redirected to
to see a progress page while waiting for the server to be ready.
If the server is not active at all,
a page will be served with a link to
Following that link will launch the requested server.
The HTTP status will be 503 in this case because a request has been made for a server that is not running.
If the server is ready, it is assumed that the proxy has not yet registered the route.
Some checks are performed and a delay is added before redirecting back to
If something is really wrong, this can result in a redirect loop.
Visiting this page will never result in triggering the spawn of servers without additional user action (i.e. clicking the link on the page).
Version changed: 1.0
Prior to 1.0, this URL itself was responsible for spawning servers. If the progress page was pending, the URL redirected it to running servers. This was useful because it made sure that the requested servers were restarted after they stopped. However, it could also be harmful because unused servers would continuously be restarted if e.g. an idle JupyterLab frontend that constantly makes polling requests was openly pointed at it.
Special handling of API requests#
/user/:username[/:servername]/api/... are assumed to be
from applications connected to stopped servers.
These requests fail with a
503 status code and an informative JSON error message
that indicates how to spawn the server.
This is meant to help applications such as JupyterLab,
that are connected to a server that has stopped.
Version changed: 1.0
JupyterHub version 0.9 failed these API requests with status
but version 1.0 uses 503.
/user-redirect/... URL is for sharing a URL that will redirect a user
to a path on their own default server.
This is useful when different users have the same file at the same URL on their servers,
and you want a single link to give to any user that will open that file on their server.
e.g. a link to
will send user
DO NOT share links to your own server with other users. This will not work in general, unless you grant those users access to your server.
Contributions welcome: The JupyterLab “shareable link” should share this link when run with JupyterHub, but it does not. See jupyterlab-hub where this should probably be done and this issue in JupyterLab that is intended to make it possible.
/hub/spawn will spawn the default server for the current user.
username and optionally
servername are specified,
then the specified server for the specified user will be spawned.
Once spawn has been requested,
the browser is redirected to
Spawner.options_form is used,
this will render a form,
and a POST request will trigger the actual spawn and redirect.
Version added: 1.0
1.0 adds the ability to specify
Prior to 1.0, only
/hub/spawn was recognized for the default server.
Version changed: 1.0
Prior to 1.0, this page redirected back to
which was responsible for triggering spawn and rendering progress, etc.
Version added: 1.0 this URL is new in JupyterHub 1.0.
This page renders the progress view for the given spawn request.
Once the server is ready,
the browser is redirected to the running server at
If this page is requested at any time after the specified server is ready, the browser will be redirected to the running server.
Requesting this page will never trigger any side effects.
If the server is not running (e.g. because the spawn has failed),
the spawn failure message (if applicable) will be displayed,
and the page will show a link back to
On this page, users can manage their JupyterHub API tokens. They can revoke access and request new tokens for writing scripts against the JupyterHub REST API.
Administrators can take various administrative actions from this page:
grant admin privileges
start/stop user servers
shutdown JupyterHub itself