Create and Use API Keys
This guide applies to multi-user Tiled deployments, i.e. deployments configured with an authentication provider.
Note
Single-user deployments run with a single API key. By default, it is randomly generated and printed to the console at server startup, like so:
$ tiled serve pyobject tiled.examples.generated:tree
Generating large example data...
Done generating example data.
Use the following URL to connect to Tiled:
"http://127.0.0.1:8000?api_key=8df0146c93e9add287d0e7f84a165ba4bd3517bbb3c7c6b4d963c3e1549d0311"
This guide does not apply to single-user deployments like this.
Why use API keys?
Users can log in to Tiled with a simple username/password or through a third party like ORICD or Google, as demonstrated in the tutorial Log into an Authenticated Tiled Server. For interactive workflows in Python, this the most convenient way to authenticate.
But in other situations, it can more convenient to generate and use an API key. An API key uses a comparatively simpler interaction, without an interactive prompt or the need to periodically “refresh” a session to keep it active. This is useful when:
The code using tiled is running in an unsupervised, automated fashion or scaled across many workers, such as in an HPC job, where is it not possible to provide credentials interactively.
You are connecting from a generic web client like a web browser or
curl
, which has no built-in integration with Tiled or with OpenAPI+OAuth2.
Create an API Key
To follow along, you may start a Tiled server with a simple authentication provider,
as shown. Alternatively, you may use an existing authenticated Tiled server, such as
https://tiled-demo.blueskyproject.io
; if you do, replace http://localhost:8000
in the example below with that address.
# You can find example_configs/ in the root of the tiled source code repository.
ALICE_PASSWORD=secret1 tiled serve config example_configs/toy_authentication.yml
Using the Tiled commandline interface, log in as alice
using the password secret1
.
$ tiled profile create http://localhost:8000
$ tiled login
Username: alice
Password:
Create a new API key.
$ tiled api_key create
48e8f8598940fa0f3e80b406def606e17e815a2c76fe21350a99d6d9935371d11533b318
This text is the API key. It should be handled as a secret.
Use the API Key in Python
We can use it in the Python client:
>>> from tiled.client import from_url
>>> API_KEY = "YOUR_KEY_HERE"
>>> c = from_uri("http://localhost:8000", api_key=API_KEY)
API keys should never be placed directly in scripts or notebooks.
Instead, set the environment variable TILED_API_KEY
.
export TILED_API_KEY=YOUR_KEY_HERE
and then start Python (or IPython, or Jupyter, or…). The Python client will use that, unless it is explicitly passed different credentials.
>>> from tiled.client import from_url
>>> c = from_uri("http://localhost:8000") # uses TILED_API_KEY, if set
Use the API Key in other web clients
We can use in other web clients as well. For example, using HTTPie we can see that unauthenticated requests are refused
$ http http://localhost:8000/api/v1/metadata/
HTTP/1.1 401 Unauthorized
content-length: 30
content-type: application/json
date: Mon, 31 Jan 2022 17:30:06 GMT
server: uvicorn
server-timing: app;dur=2.6
set-cookie: tiled_csrf=bZLhKsXVE2VirgXQncHsHn4Y0Wwwr66U0T0hqarJyfw; HttpOnly; Path=/; SameSite=lax
x-tiled-root: http://localhost:8000/api/v1
{
"detail": "Not authenticated"
}
but passing the API key in the Authorization
header as Apikey YOUR_KEY_HERE
is accepted.
(Note the use of '
quotes.)
$ http http://localhost:8000/api/v1/metadata/ 'Authorization:Apikey 48e8f8598940fa0f3e80b406def606e17e815a2c76fe21350a99d6d9935371d11533b318'
HTTP/1.1 200 OK
content-length: 320
content-type: application/json
date: Mon, 31 Jan 2022 17:34:48 GMT
etag: c7d94c99b0a3cd6e102a78520db84bef
server: uvicorn
server-timing: tok;dur=0.0, pack;dur=0.0, app;dur=15.2
set-cookie: tiled_csrf=InE4mplUO0goPxf4V07tVuLSLUvDqhgtALTHYoC3T3s; HttpOnly; Path=/; SameSite=lax
<etc.>
The API key can also be passed in the URL like
http://localhost:8000/api/v1/metadata/?api_key=YOUR_KEY_HERE
. Using the
Authorization
header is preferred (more secure) but in some situations, as in
pasting a link into a web browser, the URL is the only option.
Manage API Keys
We can use the tiled commandline interface to examine and revoke API keys as well.
$ tiled api_key
Usage: tiled api_key [OPTIONS] COMMAND [ARGS]...
Create, list, and revoke API keys.
Options:
--help Show this message and exit.
Commands:
create
list
revoke
We can see the key that we made above in the list.
$ tiled api_key list
First 8 Expires at (UTC) Latest activity Note Scopes
48e8f859 - 2022-01-31T18:32:33 inherit
By default, an API key inherits all the same access as the user who is for. If an API key will be used for a specific task, it is good security hygiene to give it only the privileges it needs for that task. It is also recommended to set a limited lifetimes so that if the key is unknowingly leaked it will not continue to work forever. For example, this command creates an API key that will expire in 10 minutes and can search/list metadata but cannot download array data.
$ tiled api_key create --expires-in 10m --scopes read:metadata
ba9af604023a829ab22edb786168d6e1b97cef68c54c6d95d7fad5e3e6347fa131263581
Expiration can be given in units of years y
, days d
, hours h
, minutes
m
, or seconds s
.
See Scopes for the full list of scopes and their capabilities.
$ tiled api_key list
First 8 Expires at (UTC) Latest activity Note Scopes
48e8f859 - 2022-01-31T18:32:33 inherit
ba9af604 2022-01-31T23:03:57 - read:metadata
Finally, the --note
option can be used to label an API key as a reminder of
where or how it is being used.
$ tiled api_key create --note="Data validation pipeline" --scopes read:metadata --scopes read:data
573928c62e53096321fb874847bcc6a90ea636eebd8acbd7c5e9d18d2859847b3bfb4f19
$ tiled api_key list
First 8 Expires at (UTC) Latest activity Note Scopes
48e8f859 - 2022-01-31T18:32:33 inherit
ba9af604 2022-01-31T23:03:57 - read:metadata
573928c6 - - Data validation pipeline read:metadata read:data
If an API key is no longer need or accidentally leaked, it should be revoked. It
can be identified by its first eight characters, as shown in the the output of
tiled api_key list
.
$ tiled api_key revoke 573928c6
$ tiled api_key list
First 8 Expires at (UTC) Latest activity Note Scopes
48e8f859 - 2022-01-31T18:32:33 inherit
ba9af604 2022-01-31T23:03:57 - read:metadata
Passing the full key also works, if you have it handy. There is no need to trim just the first eight.