Skip to content
Snippets Groups Projects
views.py 9.36 KiB
Newer Older
David Ormsbee's avatar
David Ormsbee committed
import logging
Piotr Mitros's avatar
Piotr Mitros committed
import urllib
Piotr Mitros's avatar
Piotr Mitros committed

from django.conf import settings
David Ormsbee's avatar
David Ormsbee committed
from django.core.context_processors import csrf
from django.core.urlresolvers import reverse
David Ormsbee's avatar
David Ormsbee committed
from django.contrib.auth.models import User
from django.contrib.auth.decorators import login_required
David Ormsbee's avatar
David Ormsbee committed
from django.shortcuts import redirect
from mitxmako.shortcuts import render_to_response, render_to_string
#from django.views.decorators.csrf import ensure_csrf_cookie
from django_future.csrf import ensure_csrf_cookie
from django.views.decorators.cache import cache_control
Calen Pennington's avatar
Calen Pennington committed
from module_render import toc_for_course, get_module, get_section
from student.models import UserProfile
from multicourse import multicourse_settings
from xmodule.modulestore import Location
from xmodule.modulestore.exceptions import InvalidLocationError, ItemNotFoundError, NoPathToItem
from xmodule.modulestore.django import modulestore
from xmodule.course_module import CourseDescriptor
from util.cache import cache, cache_if_anonymous
from student.models import UserTestGroup, CourseEnrollment
from courseware import grades
from courseware.courses import check_course
log = logging.getLogger("mitx.courseware")

Piotr Mitros's avatar
Piotr Mitros committed

def user_groups(user):
    if not user.is_authenticated():
        return []

    # TODO: Rewrite in Django
    key = 'user_group_names_{user.id}'.format(user=user)
    cache_expiration = 60 * 60  # one hour

    # Kill caching on dev machines -- we switch groups a lot
    group_names = cache.get(key)

    if group_names is None:
        group_names = [u.name for u in UserTestGroup.objects.filter(users=user)]
        cache.set(key, group_names, cache_expiration)

    return group_names


def format_url_params(params):
    return [urllib.quote(string.replace(' ', '_')) for string in params]
Piotr Mitros's avatar
Piotr Mitros committed

@ensure_csrf_cookie
@cache_if_anonymous
def courses(request):
    # TODO: Clean up how 'error' is done.
    courses = sorted(modulestore().get_courses(), key=lambda course: course.number)
    for course in courses:
        universities[course.org].append(course)
    return render_to_response("courses.html", {'universities': universities})

Piotr Mitros's avatar
Piotr Mitros committed
@cache_control(no_cache=True, no_store=True, must_revalidate=True)
    if 'course_admin' not in user_groups(request.user):
Piotr Mitros's avatar
Piotr Mitros committed
        raise Http404
Piotr Mitros's avatar
Piotr Mitros committed
    student_objects = User.objects.all()[:100]
    student_info = []
    for student in student_objects:
        student_module_cache = StudentModuleCache(student, course)
        course, _, _, _ = get_module(request.user, request, course.location, student_module_cache)
        student_info.append({
            'username': student.username,
            'id': student.id,
            'email': student.email,
            'grade_info': grades.grade_sheet(student, course, student_module_cache),
            'realname': UserProfile.objects.get(user=student).name
Piotr Mitros's avatar
Piotr Mitros committed

Matthew Mongeau's avatar
Matthew Mongeau committed
    return render_to_response('gradebook.html', {'students': student_info, 'course': course})
Piotr Mitros's avatar
Piotr Mitros committed

@cache_control(no_cache=True, no_store=True, must_revalidate=True)
def profile(request, course_id, student_id=None):
Piotr Mitros's avatar
Piotr Mitros committed
    ''' User profile. Show username, location, etc, as well as grades .
        We need to allow the user to change some of these settings .'''
    if student_id is None:
        if 'course_admin' not in user_groups(request.user):
        student = User.objects.get(id=int(student_id))
    user_info = UserProfile.objects.get(user=student)
Piotr Mitros's avatar
Piotr Mitros committed

    student_module_cache = StudentModuleCache(request.user, course)
    course, _, _, _ = get_module(request.user, request, course.location, student_module_cache)
    context = {'name': user_info.name,
               'username': student.username,
               'location': user_info.location,
               'language': user_info.language,
               'email': student.email,
Matthew Mongeau's avatar
Matthew Mongeau committed
               'course': course,
               'format_url_params': format_url_params,
               'csrf': csrf(request)['csrf_token']
               }
    context.update(grades.grade_sheet(student, course, student_module_cache))
Piotr Mitros's avatar
Piotr Mitros committed
    return render_to_response('profile.html', context)


def render_accordion(request, course, chapter, section):
Piotr Mitros's avatar
Piotr Mitros committed
    ''' Draws navigation bar. Takes current position in accordion as
        parameter.

        If chapter and section are '' or None, renders a default accordion.

        Returns (initialization_javascript, content)'''
    # TODO (cpennington): do the right thing with courses
    toc = toc_for_course(request.user, request, course, chapter, section)
Piotr Mitros's avatar
Piotr Mitros committed
    for i in range(len(toc)):
        if toc[i]['active']:
    context = dict([('active_chapter', active_chapter),
                    ('toc', toc),
                    ('course_name', course.title),
                    ('course_id', course.id),
                    ('format_url_params', format_url_params),
                    ('csrf', csrf(request)['csrf_token'])] + template_imports.items())
    return render_to_string('accordion.html', context)

Piotr Mitros's avatar
Piotr Mitros committed

@ensure_csrf_cookie
@cache_control(no_cache=True, no_store=True, must_revalidate=True)
def index(request, course_id, chapter=None, section=None,
    ''' Displays courseware accordion, and any associated content.
    If course, chapter, and section aren't all specified, just returns
    the accordion.  If they are specified, returns an error if they don't
    point to a valid module.

    Arguments:

     - request    : HTTP request
     - course     : coursename (str)
     - chapter    : chapter name (str)
     - section    : section name (str)
     - position   : position in module, eg of <sequential> module (str)

    Returns:

     - HTTPresponse
    '''
    def clean(s):
        ''' Fixes URLs -- we convert spaces to _ in URLs to prevent
        funny encoding characters and keep the URLs readable.  This undoes
        that transformation.
        '''
        return s.replace('_', ' ') if s is not None else None
    chapter = clean(chapter)
    section = clean(section)

    if settings.ENABLE_MULTICOURSE:
        settings.MODULESTORE['default']['OPTIONS']['data_dir'] = settings.DATA_DIR + multicourse_settings.get_course_xmlpath(course)

    context = {
        'csrf': csrf(request)['csrf_token'],
Calen Pennington's avatar
Calen Pennington committed
        'accordion': render_accordion(request, course, chapter, section),
Matthew Mongeau's avatar
Matthew Mongeau committed
        'COURSE_TITLE': course.title,
        'course': course,
    look_for_module = chapter is not None and section is not None
    if look_for_module:
        # TODO (cpennington): Pass the right course in here

        section = get_section(course, chapter, section)
        student_module_cache = StudentModuleCache(request.user, section)
        module, _, _, _ = get_module(request.user, request, section.location, student_module_cache)
    result = render_to_response('courseware.html', context)
    return result
def jump_to(request, location):
    Show the page that contains a specific location.
    If the location is invalid, return a 404.
    If the location is valid, but not present in a course, ?
    If the location is valid, but in a course the current user isn't registered for, ?
        TODO -- let the index view deal with it?
    '''
    # Complain if the location isn't valid
    try:
        location = Location(location)
    except InvalidLocationError:
        raise Http404("Invalid location")

    # Complain if there's not data for this location
    try:
        (course, chapter, section, position) = modulestore().path_to_location(location)
    except ItemNotFoundError:
        raise Http404("No data at this location: {0}".format(location))
    except NoPathToItem:
        raise Http404("This location is not in any class: {0}".format(location))
    

    return index(course, chapter, section, position)
def course_info(request, course_id):
    course = check_course(course_id)
    return render_to_response('info.html', {'course': course})
def course_about(request, course_id):
    def registered_for_course(course, user):
        if user.is_authenticated():
            return CourseEnrollment.objects.filter(user=user, course_id=course.id).exists()
    course = check_course(course_id, course_must_be_open=False)
    registered = registered_for_course(course, request.user)
    return render_to_response('portal/course_about.html', {'course': course, 'registered': registered})
def university_profile(request, org_id):
    all_courses = sorted(modulestore().get_courses(), key=lambda course: course.number)
    valid_org_ids = set(c.org for c in all_courses)
    if org_id not in valid_org_ids:
        raise Http404("University Profile not found for {0}".format(org_id))

    # Only grab courses for this org...
    courses = [c for c in all_courses if c.org == org_id]
    context = dict(courses=courses, org_id=org_id)
    template_file = "university_profile/{0}.html".format(org_id).lower()

    return render_to_response(template_file, context)