# Hooks When integrating tusd into an application, it is important to establish a communication channel between the two components. The tusd binary accomplishes this by providing a system which triggers actions when certain events happen, such as an upload being created or finished. This simple-but-powerful system enables uses ranging from logging over validation and authorization to processing the uploaded files. When a specific action happens during an upload (pre-create, post-receive, post-finish, or post-terminate), the hook system enables tusd to fire off a specific event. Tusd provides two ways of doing this: 1. Execute an arbitrary file, mirroring the Git hook system, called File Hooks. 2. Fire off an HTTP POST request to a custom endpoint, called HTTP Hooks. ## Non-Blocking Hooks If not otherwise noted, all hooks are invoked in a *non-blocking* way, meaning that tusd will not wait until the hook process has finished and exited. Therefore, the hook process is not able to influence how tusd may continue handling the current request, regardless of which exit code it may set. Furthermore, the hook process' stdout and stderr will be piped to tusd's stdout and stderr correspondingly, allowing one to use these channels for additional logging. ## Blocking Hooks On the other hand, there are a few *blocking* hooks, such as caused by the `pre-create`, `pre-get` and `pre-finish` events. Because their exit code will dictate whether tusd will accept the current incoming request, tusd will wait until the hook process has exited. Therefore, in order to keep the response times low, one should avoid to make time-consuming operations inside the processes for blocking hooks. ### Blocking File Hooks An exit code of `0` indicates that tusd should continue handling the request as normal. On the other hand, a non-zero exit code tells tusd to reject the request with a `500 Internal Server Error` response containing the process' output from stderr. For the sake of logging, the process' output from stdout will always be piped to tusd's stdout. ### Blocking HTTP(S) Hooks A successful HTTP response code (i.e. smaller than `400`) indicates that tusd should continue handling the request as normal. On the other hand, an HTTP response code greater than `400` will be forwarded to the client performing the upload, along with the body of the hook response. Only the response code will be logged by tusd. ## List of Available Hooks ### pre-create This event will be triggered before an upload is created, allowing you to run certain routines. For example, validating that specific metadata values are set, or verifying that a corresponding entity belonging to the upload (e.g. a user) exists. Because this event will result in a blocking hook, you can determine whether the upload should be created or rejected using the exit code. An exit code of `0` will allow the upload to be created and continued as usual. A non-zero exit code will reject an upload creation request, making it a good place for authentication and authorization. Please be aware that during this stage the upload ID will be an empty string and `Storage` will be null. This is because the entity has not been created and therefore this piece of information is not yet available. ### post-create This event will be triggered after an upload is created, allowing you to run certain routines. For example, notifying other parts of your system that a new upload has to be handled. At this point the upload may have received some data already since the invocation of these hooks may be delayed by a short duration. ### pre-finish This event will be triggered after an upload is fully finished but before a response has been returned to the client. This is a blocking hook, as such it can be used to validate or post-process an uploaded file. A non-zero exit code or HTTP response greater than `400` will return a HTTP 500 error to the client. ### pre-get This event will be triggered before an upload read returned to the client. This is a blocking hook, as such it can be used to validate access limits. A non-zero exit code or HTTP response greater than `400` will return a HTTP 500 error to the client. ### post-finish This event will be triggered after an upload is fully finished, meaning that all chunks have been transfered and saved in the storage. After this point, no further modifications, except possible deletion, can be made to the upload entity and it may be desirable to use the file for further processing or notify other applications of the completions of this upload. ### post-terminate This event will be triggered after an upload has been terminated, meaning that the upload has been totally stopped and all associating chunks have been fully removed from the storage. Therefore, one is not able to retrieve the upload's content anymore and one may wish to notify further applications that this upload will never be resumed nor finished. ### post-receive This event will be triggered for every running upload to indicate its current progress. It will be emitted whenever the server has received more data from the client but at most every second. The offset property will be set to the number of bytes which have been transfered to the server, at the time in total. Please be aware that this number may be higher than the number of bytes which have been stored by the data store! ## Whitelisting Hook Events The `--hooks-enabled-events` option for the tusd binary works as a whitelist for hook events and takes a comma separated list of hook events (for instance: `pre-create,post-create`). This can be useful to limit the number of hook executions and save resources if you are only interested in some events. If the `--hooks-enabled-events` option is omitted, all default hook events are enabled (pre-create, post-create, post-receive, post-terminate, post-finish). ## File Hooks ### The Hook Directory By default, the file hook system is disabled. To enable it, pass the `--hooks-dir` option to the tusd binary. The flag's value will be a path, the **hook directory**, relative to the current working directory, pointing to the folder containing the executable **hook files**: ```bash $ tusd --hooks-dir ./path/to/hooks/ [tusd] Using './path/to/hooks/' for hooks [tusd] Using './data' as directory storage. ... ``` If an event occurs, the tusd binary will look for a file, named exactly as the event, which will then be executed, as long as the object exists. In the example above, the binary `./path/to/hooks/pre-create` will be invoked, before an upload is created, which can be used to e.g. validate certain metadata. Please note, that in UNIX environments the hook file *must not* have an extension, such as `.sh` or `.py`, or else tusd will not recognize and ignore it. On Windows, however, the hook file *must* have an extension, such as `.bat` or `.exe`. ### The Hook's Environment The process of the hook files are provided with information about the event and the upload using to two methods: * The `TUS_ID` and `TUS_SIZE` environment variables will contain the upload ID and its size in bytes, which triggered the event. Please be aware, that in the `pre-create` hook the upload ID will be an empty string as the entity has not been created and therefore this piece of information is not yet available. * On `stdin` a JSON-encoded object can be read which contains more details about the corresponding event in following format: ```js { // The upload object contains the upload's details "Upload": { // The upload's ID. Will be empty during the pre-create event "ID": "14b1c4c77771671a8479bc0444bbc5ce", // The upload's total size in bytes. "Size": 46205, // The upload's current offset in bytes. "Offset": 1592, // These properties will be set to true, if the upload as a final or partial // one. See the Concatenation extension for details: // http://tus.io/protocols/resumable-upload.html#concatenation "IsFinal": false, "IsPartial": false, // If the upload is a final one, this value will be an array of upload IDs // which are concatenated to produce the upload. "PartialUploads": null, // The upload's meta data which can be supplied by the clients as it wishes. // All keys and values in this object will be strings. // Be aware that it may contain maliciously crafted values and you must not // trust it without escaping it first! "MetaData": { "filename": "transloadit.png" }, // Details about where the data store saved the uploaded file. The different // availabl keys vary depending on the used data store. "Storage": { // For example, the filestore supplies the absolute file path: "Type": "filestore", "Path": "/my/upload/directory/14b1c4c77771671a8479bc0444bbc5ce", // The S3Store and GCSStore supply the bucket name and object key: "Type": "s3store", "Bucket": "my-upload-bucket", "Key": "my-prefix/14b1c4c77771671a8479bc0444bbc5ce" } }, // Details about the HTTP request which caused this hook to be fired. // It can be used to record the client's IP address or inspect the headers. "HTTPRequest": { "Method": "PATCH", "URI": "/files/14b1c4c77771671a8479bc0444bbc5ce", "RemoteAddr": "1.2.3.4:47689", "Header": { "Host": ["myuploads.net"], "Cookies": ["..."] } } } ``` ## HTTP(S) Hooks HTTP(S) Hooks are the second type of hooks supported by tusd. Like the file hooks, it is disabled by default. To enable it, pass the `--hooks-http` option to the tusd binary. The flag's value will be an HTTP(S) URL endpoint, which the tusd binary will issue POST requests to: ```bash $ tusd --hooks-http http://localhost:8081/write [tusd] Using 'http://localhost:8081/write' as the endpoint for hooks [tusd] Using './data' as directory storage. ... ``` Note that the URL must include the `http://` or `https://` prefix! In case of a blocking hook, HTTP Status Code 400 or greater tells tusd to reject the request (in the same way as non-zero exit code for File Hooks). See also [issue #170](https://github.com/tus/tusd/issues/170) regarding further improvements. Headers from the client's upload request can be copied to the hook request with the `-hooks-http-forward-headers` flag. This is particularly useful for including authentication headers such as `Authorization` or `Cookie`. ### Usage Tusd will issue a `POST` request to the specified URL endpoint, specifying the hook name, such as pre-create or post-finish, in the `Hook-Name` header and following body: ```js { // The upload object contains the upload's details "Upload": { // The upload's ID. Will be empty during the pre-create event "ID": "14b1c4c77771671a8479bc0444bbc5ce", // The upload's total size in bytes. "Size": 46205, // The upload's current offset in bytes. "Offset": 1592, // These properties will be set to true, if the upload as a final or partial // one. See the Concatenation extension for details: // http://tus.io/protocols/resumable-upload.html#concatenation "IsFinal": false, "IsPartial": false, // If the upload is a final one, this value will be an array of upload IDs // which are concatenated to produce the upload. "PartialUploads": null, // The upload's meta data which can be supplied by the clients as it wishes. // All keys and values in this object will be strings. // Be aware that it may contain maliciously crafted values and you must not // trust it without escaping it first! "MetaData": { "filename": "transloadit.png" }, // Details about where the data store saved the uploaded file. The different // availabl keys vary depending on the used data store. "Storage": { // For example, the filestore supplies the absolute file path: "Type": "filestore", "Path": "/my/upload/directory/14b1c4c77771671a8479bc0444bbc5ce", // The S3Store and GCSStore supply the bucket name and object key: "Type": "s3store", "Bucket": "my-upload-bucket", "Key": "my-prefix/14b1c4c77771671a8479bc0444bbc5ce" } }, // Details about the HTTP request which caused this hook to be fired. // It can be used to record the client's IP address or inspect the headers. "HTTPRequest": { "Method": "PATCH", "URI": "/files/14b1c4c77771671a8479bc0444bbc5ce", "RemoteAddr": "1.2.3.4:47689", "Header": { "Host": ["myuploads.net"], "Cookies": ["..."] } } } ``` ### Configuration Tusd uses the [Pester library](https://github.com/sethgrid/pester) to issue requests and handle retries. By default, tusd will retry 3 times on a `500 Internal Server Error` response or network error, with a 1 second backoff. This can be configured with the flags `--hooks-http-retry` and `--hooks-http-backoff`, like so: ``` $ # Retrying 5 times with a 2 second backoff $ tusd --hooks-http http://localhost:8081/write --hooks-http-retry 5 --hooks-http-backoff 2 ``` ## GRPC Hooks GRPC Hooks are the third type of hooks supported by tusd. Like the others hooks, it is disabled by default. To enable it, pass the `--hooks-grpc` option to the tusd binary. The flag's value will be a gRPC endpoint, which the tusd binary will be sent to: ```bash $ tusd --hooks-grpc localhost:8080 [tusd] Using 'localhost:8080' as the endpoint for gRPC hooks [tusd] Using './data' as directory storage. ... ``` ### Usage Tusd will issue a `gRPC` request to the specified endpoint, specifying the hook name, such as pre-create or post-finish, in the `Hook-Name` header and following body: ```js { // The upload object contains the upload's details "Upload": { // The upload's ID. Will be empty during the pre-create event "ID": "14b1c4c77771671a8479bc0444bbc5ce", // The upload's total size in bytes. "Size": 46205, // The upload's current offset in bytes. "Offset": 1592, // These properties will be set to true, if the upload as a final or partial // one. See the Concatenation extension for details: // http://tus.io/protocols/resumable-upload.html#concatenation "IsFinal": false, "IsPartial": false, // If the upload is a final one, this value will be an array of upload IDs // which are concatenated to produce the upload. "PartialUploads": null, // The upload's meta data which can be supplied by the clients as it wishes. // All keys and values in this object will be strings. // Be aware that it may contain maliciously crafted values and you must not // trust it without escaping it first! "MetaData": { "filename": "transloadit.png" }, // Details about where the data store saved the uploaded file. The different // availabl keys vary depending on the used data store. "Storage": { // For example, the filestore supplies the absolute file path: "Type": "filestore", "Path": "/my/upload/directory/14b1c4c77771671a8479bc0444bbc5ce", // The S3Store and GCSStore supply the bucket name and object key: "Type": "s3store", "Bucket": "my-upload-bucket", "Key": "my-prefix/14b1c4c77771671a8479bc0444bbc5ce" } }, // Details about the HTTP request which caused this hook to be fired. // It can be used to record the client's IP address or inspect the headers. "HTTPRequest": { "Method": "PATCH", "URI": "/files/14b1c4c77771671a8479bc0444bbc5ce", "RemoteAddr": "1.2.3.4:47689" } } ``` ### Configuration By default, tusd will retry 3 times based on the gRPC status response or network error, with a 1 second backoff. This can be configured with the flags `--hooks-grpc-retry` and `--hooks-grpc-backoff`, like so: ```bash $ # Retrying 5 times with a 2 second backoff $ tusd --hooks-grpc localhost:8081/ --hooks-grpc-retry 5 --hooks-grpc-backoff 2 ```