add course models
This commit is contained in:
parent
b049f6eace
commit
51006bc773
4 changed files with 238 additions and 0 deletions
18
judge/migrations/0147_alter_profile_timezone.py
Normal file
18
judge/migrations/0147_alter_profile_timezone.py
Normal file
File diff suppressed because one or more lines are too long
|
@ -0,0 +1,58 @@
|
|||
# Generated by Django 3.2.16 on 2023-01-31 15:49
|
||||
|
||||
import django.core.validators
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('judge', '0147_alter_profile_timezone'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='Course',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(max_length=128, verbose_name='course name')),
|
||||
('about', models.TextField(verbose_name='organization description')),
|
||||
('ending_time', models.DateTimeField(verbose_name='ending time')),
|
||||
('is_public', models.BooleanField(default=False, verbose_name='publicly visible')),
|
||||
('slug', models.SlugField(help_text='Course name shown in URL', max_length=128, unique=True, validators=[django.core.validators.RegexValidator('^[-a-zA-Z0-9]+$', 'Only alphanumeric and hyphens')], verbose_name='course slug')),
|
||||
('is_open', models.BooleanField(default=False, verbose_name='public registration')),
|
||||
('image_url', models.CharField(blank=True, default='', max_length=150, verbose_name='course image')),
|
||||
('organizations', models.ManyToManyField(blank=True, help_text='If private, only these organizations may see the course', to='judge.Organization', verbose_name='organizations')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='CourseRole',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('role', models.CharField(choices=[('ST', 'Student'), ('AS', 'Assistant'), ('TE', 'Teacher')], default='ST', max_length=2)),
|
||||
('course', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to='judge.course', verbose_name='course')),
|
||||
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='user_of_course', to='judge.profile', verbose_name='user')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='CourseResource',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('files', models.FileField(blank=True, null=True, upload_to='', verbose_name='course files')),
|
||||
('description', models.CharField(blank=True, max_length=150, verbose_name='description')),
|
||||
('order', models.IntegerField(default=None, null=True)),
|
||||
('is_public', models.BooleanField(default=False, verbose_name='publicly visible')),
|
||||
('course', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to='judge.course', verbose_name='course')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='CourseAssignment',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('points', models.FloatField(verbose_name='points')),
|
||||
('contest', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to='judge.contest', verbose_name='contest')),
|
||||
('course', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to='judge.course', verbose_name='course')),
|
||||
],
|
||||
),
|
||||
]
|
|
@ -56,6 +56,7 @@ from judge.models.ticket import Ticket, TicketMessage
|
|||
from judge.models.volunteer import VolunteerProblemVote
|
||||
from judge.models.pagevote import PageVote, PageVoteVoter
|
||||
from judge.models.bookmark import BookMark, MakeBookMark
|
||||
from judge.models.course import Course
|
||||
|
||||
revisions.register(Profile, exclude=["points", "last_access", "ip", "rating"])
|
||||
revisions.register(Problem, follow=["language_limits"])
|
||||
|
@ -81,4 +82,5 @@ revisions.register(Rating)
|
|||
revisions.register(PageVoteVoter)
|
||||
revisions.register(VolunteerProblemVote)
|
||||
revisions.register(MakeBookMark)
|
||||
revisions.register(Course)
|
||||
del revisions
|
||||
|
|
160
judge/models/course.py
Normal file
160
judge/models/course.py
Normal file
|
@ -0,0 +1,160 @@
|
|||
from django.core.validators import RegexValidator
|
||||
from django.db import models
|
||||
from django.utils.translation import gettext, gettext_lazy as _
|
||||
|
||||
from judge.models import Contest
|
||||
from judge.models.profile import Organization, Profile
|
||||
|
||||
__all__ = [
|
||||
"Course",
|
||||
"CourseRole",
|
||||
"CourseResource",
|
||||
"CourseAssignment",
|
||||
]
|
||||
|
||||
course_directory_file = ""
|
||||
|
||||
class Course(models.Model):
|
||||
name = models.CharField(max_length=128, verbose_name=_("course name"),)
|
||||
about = models.TextField(verbose_name=_("organization description"))
|
||||
ending_time = models.DateTimeField(verbose_name=_("ending time"),)
|
||||
is_public = models.BooleanField(verbose_name=_("publicly visible"), default=False,)
|
||||
organizations = models.ManyToManyField(
|
||||
Organization,
|
||||
blank=True,
|
||||
verbose_name=_("organizations"),
|
||||
help_text=_("If private, only these organizations may see the course"),
|
||||
)
|
||||
slug = models.SlugField(
|
||||
max_length=128,
|
||||
verbose_name=_("course slug"),
|
||||
help_text=_("Course name shown in URL"),
|
||||
unique=True,
|
||||
validators=[
|
||||
RegexValidator("^[-a-zA-Z0-9]+$", _("Only alphanumeric and hyphens"))
|
||||
],
|
||||
)
|
||||
is_open = models.BooleanField(verbose_name=_("public registration"), default=False,)
|
||||
image_url = models.CharField(
|
||||
verbose_name=_("course image"),
|
||||
default="",
|
||||
max_length=150,
|
||||
blank=True,
|
||||
)
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
@classmethod
|
||||
def is_editable_by(course, profile):
|
||||
if (profile.is_superuser):
|
||||
return True
|
||||
userquery = CourseRole.objects.filter(course=course, user=profile)
|
||||
if userquery.exists():
|
||||
if (userquery[0].role == 'AS' or userquery[0].role == 'TE'):
|
||||
return True
|
||||
return False
|
||||
|
||||
@classmethod
|
||||
def is_accessible_by(course, profile):
|
||||
userqueryset = CourseRole.objects.filter(course=course, user=profile)
|
||||
if userqueryset.exists():
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
|
||||
@classmethod
|
||||
def get_students(course):
|
||||
return CourseRole.objects.filter(course=course, role='ST').values('user')
|
||||
|
||||
@classmethod
|
||||
def get_assistants(course):
|
||||
return CourseRole.objects.filter(course=course, role='AS').values('user')
|
||||
|
||||
@classmethod
|
||||
def get_teachers(course):
|
||||
return CourseRole.objects.filter(course=course, role='TE').values('user')
|
||||
|
||||
@classmethod
|
||||
def add_student(course, profiles):
|
||||
for profile in profiles:
|
||||
CourseRole.make_role(course=course, user=profile, role='ST')
|
||||
|
||||
@classmethod
|
||||
def add_teachers(course, profiles):
|
||||
for profile in profiles:
|
||||
CourseRole.make_role(course=course, user=profile, role='TE')
|
||||
|
||||
@classmethod
|
||||
def add_assistants(course, profiles):
|
||||
for profile in profiles:
|
||||
CourseRole.make_role(course=course, user=profile, role='AS')
|
||||
|
||||
class CourseRole(models.Model):
|
||||
course = models.OneToOneField(
|
||||
Course,
|
||||
verbose_name=_("course"),
|
||||
on_delete=models.CASCADE,
|
||||
db_index=True,
|
||||
)
|
||||
user = models.ForeignKey(
|
||||
Profile,
|
||||
verbose_name=_("user"),
|
||||
on_delete=models.CASCADE,
|
||||
related_name=_("user_of_course"),
|
||||
)
|
||||
class RoleInCourse(models.TextChoices):
|
||||
STUDENT = 'ST', _("Student")
|
||||
ASSISTANT = 'AS', _("Assistant")
|
||||
TEACHER = 'TE', _("Teacher")
|
||||
role = models.CharField(
|
||||
max_length=2,
|
||||
choices=RoleInCourse.choices,
|
||||
default=RoleInCourse.STUDENT,
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def make_role(self, course, user, role):
|
||||
userqueryset = CourseRole.objects.filter(course=course, user=user)
|
||||
if userqueryset.exists():
|
||||
userqueryset[0].role = role
|
||||
else:
|
||||
couresrole = CourseRole()
|
||||
couresrole.course = course
|
||||
couresrole.user = user
|
||||
couresrole.role = role
|
||||
couresrole.save()
|
||||
|
||||
class CourseResource(models.Model):
|
||||
course = models.OneToOneField(
|
||||
Course,
|
||||
verbose_name=_("course"),
|
||||
on_delete=models.CASCADE,
|
||||
db_index=True,
|
||||
)
|
||||
files = models.FileField(
|
||||
verbose_name=_("course files"),
|
||||
null=True,
|
||||
blank=True,
|
||||
upload_to=course_directory_file,
|
||||
)
|
||||
description = models.CharField(
|
||||
verbose_name=_("description"),
|
||||
blank=True,
|
||||
max_length=150,
|
||||
)
|
||||
order = models.IntegerField(null=True, default=None)
|
||||
is_public = models.BooleanField(verbose_name=_("publicly visible"), default=False,)
|
||||
|
||||
class CourseAssignment(models.Model):
|
||||
course = models.OneToOneField(
|
||||
Course,
|
||||
verbose_name=_("course"),
|
||||
on_delete=models.CASCADE,
|
||||
db_index=True,
|
||||
)
|
||||
contest = models.OneToOneField(
|
||||
Contest,
|
||||
verbose_name=_("contest"),
|
||||
on_delete=models.CASCADE,
|
||||
)
|
||||
points = models.FloatField(verbose_name=_("points"),)
|
Loading…
Reference in a new issue