mirror of
https://github.com/nostr-protocol/nips.git
synced 2024-12-22 08:25:53 -05:00
Merge pull request #1236 from nostr-protocol/nip96-sync
NIP-96: List files / rewording, no_transform
This commit is contained in:
commit
ff2e05d73f
126
96.md
126
96.md
|
@ -1,8 +1,6 @@
|
||||||
NIP-96
|
# NIP-96
|
||||||
======
|
|
||||||
|
|
||||||
HTTP File Storage Integration
|
## HTTP File Storage Integration
|
||||||
-----------------------------
|
|
||||||
|
|
||||||
`draft` `optional`
|
`draft` `optional`
|
||||||
|
|
||||||
|
@ -84,46 +82,46 @@ it must use the "api_url" field instead.
|
||||||
|
|
||||||
See https://github.com/aljazceru/awesome-nostr#nip-96-file-storage-servers.
|
See https://github.com/aljazceru/awesome-nostr#nip-96-file-storage-servers.
|
||||||
|
|
||||||
|
## Auth
|
||||||
|
|
||||||
|
When indicated, `clients` must add an [NIP-98](98.md) `Authorization` header (**optionally** with the encoded `payload` tag set to the base64-encoded 256-bit SHA-256 hash of the file - not the hash of the whole request body).
|
||||||
|
|
||||||
## Upload
|
## Upload
|
||||||
|
|
||||||
A file can be uploaded one at a time to `https://your-file-server.example/custom-api-path` (route from `https://your-file-server.example/.well-known/nostr/nip96.json` "api_url" field) as `multipart/form-data` content type using `POST` method with the file object set to the `file` form data field.
|
`POST $api_url` as `multipart/form-data`.
|
||||||
|
|
||||||
`Clients` must add an [NIP-98](98.md) `Authorization` header (**optionally** with the encoded `payload` tag set to the base64-encoded 256-bit SHA-256 hash of the file - not the hash of the whole request body).
|
**AUTH required**
|
||||||
If using an html form, use an `Authorization` form data field instead.
|
|
||||||
|
|
||||||
These following **optional** form data fields MAY be used by `servers` and SHOULD be sent by `clients`:
|
List of form fields:
|
||||||
- `expiration`: string of the UNIX timestamp in seconds. Empty string if file should be stored forever. The server isn't required to honor this;
|
|
||||||
- `size`: string of the file byte size. This is just a value the server can use to reject early if the file size exceeds the server limits;
|
- `file`: **REQUIRED** the file to upload
|
||||||
- `alt`: (recommended) strict description text for visibility-impaired users;
|
- `caption`: **RECOMMENDED** loose description;
|
||||||
- `caption`: loose description;
|
- `expiration`: UNIX timestamp in seconds. Empty string if file should be stored forever. The server isn't required to honor this.
|
||||||
- `media_type`: "avatar" or "banner". Informs the server if the file will be used as an avatar or banner. If absent, the server will interpret it as a normal upload, without special treatment;
|
- `size`: File byte size. This is just a value the server can use to reject early if the file size exceeds the server limits.
|
||||||
|
- `alt`: **RECOMMENDED** strict description text for visibility-impaired users.
|
||||||
|
- `media_type`: "avatar" or "banner". Informs the server if the file will be used as an avatar or banner. If absent, the server will interpret it as a normal upload, without special treatment.
|
||||||
- `content_type`: mime type such as "image/jpeg". This is just a value the server can use to reject early if the mime type isn't supported.
|
- `content_type`: mime type such as "image/jpeg". This is just a value the server can use to reject early if the mime type isn't supported.
|
||||||
|
- `no_transform`: "true" asks server not to transform the file and serve the uploaded file as is, may be rejected.
|
||||||
|
|
||||||
Others custom form data fields may be used depending on specific `server` support.
|
Others custom form data fields may be used depending on specific `server` support.
|
||||||
The `server` isn't required to store any metadata sent by `clients`.
|
The `server` isn't required to store any metadata sent by `clients`.
|
||||||
|
|
||||||
Note for `clients`: if using an HTML form, it is important for the `file` form field to be the **last** one, or be re-ordered right before sending or be appended as the last field of XHR2's FormData object.
|
|
||||||
|
|
||||||
The `filename` embedded in the file may not be honored by the `server`, which could internally store just the SHA-256 hash value as the file name, ignoring extra metadata.
|
The `filename` embedded in the file may not be honored by the `server`, which could internally store just the SHA-256 hash value as the file name, ignoring extra metadata.
|
||||||
The hash is enough to uniquely identify a file, that's why it will be used on the "download" and "delete" routes.
|
The hash is enough to uniquely identify a file, that's why it will be used on the `download` and `delete` routes.
|
||||||
|
|
||||||
The `server` MUST link the user's `pubkey` string (which is embedded in the decoded header value) as the owner of the file so to later allow them to delete the file.
|
The `server` MUST link the user's `pubkey` string as the owner of the file so to later allow them to delete the file.
|
||||||
Note that if a file with the same hash of a previously received file (so the same file) is uploaded by another user, the server doesn't need to store the new file.
|
|
||||||
It should just add the new user's `pubkey` to the list of the owners of the already stored file with said hash (if it wants to save space by keeping just one copy of the same file, because multiple uploads of the same file results in the same file hash).
|
|
||||||
|
|
||||||
The `server` MAY also store the `Authorization` header/field value (decoded or not) for accountability purpose as this proves that the user with the unique pubkey did ask for the upload of the file with a specific hash. However, storing the pubkey is sufficient to establish ownership.
|
`no_transform` can be used to replicate a file to multiple servers for redundancy, clients can use the [server list](#selecting-a-server) to find alternative servers which might contain the same file. When uploading a file and requesting `no_transform` clients should check that the hash matches in the response in order to detect if the file was modified.
|
||||||
|
|
||||||
The `server` MUST reject with 413 Payload Too Large if file size exceeds limits.
|
### Response codes
|
||||||
|
|
||||||
The `server` MUST reject with 400 Bad Request status if some fields are invalid.
|
- `200 OK`: File upload exists, but is successful (Existing hash)
|
||||||
|
- `201 Created`: File upload successful (New hash)
|
||||||
The `server` MUST reply to the upload with 200 OK status if the `payload` tag value contains an already used SHA-256 hash (if file is already owned by the same pubkey) or reject the upload with 403 Forbidden status if it isn't the same of the received file.
|
- `202 Accepted`: File upload is awaiting processing, see [Delayed Processing](#delayed-processing) section
|
||||||
|
- `413 Payload Too Large`: File size exceeds limit
|
||||||
The `server` MAY reject the upload with 402 Payment Required status if the user has a pending payment (Payment flow is not strictly required. Server owners decide if the storage is free or not. Monetization schemes may be added later to correlated NIPs.).
|
- `400 Bad Request`: Form data is invalid or not supported.
|
||||||
|
- `403 Forbidden`: User is not allowed to upload or the uploaded file hash didnt match the hash included in the `Authorization` header `payload` tag.
|
||||||
On successful uploads the `server` MUST reply with **201 Created** HTTP status code or **202 Accepted** if a `processing_url` field is added
|
- `402 Payment Required`: Payment is required by the server, **this flow is undefined**.
|
||||||
to the response so that the `client` can follow the processing status (see [Delayed Processing](#delayed-processing) section).
|
|
||||||
|
|
||||||
The upload response is a json object as follows:
|
The upload response is a json object as follows:
|
||||||
|
|
||||||
|
@ -179,11 +177,13 @@ The upload response is a json object as follows:
|
||||||
|
|
||||||
Note that if the server didn't apply any transformation to the received file, both `nip94_event.tags.*.ox` and `nip94_event.tags.*.x` fields will have the same value. The server MUST link the saved file to the SHA-256 hash of the **original** file before any server transformations (the `nip94_event.tags.*.ox` tag value). The **original** file's SHA-256 hash will be used to identify the saved file when downloading or deleting it.
|
Note that if the server didn't apply any transformation to the received file, both `nip94_event.tags.*.ox` and `nip94_event.tags.*.x` fields will have the same value. The server MUST link the saved file to the SHA-256 hash of the **original** file before any server transformations (the `nip94_event.tags.*.ox` tag value). The **original** file's SHA-256 hash will be used to identify the saved file when downloading or deleting it.
|
||||||
|
|
||||||
`Clients` may upload the same file to one or many `servers`.
|
`clients` may upload the same file to one or many `servers`.
|
||||||
After successful upload, the `client` may optionally generate and send to any set of nostr `relays` a [NIP-94](94.md) event by including the missing fields.
|
After successful upload, the `client` may optionally generate and send to any set of nostr `relays` a [NIP-94](94.md) event by including the missing fields.
|
||||||
|
|
||||||
Alternatively, instead of using NIP-94, the `client` can share or embed on a nostr note just the above url.
|
Alternatively, instead of using NIP-94, the `client` can share or embed on a nostr note just the above url.
|
||||||
|
|
||||||
|
`clients` may also use the tags from the `nip94_event` to construct an `imeta` tag
|
||||||
|
|
||||||
### Delayed Processing
|
### Delayed Processing
|
||||||
|
|
||||||
Sometimes the server may want to place the uploaded file in a processing queue for deferred file processing.
|
Sometimes the server may want to place the uploaded file in a processing queue for deferred file processing.
|
||||||
|
@ -219,7 +219,7 @@ However, for all file actions, such as download and deletion, the **original** f
|
||||||
|
|
||||||
## Download
|
## Download
|
||||||
|
|
||||||
`Servers` must make available the route `https://your-file-server.example/custom-api-path/<sha256-file-hash>(.ext)` (route taken from `https://your-file-server.example/.well-known/nostr/nip96.json` "api_url" or "download_url" field) with `GET` method for file download.
|
`GET $api_url/<sha256-hash>(.ext)`
|
||||||
|
|
||||||
The primary file download url informed at the upload's response field `nip94_event.tags.*.url`
|
The primary file download url informed at the upload's response field `nip94_event.tags.*.url`
|
||||||
can be that or not (it can be any non-standard url the server wants).
|
can be that or not (it can be any non-standard url the server wants).
|
||||||
|
@ -227,17 +227,17 @@ If not, the server still MUST also respond to downloads at the standard url
|
||||||
mentioned on the previous paragraph, to make it possible for a client
|
mentioned on the previous paragraph, to make it possible for a client
|
||||||
to try downloading a file on any NIP-96 compatible server by knowing just the SHA-256 file hash.
|
to try downloading a file on any NIP-96 compatible server by knowing just the SHA-256 file hash.
|
||||||
|
|
||||||
Note that the "\<sha256-file-hash\>" part is from the **original** file, **not** from the **transformed** file if the uploaded file went through any server transformation.
|
Note that the "\<sha256-hash\>" part is from the **original** file, **not** from the **transformed** file if the uploaded file went through any server transformation.
|
||||||
|
|
||||||
Supporting ".ext", meaning "file extension", is required for `servers`. It is optional, although recommended, for `clients` to append it to the path.
|
Supporting ".ext", meaning "file extension", is required for `servers`. It is optional, although recommended, for `clients` to append it to the path.
|
||||||
When present it may be used by `servers` to know which `Content-Type` header to send (e.g.: "Content-Type": "image/png" for ".png" extension).
|
When present it may be used by `servers` to know which `Content-Type` header to send (e.g.: "Content-Type": "image/png" for ".png" extension).
|
||||||
The file extension may be absent because the hash is the only needed string to uniquely identify a file.
|
The file extension may be absent because the hash is the only needed string to uniquely identify a file.
|
||||||
|
|
||||||
Example: `https://your-file-server.example/custom-api-path/719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b.png`
|
Example: `$api_url/719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b.png`
|
||||||
|
|
||||||
### Media Transformations
|
### Media Transformations
|
||||||
|
|
||||||
`Servers` may respond to some media transformation query parameters and ignore those they don't support by serving
|
`servers` may respond to some media transformation query parameters and ignore those they don't support by serving
|
||||||
the original media file without transformations.
|
the original media file without transformations.
|
||||||
|
|
||||||
#### Image Transformations
|
#### Image Transformations
|
||||||
|
@ -245,23 +245,23 @@ the original media file without transformations.
|
||||||
##### Resizing
|
##### Resizing
|
||||||
|
|
||||||
Upon upload, `servers` may create resized image variants, such as thumbnails, respecting the original aspect ratio.
|
Upon upload, `servers` may create resized image variants, such as thumbnails, respecting the original aspect ratio.
|
||||||
`Clients` may use the `w` query parameter to request an image version with the desired pixel width.
|
`clients` may use the `w` query parameter to request an image version with the desired pixel width.
|
||||||
`Servers` can then serve the variant with the closest width to the parameter value
|
`servers` can then serve the variant with the closest width to the parameter value
|
||||||
or an image variant generated on the fly.
|
or an image variant generated on the fly.
|
||||||
|
|
||||||
Example: `https://your-file-server.example/custom-api-path/<sha256-file-hash>.png?w=32`
|
Example: `$api_url/<sha256-hash>.png?w=32`
|
||||||
|
|
||||||
## Deletion
|
## Deletion
|
||||||
|
|
||||||
`Servers` must make available the route `https://deletion.domain/deletion-path/<sha256-file-hash>(.ext)` (route taken from `https://your-file-server.example/.well-known/nostr/nip96.json` "api_url" field) with `DELETE` method for file deletion.
|
`DELETE $api_url/<sha256-hash>(.ext)`
|
||||||
|
|
||||||
Note that the "\<sha256-file-hash\>" part is from the **original** file, **not** from the **transformed** file if the uploaded file went through any server transformation.
|
**AUTH required**
|
||||||
|
|
||||||
|
Note that the `/<sha256-hash>` part is from the **original** file, **not** from the **transformed** file if the uploaded file went through any server transformation.
|
||||||
|
|
||||||
The extension is optional as the file hash is the only needed file identification.
|
The extension is optional as the file hash is the only needed file identification.
|
||||||
|
|
||||||
`Clients` should send a `DELETE` request to the server deletion route in the above format. It must include a NIP-98 `Authorization` header.
|
The `server` should reject deletes from users other than the original uploader with the appropriate http response code (403 Forbidden).
|
||||||
|
|
||||||
The `server` should reject deletes from users other than the original uploader. The `pubkey` encoded on the header value identifies the user.
|
|
||||||
|
|
||||||
It should be noted that more than one user may have uploaded the same file (with the same hash). In this case, a delete must not really delete the file but just remove the user's `pubkey` from the file owners list (considering the server keeps just one copy of the same file, because multiple uploads of the same file results
|
It should be noted that more than one user may have uploaded the same file (with the same hash). In this case, a delete must not really delete the file but just remove the user's `pubkey` from the file owners list (considering the server keeps just one copy of the same file, because multiple uploads of the same file results
|
||||||
in the same file hash).
|
in the same file hash).
|
||||||
|
@ -275,6 +275,46 @@ The successful response is a 200 OK one with just basic JSON fields:
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Listing files
|
||||||
|
|
||||||
|
`GET $api_url?page=x&count=y`
|
||||||
|
|
||||||
|
**AUTH required**
|
||||||
|
|
||||||
|
Returns a list of files linked to the authenticated users pubkey.
|
||||||
|
|
||||||
|
Example Response:
|
||||||
|
|
||||||
|
```js
|
||||||
|
{
|
||||||
|
"count": 1, // server page size, eg. max(1, min(server_max_page_size, arg_count))
|
||||||
|
"total": 1, // total number of files
|
||||||
|
"page": 0, // the current page number
|
||||||
|
"files": [
|
||||||
|
{
|
||||||
|
"tags": [
|
||||||
|
["ox": "719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b"],
|
||||||
|
["x": "5d2899290e0e69bcd809949ee516a4a1597205390878f780c098707a7f18e3df"],
|
||||||
|
["size", "123456"],
|
||||||
|
["alt", "a meme that makes you laugh"],
|
||||||
|
["expiration", "1715691139"],
|
||||||
|
// ...other metadata
|
||||||
|
]
|
||||||
|
"content": "haha funny meme", // caption
|
||||||
|
"created_at": 1715691130 // upload timestmap
|
||||||
|
},
|
||||||
|
...
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
`files` contains an array of NIP-94 events
|
||||||
|
|
||||||
|
### Query args
|
||||||
|
|
||||||
|
- `page` page number (`offset=page*count`)
|
||||||
|
- `count` number of items per page
|
||||||
|
|
||||||
## Selecting a Server
|
## Selecting a Server
|
||||||
|
|
||||||
Note: HTTP File Storage Server developers may skip this section. This is meant for client developers.
|
Note: HTTP File Storage Server developers may skip this section. This is meant for client developers.
|
||||||
|
|
Loading…
Reference in New Issue
Block a user