summaryrefslogtreecommitdiff
path: root/client
diff options
context:
space:
mode:
authorBlaster4385 <venkatesh@tablaster.dev>2024-07-25 11:40:09 +0530
committerBlaster4385 <venkatesh@tablaster.dev>2024-07-25 17:16:07 +0530
commitb933d6ab405fdda250a26c86f23586da82f66fe9 (patch)
treef974f2b276ff4ed4a1cfb84f4bed6c91cad2c526 /client
parentf21cffacc308a9d43efca0185adce274d69e9d4d (diff)
feat: upload and store file in chunks to bypass network and postgres limits
Diffstat (limited to 'client')
-rw-r--r--client/assets/index.js83
1 files changed, 54 insertions, 29 deletions
diff --git a/client/assets/index.js b/client/assets/index.js
index 520c7f0..680b257 100644
--- a/client/assets/index.js
+++ b/client/assets/index.js
@@ -12,6 +12,7 @@ document.addEventListener('DOMContentLoaded', async () => {
});
const baseUrl = window.location.origin;
+const CHUNK_SIZE = 100* 1024 * 1024; // 100 MB
async function displayFileDetails(fileId, key) {
try {
@@ -79,39 +80,63 @@ function setupUploadForm() {
return;
}
- const formData = new FormData();
- formData.append('file', file);
-
try {
- const response = await fetch(`${baseUrl}/upload`, {
- method: 'POST',
- body: formData
- });
-
- const uploadResult = document.getElementById('upload__result');
-
- if (!response.ok) {
- uploadResult.textContent = `Error: ${response.statusText}`;
- uploadResult.classList.add('upload__result__visible');
- return;
- }
-
- const contentType = response.headers.get('Content-Type');
-
- if (contentType && contentType.includes('application/json')) {
- const result = await response.json();
- const pageUrl = `${baseUrl}/?id=${result.id}&key=${result.key}`;
- window.location.href = pageUrl;
- } else {
- const result = await response.text();
- uploadResult.textContent = result;
- }
-
- uploadResult.classList.add('upload__result__visible');
+ await uploadFileInChunks(file);
} catch (error) {
console.error('Error:', error);
document.getElementById('upload__result').textContent = 'An error occurred. Please try again.';
document.getElementById('upload__result').classList.add('upload__result__visible');
}
});
-} \ No newline at end of file
+}
+
+async function uploadFileInChunks(file) {
+ const fileSize = file.size;
+ const chunkCount = Math.ceil(fileSize / CHUNK_SIZE);
+ const uploadId = generateUploadId();
+
+ for (let chunkIndex = 0; chunkIndex < chunkCount; chunkIndex++) {
+ const start = chunkIndex * CHUNK_SIZE;
+ const end = Math.min(start + CHUNK_SIZE, fileSize);
+ const chunk = file.slice(start, end);
+
+ const formData = new FormData();
+ formData.append('chunk', chunk);
+ formData.append('uploadId', uploadId);
+ formData.append('chunkIndex', chunkIndex);
+ formData.append('chunkCount', chunkCount);
+ formData.append('fileName', file.name);
+
+ const response = await fetch(`${baseUrl}/upload_chunk`, {
+ method: 'POST',
+ body: formData,
+ });
+
+ if (!response.ok) {
+ throw new Error(`Error uploading chunk ${chunkIndex}: ${response.statusText}`);
+ }
+ }
+
+ // Call upload_complete endpoint
+ const completeFormData = new FormData();
+ completeFormData.append('uploadId', uploadId);
+ completeFormData.append('chunkCount', chunkCount);
+ completeFormData.append('fileName', file.name);
+
+ const completeResponse = await fetch(`${baseUrl}/upload_complete`, {
+ method: 'POST',
+ body: completeFormData,
+ });
+
+ if (!completeResponse.ok) {
+ throw new Error(`Error completing upload: ${completeResponse.statusText}`);
+ }
+
+ const result = await completeResponse.json();
+ const pageUrl = `${baseUrl}/?id=${result.id}&key=${result.key}`;
+ window.location.href = pageUrl;
+}
+
+function generateUploadId() {
+ return Math.random().toString(36).substr(2, 9);
+}