| import datetime |
| import os |
| import threading |
|
|
| from fastapi import FastAPI, File, Form, Request, UploadFile |
| from fastapi.responses import HTMLResponse, JSONResponse |
| from fastapi.staticfiles import StaticFiles |
| from fastapi.templating import Jinja2Templates |
| from huggingface_hub import hf_hub_download |
| from huggingface_hub.utils import disable_progress_bars |
| from huggingface_hub.utils._errors import EntryNotFoundError |
| from loguru import logger |
| from pydantic import BaseModel |
|
|
| from competitions import utils |
| from competitions.errors import AuthenticationError |
| from competitions.info import CompetitionInfo |
| from competitions.leaderboard import Leaderboard |
| from competitions.oauth import attach_oauth |
| from competitions.runner import JobRunner |
| from competitions.submissions import Submissions |
| from competitions.text import SUBMISSION_SELECTION_TEXT, SUBMISSION_TEXT |
|
|
|
|
| HF_TOKEN = os.environ.get("HF_TOKEN", None) |
| BASE_DIR = os.path.dirname(os.path.abspath(__file__)) |
| COMPETITION_ID = os.environ.get("COMPETITION_ID") |
| OUTPUT_PATH = os.environ.get("OUTPUT_PATH", "/tmp/model") |
| START_DATE = os.environ.get("START_DATE", "2000-12-31") |
| DISABLE_PUBLIC_LB = int(os.environ.get("DISABLE_PUBLIC_LB", 0)) |
| USE_OAUTH = int(os.environ.get("USE_OAUTH", 0)) |
|
|
| disable_progress_bars() |
|
|
| COMP_INFO = CompetitionInfo(competition_id=COMPETITION_ID, autotrain_token=HF_TOKEN) |
|
|
|
|
| try: |
| REQUIREMENTS_FNAME = hf_hub_download( |
| repo_id=COMPETITION_ID, |
| filename="requirements.txt", |
| token=HF_TOKEN, |
| repo_type="dataset", |
| ) |
| except EntryNotFoundError: |
| REQUIREMENTS_FNAME = None |
|
|
| if REQUIREMENTS_FNAME: |
| logger.info("Uninstalling and installing requirements") |
| utils.uninstall_requirements(REQUIREMENTS_FNAME) |
| utils.install_requirements(REQUIREMENTS_FNAME) |
|
|
|
|
| class User(BaseModel): |
| user_token: str |
|
|
|
|
| class UserSubmissionUpdate(BaseModel): |
| user_token: str |
| submission_ids: str |
|
|
|
|
| def run_job_runner(): |
| job_runner = JobRunner(token=HF_TOKEN, competition_info=COMP_INFO, output_path=OUTPUT_PATH) |
| job_runner.run() |
|
|
|
|
| thread = threading.Thread(target=run_job_runner) |
| thread.start() |
|
|
|
|
| app = FastAPI() |
| if USE_OAUTH == 1: |
| attach_oauth(app) |
|
|
| static_path = os.path.join(BASE_DIR, "static") |
| app.mount("/static", StaticFiles(directory=static_path), name="static") |
| templates_path = os.path.join(BASE_DIR, "templates") |
| templates = Jinja2Templates(directory=templates_path) |
|
|
|
|
| @app.get("/", response_class=HTMLResponse) |
| async def read_form(request: Request): |
| """ |
| This function is used to render the HTML file |
| :param request: |
| :return: |
| """ |
| logger.info(request.session.get("oauth_info")) |
| if HF_TOKEN is None: |
| return templates.TemplateResponse("error.html", {"request": request}) |
| context = { |
| "request": request, |
| "logo": COMP_INFO.logo_url, |
| "competition_type": COMP_INFO.competition_type, |
| } |
| return templates.TemplateResponse("index.html", context) |
|
|
|
|
| @app.get("/use_oauth", response_class=JSONResponse) |
| async def use_oauth(request: Request): |
| return {"response": USE_OAUTH} |
|
|
|
|
| @app.get("/competition_info", response_class=JSONResponse) |
| async def get_comp_info(request: Request): |
| info = COMP_INFO.competition_desc |
| |
| resp = {"response": info} |
| return resp |
|
|
|
|
| @app.get("/dataset_info", response_class=JSONResponse) |
| async def get_dataset_info(request: Request): |
| info = COMP_INFO.dataset_desc |
| |
| resp = {"response": info} |
| return resp |
|
|
|
|
| @app.get("/submission_info", response_class=JSONResponse) |
| async def get_submission_info(request: Request): |
| info = COMP_INFO.submission_desc |
| |
| resp = {"response": info} |
| return resp |
|
|
|
|
| @app.get("/leaderboard/{lb}", response_class=JSONResponse) |
| async def get_leaderboard(request: Request, lb: str): |
| if DISABLE_PUBLIC_LB == 1 and lb == "public": |
| return {"response": "Public leaderboard is disabled by the competition host."} |
|
|
| leaderboard = Leaderboard( |
| end_date=COMP_INFO.end_date, |
| eval_higher_is_better=COMP_INFO.eval_higher_is_better, |
| max_selected_submissions=COMP_INFO.selection_limit, |
| competition_id=COMPETITION_ID, |
| token=HF_TOKEN, |
| scoring_metric=COMP_INFO.scoring_metric, |
| ) |
| if lb == "private": |
| current_utc_time = datetime.datetime.utcnow() |
| if current_utc_time < COMP_INFO.end_date: |
| return {"response": "Private leaderboard will be available after the competition ends."} |
| df = leaderboard.fetch(private=lb == "private") |
| logger.info(df) |
| if len(df) == 0: |
| return {"response": "No teams yet. Why not make a submission?"} |
| resp = {"response": df.to_markdown(index=False)} |
| return resp |
|
|
|
|
| @app.post("/my_submissions", response_class=JSONResponse) |
| async def my_submissions(request: Request, user: User): |
| sub = Submissions( |
| end_date=COMP_INFO.end_date, |
| submission_limit=COMP_INFO.submission_limit, |
| competition_id=COMPETITION_ID, |
| token=HF_TOKEN, |
| competition_type=COMP_INFO.competition_type, |
| hardware=COMP_INFO.hardware, |
| ) |
| try: |
| subs = sub.my_submissions(user.user_token) |
| except AuthenticationError: |
| return { |
| "response": { |
| "submissions": "", |
| "submission_text": SUBMISSION_TEXT.format(COMP_INFO.submission_limit), |
| "error": "**Invalid token**", |
| } |
| } |
| subs = subs.to_dict(orient="records") |
| logger.info(subs) |
| error = "" |
| if len(subs) == 0: |
| error = "**You have not made any submissions yet.**" |
| subs = "" |
| submission_text = SUBMISSION_TEXT.format(COMP_INFO.submission_limit) |
| submission_selection_text = SUBMISSION_SELECTION_TEXT.format(COMP_INFO.selection_limit) |
|
|
| resp = { |
| "response": { |
| "submissions": subs, |
| "submission_text": submission_text + submission_selection_text, |
| "error": error, |
| } |
| } |
| return resp |
|
|
|
|
| @app.post("/new_submission", response_class=JSONResponse) |
| async def new_submission( |
| submission_file: UploadFile = File(None), |
| hub_model: str = Form(...), |
| token: str = Form(...), |
| submission_comment: str = Form(None), |
| ): |
| if submission_comment is None: |
| submission_comment = "" |
|
|
| todays_date = datetime.datetime.now() |
| start_date = datetime.datetime.strptime(START_DATE, "%Y-%m-%d") |
| if todays_date < start_date: |
| comp_org = COMPETITION_ID.split("/")[0] |
| if not utils.can_user_submit_before_start(token, comp_org): |
| return {"response": "Competition has not started yet!"} |
|
|
| sub = Submissions( |
| end_date=COMP_INFO.end_date, |
| submission_limit=COMP_INFO.submission_limit, |
| competition_id=COMPETITION_ID, |
| token=HF_TOKEN, |
| competition_type=COMP_INFO.competition_type, |
| hardware=COMP_INFO.hardware, |
| ) |
| try: |
| if COMP_INFO.competition_type == "generic": |
| resp = sub.new_submission(token, submission_file, submission_comment) |
| return {"response": f"Success! You have {resp} submissions remaining today."} |
| if COMP_INFO.competition_type == "script": |
| resp = sub.new_submission(token, hub_model, submission_comment) |
| return {"response": f"Success! You have {resp} submissions remaining today."} |
| except AuthenticationError: |
| return {"response": "Invalid token"} |
| return {"response": "Invalid competition type"} |
|
|
|
|
| @app.post("/update_selected_submissions", response_class=JSONResponse) |
| def update_selected_submissions(request: Request, user_sub: UserSubmissionUpdate): |
| sub = Submissions( |
| end_date=COMP_INFO.end_date, |
| submission_limit=COMP_INFO.submission_limit, |
| competition_id=COMPETITION_ID, |
| token=HF_TOKEN, |
| competition_type=COMP_INFO.competition_type, |
| hardware=COMP_INFO.hardware, |
| ) |
| submission_ids = user_sub.submission_ids.split(",") |
| submission_ids = [s.strip() for s in submission_ids] |
| if len(submission_ids) > COMP_INFO.selection_limit: |
| return { |
| "success": False, |
| "error": f"Please select at most {COMP_INFO.selection_limit} submissions.", |
| } |
| sub.update_selected_submissions(user_token=user_sub.user_token, selected_submission_ids=submission_ids) |
| return {"success": True, "error": ""} |
|
|