Skip to content

Commit

Permalink
feat: Add courseware language support
Browse files Browse the repository at this point in the history
  • Loading branch information
arslanashraf7 committed Jan 8, 2025
1 parent 6482515 commit 70a3859
Show file tree
Hide file tree
Showing 16 changed files with 305 additions and 7 deletions.
6 changes: 5 additions & 1 deletion cms/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,11 @@ def filter_and_sort_catalog_pages(
page.product.current_price,
page.title,
)
default_sorting_key = lambda page: (page_run_dates[page], page.title) # noqa: E731
default_sorting_key = lambda page: ( # noqa: E731
page.language.priority,
page_run_dates[page],
page.title,
)

# Best Match and Start Date sorting has same logic
sorting_key_map = defaultdict(
Expand Down
2 changes: 0 additions & 2 deletions cms/api_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -85,7 +85,6 @@ def test_filter_and_sort_catalog_pages_with_default_sorting(sort_by):
initial_program_pages = [
run.course.program.page for run in [second_program_run, first_program_run]
]

all_pages, program_pages, course_pages = filter_and_sort_catalog_pages(
initial_program_pages,
initial_course_pages,
Expand Down Expand Up @@ -114,7 +113,6 @@ def test_filter_and_sort_catalog_pages_with_default_sorting(sort_by):
assert past_run.course not in (
None if page.is_external_course_page else page.course for page in course_pages
)

# Pages should be sorted by next run date
assert [page.program for page in program_pages] == [
first_program_run.course.program,
Expand Down
11 changes: 10 additions & 1 deletion cms/factories.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,12 @@
WebinarPage,
WhoShouldEnrollPage,
)
from courses.factories import CourseFactory, PlatformFactory, ProgramFactory
from courses.factories import (
CourseFactory,
CourseLanguageFactory,
PlatformFactory,
ProgramFactory,
)

factory.Faker.add_provider(internet)

Expand All @@ -78,6 +83,7 @@ class ProgramPageFactory(wagtail_factories.PageFactory):
subhead = factory.fuzzy.FuzzyText(prefix="Subhead ")
thumbnail_image = factory.SubFactory(wagtail_factories.ImageFactory)
background_image = factory.SubFactory(wagtail_factories.ImageFactory)
language = factory.SubFactory(CourseLanguageFactory)
parent = factory.SubFactory(wagtail_factories.PageFactory)
certificate_page = factory.RelatedFactory(
"cms.factories.CertificatePageFactory", "parent"
Expand Down Expand Up @@ -108,6 +114,7 @@ class CoursePageFactory(wagtail_factories.PageFactory):
subhead = factory.fuzzy.FuzzyText(prefix="Subhead ")
thumbnail_image = factory.SubFactory(wagtail_factories.ImageFactory)
background_image = factory.SubFactory(wagtail_factories.ImageFactory)
language = factory.SubFactory(CourseLanguageFactory)
parent = factory.SubFactory(wagtail_factories.PageFactory)
certificate_page = factory.RelatedFactory(
"cms.factories.CertificatePageFactory", "parent"
Expand Down Expand Up @@ -141,6 +148,7 @@ class ExternalCoursePageFactory(wagtail_factories.PageFactory):
subhead = factory.fuzzy.FuzzyText(prefix="Subhead ")
thumbnail_image = factory.SubFactory(wagtail_factories.ImageFactory)
background_image = factory.SubFactory(wagtail_factories.ImageFactory)
language = factory.SubFactory(CourseLanguageFactory)
parent = factory.SubFactory(wagtail_factories.PageFactory)

class Meta:
Expand Down Expand Up @@ -170,6 +178,7 @@ class ExternalProgramPageFactory(wagtail_factories.PageFactory):
subhead = factory.fuzzy.FuzzyText(prefix="Subhead ")
thumbnail_image = factory.SubFactory(wagtail_factories.ImageFactory)
background_image = factory.SubFactory(wagtail_factories.ImageFactory)
language = factory.SubFactory(CourseLanguageFactory)
parent = factory.SubFactory(wagtail_factories.PageFactory)

class Meta:
Expand Down
81 changes: 81 additions & 0 deletions cms/migrations/0078_add_courseware_page_language.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
# Generated by Django 4.2.17 on 2025-01-03 08:36

import django.db.models.deletion
from django.db import migrations, models


def populate_course_language(apps, schema_editor):
"""Prepopulate the course language for the course pages"""

CoursePage = apps.get_model("cms.CoursePage")
ExternalCoursePage = apps.get_model("cms.ExternalCoursePage")
ProgramPage = apps.get_model("cms.ProgramPage")
ExternalProgramPage = apps.get_model("cms.ExternalProgramPage")

CourseLanguage = apps.get_model("courses.CourseLanguage")
# English is the default language for all the courses
course_language_english, _ = CourseLanguage.objects.get_or_create(
name="English", priority=1
)

CoursePage.objects.update(language=course_language_english)
ExternalCoursePage.objects.update(language=course_language_english)
ProgramPage.objects.update(language=course_language_english)
ExternalProgramPage.objects.update(language=course_language_english)


class Migration(migrations.Migration):
dependencies = [
("courses", "0042_add_course_language"),
("cms", "0077_alter_certificatepage_ceus"),
]

operations = [
migrations.AddField(
model_name="coursepage",
name="language",
field=models.ForeignKey(
blank=True,
help_text="The course/program language for this page",
null=True,
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.AddField(
model_name="externalcoursepage",
name="language",
field=models.ForeignKey(
blank=True,
help_text="The course/program language for this page",
null=True,
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.AddField(
model_name="externalprogrampage",
name="language",
field=models.ForeignKey(
blank=True,
help_text="The course/program language for this page",
null=True,
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.AddField(
model_name="programpage",
name="language",
field=models.ForeignKey(
blank=True,
help_text="The course/program language for this page",
null=True,
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.RunPython(
populate_course_language, reverse_code=migrations.RunPython.noop
),
]
53 changes: 53 additions & 0 deletions cms/migrations/0079_make_language_non_nullable.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
# Generated by Django 4.2.17 on 2025-01-03 08:39

import django.db.models.deletion
from django.db import migrations, models


class Migration(migrations.Migration):
dependencies = [
("cms", "0078_add_courseware_page_language"),
]

operations = [
migrations.AlterField(
model_name="coursepage",
name="language",
field=models.ForeignKey(
help_text="The course/program language for this page",
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.AlterField(
model_name="externalcoursepage",
name="language",
field=models.ForeignKey(
help_text="The course/program language for this page",
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
),
migrations.AlterField(
model_name="externalprogrampage",
name="language",
field=models.ForeignKey(
default="",
help_text="The course/program language for this page",
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
preserve_default=False,
),
migrations.AlterField(
model_name="programpage",
name="language",
field=models.ForeignKey(
default="",
help_text="The course/program language for this page",
on_delete=django.db.models.deletion.PROTECT,
to="courses.courselanguage",
),
preserve_default=False,
),
]
9 changes: 9 additions & 0 deletions cms/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -922,6 +922,14 @@ class ProductPage(MetadataPageMixin, WagtailCachedPageMixin, Page):
class Meta:
abstract = True

language = models.ForeignKey(
"courses.CourseLanguage",
null=False,
blank=False,
on_delete=models.PROTECT,
help_text="The course/program language for this page",
)

description = RichTextField(
blank=True, help_text="The description shown on the product page"
)
Expand Down Expand Up @@ -1030,6 +1038,7 @@ class Meta:
use_json_field=True,
)
content_panels = Page.content_panels + [ # noqa: RUF005
FieldPanel("language"),
FieldPanel("external_marketing_url"),
FieldPanel("marketing_hubspot_form_id"),
FieldPanel("subhead"),
Expand Down
4 changes: 3 additions & 1 deletion cms/models_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@
from cms.wagtail_hooks import create_product_and_versions_for_courseware_pages
from courses.factories import (
CourseFactory,
CourseLanguageFactory,
CourseRunCertificateFactory,
CourseRunFactory,
ProgramCertificateFactory,
Expand Down Expand Up @@ -2173,7 +2174,7 @@ def _create_external_course_page(superuser_client, course_id, slug):
Asserts:
Response status code is 302 (successful redirection).
"""

language = CourseLanguageFactory.create()
post_data = {
"course": course_id,
"title": "Icon Grid #6064",
Expand All @@ -2182,6 +2183,7 @@ def _create_external_course_page(superuser_client, course_id, slug):
"content-count": 0,
"slug": slug,
"action-publish": "action-publish",
"language": language.id,
}
response = superuser_client.post(
reverse(
Expand Down
10 changes: 10 additions & 0 deletions courses/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@

from .models import (
Course,
CourseLanguage,
CourseRun,
CourseRunCertificate,
CourseRunEnrollment,
Expand Down Expand Up @@ -422,3 +423,12 @@ class PlatformAdmin(TimestampedModelAdmin):
model = Platform
list_display = ["id", "name", "created_on", "updated_on"]
search_fields = ["name"]


@admin.register(CourseLanguage)
class CourseLanguageAdmin(admin.ModelAdmin):
"""Admin for CourseLanguage"""

model = CourseLanguage
list_display = ["id", "name", "priority"]
search_fields = ["name"]
10 changes: 10 additions & 0 deletions courses/factories.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@

from .models import (
Course,
CourseLanguage,
CourseRun,
CourseRunCertificate,
CourseRunEnrollment,
Expand All @@ -38,6 +39,15 @@ class Meta:
model = Company


class CourseLanguageFactory(DjangoModelFactory):
"""Factory for Course Language"""

name = factory.Sequence(lambda n: f"Language_{n}")

class Meta:
model = CourseLanguage


class PlatformFactory(DjangoModelFactory):
"""Factory for Platform"""

Expand Down
73 changes: 73 additions & 0 deletions courses/migrations/0042_add_course_language.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
# Generated by Django 4.2.17 on 2025-01-07 13:31

import django.core.validators
import django.db.models.functions.text
from django.db import migrations, models


def add_default_supported_languages(apps, schema_editor):
"""Add the languages that xPRO platform will support as default"""
CourseLanguage = apps.get_model("courses.CourseLanguage")
CourseLanguage.objects.bulk_create(
[
CourseLanguage(name="English", priority=1),
CourseLanguage(name="Spanish"),
CourseLanguage(name="Portuguese"),
CourseLanguage(name="Mandarin"),
CourseLanguage(name="Italian"),
CourseLanguage(name="French"),
]
)


def remove_all_languages(apps, schema_editor):
"""Remove all languages"""
CourseLanguage = apps.get_model("courses.CourseLanguage")
CourseLanguage.objects.all().delete()


class Migration(migrations.Migration):
dependencies = [
("courses", "0041_platform_sync_daily"),
]

operations = [
migrations.CreateModel(
name="CourseLanguage",
fields=[
(
"id",
models.AutoField(
auto_created=True,
primary_key=True,
serialize=False,
verbose_name="ID",
),
),
("created_on", models.DateTimeField(auto_now_add=True)),
("updated_on", models.DateTimeField(auto_now=True)),
("name", models.CharField(max_length=255, unique=True)),
(
"priority",
models.PositiveSmallIntegerField(
blank=True,
default=100,
help_text="The priority of this language in the course/program sorting.",
null=True,
validators=[django.core.validators.MinValueValidator(1)],
),
),
],
),
migrations.AddConstraint(
model_name="courselanguage",
constraint=models.UniqueConstraint(
django.db.models.functions.text.Lower("name"),
name="unique_language_name",
),
),
migrations.RunPython(
code=add_default_supported_languages,
reverse_code=remove_all_languages,
),
]
Loading

0 comments on commit 70a3859

Please sign in to comment.