TGDRIVE / main.py
dragxd's picture
Fix: Update port to 7860 for Hugging Face Spaces and improve offline mode logging
f2d1bff
from utils.downloader import (
download_file,
get_file_info_from_url,
)
import asyncio
from pathlib import Path
from contextlib import asynccontextmanager
import aiofiles
from fastapi import FastAPI, HTTPException, Request, File, UploadFile, Form, Response
from fastapi.responses import FileResponse, JSONResponse
from config import ADMIN_PASSWORD, MAX_FILE_SIZE, STORAGE_CHANNEL
from utils.clients import initialize_clients
from utils.directoryHandler import getRandomID
from utils.extra import auto_ping_website, convert_class_to_dict, reset_cache_dir
from utils.streamer import media_streamer
from utils.uploader import start_file_uploader
from utils.logger import Logger
import urllib.parse
# Startup Event
@asynccontextmanager
async def lifespan(app: FastAPI):
# Reset the cache directory, delete cache files
reset_cache_dir()
# Initialize the clients
clients_initialized = await initialize_clients()
# If clients failed to initialize, initialize DRIVE_DATA in offline mode
if not clients_initialized:
logger.warning("No Telegram clients available - Initializing in OFFLINE MODE")
from utils.directoryHandler import initDriveDataWithoutClients
await initDriveDataWithoutClients()
logger.warning("✅ Website is running in OFFLINE MODE - Telegram features are disabled")
logger.info("✅ Website UI is available - File operations requiring Telegram will show errors")
# Start the website auto ping task
asyncio.create_task(auto_ping_website())
yield
app = FastAPI(docs_url=None, redoc_url=None, lifespan=lifespan)
logger = Logger(__name__)
@app.get("/")
async def home_page():
return FileResponse("website/home.html")
@app.get("/stream")
async def home_page():
return FileResponse("website/VideoPlayer.html")
@app.get("/static/{file_path:path}")
async def static_files(file_path):
if "apiHandler.js" in file_path:
with open(Path("website/static/js/apiHandler.js")) as f:
content = f.read()
content = content.replace("MAX_FILE_SIZE__SDGJDG", str(MAX_FILE_SIZE))
return Response(content=content, media_type="application/javascript")
return FileResponse(f"website/static/{file_path}")
@app.get("/file")
async def dl_file(request: Request):
from utils.directoryHandler import DRIVE_DATA
from utils.clients import has_clients
if not has_clients():
raise HTTPException(status_code=503, detail="Telegram clients not available - Service running in offline mode")
path = request.query_params["path"]
file = DRIVE_DATA.get_file(path)
return await media_streamer(STORAGE_CHANNEL, file.file_id, file.name, request)
# Api Routes
@app.post("/api/checkPassword")
async def check_password(request: Request):
data = await request.json()
if data["pass"] == ADMIN_PASSWORD:
return JSONResponse({"status": "ok"})
return JSONResponse({"status": "Invalid password"})
@app.post("/api/createNewFolder")
async def api_new_folder(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"createNewFolder {data}")
folder_data = DRIVE_DATA.get_directory(data["path"]).contents
for id in folder_data:
f = folder_data[id]
if f.type == "folder":
if f.name == data["name"]:
return JSONResponse(
{
"status": "Folder with the name already exist in current directory"
}
)
DRIVE_DATA.new_folder(data["path"], data["name"])
return JSONResponse({"status": "ok"})
@app.post("/api/getDirectory")
async def api_get_directory(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] == ADMIN_PASSWORD:
is_admin = True
else:
is_admin = False
auth = data.get("auth")
logger.info(f"getFolder {data}")
if data["path"] == "/trash":
data = {"contents": DRIVE_DATA.get_trashed_files_folders()}
folder_data = convert_class_to_dict(data, isObject=False, showtrash=True)
elif "/search_" in data["path"]:
query = urllib.parse.unquote(data["path"].split("_", 1)[1])
print(query)
data = {"contents": DRIVE_DATA.search_file_folder(query)}
print(data)
folder_data = convert_class_to_dict(data, isObject=False, showtrash=False)
print(folder_data)
elif "/share_" in data["path"]:
path = data["path"].split("_", 1)[1]
folder_data, auth_home_path = DRIVE_DATA.get_directory(path, is_admin, auth)
auth_home_path= auth_home_path.replace("//", "/") if auth_home_path else None
folder_data = convert_class_to_dict(folder_data, isObject=True, showtrash=False)
return JSONResponse(
{"status": "ok", "data": folder_data, "auth_home_path": auth_home_path}
)
else:
folder_data = DRIVE_DATA.get_directory(data["path"])
folder_data = convert_class_to_dict(folder_data, isObject=True, showtrash=False)
return JSONResponse({"status": "ok", "data": folder_data, "auth_home_path": None})
SAVE_PROGRESS = {}
@app.post("/api/upload")
async def upload_file(
file: UploadFile = File(...),
path: str = Form(...),
password: str = Form(...),
id: str = Form(...),
total_size: str = Form(...),
):
global SAVE_PROGRESS
from utils.clients import has_clients
if password != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
if not has_clients():
return JSONResponse({"status": "Telegram clients not available - Service running in offline mode"})
total_size = int(total_size)
SAVE_PROGRESS[id] = ("running", 0, total_size)
ext = file.filename.lower().split(".")[-1]
cache_dir = Path("./cache")
cache_dir.mkdir(parents=True, exist_ok=True)
file_location = cache_dir / f"{id}.{ext}"
file_size = 0
async with aiofiles.open(file_location, "wb") as buffer:
while chunk := await file.read(1024 * 1024): # Read file in chunks of 1MB
SAVE_PROGRESS[id] = ("running", file_size, total_size)
file_size += len(chunk)
if file_size > MAX_FILE_SIZE:
await buffer.close()
file_location.unlink() # Delete the partially written file
raise HTTPException(
status_code=400,
detail=f"File size exceeds {MAX_FILE_SIZE} bytes limit",
)
await buffer.write(chunk)
SAVE_PROGRESS[id] = ("completed", file_size, file_size)
asyncio.create_task(
start_file_uploader(file_location, id, path, file.filename, file_size)
)
return JSONResponse({"id": id, "status": "ok"})
@app.post("/api/getSaveProgress")
async def get_save_progress(request: Request):
global SAVE_PROGRESS
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"getUploadProgress {data}")
try:
progress = SAVE_PROGRESS[data["id"]]
return JSONResponse({"status": "ok", "data": progress})
except:
return JSONResponse({"status": "not found"})
@app.post("/api/getUploadProgress")
async def get_upload_progress(request: Request):
from utils.uploader import PROGRESS_CACHE
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"getUploadProgress {data}")
try:
progress = PROGRESS_CACHE[data["id"]]
return JSONResponse({"status": "ok", "data": progress})
except:
return JSONResponse({"status": "not found"})
@app.post("/api/cancelUpload")
async def cancel_upload(request: Request):
from utils.uploader import STOP_TRANSMISSION
from utils.downloader import STOP_DOWNLOAD
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"cancelUpload {data}")
STOP_TRANSMISSION.append(data["id"])
STOP_DOWNLOAD.append(data["id"])
return JSONResponse({"status": "ok"})
@app.post("/api/renameFileFolder")
async def rename_file_folder(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"renameFileFolder {data}")
DRIVE_DATA.rename_file_folder(data["path"], data["name"])
return JSONResponse({"status": "ok"})
@app.post("/api/trashFileFolder")
async def trash_file_folder(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"trashFileFolder {data}")
DRIVE_DATA.trash_file_folder(data["path"], data["trash"])
return JSONResponse({"status": "ok"})
@app.post("/api/deleteFileFolder")
async def delete_file_folder(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"deleteFileFolder {data}")
DRIVE_DATA.delete_file_folder(data["path"])
return JSONResponse({"status": "ok"})
@app.post("/api/getFileInfoFromUrl")
async def getFileInfoFromUrl(request: Request):
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"getFileInfoFromUrl {data}")
try:
file_info = await get_file_info_from_url(data["url"])
return JSONResponse({"status": "ok", "data": file_info})
except Exception as e:
return JSONResponse({"status": str(e)})
@app.post("/api/startFileDownloadFromUrl")
async def startFileDownloadFromUrl(request: Request):
from utils.clients import has_clients
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
if not has_clients():
return JSONResponse({"status": "Telegram clients not available - Service running in offline mode"})
logger.info(f"startFileDownloadFromUrl {data}")
try:
id = getRandomID()
asyncio.create_task(
download_file(data["url"], id, data["path"], data["filename"], data["singleThreaded"])
)
return JSONResponse({"status": "ok", "id": id})
except Exception as e:
return JSONResponse({"status": str(e)})
@app.post("/api/getFileDownloadProgress")
async def getFileDownloadProgress(request: Request):
from utils.downloader import DOWNLOAD_PROGRESS
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"getFileDownloadProgress {data}")
try:
progress = DOWNLOAD_PROGRESS[data["id"]]
return JSONResponse({"status": "ok", "data": progress})
except:
return JSONResponse({"status": "not found"})
@app.post("/api/getFolderShareAuth")
async def getFolderShareAuth(request: Request):
from utils.directoryHandler import DRIVE_DATA
data = await request.json()
if data["password"] != ADMIN_PASSWORD:
return JSONResponse({"status": "Invalid password"})
logger.info(f"getFolderShareAuth {data}")
try:
auth = DRIVE_DATA.get_folder_auth(data["path"])
return JSONResponse({"status": "ok", "auth": auth})
except:
return JSONResponse({"status": "not found"})