Skip to content

Instantly share code, notes, and snippets.

@benixal
Last active February 11, 2025 10:16
Show Gist options
  • Save benixal/b3ff3666afe5055d57fa6ab0fc1c1183 to your computer and use it in GitHub Desktop.
Save benixal/b3ff3666afe5055d57fa6ab0fc1c1183 to your computer and use it in GitHub Desktop.
Free Telegram Upload Center
/*
https://github.com/benixal
https://www.youtube.com/@benixal
*/
export default {
// @ts-ignore
async fetch(request, env, ctx) {
const botToken = '<bot-token>';
const chatId = '<chat-id>';
const url = new URL(request.url);
const domain = url.hostname;
const pathParts = url.pathname.split('/');
if (request.method === 'GET' && url.pathname === '/') {
// @ts-ignore
const isBotTokenInvalid = !botToken || botToken === '<bot-token>';
// @ts-ignore
const isChatIdInvalid = !chatId || chatId === '<chat-id>';
let warningMessage = '';
if (isBotTokenInvalid) {
warningMessage += '<p style="color: red;">Warning: botToken is not defined</p>';
}
if (isChatIdInvalid) {
warningMessage += '<p style="color: red;">Warning: chatId is not defined</p>';
}
const htmlForm = `<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>File Upload</title>
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
background-color: #15181e;
color: white;
font-family: 'Arial', sans-serif;
display: flex;
justify-content: center;
align-items: center;
height: 100vh;
padding: 20px;
}
.container {
background-color: #1f2229;
padding: 40px;
border-radius: 10px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.5);
text-align: center;
}
h1 {
font-size: 2rem;
margin-bottom: 20px;
}
input[type="file"] {
display: block;
margin: 20px auto;
padding: 1rem;
font-size: 1.2rem;
background-color: #2b3038;
color: white;
border: none;
border-radius: 5px;
width: 100%;
max-width: 400px;
cursor: pointer;
}
button {
font-size: 1.5rem;
padding: 1rem 2rem;
margin-top: 20px;
background-color: #4CAF50;
color: white;
border: none;
border-radius: 5px;
cursor: pointer;
transition: background-color 0.3s ease;
}
button:hover {
background-color: #45a049;
}
input[type="file"]:hover {
background-color: #3b4048;
}
</style>
</head>
<body>
<div class="container">
<h1>Upload a File</h1>
${warningMessage}
<form action="/upload" method="POST" enctype="multipart/form-data">
<input type="file" name="file" id="fileInput" accept="image/*,video/*" required />
<div id="previewContainer" style="margin: 20px 0; text-align: center;">
<img id="imagePreview" style="max-width: 100%; max-height: 300px; display: none;" />
<video id="videoPreview" controls style="max-width: 100%; max-height: 300px; display: none;"></video>
</div>
<button type="submit">Upload</button>
</form>
<script>
const fileInput = document.getElementById('fileInput');
const imagePreview = document.getElementById('imagePreview');
const videoPreview = document.getElementById('videoPreview');
const previewContainer = document.getElementById('previewContainer');
fileInput.addEventListener('change', () => {
const file = fileInput.files[0];
if (!file) return;
const fileType = file.type;
const previewURL = URL.createObjectURL(file);
// Reset previews
imagePreview.style.display = 'none';
videoPreview.style.display = 'none';
if (fileType.startsWith('image/')) {
imagePreview.src = previewURL;
imagePreview.style.display = 'block';
} else if (fileType.startsWith('video/')) {
videoPreview.src = previewURL;
videoPreview.style.display = 'block';
}
});
</script>
</div>
</body>
</html>
`;
return new Response(htmlForm, {
headers: { 'Content-Type': 'text/html' }
});
}
if (request.method === 'GET' && pathParts[1] === 'init') {
const telegramResponse = await postReq("setWebhook", [
{ "url": `https://${domain}/hook` }
]);
const telegramResult = await telegramResponse.text();
return new Response(telegramResult);
}
// Handle file download request
if (pathParts[1] === 'download' && pathParts[2]) {
const fileResponse = await postReq(`getFile`, [
{ "file_id": pathParts[2] }
]);
const fileData = await fileResponse.json();
const telegramFileResponse = await fetch(`https://api.telegram.org/file/bot${botToken}/${fileData.result.file_path}`);
// Modify response headers
const newHeaders = new Headers(telegramFileResponse.headers);
// Optionally change Content-Type if it's 'application/octet-stream'
const contentType = newHeaders.get('Content-Type');
if (contentType === 'application/octet-stream') {
newHeaders.set('Content-Type', ''); // Adjust or leave blank
}
newHeaders.delete('Content-Disposition'); // Prevent forced download
return new Response(telegramFileResponse.body, { headers: newHeaders });
}
function extractFileIds(obj) {
const fileIds = [];
function searchForFileIds(item) {
if (item && typeof item === 'object') {
// Check if the item has 'file_id'
if (item.file_id) fileIds.push(item.file_id);
// Recursively search through all object properties
Object.values(item).forEach(searchForFileIds);
} else if (Array.isArray(item)) {
item.forEach(searchForFileIds);
}
}
searchForFileIds(obj);
const uniquefileIds = [...new Set(fileIds)];
return uniquefileIds;
}
async function postReq(url, fields) {
const tgFormData = new FormData();
fields.forEach(obj => {
for (let key in obj) {
tgFormData.append(key, obj[key]);
}
});
const telegramResponse = await fetch(`https://api.telegram.org/bot${botToken}/${url}`, {
method: 'POST',
body: tgFormData,
});
return await telegramResponse;
}
if (url.pathname === '/hook' && ['POST', 'PUT'].includes(request.method)) {
const json = await request.json();
const fileIds = extractFileIds(json);
if (fileIds.length > 0) {
const downloadLinks = await Promise.all(
fileIds.map(async (fid) => {
const fileResponse = await postReq(`getFile`, [
{ "file_id": fid }
]);
const fileData = await fileResponse.json();
if (fileData.ok) {
return `https://${domain}/download/${fid}/${fileData.result.file_path}`;
} else {
await postReq(`sendMessage`, [
{ "chat_id": json.message.from.id },
{ "text": "error" },
{ "parse_mode": "MarkdownV2" },
{ "reply_to_message_id": json.message.message_id }
])
return false;
}
})
);
let msg = [];
for (const item of downloadLinks) {
if (item) {
msg.push(`Download Link: \`${item}\``);
}
}
if (msg.length > 0) {
await postReq(`sendMessage`, [
{ "chat_id": json.message.from.id },
{ "text": msg.join("\n\n") },
{ "parse_mode": "MarkdownV2" },
{ "reply_to_message_id": json.message.message_id }
])
}
} else {
if ('text' in json.message && json.message.text.toLowerCase().includes('chatid')) {
await postReq(`sendMessage`, [
{ "chat_id": json.message.from.id },
{ "text": `your chatId is: \`${json.message.from.id}\`` },
{ "parse_mode": "MarkdownV2" },
{ "reply_to_message_id": json.message.message_id }
])
} else if ('text' in json.message && json.message.text.includes('/start')) {
await postReq(`sendMessage`, [
{ "chat_id": json.message.from.id },
{ "text": "Welcome!" },
])
} else {
await postReq(`sendMessage`, [
{ "chat_id": json.message.from.id },
{ "text": "send me a file" },
{ "reply_to_message_id": json.message.message_id }
])
}
}
return new Response("");
}
if (url.pathname === '/upload' && request.method === 'POST') {
const formData = await request.formData();
const file = formData.get('file');
if (file) {
const sendFileToChat = await postReq("sendDocument", [
{ "chat_id": chatId },
{ "document": file }
])
const sendFileToChatResponse = await sendFileToChat.json();
if (sendFileToChat.ok) {
const fileIds = extractFileIds(sendFileToChatResponse);
const downloadLinks = await Promise.all(
fileIds.map(async (fid) => {
const fileResponse = await postReq(`getFile`, [
{ "file_id": fid }
]);
const fileData = await fileResponse.json();
if (fileData.ok) {
return `https://${domain}/download/${fid}/${fileData.result.file_path}`;
} else {
await postReq(`sendMessage`, [
{ "chat_id": chatId },
{ "text": "error" },
{ "parse_mode": "MarkdownV2" }
])
return false;
}
})
);
let msg = [];
for (const itemx of downloadLinks) {
if (itemx) {
msg.push(`Download Link: \`${itemx}\``);
}
}
if (msg.length > 0) {
await postReq("editMessageCaption", [
{ "chat_id": chatId },
{ "message_id": sendFileToChatResponse['result']['message_id'] },
{ "parse_mode": "MarkdownV2" },
{ "caption": msg.join("\n\n") }
])
}
const html = `
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Uploads</title>
<style>
body {
background-color: #15181e;
}
.container {
display: flex;
flex-direction: column;
gap: 1rem;
padding: 1rem;
}
.item {
display: flex;
flex-direction: column;
text-align: left;
padding: 1rem;
color: white;
border-radius: 8px;
border: 1px solid green;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
background-color: white;
}
.link {
color: black;
text-align: left;
padding: 1rem;
letter-spacing: 1px;
font-size: 14px;
}
.dllink {
color: green;
margin: 1rem 1rem 0rem 1rem;
font-weight: bold;
font-size: 1.5rem;
}
.image-preview {
max-width: 100px;
max-height: 100px;
border: 1px solid #ddd;
border-radius: 8px;
}
.image-size {
font-size: 12px;
color: black;
margin-top: 0.5rem;
}
</style>
</head>
<body>
<h1 style="color:#c0b6ff;margin-left:2rem">My Uploads</h1>
<div class="container">
${downloadLinks.map(url => {
// Check file extension for image types
const fileExtension = url.split('.').pop().toLowerCase();
const imageExtensions = ['jpg', 'jpeg', 'png', 'webp'];
const isImage = imageExtensions.includes(fileExtension);
// Set the preview image, either the actual image or a generic icon
const previewImage = isImage ? url : '';
return `
<div class="item">
<img src="${previewImage}" class="image-preview" alt="File Preview" onload="displayImageSize(this)">
<div class="image-size" data-size></div>
<div class="dllink">Download Link:</div>
<div class="link">${url}</div>
</div>`;
}).join('')}
</div>
<script>
function displayImageSize(imgElement) {
if (imgElement.src.startsWith('data:image/png;base64')) return; // Skip for non-images
const width = imgElement.naturalWidth;
const height = imgElement.naturalHeight;
const sizeElement = imgElement.parentElement.querySelector('[data-size]');
sizeElement.textContent = \`\${width} X \${height}\`;
}
</script>
</body>
</html>
`;
return new Response(html, {
headers: { 'content-type': 'text/html;charset=UTF-8' },
});
} else {
return new Response(`Failed`, { status: 500 });
}
} else {
return new Response('No file uploaded', { status: 400 });
}
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment