File size: 7,452 Bytes
42c2745 9b8659c 1094cbb 9b8659c 09d288e 936d8d9 9b8659c f5d4c87 42c2745 f5d4c87 a2fa160 936d8d9 9b8659c f5d4c87 42c2745 9b8659c 1094cbb 936d8d9 09d288e 9b8659c 1094cbb 42c2745 9b8659c 936d8d9 f5d4c87 936d8d9 6aa12a9 1094cbb 9b8659c 936d8d9 9b8659c 09d288e 9b8659c 42c2745 9b8659c 42c2745 9b8659c a2fa160 9b8659c 936d8d9 a2fa160 2a67404 936d8d9 42c2745 3b19076 42c2745 3b19076 42c2745 09d288e 6aa12a9 3b19076 6aa12a9 3b19076 936d8d9 3b19076 09d288e 3b19076 09d288e 9b8659c 936d8d9 a2fa160 936d8d9 3b19076 936d8d9 3b19076 936d8d9 a2fa160 2a67404 936d8d9 42c2745 9e10e0f 42c2745 936d8d9 6aa12a9 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 |
import datetime
import os
import threading
import pandas as pd
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.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.getenv("COMPETITION_ID")
OUTPUT_PATH = os.getenv("OUTPUT_PATH", "/tmp/model")
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()
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:
"""
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("/competition_info", response_class=JSONResponse)
async def get_comp_info(request: Request):
info = COMP_INFO.competition_desc
# info = markdown.markdown(info)
resp = {"response": info}
return resp
@app.get("/dataset_info", response_class=JSONResponse)
async def get_dataset_info(request: Request):
info = COMP_INFO.dataset_desc
# info = markdown.markdown(info)
resp = {"response": info}
return resp
@app.get("/submission_info", response_class=JSONResponse)
async def get_submission_info(request: Request):
info = COMP_INFO.submission_desc
# info = markdown.markdown(info)
resp = {"response": info}
return resp
@app.get("/leaderboard/{lb}", response_class=JSONResponse)
async def get_leaderboard(request: Request, lb: str):
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,
)
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)
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:
success_subs, failed_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 = pd.concat([success_subs, failed_subs], axis=0)
subs = subs.to_dict(orient="records")
error = ""
if len(subs) == 0:
error = "**You have not made any submissions yet.**"
failed_subs = ""
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 = ""
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."}
elif 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": ""}
|