You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
234 lines
8.1 KiB
234 lines
8.1 KiB
"""Создание/правка теста, fork версии при наличии попыток (порт `testDraftService.js`).""" |
|
from __future__ import annotations |
|
|
|
from typing import Any |
|
|
|
from sqlalchemy import text |
|
|
|
from ..db import get_engine |
|
from ..messages import RU |
|
from .test_access import is_test_author |
|
from .test_chain import has_any_attempt_for_test |
|
|
|
|
|
class HttpError(Exception): |
|
def __init__(self, status: int, message: str): |
|
super().__init__(message) |
|
self.status = status |
|
self.message = message |
|
|
|
|
|
def create_test_with_version(author_id: str, *, title: str, description: str | None) -> dict: |
|
eng = get_engine() |
|
with eng.begin() as conn: |
|
t = conn.execute( |
|
text( |
|
""" |
|
INSERT INTO tests (title, description, created_by, is_active, is_versioned) |
|
VALUES (:title, :desc, :uid, true, true) RETURNING id |
|
""" |
|
), |
|
{'title': title, 'desc': description or None, 'uid': author_id}, |
|
).mappings().first() |
|
test_id = t['id'] |
|
v = conn.execute( |
|
text( |
|
""" |
|
INSERT INTO test_versions (test_id, version, is_active, parent_id) |
|
VALUES (:tid, 1, true, NULL) RETURNING id |
|
""" |
|
), |
|
{'tid': test_id}, |
|
).mappings().first() |
|
return {'testId': str(test_id), 'versionId': str(v['id'])} |
|
|
|
|
|
def _get_active_version_row(conn, test_id: str) -> dict | None: |
|
row = conn.execute( |
|
text( |
|
'SELECT * FROM test_versions WHERE test_id = :id AND is_active = true LIMIT 1' |
|
), |
|
{'id': test_id}, |
|
).mappings().first() |
|
return dict(row) if row else None |
|
|
|
|
|
def _copy_question_tree(conn, from_version_id, to_version_id) -> None: |
|
questions = conn.execute( |
|
text( |
|
'SELECT id, text, question_order, has_multiple_answers ' |
|
'FROM questions WHERE test_version_id = :v ORDER BY question_order' |
|
), |
|
{'v': from_version_id}, |
|
).mappings().all() |
|
for q in questions: |
|
new_q = conn.execute( |
|
text( |
|
""" |
|
INSERT INTO questions (test_version_id, text, question_order, has_multiple_answers) |
|
VALUES (:v, :text, :ord, :multi) RETURNING id |
|
""" |
|
), |
|
{ |
|
'v': to_version_id, |
|
'text': q['text'], |
|
'ord': q['question_order'], |
|
'multi': q['has_multiple_answers'], |
|
}, |
|
).mappings().first() |
|
nqid = new_q['id'] |
|
opts = conn.execute( |
|
text( |
|
'SELECT text, is_correct, option_order FROM answer_options ' |
|
'WHERE question_id = :q ORDER BY option_order' |
|
), |
|
{'q': q['id']}, |
|
).mappings().all() |
|
for o in opts: |
|
conn.execute( |
|
text( |
|
""" |
|
INSERT INTO answer_options (question_id, text, is_correct, option_order) |
|
VALUES (:q, :text, :ic, :ord) |
|
""" |
|
), |
|
{'q': nqid, 'text': o['text'], 'ic': o['is_correct'], 'ord': o['option_order']}, |
|
) |
|
|
|
|
|
def _replace_version_content(conn, test_version_id, payload: dict) -> None: |
|
conn.execute( |
|
text( |
|
""" |
|
DELETE FROM answer_options WHERE question_id IN ( |
|
SELECT id FROM questions WHERE test_version_id = :v |
|
) |
|
""" |
|
), |
|
{'v': test_version_id}, |
|
) |
|
conn.execute( |
|
text('DELETE FROM questions WHERE test_version_id = :v'), |
|
{'v': test_version_id}, |
|
) |
|
questions = payload.get('questions') or [] |
|
for i, q in enumerate(questions): |
|
ins_q = conn.execute( |
|
text( |
|
""" |
|
INSERT INTO questions (test_version_id, text, question_order, has_multiple_answers) |
|
VALUES (:v, :text, :ord, :multi) RETURNING id |
|
""" |
|
), |
|
{ |
|
'v': test_version_id, |
|
'text': q.get('text'), |
|
'ord': q.get('question_order') or (i + 1), |
|
'multi': bool(q.get('hasMultipleAnswers')), |
|
}, |
|
).mappings().first() |
|
qid = ins_q['id'] |
|
opts = q.get('options') or [] |
|
for j, o in enumerate(opts): |
|
conn.execute( |
|
text( |
|
""" |
|
INSERT INTO answer_options (question_id, text, is_correct, option_order) |
|
VALUES (:q, :text, :ic, :ord) |
|
""" |
|
), |
|
{ |
|
'q': qid, |
|
'text': o.get('text'), |
|
'ic': bool(o.get('isCorrect')), |
|
'ord': o.get('option_order') or (j + 1), |
|
}, |
|
) |
|
|
|
|
|
def _fork_new_version(conn, test_id: str) -> dict: |
|
av = _get_active_version_row(conn, test_id) |
|
if not av: |
|
raise HttpError(500, RU['internal']) # invariant: должна быть активная версия |
|
mx = conn.execute( |
|
text( |
|
'SELECT COALESCE(MAX(version), 0) AS v FROM test_versions WHERE test_id = :t' |
|
), |
|
{'t': test_id}, |
|
).mappings().first() |
|
next_v = (mx['v'] or 0) + 1 |
|
conn.execute( |
|
text('UPDATE test_versions SET is_active = false WHERE test_id = :t'), |
|
{'t': test_id}, |
|
) |
|
nv = conn.execute( |
|
text( |
|
""" |
|
INSERT INTO test_versions (test_id, version, is_active, parent_id) |
|
VALUES (:t, :ver, true, :parent) RETURNING * |
|
""" |
|
), |
|
{'t': test_id, 'ver': next_v, 'parent': av['id']}, |
|
).mappings().first() |
|
_copy_question_tree(conn, av['id'], nv['id']) |
|
return dict(nv) |
|
|
|
|
|
def save_test_draft(author_id: str, test_id: str, payload: dict) -> dict: |
|
if not isinstance(payload, dict): |
|
payload = {} |
|
eng = get_engine() |
|
with eng.begin() as conn: |
|
t = conn.execute( |
|
text('SELECT id, created_by FROM tests WHERE id = :id'), |
|
{'id': test_id}, |
|
).mappings().first() |
|
if not t: |
|
raise HttpError(404, RU['testNotFound'] if 'testNotFound' in RU else 'Тест не найден.') |
|
if not is_test_author(t['created_by'], author_id): |
|
raise HttpError(403, 'Доступ запрещён.') |
|
|
|
if payload.get('title') is not None or payload.get('description') is not None: |
|
conn.execute( |
|
text( |
|
""" |
|
UPDATE tests |
|
SET title = COALESCE(:title, title), |
|
description = COALESCE(:desc, description), |
|
updated_at = CURRENT_TIMESTAMP |
|
WHERE id = :id |
|
""" |
|
), |
|
{ |
|
'title': payload.get('title'), |
|
'desc': payload.get('description'), |
|
'id': test_id, |
|
}, |
|
) |
|
if payload.get('passingThreshold') is not None: |
|
try: |
|
raw = float(payload['passingThreshold']) |
|
pt = max(0, min(100, round(raw))) |
|
conn.execute( |
|
text( |
|
'UPDATE tests SET passing_threshold = :pt, updated_at = CURRENT_TIMESTAMP WHERE id = :id' |
|
), |
|
{'pt': pt, 'id': test_id}, |
|
) |
|
except (TypeError, ValueError): |
|
pass |
|
|
|
has_attempts = has_any_attempt_for_test(conn, test_id) |
|
version_row = _get_active_version_row(conn, test_id) |
|
if not version_row: |
|
raise HttpError(500, 'Нет активной версии теста.') |
|
|
|
forked = False |
|
if has_attempts and 'questions' in payload and payload.get('questions') is not None: |
|
version_row = _fork_new_version(conn, test_id) |
|
forked = True |
|
|
|
if payload.get('questions') is not None: |
|
_replace_version_content(conn, version_row['id'], payload) |
|
|
|
return {'testId': test_id, 'versionId': str(version_row['id']), 'forked': forked}
|
|
|