sprint/Main/models/solution.py
2022-05-09 19:51:11 +03:00

104 lines
3.2 KiB
Python

from functools import cached_property
from subprocess import call
from django.contrib.auth.models import User
from django.db import models
from django.utils import timezone
from Main.models import Set, SetTask
from Main.models.solution_file import SolutionFile
from Main.models.task import Task
from Sprint.settings import CONSTS
from SprintLib.language import languages
class Solution(models.Model):
task = models.ForeignKey(Task, on_delete=models.CASCADE)
user = models.ForeignKey(User, on_delete=models.CASCADE)
language_id = models.IntegerField(default=0)
time_sent = models.DateTimeField(default=timezone.now)
result = models.TextField(default=CONSTS["in_queue_status"])
test = models.IntegerField(default=None, null=True, blank=True)
set = models.ForeignKey(Set, null=True, blank=True, on_delete=models.SET_NULL)
extras = models.JSONField(default=dict)
_solutionfiles = None
class Meta:
indexes = [
models.Index(fields=['task', 'user', '-time_sent']),
models.Index(fields=['task', '-time_sent']),
models.Index(fields=['set', '-time_sent']),
]
@property
def solutionfiles(self):
if self._solutionfiles is not None:
return self._solutionfiles
return SolutionFile.objects.filter(solution=self)
@solutionfiles.setter
def solutionfiles(self, value):
self._solutionfiles = value
@cached_property
def settask(self):
return SetTask.objects.filter(set=self.set, task=self.task).first()
@property
def percentage_done(self):
if self.test is None:
return 0
return self.test * 100 // self.task.tests_count
@property
def language(self):
return languages[self.language_id]
@cached_property
def files(self):
data = []
for file in SolutionFile.objects.filter(solution=self):
try:
text = file.text
except:
continue
entity = {"filename": file.path, "text": text}
end = file.path.split(".")[-1]
language = None
for l in languages:
if l.file_type == end:
language = l
break
if language is None:
highlight = "nohighlight"
else:
highlight = "language-" + language.highlight
entity["highlight"] = highlight
data.append(entity)
data.sort(key=lambda x: x["filename"])
return data
@property
def number_result(self):
if self.test is None:
return self.result
return f"{self.result} ({self.test})"
@property
def badge_style(self):
if self.result == CONSTS["in_queue_status"]:
return "secondary"
if self.result == CONSTS["ok_status"]:
return "success"
if self.result == CONSTS["testing_status"]:
return "info"
return "danger"
def exec_command(self, command, working_directory="app", timeout=None):
return call(
f'docker exec -i solution_{self.id} sh -c "cd {working_directory} && {command}"',
shell=True,
timeout=timeout,
)