diff --git a/tus-uploads/.env.example b/tus-uploads/.env.example
new file mode 100644
index 00000000..8ef578c9
--- /dev/null
+++ b/tus-uploads/.env.example
@@ -0,0 +1 @@
+PRIVATE_KEY_GOOGLE
\ No newline at end of file
diff --git a/tus-uploads/.gitignore b/tus-uploads/.gitignore
new file mode 100644
index 00000000..b57e31b8
--- /dev/null
+++ b/tus-uploads/.gitignore
@@ -0,0 +1,3 @@
+/uploads
+.env
+/app/config.ts
\ No newline at end of file
diff --git a/tus-uploads/.nvmrc b/tus-uploads/.nvmrc
new file mode 100644
index 00000000..9aef5aab
--- /dev/null
+++ b/tus-uploads/.nvmrc
@@ -0,0 +1 @@
+v20.17.0
\ No newline at end of file
diff --git a/tus-uploads/README.md b/tus-uploads/README.md
new file mode 100644
index 00000000..544004c7
--- /dev/null
+++ b/tus-uploads/README.md
@@ -0,0 +1,67 @@
+# TUS Integration Resumable File Uploads
+
+This is an example to use the Tus Protocol to upload files to either a Google Cloud Bucket or your internal file storge
+
+The relevent files are:
+
+```
+├── app
+|   ├── bucket.server.tsx // if using google cloud bucket - store your credentials here
+|   ├── tusCloudBucketHandler.server.tsx // this file handles different request methods used by Tus Protocol for cloud bucket integration
+|   ├── tusFileStoreHanlder.server.tsx // this file handles different request methods used by Tus Protocol for uploading files to an internal file storage/directory within your 
+|   ├── routes
+|   │   └── _index.tsx  // front end end file with basic form and tus-js client that uploads file to a designated route
+|   │   └── api.tus-cloud-uploads.tsx  // initial route Tus uses to POST and create file for cloud bucket integration
+|   │   └── api.tus-cloud-uploads.$fileId.tsx  // Afte file is created Tus makes patch requests to the file that was created using the POST request to update the file in "chunks" for cloud bucket integration
+|   │   └── api.tus-native-uploads.tsx  // initial route Tus uses to POST and create file on local files system
+|   │   └── api.tus-native-uploads.$fileId.tsx  // Afte file is created Tus makes patch requests to the file that was created using the POST request to update the file in "chunks" for local file system integration
+└── .env // hold cloud bucket credentials secret key
+```
+
+## Setup
+
+1. Copy `.env.example` to create a new file `.env`:
+
+```sh
+cp .env.example .env
+```
+
+## Example
+
+Servers like Cloud Run usually have a fixed limit ~32 mb of what data your upload to the server at one time,  The Tus Protocol solves these limits by uploading files in chunks, when large files are uploaded there can be network issues but when files are uploaded in chuunks in tus prootcol tus keeps track of when a file stopped uploading and can resume the upload.
+
+## Related Links
+
+Tus Protocol generally utilizes a front end and a back end, while integrating Tus-Js-Client npm package was relatively easy in a remix application - integrating Tus Server required either an implemented Node/Expres server that didn't quite fit into the remix architecture of using web fetch Api, rather it uses the native req, res objects in Express, instead of using the TusServer npm package which is tighly couple to Express/Node, the tusHanlerServer files basically implement the tus Server request methods while not being confined to using Express.  The TusHandler handles the same request methods required by the tus protocol "POST" - creation of file, "PATCH" - updates to File - "HEAD" - when needed to retrieve metadata for file.  
+npm package for tus-js-client - https://github.com/tus/tus-js-client  
+npm package for gcs-store tus-node-server - https://github.com/tus/tus-node-server/tree/main/packages/gcs-store  
+npm pacakge for file-store tus-node-server - https://github.com/tus/tus-node-server/tree/main/packages/file-store  
+code inspiration for request handlers - https://github.com/tus/tus-node-server/tree/main/packages/server    
+
+
+
+## Production
+On an environment like cloud run you may need to set content security policy header
+```
+if (process.env.NODE_ENV === "production") {
+  app.use((req, res, next) => {
+    res.setHeader("Content-Security-Policy", "upgrade-insecure-requests");
+    next();
+  });
+}
+```  
+see issue here - https://github.com/tus/tus-js-client/issues/186
+
+
+## To Run exmaple  
+`npm i`  
+in `_index.tsx` when tusClientUploader is invoked you have th option to call either `/api/tus-cloud-uploads` endpoint or the `/api/tus-native-uploads` endpoint when calling the cloud-uploads endpoint you must provide a bucketName `${bucketName}` the other endpoint requires a directory path like `./uploads/tus`  
+`npm run dev`
+use ux to upload file and watch the magic happen
+
+## Process  
+The typical flow for tus-js-client involves:
+- An initial POST request to create the upload resource.
+- One or more PATCH requests to upload the file data.
+- HEAD requests as needed to check the status of the existing upload/resource.
+
diff --git a/tus-uploads/app/bucket.server.tsx b/tus-uploads/app/bucket.server.tsx
new file mode 100644
index 00000000..db92ee5d
--- /dev/null
+++ b/tus-uploads/app/bucket.server.tsx
@@ -0,0 +1,24 @@
+import {Storage} from "@google-cloud/storage"
+
+
+export function getCloudStorage() {
+  const projectId = "";
+
+  let private_key = ''
+  const private_key_string = process.env.PRIVATE_KEY_GOOGLE || "";
+  if (private_key_string.length) {
+    private_key = private_key_string.split(String.raw`\n`).join("\n");
+  }
+  return new Storage({
+    projectId,
+    credentials: {
+      type: "",
+      project_id: "",
+      private_key_id: "",
+      private_key: `-----BEGIN PRIVATE KEY-----\n${private_key}\n-----END PRIVATE KEY-----\n`,
+      client_email: "",
+      client_id: "",
+      universe_domain: "",
+    },
+  });
+}
diff --git a/tus-uploads/app/root.tsx b/tus-uploads/app/root.tsx
new file mode 100644
index 00000000..e82f26fd
--- /dev/null
+++ b/tus-uploads/app/root.tsx
@@ -0,0 +1,29 @@
+import {
+  Links,
+  Meta,
+  Outlet,
+  Scripts,
+  ScrollRestoration,
+} from "@remix-run/react";
+
+export function Layout({ children }: { children: React.ReactNode }) {
+  return (
+    
+      
+        
+        
+        
+        
+      
+      
+        {children}
+        
+        
+      
+    
+  );
+}
+
+export default function App() {
+  return ;
+}
diff --git a/tus-uploads/app/routes/_index.tsx b/tus-uploads/app/routes/_index.tsx
new file mode 100644
index 00000000..26d190a9
--- /dev/null
+++ b/tus-uploads/app/routes/_index.tsx
@@ -0,0 +1,108 @@
+import { Form } from "@remix-run/react";
+import { useState } from "react";
+import * as tus from "tus-js-client";
+export default function Index() {
+
+  const [file, setFile] = useState(null as File | null);
+
+  //EDIT THIS
+  const bucketName = "you_bucket_name"
+
+  const startuploadProcess = async (e:any) => {
+    e.preventDefault();
+    if (file) {
+      // This will upload to a cloud storage buket
+      await tusClientUploader(file, "/api/tus-cloud-uploads", bucketName);
+      // this will upload to a directory on your file system
+      
+      // await tusClientUploader(file, "/api/tus-native-uploads", "./uploads/tus");
+    }
+
+  }
+
+  async function tusClientUploader(file:File, endpoint: string, destination: string, title?: string) {
+    let tusUpload:any;
+    const metadata = {
+      name: title || file.name,
+      filename: file.name,
+      filetype: file.type,
+      contentType: file.type,
+      destination,
+    };
+    return new Promise((resolve, reject) => {
+      const options = {
+        endpoint: endpoint,
+        chunkSize: Infinity,
+        metadata: metadata,
+        onError(tusError:Error) {
+          console.log({ tusError });
+          reject({
+            success: false,
+            error: tusError,
+            message: `error uploading file ${metadata.name}`,
+          });
+        },
+        onSuccess() {
+          console.log("onsuccess");
+          const url = new URL(tusUpload.url);
+          const id = url.pathname.split("/").pop();
+          const bucketName = url.searchParams.get("bucket");
+          if (bucketName) {
+            const encodedFormat = encodeURIComponent(tusUpload.options.metadata.contentType);
+            fetch(`/api/tus-cloud-uploads?id=${id}&mediaType=${encodedFormat}&bucketName=${bucketName}`)
+            .then((response) => {
+              return response.json();
+            })
+            .then((json) => {
+              console.log({ json });
+              resolve({
+                success: true,
+                url: url,
+                id,
+                fileUpdated: true,
+              });
+            })
+            .catch((error) => {
+              console.error({ error });
+              resolve({
+                success: true,
+                url: url,
+                id,
+                fileUpdated: false,
+                encodedFormat,
+                bucketName,
+              });
+            });
+          }
+
+
+        },
+        onProgress(bytesUploaded:number) {
+          const progress = (bytesUploaded / file.size) * 100;
+          console.log(progress + "%");
+        },
+      };
+  
+      tusUpload = new tus.Upload(file, options);
+      console.log({ tusUpload });
+      tusUpload.start();
+    });
+  }
+
+  return (
+    
+      
Uploading Files using Tus Protocol
+      
+    
+  );
+}
diff --git a/tus-uploads/app/routes/api.tus-cloud-uploads.$fileId.tsx b/tus-uploads/app/routes/api.tus-cloud-uploads.$fileId.tsx
new file mode 100644
index 00000000..178993b0
--- /dev/null
+++ b/tus-uploads/app/routes/api.tus-cloud-uploads.$fileId.tsx
@@ -0,0 +1,17 @@
+import { json } from "@remix-run/node";
+import type { LoaderFunctionArgs, ActionFunctionArgs } from "@remix-run/server-runtime";
+import { handleTusRequest } from "../tusCloudBucketHandler.server";
+
+export async function action({ request }: ActionFunctionArgs) {
+  const {method} = request;
+  if (method !== "PATCH") {
+    return json({ message: "Method not allowed" }, 405);
+  }
+
+  return handleTusRequest(request, method);
+}
+
+export const loader = async ({ request }: LoaderFunctionArgs) => {
+  const method = request.method;
+  return handleTusRequest(request, method);
+}
diff --git a/tus-uploads/app/routes/api.tus-cloud-uploads.tsx b/tus-uploads/app/routes/api.tus-cloud-uploads.tsx
new file mode 100644
index 00000000..d3f6c34b
--- /dev/null
+++ b/tus-uploads/app/routes/api.tus-cloud-uploads.tsx
@@ -0,0 +1,19 @@
+import type { LoaderFunctionArgs, ActionFunctionArgs } from "@remix-run/server-runtime";
+import { handleTusRequest } from "../tusCloudBucketHandler.server";
+import { json } from "@remix-run/node";
+
+export async function action({ request }: ActionFunctionArgs) {
+  const {method} = request;
+  return handleTusRequest(request, method);
+}
+
+export const loader = async ({ request }: LoaderFunctionArgs) => {
+  const method = request.method;
+  const urlToSearch = new URL(request.url);
+  const id = urlToSearch.searchParams.get("id") || "";
+  const mediaType = urlToSearch.searchParams.get("mediaType") || "";
+  const bucketName = urlToSearch.searchParams.get("bucketName") || "";
+  const data = await handleTusRequest(request, method, id, mediaType, bucketName);
+
+  return json({ data });
+};
diff --git a/tus-uploads/app/routes/api.tus-native-uploads.$fileId.tsx b/tus-uploads/app/routes/api.tus-native-uploads.$fileId.tsx
new file mode 100644
index 00000000..434a2bad
--- /dev/null
+++ b/tus-uploads/app/routes/api.tus-native-uploads.$fileId.tsx
@@ -0,0 +1,17 @@
+import { json } from "@remix-run/node";
+import type { LoaderFunctionArgs, ActionFunctionArgs } from "@remix-run/server-runtime";
+import { handleTusRequest } from "../tusFileStoreHandler.server";
+
+export async function action({ request }: ActionFunctionArgs) {
+  const {method} = request;
+  if (method !== "PATCH") {
+    return json({ message: "Method not allowed" }, 405);
+  }
+
+  return handleTusRequest(request, method);
+}
+
+export const loader = async ({ request }: LoaderFunctionArgs) => {
+  const {method} = request;
+  return handleTusRequest(request, method);
+}
diff --git a/tus-uploads/app/routes/api.tus-native-uploads.tsx b/tus-uploads/app/routes/api.tus-native-uploads.tsx
new file mode 100644
index 00000000..b7a02359
--- /dev/null
+++ b/tus-uploads/app/routes/api.tus-native-uploads.tsx
@@ -0,0 +1,12 @@
+import type { LoaderFunctionArgs, ActionFunctionArgs } from "@remix-run/server-runtime";
+import { handleTusRequest } from "../tusFileStoreHandler.server"
+
+export async function action({ request }: ActionFunctionArgs) {
+  const {method} = request;
+  return handleTusRequest(request, method);
+}
+
+export const loader = async ({ request }: LoaderFunctionArgs) => {
+  const {method} = request;
+  return handleTusRequest(request, method);
+};
diff --git a/tus-uploads/app/tusCloudBucketHandler.server.tsx b/tus-uploads/app/tusCloudBucketHandler.server.tsx
new file mode 100644
index 00000000..c136835a
--- /dev/null
+++ b/tus-uploads/app/tusCloudBucketHandler.server.tsx
@@ -0,0 +1,195 @@
+import { GCSStore } from "@tus/gcs-store";
+import { getCloudStorage } from "./bucket.server"
+import { Readable } from "stream";
+const { randomUUID } = await import("node:crypto");
+
+const storage = getCloudStorage();
+
+
+export async function handleTusRequest(request:any, method: string, id?:string, fileType?: string, nameOfBucket?: string) {
+
+  // we only do this on a GET request to update metadata of content type
+  if (id && nameOfBucket && fileType) {
+    const bucket = storage.bucket(nameOfBucket);
+    const file = bucket.file(id);
+    const fileMetadata = await file.setMetadata({
+      contentType: fileType,
+    });
+    return fileMetadata;
+  }
+
+  switch (method) {
+    case "POST":
+      return handlePostRequest(request);
+    case "PATCH":
+      return handlePatchRequest(request);
+    case "HEAD":
+      return handleHeadRequest(request);
+    default:
+      return new Response("Method Not Allowed", { status: 405 });
+  }
+}
+
+async function handlePostRequest(request: any) {
+  const uploadLength = request.headers.get("Upload-Length");
+  const metadata = request.headers.get("Upload-Metadata");
+  const parsedMetaData = parseMetadata(metadata);
+
+  const { destination } = parsedMetaData;
+  const targetBucket = destination;
+
+  const store = new GCSStore({
+    bucket: storage.bucket(targetBucket),
+  });
+
+  let id;
+
+  if (!uploadLength) {
+    return new Response("Upload-Length header is required", { status: 400 });
+  }
+  console.log({ uploadLength });
+
+  const genereatedId = randomUUID();
+  try {
+    const file = await store.create({
+        metadata: parsedMetaData,
+        id: genereatedId,
+        offset: parseInt(uploadLength),
+        size: uploadLength,
+        creation_date: undefined,
+        storage: undefined,
+        sizeIsDeferred: false
+    });
+
+    id = file.id;
+  } catch (error) {
+    console.error("Error creating file:", error);
+  }
+
+  const location = new URL(request.url);
+  location.pathname += `/${id}`;
+  location.search = `?bucket=${targetBucket}`;
+
+  return new Response(null, {
+    status: 201,
+    headers: {
+      Location: location.toString(),
+      "Tus-Resumable": "1.0.0",
+    },
+  });
+}
+
+async function handlePatchRequest(request: any) {
+  const url = new URL(request.url);
+  const fileId = url.pathname.split("/").pop();
+  const bucketName = url.searchParams.get("bucket");
+  const offset = parseInt(request.headers.get("Upload-Offset") || "0");
+
+  if (!bucketName) {
+    return new Response("bucket name not provided", { status: 404 });
+  }
+
+  const store = new GCSStore({
+    bucket: storage.bucket(bucketName),
+  });
+
+  if (!store) {
+    return new Response("Upload not found", { status: 404 });
+  }
+
+  if (!fileId) {
+    return new Response("File ID is required", { status: 400 });
+  }
+  try {
+    const body = await request.arrayBuffer();
+    if (!body) {
+      return new Response("Request body is missing", { status: 400 });
+    }
+
+    const buffer = Buffer.from(body);
+    const readable = Readable.from(buffer);
+    const newOffset = await store.write(readable, fileId, offset);
+
+    console.log({ newOffset });
+
+    return new Response(null, {
+      status: 204,
+      headers: {
+        "Upload-Offset": newOffset.toString(),
+        "Tus-Resumable": "1.0.0",
+      },
+    });
+  } catch (error) {
+    console.error("Error handling PATCH request:", error);
+    return new Response("Internal Server Error", { status: 500 });
+  }
+}
+
+async function handleHeadRequest(request:any) {
+  const url = new URL(request.url);
+  const fileId = url.pathname.split("/").pop();
+  const bucketName = url.searchParams.get("bucket");
+
+  if (!fileId) {
+    return new Response("File ID is required", { status: 400 });
+  }
+
+  if (!bucketName) {
+    return new Response("bucket name not provided", { status: 404 });
+  }
+
+  const bucket = storage.bucket(bucketName);
+  const file = bucket.file(fileId);
+
+  try {
+    const [metadata] = await file.getMetadata();
+    console.log({ metadata });
+
+    const thirdTierMetaData = metadata?.metadata?.metadata;
+
+    const stringifyMetaData = stringify(thirdTierMetaData);
+
+    const metaDataOffset =  metadata?.metadata?.["offset"];
+    const uploadSizeMetaData = metadata?.metadata?.["size"];
+
+    const uploadOffset = parseInt(metaDataOffset as any, 10);
+    const uploadSize = parseInt(uploadSizeMetaData as any, 10);
+
+    return new Response(null, {
+      status: 200,
+      headers: {
+        "Upload-Offset": uploadOffset.toString(),
+        "Upload-Length": uploadSize.toString(),
+        "Tus-Resumable": "1.0.0",
+        "Cache-Control": "no-store",
+        "Upload-Metadata": stringifyMetaData,
+      },
+    });
+  } catch (error) {
+    console.error("Error retrieving upload info:", error);
+    throw error;
+  }
+}
+
+function parseMetadata(metadata: any) {
+  if (!metadata) return {};
+  const data = metadata.split(",").reduce((acc: { [x: string]: string; }, pair: { split: (arg0: string) => [any, any]; }) => {
+    const [key, value] = pair.split(" ");
+    acc[key] = Buffer.from(value, "base64").toString("utf-8");
+    return acc;
+  }, {});
+  return data;
+}
+
+function stringify(metadata: NonNullable): string {
+  return Object.entries(metadata)
+    .map(([key, value]) => {
+      if (value === null) {
+        return key
+      }
+
+      const encodedValue = Buffer.from(value as any, 'utf8').toString('base64')
+      return `${key} ${encodedValue}`
+    })
+    .join(',')
+}
diff --git a/tus-uploads/app/tusFileStoreHandler.server.tsx b/tus-uploads/app/tusFileStoreHandler.server.tsx
new file mode 100644
index 00000000..a1803101
--- /dev/null
+++ b/tus-uploads/app/tusFileStoreHandler.server.tsx
@@ -0,0 +1,171 @@
+import { Readable } from "stream";
+const { randomUUID } = await import("node:crypto");
+import { FileStore } from "@tus/file-store";
+
+
+export async function handleTusRequest(request:any, method: string) {
+
+  switch (method) {
+    case "POST":
+      return handlePostRequest(request);
+    case "PATCH":
+      return handlePatchRequest(request);
+    case "HEAD":
+      return handleHeadRequest(request);
+    default:
+      return new Response("Method Not Allowed", { status: 405 });
+  }
+}
+
+async function handlePostRequest(request: any) {
+  const uploadLength = request.headers.get("Upload-Length");
+  const metadata = request.headers.get("Upload-Metadata");
+  const parsedMetaData = parseMetadata(metadata);
+
+  const { destination } = parsedMetaData;
+  const directory = destination;
+
+  const dataStore = new FileStore({directory});
+
+  let id;
+
+  if (!uploadLength) {
+    return new Response("Upload-Length header is required", { status: 400 });
+  }
+  console.log({ uploadLength });
+
+  const genereatedId = randomUUID();
+  try {
+    const file = await dataStore.create({
+        metadata: parsedMetaData,
+        id: genereatedId,
+        offset: parseInt(uploadLength),
+        size: uploadLength,
+        creation_date: undefined,
+        storage: undefined,
+        sizeIsDeferred: false
+    });
+    id = file.id;
+  } catch (error) {
+    console.error("Error creating file:", error);
+  }
+
+  const location = new URL(request.url);
+  location.pathname += `/${id}`;
+  location.search = `?directory=${directory}`;
+
+  return new Response(null, {
+    status: 201,
+    headers: {
+      Location: location.toString(),
+      "Tus-Resumable": "1.0.0",
+    },
+  });
+}
+
+async function handlePatchRequest(request: any) {
+  const url = new URL(request.url);
+  const fileId = url.pathname.split("/").pop();
+  const directory = url.searchParams.get("directory");
+  const offset = parseInt(request.headers.get("Upload-Offset") || "0");
+
+  if (!directory) {
+    return new Response("bucket name not provided", { status: 404 });
+  }
+
+  const dataStore = new FileStore({directory});
+
+  if (!dataStore) {
+    return new Response("Upload not found", { status: 404 });
+  }
+
+  if (!fileId) {
+    return new Response("File ID is required", { status: 400 });
+  }
+  try {
+    const body = await request.arrayBuffer();
+    if (!body) {
+      return new Response("Request body is missing", { status: 400 });
+    }
+
+    const buffer = Buffer.from(body);
+    const readable = Readable.from(buffer);
+    const newOffset = await dataStore.write(readable, fileId, offset);
+
+    console.log({ newOffset });
+
+    return new Response(null, {
+      status: 204,
+      headers: {
+        "Upload-Offset": newOffset.toString(),
+        "Tus-Resumable": "1.0.0",
+      },
+    });
+  } catch (error) {
+    console.error("Error handling PATCH request:", error);
+    return new Response("Internal Server Error", { status: 500 });
+  }
+}
+
+async function handleHeadRequest(request:any) {
+  const url = new URL(request.url);
+  const fileId = url.pathname.split("/").pop();
+  const directory = url.searchParams.get("directory");
+
+  if (!fileId) {
+    return new Response("File ID is required", { status: 400 });
+  }
+
+  if (!directory) {
+    return new Response("bucket name not provided", { status: 404 });
+  }
+
+  const dataStore = new FileStore({directory});
+
+  try {
+    const file = await dataStore.getUpload(fileId);
+    const uploadOffset = file.offset
+    const uploadSize = file.size as number;
+    let stringifyMetaData = ""
+    if (file.metadata !== undefined) {
+        stringifyMetaData = stringify(file.metadata);
+    }
+
+    return new Response(null, {
+      status: 200,
+      headers: {
+        "Upload-Offset": uploadOffset.toString(),
+        "Upload-Length": uploadSize.toString(),
+        "Tus-Resumable": "1.0.0",
+        "Cache-Control": "no-store",
+        "Upload-Metadata": stringifyMetaData,
+      },
+    });
+  } catch (error) {
+    console.error("Error retrieving upload info:", error);
+    throw error;
+  }
+}
+
+function parseMetadata(metadata: any) {
+  if (!metadata) return {};
+  const data = metadata.split(",").reduce((acc: { [x: string]: string; }, pair: { split: (arg0: string) => [any, any]; }) => {
+    const [key, value] = pair.split(" ");
+    acc[key] = Buffer.from(value, "base64").toString("utf-8");
+    return acc;
+  }, {});
+  return data;
+}
+
+export function stringify(metadata: NonNullable): string {
+  return Object.entries(metadata)
+    .map(([key, value]) => {
+      if (value === null) {
+        return key
+      }
+
+      const encodedValue = Buffer.from(value as any, 'utf8').toString('base64')
+      return `${key} ${encodedValue}`
+    })
+    .join(',')
+}
diff --git a/tus-uploads/package.json b/tus-uploads/package.json
new file mode 100644
index 00000000..d416296b
--- /dev/null
+++ b/tus-uploads/package.json
@@ -0,0 +1,43 @@
+{
+  "name": "template",
+  "private": true,
+  "sideEffects": false,
+  "type": "module",
+  "scripts": {
+    "build": "remix vite:build",
+    "dev": "remix vite:dev",
+    "lint": "eslint --ignore-path .gitignore --cache --cache-location ./node_modules/.cache/eslint .",
+    "start": "remix-serve ./build/server/index.js",
+    "typecheck": "tsc"
+  },
+  "dependencies": {
+    "@remix-run/node": "^2.9.2",
+    "@remix-run/react": "^2.9.2",
+    "@remix-run/serve": "^2.9.2",
+    "@tus/file-store": "^1.5.0",
+    "@tus/gcs-store": "^1.4.0",
+    "isbot": "^4.1.0",
+    "react": "^18.2.0",
+    "react-dom": "^18.2.0",
+    "tus-js-client": "^4.2.3"
+  },
+  "devDependencies": {
+    "@remix-run/dev": "^2.9.2",
+    "@types/react": "^18.2.20",
+    "@types/react-dom": "^18.2.7",
+    "@typescript-eslint/eslint-plugin": "^6.7.4",
+    "@typescript-eslint/parser": "^6.7.4",
+    "eslint": "^8.38.0",
+    "eslint-import-resolver-typescript": "^3.6.1",
+    "eslint-plugin-import": "^2.28.1",
+    "eslint-plugin-jsx-a11y": "^6.7.1",
+    "eslint-plugin-react": "^7.33.2",
+    "eslint-plugin-react-hooks": "^4.6.0",
+    "typescript": "^5.1.6",
+    "vite": "^5.1.0",
+    "vite-tsconfig-paths": "^4.2.1"
+  },
+  "engines": {
+    "node": ">=20.0.0"
+  }
+}
diff --git a/tus-uploads/public/favicon.ico b/tus-uploads/public/favicon.ico
new file mode 100644
index 00000000..8830cf68
Binary files /dev/null and b/tus-uploads/public/favicon.ico differ
diff --git a/tus-uploads/sandbox.config.json b/tus-uploads/sandbox.config.json
new file mode 100644
index 00000000..f92e0250
--- /dev/null
+++ b/tus-uploads/sandbox.config.json
@@ -0,0 +1,7 @@
+{
+  "hardReloadOnChange": true,
+  "template": "remix",
+  "container": {
+    "port": 3000
+  }
+}
diff --git a/tus-uploads/tsconfig.json b/tus-uploads/tsconfig.json
new file mode 100644
index 00000000..9d87dd37
--- /dev/null
+++ b/tus-uploads/tsconfig.json
@@ -0,0 +1,32 @@
+{
+  "include": [
+    "**/*.ts",
+    "**/*.tsx",
+    "**/.server/**/*.ts",
+    "**/.server/**/*.tsx",
+    "**/.client/**/*.ts",
+    "**/.client/**/*.tsx"
+  ],
+  "compilerOptions": {
+    "lib": ["DOM", "DOM.Iterable", "ES2022"],
+    "types": ["@remix-run/node", "vite/client"],
+    "isolatedModules": true,
+    "esModuleInterop": true,
+    "jsx": "react-jsx",
+    "module": "ESNext",
+    "moduleResolution": "Bundler",
+    "resolveJsonModule": true,
+    "target": "ES2022",
+    "strict": true,
+    "allowJs": true,
+    "skipLibCheck": true,
+    "forceConsistentCasingInFileNames": true,
+    "baseUrl": ".",
+    "paths": {
+      "~/*": ["./app/*"]
+    },
+
+    // Vite takes care of building everything, not tsc.
+    "noEmit": true
+  }
+}
diff --git a/tus-uploads/vite.config.ts b/tus-uploads/vite.config.ts
new file mode 100644
index 00000000..54066fb7
--- /dev/null
+++ b/tus-uploads/vite.config.ts
@@ -0,0 +1,16 @@
+import { vitePlugin as remix } from "@remix-run/dev";
+import { defineConfig } from "vite";
+import tsconfigPaths from "vite-tsconfig-paths";
+
+export default defineConfig({
+  plugins: [
+    remix({
+      future: {
+        v3_fetcherPersist: true,
+        v3_relativeSplatPath: true,
+        v3_throwAbortReason: true,
+      },
+    }),
+    tsconfigPaths(),
+  ],
+});