| |
| import os |
| import json |
| import logging |
| import requests |
| from urllib3.util.retry import Retry |
| from requests.adapters import HTTPAdapter |
|
|
| |
| BACKEND = (os.getenv("BACKEND_URL") or "").strip().rstrip("/") |
| if not BACKEND: |
| raise RuntimeError("BACKEND_URL is not set in Space secrets.") |
|
|
| TOKEN = (os.getenv("BACKEND_TOKEN") or os.getenv("HF_TOKEN") or "").strip() |
| DEFAULT_TIMEOUT = int(os.getenv("BACKEND_TIMEOUT", "120")) |
|
|
| _session = requests.Session() |
| retry = Retry( |
| total=3, |
| connect=3, |
| read=3, |
| backoff_factor=0.5, |
| status_forcelist=(429, 500, 502, 503, 504), |
| allowed_methods=frozenset(["GET", "POST", "PUT", "PATCH", "DELETE"]), |
| ) |
| _session.mount("https://", HTTPAdapter(max_retries=retry)) |
| _session.mount("http://", HTTPAdapter(max_retries=retry)) |
| _session.headers.update({ |
| "Accept": "application/json, */*;q=0.1", |
| "User-Agent": "FInFront/1.0 (+spaces)", |
| }) |
| if TOKEN: |
| _session.headers["Authorization"] = f"Bearer {TOKEN}" |
|
|
| logging.basicConfig(level=logging.INFO) |
| logger = logging.getLogger(__name__) |
|
|
| |
| API_PREFIX_ENV = (os.getenv("BACKEND_API_PREFIX") or "").strip().rstrip("/") |
| def _prefixes(): |
| return ["/" + API_PREFIX_ENV.strip("/"), ""] if API_PREFIX_ENV else [""] |
|
|
| |
| def _json_or_raise(resp: requests.Response): |
| ctype = (resp.headers.get("content-type") or "").lower() |
| if "application/json" in ctype: |
| return resp.json() |
| try: |
| return resp.json() |
| except Exception: |
| snippet = (resp.text or "")[:300] |
| raise RuntimeError(f"Expected JSON but got {ctype or 'unknown'}:\n{snippet}") |
|
|
| def _req(method: str, path: str, **kw): |
| if not path.startswith("/"): |
| path = "/" + path |
| for pref in _prefixes(): |
| url = f"{BACKEND}{pref}{path}" |
| try: |
| r = _session.request(method, url, timeout=180, **kw) |
| r.raise_for_status() |
| return r |
| except requests.HTTPError as e: |
| status = getattr(e.response, "status_code", "?") |
| |
| if status == 404: |
| continue |
|
|
| |
| if status in (401, 403): |
| |
| if path.endswith("/auth/login"): |
| raise RuntimeError("Incorrect email or password.") from e |
| |
| raise RuntimeError("Authentication failed. Please sign in again.") from e |
| |
|
|
| |
| body = "" |
| try: |
| body = e.response.text[:500] |
| except Exception: |
| pass |
| raise RuntimeError(f"{method} {url} failed [{status}]: {body}") from e |
| except requests.RequestException: |
| |
| continue |
| raise RuntimeError(f"No matching endpoint for {method} {path} with prefixes {list(_prefixes())}") |
|
|
| |
| def health(): |
| try: |
| return _json_or_raise(_req("GET", "/health")) |
| except Exception: |
| try: |
| _req("GET", "/") |
| return {"ok": True} |
| except Exception: |
| return {"ok": False} |
|
|
| |
| def login(email: str, password: str): |
| return _json_or_raise(_req("POST", "/auth/login", json={"email": email, "password": password})) |
|
|
| def signup_student(name: str, email: str, password: str, level_label: str, country_label: str): |
| payload = { |
| "name": name, "email": email, "password": password, |
| "level_label": level_label, "country_label": country_label |
| } |
| return _json_or_raise(_req("POST", "/auth/signup/student", json=payload)) |
|
|
| def signup_teacher(title: str, name: str, email: str, password: str): |
| payload = {"title": title, "name": name, "email": email, "password": password} |
| return _json_or_raise(_req("POST", "/auth/signup/teacher", json=payload)) |
|
|
| |
| def create_class(teacher_id: int, name: str): |
| return _json_or_raise(_req("POST", f"/teachers/{teacher_id}/classes", json={"name": name})) |
|
|
| def list_classes_by_teacher(teacher_id: int): |
| return _json_or_raise(_req("GET", f"/teachers/{teacher_id}/classes")) |
|
|
| def join_class_by_code(student_id: int, code: str): |
| d = _json_or_raise(_req("POST", f"/students/{student_id}/classes/join", json={"code": code})) |
| return d.get("class_id", d) |
|
|
| def leave_class(student_id: int, class_id: int): |
| _json_or_raise(_req("POST", f"/classes/{class_id}/leave", json={"student_id": student_id})) |
| return True |
|
|
| def list_students_in_class(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/students")) |
|
|
| def class_content_counts(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/content_counts")) |
|
|
| def list_class_assignments(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/assignments")) |
|
|
| def class_analytics(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/analytics")) |
|
|
| def class_student_metrics(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/students/metrics")) |
|
|
| def class_weekly_activity(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/activity/weekly")) |
|
|
| def class_progress_overview(class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/progress")) |
|
|
| def class_recent_activity(class_id: int, limit=6, days=30): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/activity/recent", |
| params={"limit": limit, "days": days})) |
|
|
| def student_class_progress(student_id: int, class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/students/{student_id}/progress")) |
|
|
| def student_assignments_for_class(student_id: int, class_id: int): |
| return _json_or_raise(_req("GET", f"/classes/{class_id}/students/{student_id}/assignments")) |
|
|
| |
| def list_lessons_by_teacher(teacher_id: int): |
| return _json_or_raise(_req("GET", f"/teachers/{teacher_id}/lessons")) |
|
|
| def create_lesson(teacher_id: int, title: str, description: str, |
| subject: str, level: str, sections: list[dict]): |
| payload = { |
| "title": title, "description": description, |
| "subject": subject, "level": level, "sections": sections, |
| } |
| d = _json_or_raise(_req("POST", f"/teachers/{teacher_id}/lessons", json=payload)) |
| return d.get("lesson_id", d.get("id", d)) |
|
|
| def get_lesson(lesson_id: int): |
| return _json_or_raise(_req("GET", f"/lessons/{lesson_id}")) |
|
|
| def update_lesson(lesson_id: int, teacher_id: int, title: str, description: str, |
| subject: str, level: str, sections: list[dict]): |
| d = _json_or_raise(_req("PUT", f"/lessons/{lesson_id}", json={ |
| "teacher_id": teacher_id, "title": title, "description": description, |
| "subject": subject, "level": level, "sections": sections |
| })) |
| return bool(d.get("ok", True)) |
|
|
| def delete_lesson(lesson_id: int, teacher_id: int): |
| d = _json_or_raise(_req("DELETE", f"/lessons/{lesson_id}", json={"teacher_id": teacher_id})) |
| return bool(d.get("ok", True)), d.get("message", "") |
|
|
| def list_assigned_students_for_lesson(lesson_id: int): |
| return _json_or_raise(_req("GET", f"/lessons/{lesson_id}/assignees")) |
|
|
| |
| def list_quizzes_by_teacher(teacher_id: int): |
| return _json_or_raise(_req("GET", f"/teachers/{teacher_id}/quizzes")) |
|
|
| def get_quiz(quiz_id: int): |
| return _json_or_raise(_req("GET", f"/quizzes/{quiz_id}")) |
|
|
| def create_quiz(lesson_id: int, title: str, items: list[dict], settings: dict): |
| d = _json_or_raise(_req("POST", "/quizzes", json={ |
| "lesson_id": lesson_id, "title": title, "items": items, "settings": settings |
| })) |
| return d.get("quiz_id", d.get("id", d)) |
|
|
| def update_quiz(quiz_id: int, teacher_id: int, title: str, items: list[dict], settings: dict): |
| d = _json_or_raise(_req("PUT", f"/quizzes/{quiz_id}", json={ |
| "teacher_id": teacher_id, "title": title, "items": items, "settings": settings |
| })) |
| return bool(d.get("ok", True)) |
|
|
| def delete_quiz(quiz_id: int, teacher_id: int): |
| d = _json_or_raise(_req("DELETE", f"/quizzes/{quiz_id}", json={"teacher_id": teacher_id})) |
| return bool(d.get("ok", True)), d.get("message", "") |
|
|
| def generate_quiz_from_text(content: str, n_questions: int = 5, subject: str = "finance", level: str = "beginner"): |
| return _json_or_raise(_req("POST", "/quiz/generate", json={ |
| "content": content, "n_questions": n_questions, "subject": subject, "level": level |
| })) |
|
|
| def generate_quiz(*, lesson_id: int | None, level_slug: str | None, lesson_title: str | None): |
| payload = { |
| "lesson_id": int(lesson_id) if lesson_id is not None else None, |
| "level_slug": level_slug, |
| "lesson_title": lesson_title, |
| } |
| resp = _req("POST", "/quiz/auto", json=payload) |
| result = _json_or_raise(resp) |
| if isinstance(result, dict) and "items" in result: |
| return result["items"] |
| return result if isinstance(result, list) else [] |
|
|
| def submit_quiz(*, student_id: int, lesson_id: int, level_slug: str, |
| user_answers: list[dict], original_quiz: list[dict], assignment_id: int | None = None): |
| payload = { |
| "student_id": student_id, |
| "lesson_id": lesson_id, |
| "level_slug": level_slug, |
| "user_answers": user_answers, |
| "original_quiz": original_quiz, |
| "assignment_id": assignment_id, |
| } |
| return _json_or_raise(_req("POST", "/quiz/grade", json=payload)) |
|
|
| |
| def tutor_explain(lesson_id: int, level_slug: str, wrong: list[dict]): |
| payload = {"lesson_id": lesson_id, "level_slug": level_slug, "wrong": wrong} |
| return _json_or_raise(_req("POST", "/tutor/explain", json=payload, timeout=60)) |
|
|
| |
| def user_stats(student_id: int): |
| return _json_or_raise(_req("GET", f"/students/{student_id}/stats")) |
|
|
| def list_assignments_for_student(student_id: int): |
| return _json_or_raise(_req("GET", f"/students/{student_id}/assignments")) |
|
|
| def student_quiz_average(student_id: int): |
| d = _json_or_raise(_req("GET", f"/students/{student_id}/quiz_avg")) |
| |
| for k in ("avg_pct", "avg", "average", "score_pct", "score", "value"): |
| if k in d: |
| try: |
| return int(round(float(str(d[k]).strip().rstrip("%")))) |
| except Exception: |
| break |
| return 0 |
|
|
| def recent_lessons_for_student(student_id: int, limit: int = 5): |
| return _json_or_raise(_req("GET", f"/students/{student_id}/recent", params={"limit": limit})) |
|
|
| def list_classes_for_student(student_id: int): |
| return _json_or_raise(_req("GET", f"/students/{student_id}/classes")) |
|
|
| def level_from_xp(xp: int): |
| return _json_or_raise(_req("GET", "/level-from-xp", params={"xp": xp}))["level"] |
|
|
| |
| def record_money_match_play(user_id: int, target: int, total: int, |
| elapsed_ms: int, matched: bool, gained_xp: int | None = None): |
| payload = { |
| "user_id": user_id, |
| "target": target, |
| "total": total, |
| "elapsed_ms": elapsed_ms, |
| "matched": bool(matched), |
| "gained_xp": gained_xp, |
| } |
| return _json_or_raise(_req("POST", "/games/money_match/record", json=payload)) |
|
|
| def record_budget_builder_play(user_id: int, weekly_allowance: int, |
| allocations: list[dict], gained_xp: int | None = None): |
| payload = { |
| "user_id": user_id, |
| "weekly_allowance": weekly_allowance, |
| "allocations": allocations, |
| "gained_xp": gained_xp, |
| } |
| return _json_or_raise(_req("POST", "/games/budget_builder/record", json=payload)) |
|
|
| def record_debt_dilemma_play(user_id: int, *, level: int, round_no: int, |
| wallet: int, health: int, happiness: int, credit_score: int, |
| event_json: dict, outcome: str, gained_xp: int | None = None): |
| payload = { |
| "user_id": user_id, |
| "level": level, |
| "round_no": round_no, |
| "wallet": wallet, |
| "health": health, |
| "happiness": happiness, |
| "credit_score": credit_score, |
| "event_json": event_json, |
| "outcome": outcome, |
| "gained_xp": gained_xp, |
| } |
| return _json_or_raise(_req("POST", "/games/debt_dilemma/record", json=payload)) |
|
|
| def record_profit_puzzler_play(user_id: int, scenario_id: str, title: str, |
| units: int, price: int, cost: int, |
| user_answer: float, actual_profit: float, is_correct: bool, |
| gained_xp: int | None = None): |
| payload = { |
| "user_id": user_id, "scenario_id": scenario_id, "title": title, |
| "units": units, "price": price, "cost": cost, |
| "user_answer": user_answer, "actual_profit": actual_profit, |
| "is_correct": bool(is_correct), "gained_xp": gained_xp, |
| } |
| return _json_or_raise(_req("POST", "/games/profit_puzzler/record", json=payload)) |
|
|
| |
| def retrieve(query: str, lesson_id: int, level_slug: str = "beginner", k: int = 3) -> str: |
| try: |
| d = _json_or_raise(_req("GET", "/retrieve", |
| params={"query": query, "lesson_id": lesson_id, "level_slug": level_slug, "k": k})) |
| return d.get("context", "") |
| except Exception as e: |
| return f"(retrieval failed: {e})" |
|
|
| def chat_ai(query: str, lesson_id: int, level_slug: str, history=None) -> str: |
| payload = { |
| "query": query, |
| "lesson_id": int(lesson_id) if lesson_id is not None else 0, |
| "level_slug": level_slug or "beginner", |
| "history": history or [], |
| } |
| try: |
| url = f"{BACKEND}/chat" |
| with requests.Session() as s: |
| s.headers.update(_session.headers) |
| r = s.post(url, json=payload, timeout=DEFAULT_TIMEOUT) |
| r.raise_for_status() |
| return r.json().get("answer", "") |
| except Exception as e: |
| return f"(chat failed: {e})" |
| |
| |
|
|
| def signup_student(name: str, email: str, password: str, level_label: str, country_label: str): |
| payload = { |
| "name": name, |
| "email": email, |
| "password": password, |
| "level_label": level_label, |
| "country_label": country_label, |
| } |
| try: |
| url = f"{BACKEND}/auth/signup/student" |
| r = _session.post(url, json=payload, timeout=DEFAULT_TIMEOUT) |
| r.raise_for_status() |
| return r.json() |
| except Exception as e: |
| raise RuntimeError(f"Signup failed: {e}") |
|
|
| def signup_teacher(title: str, name: str, email: str, password: str): |
| payload = { |
| "title": title, |
| "name": name, |
| "email": email, |
| "password": password, |
| } |
| try: |
| url = f"{BACKEND}/auth/signup/teacher" |
| r = _session.post(url, json=payload, timeout=DEFAULT_TIMEOUT) |
| r.raise_for_status() |
| return r.json() |
| except Exception as e: |
| raise RuntimeError(f"Signup failed: {e}") |
|
|
| def login(email: str, password: str): |
| payload = { |
| "email": email, |
| "password": password, |
| } |
| try: |
| url = f"{BACKEND}/auth/login" |
| r = _session.post(url, json=payload, timeout=DEFAULT_TIMEOUT) |
| r.raise_for_status() |
| return r.json() |
| except Exception as e: |
| raise RuntimeError(f"Login failed: {e}") |
|
|
|
|
| |
| def fetch_lesson_content(lesson: str, module: str, topic: str): |
| d = _json_or_raise(_req("POST", "/lesson", json={"lesson": lesson, "module": module, "topic": topic})) |
| return d.get("lesson_content", "") |
|
|
| def submit_lesson_quiz(lesson: str, module: str, topic: str, responses: dict): |
| return _json_or_raise(_req("POST", "/lesson-quiz", |
| json={"lesson": lesson, "module": module, "topic": topic, "responses": responses})) |
|
|
| def submit_practice_quiz(lesson: str, responses: dict): |
| return _json_or_raise(_req("POST", "/practice-quiz", json={"lesson": lesson, "responses": responses})) |
|
|