open-webui/backend/apps/web/routers/utils.py

164 lines
4.9 KiB
Python
Raw Normal View History

2023-12-23 23:38:52 +00:00
from fastapi import APIRouter, UploadFile, File, BackgroundTasks
from fastapi import Depends, HTTPException, status
from starlette.responses import StreamingResponse
from pydantic import BaseModel
import requests
import os
2023-12-24 07:40:14 +00:00
import aiohttp
2023-12-23 23:38:52 +00:00
import json
2023-12-24 07:40:14 +00:00
from utils.misc import calculate_sha256
2023-12-23 23:38:52 +00:00
from config import OLLAMA_API_BASE_URL
router = APIRouter()
class UploadBlobForm(BaseModel):
filename: str
2023-12-24 07:05:52 +00:00
from urllib.parse import urlparse
def parse_huggingface_url(hf_url):
2023-12-24 08:27:04 +00:00
try:
# Parse the URL
parsed_url = urlparse(hf_url)
2023-12-24 07:05:52 +00:00
2023-12-24 08:27:04 +00:00
# Get the path and split it into components
path_components = parsed_url.path.split("/")
2023-12-24 07:05:52 +00:00
2023-12-24 08:27:04 +00:00
# Extract the desired output
user_repo = "/".join(path_components[1:3])
model_file = path_components[-1]
2023-12-24 07:05:52 +00:00
2023-12-24 08:27:04 +00:00
return model_file
except ValueError:
return None
2023-12-24 07:05:52 +00:00
2023-12-24 07:40:14 +00:00
async def download_file_stream(url, file_path, chunk_size=1024 * 1024):
2023-12-24 07:05:52 +00:00
done = False
if os.path.exists(file_path):
current_size = os.path.getsize(file_path)
else:
current_size = 0
headers = {"Range": f"bytes={current_size}-"} if current_size > 0 else {}
2023-12-24 08:27:04 +00:00
timeout = aiohttp.ClientTimeout(total=600) # Set the timeout
2023-12-24 07:40:14 +00:00
async with aiohttp.ClientSession(timeout=timeout) as session:
async with session.get(url, headers=headers) as response:
total_size = int(response.headers.get("content-length", 0)) + current_size
with open(file_path, "ab+") as file:
async for data in response.content.iter_chunked(chunk_size):
current_size += len(data)
file.write(data)
done = current_size == total_size
progress = round((current_size / total_size) * 100, 2)
2023-12-24 08:27:04 +00:00
yield f'data: {{"progress": {progress}, "completed": {current_size}, "total": {total_size}}}\n\n'
2023-12-24 07:40:14 +00:00
if done:
file.seek(0)
hashed = calculate_sha256(file)
file.seek(0)
2023-12-24 07:05:52 +00:00
2023-12-24 07:40:14 +00:00
url = f"{OLLAMA_API_BASE_URL}/blobs/sha256:{hashed}"
response = requests.post(url, data=file)
2023-12-24 07:05:52 +00:00
2023-12-24 07:40:14 +00:00
if response.ok:
res = {
"done": done,
"blob": f"sha256:{hashed}",
2023-12-24 08:27:04 +00:00
"name": file.name,
2023-12-24 07:40:14 +00:00
}
os.remove(file_path)
2023-12-24 07:05:52 +00:00
2023-12-24 07:40:14 +00:00
yield f"data: {json.dumps(res)}\n\n"
else:
raise "Ollama: Could not create blob, Please try again."
2023-12-24 07:05:52 +00:00
@router.get("/download")
async def download(
2023-12-24 08:27:04 +00:00
url: str,
2023-12-24 07:05:52 +00:00
):
2023-12-24 08:27:04 +00:00
# url = "https://huggingface.co/TheBloke/stablelm-zephyr-3b-GGUF/resolve/main/stablelm-zephyr-3b.Q2_K.gguf"
model_file = parse_huggingface_url(url)
2023-12-24 07:05:52 +00:00
2023-12-24 08:27:04 +00:00
if model_file:
os.makedirs("./uploads", exist_ok=True)
file_path = os.path.join("./uploads", f"{model_file}")
2023-12-24 07:05:52 +00:00
2023-12-24 08:27:04 +00:00
return StreamingResponse(
download_file_stream(url, file_path), media_type="text/event-stream"
)
else:
return None
2023-12-24 07:05:52 +00:00
2023-12-23 23:38:52 +00:00
@router.post("/upload")
async def upload(file: UploadFile = File(...)):
os.makedirs("./uploads", exist_ok=True)
file_path = os.path.join("./uploads", file.filename)
2023-12-24 07:05:52 +00:00
async def file_write_stream():
2023-12-23 23:38:52 +00:00
total = 0
total_size = file.size
chunk_size = 1024 * 1024
done = False
try:
2023-12-24 07:05:52 +00:00
with open(file_path, "wb+") as f:
2023-12-23 23:38:52 +00:00
while True:
chunk = file.file.read(chunk_size)
if not chunk:
break
f.write(chunk)
total += len(chunk)
done = total_size == total
2023-12-24 08:27:04 +00:00
progress = round((total / total_size) * 100, 2)
2023-12-23 23:38:52 +00:00
res = {
2023-12-24 08:27:04 +00:00
"progress": progress,
2023-12-23 23:38:52 +00:00
"total": total_size,
2023-12-24 08:27:04 +00:00
"completed": total,
2023-12-23 23:38:52 +00:00
}
yield f"data: {json.dumps(res)}\n\n"
if done:
2023-12-24 07:05:52 +00:00
f.seek(0)
hashed = calculate_sha256(f)
f.seek(0)
url = f"{OLLAMA_API_BASE_URL}/blobs/sha256:{hashed}"
response = requests.post(url, data=f)
if response.ok:
res = {
"done": done,
"blob": f"sha256:{hashed}",
2023-12-24 08:27:04 +00:00
"name": file.filename,
2023-12-24 07:05:52 +00:00
}
os.remove(file_path)
yield f"data: {json.dumps(res)}\n\n"
else:
raise "Ollama: Could not create blob, Please try again."
2023-12-23 23:38:52 +00:00
except Exception as e:
res = {"error": str(e)}
yield f"data: {json.dumps(res)}\n\n"
return StreamingResponse(file_write_stream(), media_type="text/event-stream")