Files
edx-platform/lms/urls.py
Deborah Kaplan 3136134be8 chore: move the program dashboard APIs (#36420)
Moves the Program Dashboard APIs out of the deprecated remnants of the legacy learner dashboard, into the Programs djangoapp.

Keeps the old legacy routes for this API, left over from the deprecated remnants of the legacy learner dashboard, alongside future-proofed routes which will work when the deprecated, legacy Program Dashboard is eventually replaced with functionality in the Learner Dashboard MFE.

FIXES: APER-3949
2025-03-24 12:06:52 -04:00

1050 lines
35 KiB
Python

"""
URLs for LMS
"""
from config_models.views import ConfigurationModelCurrentAPIView
from django.conf import settings
from django.conf.urls.static import static
from django.contrib import admin
from django.contrib.admin import autodiscover as django_autodiscover
from django.urls import include, path, re_path
from django.utils.translation import gettext_lazy as _
from django.views.generic.base import RedirectView
from edx_api_doc_tools import make_docs_urls
from edx_django_utils.plugins import get_plugin_url_patterns
from common.djangoapps.student import views as student_views
from common.djangoapps.util import views as util_views
from lms.djangoapps.branding import views as branding_views
from lms.djangoapps.courseware.masquerade import MasqueradeView
from lms.djangoapps.courseware.block_render import (
handle_xblock_callback,
handle_xblock_callback_noauth,
xblock_view,
xqueue_callback
)
from lms.djangoapps.courseware.views import views as courseware_views
from lms.djangoapps.courseware.views.index import CoursewareIndex
from lms.djangoapps.courseware.views.views import CourseTabView, EnrollStaffView, StaticCourseTabView
from lms.djangoapps.debug import views as debug_views
from lms.djangoapps.discussion import views as discussion_views
from lms.djangoapps.discussion.config.settings import is_forum_daily_digest_enabled
from lms.djangoapps.discussion.notification_prefs import views as notification_prefs_views
from lms.djangoapps.instructor.views import instructor_dashboard as instructor_dashboard_views
from lms.djangoapps.instructor_task import views as instructor_task_views
from lms.djangoapps.static_template_view import views as static_template_view_views
from lms.djangoapps.staticbook import views as staticbook_views
from openedx.core.apidocs import api_info
from openedx.core.djangoapps.auth_exchange.views import LoginWithAccessTokenView
from openedx.core.djangoapps.catalog.models import CatalogIntegration
from openedx.core.djangoapps.common_views.xblock import xblock_resource
from openedx.core.djangoapps.cors_csrf import views as cors_csrf_views
from openedx.core.djangoapps.course_groups import views as course_groups_views
from openedx.core.djangoapps.debug import views as openedx_debug_views
from openedx.core.djangoapps.django_comment_common.models import ForumsConfig
from openedx.core.djangoapps.lang_pref import views as lang_pref_views
from openedx.core.djangoapps.password_policy import compliance as password_policy_compliance
from openedx.core.djangoapps.password_policy.forms import PasswordPolicyAwareAdminAuthForm
from openedx.core.djangoapps.plugins.constants import ProjectType
from openedx.core.djangoapps.programs.models import ProgramsApiConfig
from openedx.core.djangoapps.site_configuration import helpers as configuration_helpers
from openedx.core.djangoapps.user_authn.views.login import redirect_to_lms_login
from openedx.features.enterprise_support.api import enterprise_enabled
RESET_COURSE_DEADLINES_NAME = 'reset_course_deadlines'
RENDER_XBLOCK_NAME = 'render_xblock'
RENDER_VIDEO_XBLOCK_NAME = 'render_public_video_xblock'
RENDER_VIDEO_XBLOCK_EMBED_NAME = 'render_public_video_xblock_embed'
COURSE_PROGRESS_NAME = 'progress'
if settings.DEBUG or settings.FEATURES.get('ENABLE_DJANGO_ADMIN_SITE'):
django_autodiscover()
admin.site.site_header = _('LMS Administration')
admin.site.site_title = admin.site.site_header
if password_policy_compliance.should_enforce_compliance_on_login():
admin.site.login_form = PasswordPolicyAwareAdminAuthForm
# Custom error pages
# These are used by Django to render these error codes. Do not remove.
# pylint: disable=invalid-name
handler403 = static_template_view_views.render_403
handler404 = static_template_view_views.render_404
handler429 = static_template_view_views.render_429
handler500 = static_template_view_views.render_500
notification_prefs_urls = [
path('notification_prefs/enable/', notification_prefs_views.ajax_enable),
path('notification_prefs/disable/', notification_prefs_views.ajax_disable),
path('notification_prefs/status/', notification_prefs_views.ajax_status),
re_path(
r'^notification_prefs/unsubscribe/(?P<token>[a-zA-Z0-9-_=]+)/',
notification_prefs_views.set_subscription,
{'subscribe': False},
name='unsubscribe_forum_update',
),
re_path(
r'^notification_prefs/resubscribe/(?P<token>[a-zA-Z0-9-_=]+)/',
notification_prefs_views.set_subscription,
{'subscribe': True},
name='resubscribe_forum_update',
),
]
urlpatterns = [
path('', branding_views.index, name='root'), # Main marketing page, or redirect to courseware
path('', include('common.djangoapps.student.urls')),
# TODO: Move lms specific student views out of common code
re_path(r'^dashboard/?$', student_views.student_dashboard, name='dashboard'),
path('change_enrollment', student_views.change_enrollment, name='change_enrollment'),
# Event tracking endpoints
path('', include('common.djangoapps.track.urls')),
# Static template view endpoints like blog, faq, etc.
path('', include('lms.djangoapps.static_template_view.urls')),
path('heartbeat', include('openedx.core.djangoapps.heartbeat.urls')),
path('i18n/', include('django.conf.urls.i18n')),
# Course assets
path('', include('openedx.core.djangoapps.contentserver.urls')),
# Enrollment API RESTful endpoints
path('api/enrollment/v1/', include('openedx.core.djangoapps.enrollments.urls')),
# Agreements API RESTful endpoints
path('api/agreements/v1/', include('openedx.core.djangoapps.agreements.urls')),
# Entitlement API RESTful endpoints
path(
'api/entitlements/',
include(
('common.djangoapps.entitlements.rest_api.urls', 'common.djangoapps.entitlements'),
namespace='entitlements_api',
),
),
# Courseware search endpoints
path('search/', include('search.urls')),
# Course API
path('api/courses/', include('lms.djangoapps.course_api.urls')),
# User API endpoints
path('api/user/', include('openedx.core.djangoapps.user_api.urls')),
# Note: these are older versions of the User API that will eventually be
# subsumed by api/user listed above.
path('', include('openedx.core.djangoapps.user_api.legacy_urls')),
# Profile Images API endpoints
path('api/profile_images/', include('openedx.core.djangoapps.profile_images.urls')),
# Video Abstraction Layer used to allow video teams to manage video assets
# independently of courseware. https://github.com/openedx/edx-val
path('api/val/v0/', include('edxval.urls')),
path(
'api/commerce/',
include(
('lms.djangoapps.commerce.api.urls', 'lms.djangoapps.commerce'),
namespace='commerce_api',
),
),
path('api/credit/', include('openedx.core.djangoapps.credit.urls')),
path('api/toggles/', include('openedx.core.djangoapps.waffle_utils.urls')),
path('rss_proxy/', include('lms.djangoapps.rss_proxy.urls')),
path('api/organizations/', include('organizations.urls', namespace='organizations')),
path('catalog/', include(('openedx.core.djangoapps.catalog.urls', 'openedx.core.djangoapps.catalog'),
namespace='catalog')),
# Update session view
path('lang_pref/update_language', lang_pref_views.update_language, name='update_language'),
# Multiple course modes and identity verification
path(
'course_modes/',
include('common.djangoapps.course_modes.urls'),
),
path(
'api/course_modes/',
include(
('common.djangoapps.course_modes.rest_api.urls', 'common.djangoapps.course_mods'),
namespace='course_modes_api',
)
),
path('verify_student/', include('lms.djangoapps.verify_student.urls')),
# URLs for managing dark launches of languages
path('update_lang/', include(('openedx.core.djangoapps.dark_lang.urls', 'openedx.core.djangoapps.dark_lang'),
namespace='dark_lang')),
# For redirecting to help pages.
path('help_token/', include('help_tokens.urls')),
# URLs for API access management
path('api-admin/', include(('openedx.core.djangoapps.api_admin.urls', 'openedx.core.djangoapps.api_admin'),
namespace='api_admin')),
# Learner Home and Program Dashboard
path('api/learner_home/', include('lms.djangoapps.learner_home.urls', namespace='learner_home')),
path('dashboard/', include('lms.djangoapps.learner_dashboard.urls')),
# This is the legacy URL for the program dashboard API when the legacy learner dashboard existed.
# Current-and-future advertised URLs for this API will be under 'api/learner_home'
path('api/dashboard/', include('openedx.core.djangoapps.programs.rest_api.urls', namespace='dashboard_api')),
path(
'api/experiments/',
include(
('lms.djangoapps.experiments.urls', 'lms.djangoapps.experiments'),
namespace='api_experiments',
),
),
path('api/discounts/', include(('openedx.features.discounts.urls', 'openedx.features.discounts'),
namespace='api_discounts')),
# Provide URLs where we can see the rendered error pages without having to force an error.
path('403', handler403, name='render_403'),
path('404', handler404, name='render_404'),
path('429', handler429, name='render_429'),
path('500', handler500, name='render_500'),
]
if settings.FEATURES.get('ENABLE_MOBILE_REST_API'):
urlpatterns += [
re_path(r'^api/mobile/(?P<api_version>v(4|3|2|1|0.5))/', include('lms.djangoapps.mobile_api.urls')),
]
urlpatterns += [
path('openassessment/fileupload/', include('openassessment.fileupload.urls')),
]
urlpatterns += [
path('support/', include('lms.djangoapps.support.urls')),
]
# Favicon
favicon_path = configuration_helpers.get_value('favicon_path', settings.FAVICON_PATH) # pylint: disable=invalid-name
urlpatterns += [
re_path(r'^favicon\.ico$', RedirectView.as_view(url=settings.STATIC_URL + favicon_path, permanent=True)),
]
# Multicourse wiki (Note: wiki urls must be above the courseware ones because of
# the custom tab catch-all)
if settings.WIKI_ENABLED:
from django_notify.urls import get_pattern as notify_pattern
from wiki.urls import get_pattern as wiki_pattern
from lms.djangoapps.course_wiki import views as course_wiki_views
wiki_url_patterns, wiki_app_name = wiki_pattern()
notify_url_patterns, notify_app_name = notify_pattern()
urlpatterns += [
# First we include views from course_wiki that we use to override the default views.
# They come first in the urlpatterns so they get resolved first
path('wiki/create-root/', course_wiki_views.root_create, name='root_create'),
path('wiki/', include((wiki_url_patterns, wiki_app_name), namespace='wiki')),
path('notify/', include((notify_url_patterns, notify_app_name), namespace='notify')),
# These urls are for viewing the wiki in the context of a course. They should
# never be returned by a reverse() so they come after the other url patterns
re_path(fr'^courses/{settings.COURSE_ID_PATTERN}/course_wiki/?$',
course_wiki_views.course_wiki_redirect, name='course_wiki'),
re_path(fr'^courses/{settings.COURSE_KEY_REGEX}/wiki/',
include((wiki_url_patterns, 'course_wiki_do_not_reverse'), namespace='course_wiki_do_not_reverse')),
]
urlpatterns += [
# jump_to URLs for direct access to a location in the course
re_path(
r'^courses/{}/jump_to/(?P<location>.*)$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.jump_to,
name='jump_to',
),
re_path(
r'^courses/{}/jump_to_id/(?P<module_id>.*)$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.jump_to_id,
name='jump_to_id',
),
# xblock Handler APIs
re_path(
r'^courses/{course_key}/xblock/{usage_key}/handler/(?P<handler>[^/]*)(?:/(?P<suffix>.*))?$'.format(
course_key=settings.COURSE_ID_PATTERN,
usage_key=settings.USAGE_ID_PATTERN,
),
handle_xblock_callback,
name='xblock_handler',
),
re_path(
r'^courses/{course_key}/xblock/{usage_key}/handler_noauth/(?P<handler>[^/]*)(?:/(?P<suffix>.*))?$'.format(
course_key=settings.COURSE_ID_PATTERN,
usage_key=settings.USAGE_ID_PATTERN,
),
handle_xblock_callback_noauth,
name='xblock_handler_noauth',
),
# xblock View API
# (unpublished) API that returns JSON with the HTML fragment and related resources
# for the xBlock's requested view.
re_path(
r'^courses/{course_key}/xblock/{usage_key}/view/(?P<view_name>[^/]*)$'.format(
course_key=settings.COURSE_ID_PATTERN,
usage_key=settings.USAGE_ID_PATTERN,
),
xblock_view,
name='xblock_view',
),
# xblock Rendering View URL
# URL to provide an HTML view of an xBlock. The view type (e.g., student_view) is
# passed as a 'view' parameter to the URL.
# Note: This is not an API. Compare this with the xblock_view API above.
re_path(
fr'^xblock/{settings.USAGE_KEY_PATTERN}$',
courseware_views.render_xblock,
name=RENDER_XBLOCK_NAME,
),
re_path(
fr'^videos/embed/{settings.USAGE_KEY_PATTERN}$',
courseware_views.PublicVideoXBlockEmbedView.as_view(),
name=RENDER_VIDEO_XBLOCK_EMBED_NAME,
),
re_path(
fr'^videos/{settings.USAGE_KEY_PATTERN}$',
courseware_views.PublicVideoXBlockView.as_view(),
name=RENDER_VIDEO_XBLOCK_NAME,
),
# xblock Resource URL
re_path(
r'xblock/resource/(?P<block_type>[^/]+)/(?P<uri>.*)$',
xblock_resource,
name='xblock_resource_url',
),
# New (Learning-Core-based) XBlock REST API
path('', include(('openedx.core.djangoapps.xblock.rest_api.urls', 'openedx.core.djangoapps.xblock'),
namespace='xblock_api')),
re_path(
r'^courses/{}/xqueue/(?P<userid>[^/]*)/(?P<mod_id>.*?)/(?P<dispatch>[^/]*)$'.format(
settings.COURSE_ID_PATTERN,
),
xqueue_callback,
name='xqueue_callback',
),
# TODO: These views need to be updated before they work
path('calculate', util_views.calculate),
path(
'reset_deadlines',
util_views.reset_course_deadlines,
name=RESET_COURSE_DEADLINES_NAME,
),
re_path(r'^courses/?$', branding_views.courses, name='courses'),
# About the course
re_path(
r'^courses/{}/about$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.course_about,
name='about_course',
),
path(
'courses/yt_video_metadata',
courseware_views.yt_video_metadata,
name='yt_video_metadata',
),
re_path(
r'^courses/{}/enroll_staff$'.format(
settings.COURSE_ID_PATTERN,
),
EnrollStaffView.as_view(),
name='enroll_staff',
),
# Inside the course
re_path(
r'^courses/{}/$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.course_about,
name='course_root',
),
# TODO arjun remove when custom tabs in place, see courseware/courses.py
re_path(
r'^courses/{}/syllabus$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.syllabus,
name='syllabus',
),
# Survey associated with a course
re_path(
r'^courses/{}/survey$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.course_survey,
name='course_survey',
),
re_path(
r'^courses/{}/book/(?P<book_index>\d+)/$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.index,
name='book',
),
re_path(
r'^courses/{}/book/(?P<book_index>\d+)/(?P<page>\d+)$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.index,
name='book',
),
re_path(
r'^courses/{}/pdfbook/(?P<book_index>\d+)/$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.pdf_index,
name='pdf_book',
),
re_path(
r'^courses/{}/pdfbook/(?P<book_index>\d+)/(?P<page>\d+)$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.pdf_index,
name='pdf_book',
),
re_path(
r'^courses/{}/pdfbook/(?P<book_index>\d+)/chapter/(?P<chapter>\d+)/$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.pdf_index,
name='pdf_book',
),
re_path(
r'^courses/{}/pdfbook/(?P<book_index>\d+)/chapter/(?P<chapter>\d+)/(?P<page>\d+)$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.pdf_index,
name='pdf_book',
),
re_path(
r'^courses/{}/htmlbook/(?P<book_index>\d+)/$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.html_index,
name='html_book',
),
re_path(
r'^courses/{}/htmlbook/(?P<book_index>\d+)/chapter/(?P<chapter>\d+)/$'.format(
settings.COURSE_ID_PATTERN,
),
staticbook_views.html_index,
name='html_book',
),
re_path(
r'^courses/{}/courseware/?$'.format(
settings.COURSE_ID_PATTERN,
),
CoursewareIndex.as_view(),
name='courseware',
),
re_path(
r'^courses/{}/courseware/(?P<chapter>[^/]*)/$'.format(
settings.COURSE_ID_PATTERN,
),
CoursewareIndex.as_view(),
name='courseware_chapter',
),
re_path(
r'^courses/{}/courseware/(?P<chapter>[^/]*)/(?P<section>[^/]*)/$'.format(
settings.COURSE_ID_PATTERN,
),
CoursewareIndex.as_view(),
name='courseware_section',
),
re_path(
r'^courses/{}/courseware/(?P<chapter>[^/]*)/(?P<section>[^/]*)/(?P<position>[^/]*)/?$'.format(
settings.COURSE_ID_PATTERN,
),
CoursewareIndex.as_view(),
name='courseware_position',
),
# progress page
re_path(
r'^courses/{}/progress$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.progress,
name=COURSE_PROGRESS_NAME,
),
# dates page (no longer functional, just redirects to MFE)
re_path(r'^courses/{}/dates'.format(settings.COURSE_ID_PATTERN), courseware_views.dates, name='dates'),
# Takes optional student_id for instructor use--shows profile as that student sees it.
re_path(
r'^courses/{}/progress/(?P<student_id>[^/]*)/$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.progress,
name='student_progress',
),
re_path(
r'^programs/{}/about'.format(
r'(?P<program_uuid>[0-9a-f-]+)',
),
courseware_views.program_marketing,
name='program_marketing_view',
),
# For the instructor
re_path(
r'^courses/{}/instructor$'.format(
settings.COURSE_ID_PATTERN,
),
instructor_dashboard_views.instructor_dashboard_2,
name='instructor_dashboard',
),
re_path(
r'^courses/{}/set_course_mode_price$'.format(
settings.COURSE_ID_PATTERN,
),
instructor_dashboard_views.set_course_mode_price,
name='set_course_mode_price',
),
# Discussions Management
re_path(
r'^courses/{}/discussions/settings$'.format(
settings.COURSE_KEY_PATTERN,
),
discussion_views.course_discussions_settings_handler,
name='course_discussions_settings',
),
# Cohorts management API
path('api/cohorts/', include(
('openedx.core.djangoapps.course_groups.urls', 'openedx.core.djangoapps.course_groups'),
namespace='api_cohorts')),
# Cohorts management
re_path(
r'^courses/{}/cohorts/settings$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.course_cohort_settings_handler,
name='course_cohort_settings',
),
re_path(
r'^courses/{}/cohorts/(?P<cohort_id>[0-9]+)?$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.cohort_handler,
name='cohorts',
),
re_path(
r'^courses/{}/cohorts/(?P<cohort_id>[0-9]+)$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.users_in_cohort,
name='list_cohort',
),
re_path(
r'^courses/{}/cohorts/(?P<cohort_id>[0-9]+)/add$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.add_users_to_cohort,
name='add_to_cohort',
),
re_path(
r'^courses/{}/cohorts/(?P<cohort_id>[0-9]+)/delete$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.remove_user_from_cohort,
name='remove_from_cohort',
),
re_path(
r'^courses/{}/cohorts/debug$'.format(
settings.COURSE_KEY_PATTERN,
),
course_groups_views.debug_cohort_mgmt,
name='debug_cohort_mgmt',
),
re_path(
r'^courses/{}/discussion/topics$'.format(
settings.COURSE_KEY_PATTERN,
),
discussion_views.discussion_topics,
name='discussion_topics',
),
# LTI endpoints listing
re_path(
r'^courses/{}/lti_rest_endpoints/'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.get_course_lti_endpoints,
name='lti_rest_endpoints',
),
# Student Notes
re_path(
r'^courses/{}/edxnotes/'.format(
settings.COURSE_ID_PATTERN,
),
include('lms.djangoapps.edxnotes.urls'),
name='edxnotes_endpoints',
),
# Student Notes API
path(
'api/edxnotes/v1/',
include('lms.djangoapps.edxnotes.api_urls'),
),
# Branding API
path(
'api/branding/v1/',
include('lms.djangoapps.branding.api_urls')
),
# Course experience
re_path(
r'^courses/{}/course/'.format(
settings.COURSE_ID_PATTERN,
),
include('openedx.features.course_experience.urls'),
),
# Course bookmarks UI in LMS
re_path(
r'^courses/{}/bookmarks/'.format(
settings.COURSE_ID_PATTERN,
),
include('openedx.features.course_bookmarks.urls'),
),
# Calendar Sync UI in LMS
re_path(
fr'^courses/{settings.COURSE_ID_PATTERN}/',
include('openedx.features.calendar_sync.urls'),
),
# Survey Report
re_path(
fr'^survey_report/',
include('openedx.features.survey_report.urls'),
),
]
if settings.FEATURES.get('ENABLE_TEAMS'):
# Teams endpoints
urlpatterns += [
path(
'api/team/',
include('lms.djangoapps.teams.api_urls')
),
re_path(
r'^courses/{}/teams/'.format(
settings.COURSE_ID_PATTERN,
),
include('lms.djangoapps.teams.urls'),
name='teams_endpoints',
),
]
# allow course staff to change to student view of courseware
if settings.FEATURES.get('ENABLE_MASQUERADE'):
urlpatterns += [
re_path(
r'^courses/{}/masquerade$'.format(
settings.COURSE_KEY_PATTERN,
),
MasqueradeView.as_view(),
name='masquerade_update',
),
]
urlpatterns += [
re_path(
r'^courses/{}/generate_user_cert'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.generate_user_cert,
name='generate_user_cert',
),
]
# discussion forums live within courseware, so courseware must be enabled first
if settings.FEATURES.get('ENABLE_DISCUSSION_SERVICE'):
urlpatterns += [
path(
'api/discussion/',
include('lms.djangoapps.discussion.rest_api.urls')
),
re_path(
r'^courses/{}/discussion/'.format(
settings.COURSE_ID_PATTERN,
),
include('lms.djangoapps.discussion.django_comment_client.urls')
),
]
if is_forum_daily_digest_enabled():
urlpatterns += notification_prefs_urls
urlpatterns += [
path('bulk_email/', include('lms.djangoapps.bulk_email.urls')),
]
urlpatterns += [
re_path(
r'^courses/{}/tab/(?P<tab_type>[^/]+)/$'.format(
settings.COURSE_ID_PATTERN,
),
CourseTabView.as_view(),
name='course_tab_view',
),
]
urlpatterns += [
re_path(
r'^courses/{}/courseware-search/enabled/$'.format(
settings.COURSE_ID_PATTERN,
),
courseware_views.courseware_mfe_search_enabled,
name='courseware_search_enabled_view',
),
re_path(
fr'^courses/{settings.COURSE_ID_PATTERN}/courseware-navigation-sidebar/toggles/$',
courseware_views.courseware_mfe_navigation_sidebar_toggles,
name='courseware_navigation_sidebar_toggles_view',
),
]
urlpatterns += [
re_path(
r'^courses/{}/lti_tab/(?P<provider_uuid>[^/]+)/$'.format(
settings.COURSE_ID_PATTERN,
),
CourseTabView.as_view(),
name='lti_course_tab',
kwargs={'tab_type': 'lti_tab'},
),
]
urlpatterns += [
# This MUST be the last view in the courseware--it's a catch-all for custom tabs.
re_path(
r'^courses/{}/(?P<tab_slug>[^/]+)/$'.format(
settings.COURSE_ID_PATTERN,
),
StaticCourseTabView.as_view(),
name='static_tab',
),
]
if settings.FEATURES.get('ENABLE_STUDENT_HISTORY_VIEW'):
urlpatterns += [
re_path(
r'^courses/{}/submission_history/(?P<learner_identifier>[^/]*)/(?P<location>.*?)$'.format(
settings.COURSE_ID_PATTERN
),
courseware_views.submission_history,
name='submission_history',
),
]
if settings.DEBUG or settings.FEATURES.get('ENABLE_DJANGO_ADMIN_SITE'):
# Jasmine and admin
# The password pages in the admin tool are disabled so that all password
# changes go through our user portal and follow complexity requirements.
# The form to change another user's password is conditionally enabled
# for backwards compatibility.
if not settings.FEATURES.get('ENABLE_CHANGE_USER_PASSWORD_ADMIN'):
urlpatterns += [
re_path(r'^admin/auth/user/\d+/password/$', handler404),
]
urlpatterns += [
path('admin/password_change/', handler404),
# We are enforcing users to login through third party auth in site's
# login page so we are disabling the admin panel's login page.
path('admin/login/', redirect_to_lms_login),
path('admin/', admin.site.urls),
]
if configuration_helpers.get_value('ENABLE_BULK_ENROLLMENT_VIEW', settings.FEATURES.get('ENABLE_BULK_ENROLLMENT_VIEW')):
urlpatterns += [
path('api/bulk_enroll/v1/', include('lms.djangoapps.bulk_enroll.urls')),
]
# Embargo
if settings.FEATURES.get('EMBARGO'):
urlpatterns += [
path('embargo/', include(('openedx.core.djangoapps.embargo.urls', 'openedx.core.djangoapps.embargo'),
namespace='embargo')),
path('api/embargo/', include(('openedx.core.djangoapps.embargo.urls', 'openedx.core.djangoapps.embargo'),
namespace='api_embargo')),
]
# Survey Djangoapp
urlpatterns += [
path('survey/', include('lms.djangoapps.survey.urls')),
]
if settings.FEATURES.get('ENABLE_OAUTH2_PROVIDER'):
urlpatterns += [
# These URLs dispatch to django-oauth-toolkit or django-oauth2-provider as appropriate.
# Developers should use these routes, to maintain compatibility for existing client code
path('oauth2/', include('openedx.core.djangoapps.oauth_dispatch.urls')),
# The /_o/ prefix exists to provide a target for code in django-oauth-toolkit that
# uses reverse() with the 'oauth2_provider' namespace. Developers should not access these
# views directly, but should rather use the wrapped views at /oauth2/
path('_o/', include('oauth2_provider.urls', namespace='oauth2_provider')),
]
if settings.FEATURES.get('ENABLE_SERVICE_STATUS'):
urlpatterns += [
path('status/', include('openedx.core.djangoapps.service_status.urls')),
]
if settings.FEATURES.get('ENABLE_INSTRUCTOR_BACKGROUND_TASKS'):
urlpatterns += [
path(
'instructor_task_status/',
instructor_task_views.instructor_task_status,
name='instructor_task_status'
),
]
if settings.FEATURES.get('ENABLE_DEBUG_RUN_PYTHON'):
urlpatterns += [
path('debug/run_python', debug_views.run_python),
]
urlpatterns += [
path('debug/show_parameters', debug_views.show_parameters),
]
# Third-party auth.
if settings.FEATURES.get('ENABLE_THIRD_PARTY_AUTH'):
urlpatterns += [
path('', include('common.djangoapps.third_party_auth.urls')),
path('api/third_party_auth/', include('common.djangoapps.third_party_auth.api.urls')),
]
# Enterprise
if enterprise_enabled():
urlpatterns += [
path('', include('enterprise.urls')),
]
# OAuth token exchange
if settings.FEATURES.get('ENABLE_OAUTH2_PROVIDER'):
urlpatterns += [
path(
'oauth2/login/',
LoginWithAccessTokenView.as_view(),
name='login_with_access_token'
),
]
# Certificates
urlpatterns += [
path('certificates/', include('lms.djangoapps.certificates.urls')),
# REST APIs
path('api/certificates/', include(('lms.djangoapps.certificates.apis.urls', 'lms.djangoapps.certificates'),
namespace='certificates_api')),
]
# XDomain proxy
urlpatterns += [
re_path(r'^xdomain_proxy.html$', cors_csrf_views.xdomain_proxy, name='xdomain_proxy'),
]
# Custom courses on edX (CCX) URLs
if settings.FEATURES.get('CUSTOM_COURSES_EDX'):
urlpatterns += [
re_path(fr'^courses/{settings.COURSE_ID_PATTERN}/', include('lms.djangoapps.ccx.urls')),
path('api/ccx/', include(('lms.djangoapps.ccx.api.urls', 'lms.djangoapps.ccx'), namespace='ccx_api')),
]
# Access to courseware as an LTI provider
if settings.FEATURES.get('ENABLE_LTI_PROVIDER'):
urlpatterns += [
path('lti_provider/', include('lms.djangoapps.lti_provider.urls')),
]
urlpatterns += [
path('config/programs', ConfigurationModelCurrentAPIView.as_view(model=ProgramsApiConfig)),
path('config/catalog', ConfigurationModelCurrentAPIView.as_view(model=CatalogIntegration)),
path('config/forums', ConfigurationModelCurrentAPIView.as_view(model=ForumsConfig)),
]
if settings.DEBUG:
urlpatterns += static(settings.STATIC_URL, document_root=settings.STATIC_ROOT)
# profile image urls must come before the media url to work
urlpatterns += static(
settings.PROFILE_IMAGE_BACKEND['options']['base_url'],
document_root=settings.PROFILE_IMAGE_BACKEND['options']['location']
)
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
# UX reference templates
urlpatterns += [
path('template/<path:template>', openedx_debug_views.show_reference_template),
]
if 'debug_toolbar' in settings.INSTALLED_APPS:
import debug_toolbar
urlpatterns += [
path('__debug__/', include(debug_toolbar.urls)),
]
if settings.FEATURES.get('ENABLE_FINANCIAL_ASSISTANCE_FORM'):
urlpatterns += [
path(
'financial-assistance/',
courseware_views.financial_assistance,
name='financial_assistance'
),
path(
'financial-assistance/apply/',
courseware_views.financial_assistance_form,
name='financial_assistance_form'
),
path(
'financial-assistance/submit/',
courseware_views.financial_assistance_request,
name='submit_financial_assistance_request'
),
path(
'financial-assistance_v2/submit/',
courseware_views.financial_assistance_request_v2,
name='submit_financial_assistance_request_v2'
),
re_path(
fr'financial-assistance/{settings.COURSE_ID_PATTERN}/apply/',
courseware_views.financial_assistance_form,
name='financial_assistance_form_v2'
),
re_path(
fr'financial-assistance/{settings.COURSE_ID_PATTERN}',
courseware_views.financial_assistance,
name='financial_assistance_v2'
)
]
# API docs.
urlpatterns += make_docs_urls(api_info)
# edx-drf-extensions csrf app
urlpatterns += [
path('', include('csrf.urls')),
]
if 'openedx.testing.coverage_context_listener' in settings.INSTALLED_APPS:
urlpatterns += [
path('coverage_context', include('openedx.testing.coverage_context_listener.urls'))
]
urlpatterns.append(
path(
'api/learning_sequences/',
include(
('openedx.core.djangoapps.content.learning_sequences.urls', 'learning_sequences'),
namespace='learning_sequences'
),
),
)
urlpatterns.extend(get_plugin_url_patterns(ProjectType.LMS))
# Course Home API urls
urlpatterns += [
# This is a BFF ("backend for frontend") djangoapp for the Learning MFE (like courseware_api).
# It will change and morph as needed for the frontend, and is not a stable API on which other code can rely.
path('api/course_home/', include(('lms.djangoapps.course_home_api.urls', 'course-home'))),
# This v1 version is just kept for transitional reasons and is going away as soon as the MFE stops referencing it.
# We don't promise any sort of versioning stability.
path('api/course_home/v1/', include(('lms.djangoapps.course_home_api.urls', 'course-home-v1'))),
]
# User Tour API urls
urlpatterns += [
path('api/user_tours/', include('lms.djangoapps.user_tours.urls')),
]
# Course Experience API urls
urlpatterns += [
path('api/course_experience/', include('openedx.features.course_experience.api.v1.urls')),
]
# Bulk User Retirement API urls
if settings.FEATURES.get('ENABLE_BULK_USER_RETIREMENT'):
urlpatterns += [
path('', include('lms.djangoapps.bulk_user_retirement.urls')),
]
# Provider States urls
if getattr(settings, 'PROVIDER_STATES_URL', None):
from lms.djangoapps.courseware.tests.pacts.views import provider_state as courseware_xblock_handler_provider_state
urlpatterns += [
path(
'courses/xblock/handler/provider_states',
courseware_xblock_handler_provider_state,
name='courseware_xblock_handler_provider_state',
)
]
# Enhanced Staff Grader (ESG) URLs
urlpatterns += [
path('api/ora_staff_grader/', include('lms.djangoapps.ora_staff_grader.urls', 'ora-staff-grader')),
]
# Scheduled Bulk Email (Instructor Task) URLs
urlpatterns += [
path('api/instructor_task/', include('lms.djangoapps.instructor_task.rest_api.urls')),
]
# MFE API urls
urlpatterns += [
path('api/mfe_config/v1', include(('lms.djangoapps.mfe_config_api.urls', 'lms.djangoapps.mfe_config_api'), namespace='mfe_config_api'))
]
urlpatterns += [
path('api/notifications/', include('openedx.core.djangoapps.notifications.urls')),
]