from fastapi import FastAPI from fastapi.middleware.cors import CORSMiddleware # CORS import json import random from pathlib import Path from datetime import datetime from settings import WAVE, END_TEST_PASSWORD, QUIZ_LENGTH app = FastAPI() origins = [ # CORS "*", ] app.add_middleware( # CORS CORSMiddleware, allow_origins=origins, allow_credentials=True, allow_methods=["*"], allow_headers=["*"], ) # student_answers_example = { # "version": 1, # "student": "asd", # "wave": 0, # "questions": [ # { # "id": 4, # "question": "Сколько винтов у теодолита?", # "student_answer": 100 # } # ] # } # open files once and use variables after with open("questions.json", "r", encoding="UTF-8") as f: content = json.load(f) all_questions = content["questions"] topics_questions = all_questions # don't use topics for now. see next to lines to use topics # topics = ["теодолит"] # move topics to VARIABLES # topics_questions = [q for q in all_questions if q.get("topic") in topics] remove_keys = ["author", "answer", "topic"] quiz_questions = [{key: value for key, value in q.items() if key not in remove_keys} for q in topics_questions] with open("students.json", "r", encoding="UTF-8") as f: content = json.load(f) students = content["students"] # create folders if they don't exist Path("answers").mkdir(parents=True, exist_ok=True) Path("results").mkdir(parents=True, exist_ok=True) def check_answers(student_answers: dict): checked_answers = student_answers for a in checked_answers["questions"]: question_id = a["id"] a["correct_answer"] = next(question["answer"] for question in topics_questions if question["id"] == question_id) a["is_correct"] = a["student_answer"] == a["correct_answer"] checked_answers["correct"] = sum([a["is_correct"] for a in checked_answers["questions"]]) checked_answers["correct_percent"] = round(checked_answers["correct"] * 100 / len(checked_answers["questions"])) return checked_answers @app.get("/students") def show_students(): return students @app.get("/get_quiz") def get_quiz(student_id, student: str): """Get json-like quiz Args: student_id (int): indentificator of a student student (str): student name Returns: obj: json-like quiz """ quiz_length = QUIZ_LENGTH questions_for_student = random.sample(quiz_questions, len(quiz_questions))[:quiz_length] # random order and only first n questions return { "version": 1, "student_id": student_id, "student": student, "wave": WAVE, # волна сдачи теста "start_time": datetime.now().strftime("%Y-%m-%dT%H-%M-%S"), "questions": questions_for_student } @app.get("/save_student_answers") def send_student_answers(student_answers: str): """Save answers as-is and check answers Args: student_answers (str): json-like string with student's answers Returns: str: student name """ json_answers = json.loads(student_answers) timestamp_str = datetime.now().strftime("%Y-%m-%dT%H-%M-%S") json_answers["end_time"] = timestamp_str path_to_answers = f'answers/{json_answers["student"]}_{json_answers["wave"]}_{timestamp_str}.json' with open(path_to_answers, 'w', encoding='utf-8') as f: json.dump(json_answers, f, ensure_ascii=False) path_to_results = f'results/{json_answers["student"]}_{json_answers["wave"]}_{timestamp_str}.json' with open(path_to_results, 'w', encoding='utf-8') as f: json.dump(check_answers(json_answers), f, ensure_ascii=False) # TODO move checking to background? return json_answers["student"] @app.get("/end_quiz") def end_test(password: str): """End test and save results to csv Args: password (str): password to end test """ if password == END_TEST_PASSWORD: csv_string = "student,percent,start,end\n" for file_in_results in Path("results").iterdir(): if file_in_results.is_file() and file_in_results.suffix == ".json": print(file_in_results) with open(file_in_results, "r", encoding="UTF-8") as f: content = json.load(f) csv_string += f"{content['student']},{content['correct_percent']},{content['start_time']},{content['end_time']}\n" timestamp = datetime.now().strftime("%Y-%m-%dT%H-%M-%S") results_path = Path(f"results/results_{timestamp}.csv") with open(results_path, "w", encoding='utf-8') as f: f.write(csv_string) return(f"Тестирование завершено. Сводные результаты сохранены в {results_path.resolve()}") else: return("Неверный пароль")