upleb.uk

Public git repos — served from a NIP-34 GRASP relay at git.upleb.uk

summaryrefslogtreecommitdiff
path: root/96.md
diff options
context:
space:
mode:
authorarthurfranca <arthur.a.franca@gmail.com>2024-01-08 14:05:01 -0300
committerGitHub <noreply@github.com>2024-01-08 14:05:01 -0300
commitb0e6c01321633f5e0de6ebf6676f018a218459a9 (patch)
tree1457552431896855c7391be4dc18fd89fbfd8ecc /96.md
parente489ed468b528f8022f420395b49a7615722aa57 (diff)
NIP-96 - HTTP File Storage Integration (#547)
* Add NIP-95 - File Storage * Add missing response info * Make it clear that is is an HTTP file storage server integration * Add monetization suggestion * Use zap split tags for monetization suggestion * Add resize option * Add Zap Gates Integration * Replace /nip96 convention with /.well-known/nostr.json configuration * Relays can choose to also act as HTTP file storage server * Remove nip96 tag in favor of x tags third element * Fix typo * Remove redirect cooperation * Replaced 422 with 400 status code * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Update 96.md Co-authored-by: Jon Staab <jstaab@protonmail.com> * Make file expiration a range and add terms_of_service * Add optional content_type field * Add plans and tos * Remove monetization * Apply minor fixes * Update 96.md Co-authored-by: Semisol <45574030+Semisol@users.noreply.github.com> * Fix after review * Add kind 10096 * Apply suggestions * Add suggestions * Remove duplicate field * Add optional is_nip98_required plan config * Add suggestions * Replace x with ox tag for original file hash * Make minor changes * Remove nip96 namespace response field * Add note about alternative file processing flow * Simplify processing flow * Add nostrcheck to server list * Add audio/* example * Explain what metadata to show before processing is done * Add nostrage to list * Add eta * Add sove to list and replace eta with percentage * Fix status code * Add nostr.build to list * Add sovbit * Add optional extra http servers to ox tag * Add void.cat to list * Small fix * Remove ox third array element --------- Co-authored-by: Jon Staab <jstaab@protonmail.com> Co-authored-by: Semisol <45574030+Semisol@users.noreply.github.com>
Diffstat (limited to '96.md')
-rw-r--r--96.md302
1 files changed, 302 insertions, 0 deletions
diff --git a/96.md b/96.md
new file mode 100644
index 0000000..2641b2a
--- /dev/null
+++ b/96.md
@@ -0,0 +1,302 @@
1NIP-96
2======
3
4HTTP File Storage Integration
5-----------------------------
6
7`draft` `optional` `author:arthurfranca` `author:Semisol` `author:staab` `author:v0l` `author:bndw` `author:michaelhall923` `author:fishcakeday` `author:quentintaranpino`
8
9## Introduction
10
11This NIP defines a REST API for HTTP file storage servers intended to be used in conjunction with the nostr network.
12The API will enable nostr users to upload files and later reference them by url on nostr notes.
13
14The spec DOES NOT use regular nostr events through websockets for
15storing, requesting nor retrieving data because, for simplicity, the server
16will not have to learn anything about nostr relays.
17
18## Server Adaptation
19
20File storage servers wishing to be accessible by nostr users should opt-in by making available an https route at `/.well-known/nostr/nip96.json` with `api_url`:
21
22```js
23{
24 // Required
25 // File upload and deletion are served from this url
26 // Also downloads if "download_url" field is absent or empty string
27 "api_url": "https://your-file-server.example/custom-api-path",
28 // Optional
29 // If absent, downloads are served from the api_url
30 "download_url": "https://a-cdn.example/a-path",
31 // Optional
32 // Note: This field is not meant to be set by HTTP Servers.
33 // Use this if you are a nostr relay using your /.well-known/nostr/nip96.json
34 // just to redirect to someone else's http file storage server's /.well-known/nostr/nip96.json
35 // In this case, "api_url" field must be an empty string
36 "delegated_to_url": "https://your-file-server.example",
37 // Optional
38 "supported_nips": [60],
39 // Optional
40 "tos_url": "https://your-file-server.example/terms-of-service",
41 // Optional
42 "content_types": ["image/jpeg", "video/webm", "audio/*"],
43 // Optional
44 "plans": {
45 // "free" is the only standardized plan key and
46 // clients may use its presence to learn if server offers free storage
47 "free": {
48 "name": "Free Tier",
49 // Default is true
50 // All plans MUST support NIP-98 uploads
51 // but some plans may also allow uploads without it
52 "is_nip98_required": true,
53 "url": "https://...", // plan's landing page if there is one
54 "max_byte_size": 10485760,
55 // Range in days / 0 for no expiration
56 // [7, 0] means it may vary from 7 days to unlimited persistence,
57 // [0, 0] means it has no expiration
58 // early expiration may be due to low traffic or any other factor
59 "file_expiration": [14, 90],
60 "media_transformations": {
61 "image": [
62 'resizing'
63 ]
64 }
65 }
66 }
67}
68```
69
70### Relay Hints
71
72Note: This section is not meant to be used by HTTP Servers.
73
74A nostr relay MAY redirect to someone else's HTTP file storage server by
75adding a `/.well-known/nostr/nip96.json` with "delegated_to_url" field
76pointing to the url where the server hosts its own
77`/.well-known/nostr/nip96.json`. In this case, the "api_url" field must
78be an empty string and all other fields must be absent.
79
80If the nostr relay is also an HTTP file storage server,
81it must use the "api_url" field instead.
82
83### List of Supporting File Storage Servers
84
85| Name | Domain |
86| ------------- | ------------------------- |
87| nostrcheck.me | https://nostrcheck.me |
88| nostrage | https://nostrage.com |
89| sove | https://sove.rent |
90| nostr.build | https://nostr.build |
91| sovbit | https://files.sovbit.host |
92| void.cat | https://void.cat |
93
94## Upload
95
96A 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.
97
98`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).
99If using an html form, use an `Authorization` form data field instead.
100
101These following **optional** form data fields MAY be used by `servers` and SHOULD be sent by `clients`:
102- `expiration`: string of the UNIX timestamp in seconds. Empty string if file should be stored forever. The server isn't required to honor this;
103- `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;
104- `alt`: (recommended) strict description text for visibility-impaired users;
105- `caption`: loose description;
106- `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;
107- `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.
108
109
110Others custom form data fields may be used depending on specific `server` support.
111The `server` isn't required to store any metadata sent by `clients`.
112
113Note 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.
114
115The `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.
116The hash is enough to uniquely identify a file, that's why it will be used on the "download" and "delete" routes.
117
118The `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.
119Note 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.
120It 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).
121
122The `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.
123
124The `server` MUST reject with 413 Payload Too Large if file size exceeds limits.
125
126The `server` MUST reject with 400 Bad Request status if some fields are invalid.
127
128The `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.
129
130The `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.).
131
132On successful uploads the `server` MUST reply with **201 Created** HTTP status code or **202 Accepted** if a `processing_url` field is added
133to the response so that the `client` can follow the processing status (see [Delayed Processing](#delayed-processing) section).
134
135The upload response is a json object as follows:
136
137```js
138{
139 // "success" if successful or "error" if not
140 status: "success",
141 // Free text success, failure or info message
142 message: "Upload successful.",
143 // Optional. See "Delayed Processing" section
144 processing_url: "...",
145 // This uses the NIP-94 event format but DO NOT need
146 // to fill some fields like "id", "pubkey", "created_at" and "sig"
147 //
148 // This holds the download url ("url"),
149 // the ORIGINAL file hash before server transformations ("ox")
150 // and, optionally, all file metadata the server wants to make available
151 //
152 // nip94_event field is absent if unsuccessful upload
153 nip94_event: {
154 // Required tags: "url" and "ox"
155 tags: [
156 // Can be same from /.well-known/nostr/nip96.json's "download_url" field
157 // (or "api_url" field if "download_url" is absent or empty) with appended
158 // original file hash.
159 //
160 // Note we appended .png file extension to the `ox` value
161 // (it is optional but extremely recommended to add the extension as it will help nostr clients
162 // with detecting the file type by using regular expression)
163 //
164 // Could also be any url to download the file
165 // (using or not using the /.well-known/nostr/nip96.json's "download_url" prefix),
166 // for load balancing purposes for example.
167 ["url", "https://your-file-server.example/custom-api-path/719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b.png"],
168 // SHA-256 hash of the ORIGINAL file, before transformations.
169 // The server MUST store it even though it represents the ORIGINAL file because
170 // users may try to download/delete the transformed file using this value
171 ["ox", "719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b"],
172 // Optional. SHA-256 hash of the saved file after any server transformations.
173 // The server can but does not need to store this value.
174 ["x", "543244319525d9d08dd69cb716a18158a249b7b3b3ec4bbde5435543acb34443"],
175 // Optional. Recommended for helping clients to easily know file type before downloading it.
176 ["m", "image/png"]
177 // Optional. Recommended for helping clients to reserve an adequate UI space to show the file before downloading it.
178 ["dim", "800x600"]
179 // ... other optional NIP-94 tags
180 ],
181 content: ""
182 },
183 // ... other custom fields (please consider adding them to this NIP or to NIP-94 tags)
184}
185```
186
187Note 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.
188
189`Clients` may upload the same file to one or many `servers`.
190After 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.
191
192Alternatively, instead of using NIP-94, the `client` can share or embed on a nostr note just the above url with added "ox" [NIP-54](54.md) inline metadata field and optionally other ones.
193
194### Delayed Processing
195
196Sometimes the server may want to place the uploaded file in a processing queue for deferred file processing.
197
198In that case, the server MUST serve the original file while the processing isn't done, then swap the original file for the processed one when the processing is over. The upload response is the same as usual but some optional metadata like `nip94_event.tags.*.x` and `nip94_event.tags.*.size` won't be available.
199
200The expected resulting metadata that is known in advance should be returned on the response.
201For example, if the file processing would change a file from "jpg" to "webp",
202use ".webp" extension on the `nip94_event.tags.*.url` field value and set "image/webp" to the `nip94_event.tags.*.m` field.
203If some metadata are unknown before processing ends, omit them from the response.
204
205The upload response MAY include a `processing_url` field informing a temporary url that may be used by clients to check if
206the file processing is done.
207
208If the processing isn't done, the server should reply at the `processing_url` url with **200 OK** and the following JSON:
209
210```
211{
212 // It should be "processing". If "error" it would mean the processing failed.
213 status: "processing",
214 message: "Processing. Please check again later for updated status.",
215 percentage: 15 // Processing percentage. An integer between 0 and 100.
216}
217```
218
219When the processing is over, the server replies at the `processing_url` url with **201 Created** status and a regular successful JSON response already mentioned before (now **without** a `processing_url` field), possibly including optional metadata at `nip94_event.tags.*` fields
220that weren't available before processing.
221
222### File compression
223
224File compression and other transformations like metadata stripping can be applied by the server.
225However, for all file actions, such as download and deletion, the **original** file SHA-256 hash is what identifies the file in the url string.
226
227## Download
228
229`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.
230
231The primary file download url informed at the upload's response field `nip94_event.tags.*.url`
232can be that or not (it can be any non-standard url the server wants).
233If not, the server still MUST also respond to downloads at the standard url
234mentioned on the previous paragraph, to make it possible for a client
235to try downloading a file on any NIP-96 compatible server by knowing just the SHA-256 file hash.
236
237Note 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.
238
239Supporting ".ext", meaning "file extension", is required for `servers`. It is optional, although recommended, for `clients` to append it to the path.
240When present it may be used by `servers` to know which `Content-Type` header to send (e.g.: "Content-Type": "image/png" for ".png" extension).
241The file extension may be absent because the hash is the only needed string to uniquely identify a file.
242
243Example: `https://your-file-server.example/custom-api-path/719171db19525d9d08dd69cb716a18158a249b7b3b3ec4bbdec5698dca104b7b.png`
244
245### Media Transformations
246
247`Servers` may respond to some media transformation query parameters and ignore those they don't support by serving
248the original media file without transformations.
249
250#### Image Transformations
251
252##### Resizing
253
254Upon upload, `servers` may create resized image variants, such as thumbnails, respecting the original aspect ratio.
255`Clients` may use the `w` query parameter to request an image version with the desired pixel width.
256`Servers` can then serve the variant with the closest width to the parameter value
257or an image variant generated on the fly.
258
259Example: `https://your-file-server.example/custom-api-path/<sha256-file-hash>.png?w=32`
260
261## Deletion
262
263`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.
264
265Note 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.
266
267The extension is optional as the file hash is the only needed file identification.
268
269`Clients` should send a `DELETE` request to the server deletion route in the above format. It must include a NIP-98 `Authorization` header.
270
271The `server` should reject deletes from users other than the original uploader. The `pubkey` encoded on the header value identifies the user.
272
273It 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
274in the same file hash).
275
276The successfull response is a 200 OK one with just basic JSON fields:
277
278```
279{
280 status: "success",
281 message: "File deleted."
282}
283```
284
285## Selecting a Server
286
287Note: HTTP File Storage Server developers may skip this section. This is meant for client developers.
288
289A File Server Preference event is a kind 10096 replaceable event meant to select one or more servers the user wants
290to upload files to. Servers are listed as `server` tags:
291
292```js
293{
294 // ...
295 "kind": 10096,
296 "content": "",
297 "tags": [
298 ["server", "https://file.server.one"],
299 ["server", "https://file.server.two"]
300 ]
301}
302```