| from fastapi import APIRouter, UploadFile, File, Form, HTTPException |
| from fastapi.responses import FileResponse, JSONResponse |
| import os |
| import uuid |
| import aiofiles |
| from app.core.config import settings |
| from custom_logger import logger_config as logger |
| from app.db import crud |
| from app.services.worker import start_worker, is_worker_running |
|
|
| router = APIRouter() |
|
|
| def allowed_file(filename): |
| return '.' in filename and filename.rsplit('.', 1)[1].lower() in settings.ALLOWED_EXTENSIONS |
|
|
| @router.get("/") |
| async def index(): |
| return FileResponse('index.html') |
|
|
| @router.post("/api/tasks/upload") |
| async def upload_task(image: UploadFile = File(...), hide_from_ui: str = Form("")): |
| if not image.filename: |
| raise HTTPException(status_code=400, detail="No file selected") |
| |
| if not allowed_file(image.filename): |
| raise HTTPException(status_code=400, detail="Invalid file type") |
| |
| task_id = str(uuid.uuid4()) |
| filename = image.filename |
| filepath = os.path.join(settings.UPLOAD_FOLDER, f"{task_id}_{filename}") |
| |
| try: |
| async with aiofiles.open(filepath, 'wb') as out_file: |
| content = await image.read() |
| await out_file.write(content) |
| logger.info(f"File uploaded successfully: {filename} -> {filepath}") |
| except Exception as e: |
| logger.error(f"Error saving uploaded file {filename}: {e}") |
| raise HTTPException(status_code=500, detail="Could not save file") |
| |
| hide_from_ui_val = 1 if hide_from_ui.lower() in ['true', '1'] else 0 |
| |
| await crud.insert_task(task_id, filename, filepath, 'not_started', hide_from_ui_val) |
| |
| await start_worker() |
| |
| return JSONResponse(status_code=201, content={ |
| 'id': task_id, |
| 'filename': filename, |
| 'status': 'not_started', |
| 'message': 'File uploaded successfully' |
| }) |
|
|
| @router.get("/api/tasks") |
| async def get_tasks(): |
| rows, queue_ids, processing_count, avg_time = await crud.get_all_tasks() |
| |
| tasks = [] |
| for row in rows: |
| queue_position = None |
| estimated_start_seconds = None |
| |
| if row['status'] == 'not_started' and row['id'] in queue_ids: |
| queue_position = queue_ids.index(row['id']) + 1 |
| tasks_ahead = queue_position - 1 + processing_count |
| estimated_start_seconds = round(tasks_ahead * avg_time) |
| |
| tasks.append({ |
| 'id': row['id'], |
| 'filename': row['filename'], |
| 'status': row['status'], |
| 'result': "HIDDEN_IN_LIST_VIEW", |
| 'created_at': row['created_at'], |
| 'processed_at': row['processed_at'], |
| 'progress': row['progress'] or 0, |
| 'progress_text': row['progress_text'], |
| 'queue_position': queue_position, |
| 'estimated_start_seconds': estimated_start_seconds |
| }) |
| |
| return tasks |
|
|
| @router.get("/api/tasks/{task_id}") |
| async def get_task(task_id: str): |
| result = await crud.get_task_by_id(task_id) |
| if not result: |
| raise HTTPException(status_code=404, detail="Task not found") |
| |
| row, queue_position, estimated_start_seconds = result |
| |
| return { |
| 'id': row['id'], |
| 'filename': row['filename'], |
| 'status': row['status'], |
| 'result': row['result'], |
| 'created_at': row['created_at'], |
| 'processed_at': row['processed_at'], |
| 'progress': row['progress'] or 0, |
| 'progress_text': row['progress_text'], |
| 'queue_position': queue_position, |
| 'estimated_start_seconds': estimated_start_seconds |
| } |
|
|
| @router.get("/health") |
| async def health(): |
| return { |
| 'status': 'healthy', |
| 'service': 'ocr-runner', |
| 'worker_running': is_worker_running() |
| } |
|
|