Source code for nbgrader.api

from __future__ import division

from . import utils

from sqlalchemy import (create_engine, ForeignKey, Column, String, Text,
    DateTime, Interval, Float, Enum, UniqueConstraint, Boolean)
from sqlalchemy.orm import sessionmaker, scoped_session, relationship, column_property
from sqlalchemy.orm.exc import NoResultFound, FlushError
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.ext.associationproxy import association_proxy
from sqlalchemy.exc import IntegrityError
from sqlalchemy.sql import and_
from sqlalchemy import select, func, exists, case, literal_column

from uuid import uuid4

Base = declarative_base()

class InvalidEntry(ValueError):
    pass
class MissingEntry(ValueError):
    pass


def new_uuid():
    return uuid4().hex


[docs]class Assignment(Base): """Database representation of the master/source version of an assignment.""" __tablename__ = "assignment" #: Unique id of the assignment (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique human-readable name for the assignment, such as "Problem Set 1" name = Column(String(128), unique=True, nullable=False) #: (Optional) Duedate for the assignment in datetime format, with UTC timezone duedate = Column(DateTime()) #: A collection of notebooks contained in this assignment, represented #: by :class:`~nbgrader.api.Notebook` objects notebooks = relationship("Notebook", backref="assignment", order_by="Notebook.name") #: A collection of submissions of this assignment, represented by #: :class:`~nbgrader.api.SubmittedAssignment` objects. submissions = relationship("SubmittedAssignment", backref="assignment") #: The number of submissions of this assignment num_submissions = None #: Maximum score achievable on this assignment, automatically calculated #: from the :attr:`~nbgrader.api.Notebook.max_score` of each notebook max_score = None #: Maximum coding score achievable on this assignment, automatically #: calculated from the :attr:`~nbgrader.api.Notebook.max_code_score` of #: each notebook max_code_score = None #: Maximum written score achievable on this assignment, automatically #: calculated from the :attr:`~nbgrader.api.Notebook.max_written_score` of #: each notebook max_written_score = None
[docs] def to_dict(self): """Convert the assignment object to a JSON-friendly dictionary representation. """ return { "id": self.id, "name": self.name, "duedate": self.duedate.isoformat() if self.duedate is not None else None, "num_submissions": self.num_submissions, "max_score": self.max_score, "max_code_score": self.max_code_score, "max_written_score": self.max_written_score, }
def __repr__(self): return "Assignment<{}>".format(self.name)
[docs]class Notebook(Base): """Database representation of the master/source version of a notebook.""" __tablename__ = "notebook" __table_args__ = (UniqueConstraint('name', 'assignment_id'),) #: Unique id of the notebook (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique human-readable name for the notebook, such as "Problem 1". Note #: the uniqueness is only constrained within assignments (e.g. it is ok for #: two different assignments to both have notebooks called "Problem 1", but #: the same assignment cannot have two notebooks with the same name). name = Column(String(128), nullable=False) #: The :class:`~nbgrader.api.Assignment` object that this notebook is a #: part of assignment = None #: Unique id of :attr:`~nbgrader.api.Notebook.assignment` assignment_id = Column(String(32), ForeignKey('assignment.id')) #: A collection of grade cells contained within this notebook, represented #: by :class:`~nbgrader.api.GradeCell` objects grade_cells = relationship("GradeCell", backref="notebook") #: A collection of solution cells contained within this notebook, represented #: by :class:`~nbgrader.api.SolutionCell` objects solution_cells = relationship("SolutionCell", backref="notebook") #: A collection of source cells contained within this notebook, represented #: by :class:`~nbgrader.api.SourceCell` objects source_cells = relationship("SourceCell", backref="notebook") #: A collection of submitted versions of this notebook, represented by #: :class:`~nbgrader.api.SubmittedNotebook` objects submissions = relationship("SubmittedNotebook", backref="notebook") #: The number of submissions of this notebook num_submissions = None #: Maximum score achievable on this notebook, automatically calculated #: from the :attr:`~nbgrader.api.GradeCell.max_score` of each grade cell max_score = None #: Maximum coding score achievable on this notebook, automatically #: calculated from the :attr:`~nbgrader.api.GradeCell.max_score` and #: :attr:`~nbgrader.api.GradeCell.cell_type` of each grade cell max_code_score = None #: Maximum written score achievable on this notebook, automatically #: calculated from the :attr:`~nbgrader.api.GradeCell.max_score` and #: :attr:`~nbgrader.api.GradeCell.cell_type` of each grade cell max_written_score = None #: Whether there are any submitted versions of this notebook that need to #: be manually graded, automatically determined from the #: :attr:`~nbgrader.api.SubmittedNotebook.needs_manual_grade` attribute of #: each submitted notebook needs_manual_grade = None
[docs] def to_dict(self): """Convert the notebook object to a JSON-friendly dictionary representation. """ return { "id": self.id, "name": self.name, "num_submissions": self.num_submissions, "max_score": self.max_score, "max_code_score": self.max_code_score, "max_written_score": self.max_written_score, "needs_manual_grade": self.needs_manual_grade }
def __repr__(self): return "Notebook<{}/{}>".format(self.assignment.name, self.name)
[docs]class GradeCell(Base): """Database representation of the master/source version of a grade cell.""" __tablename__ = "grade_cell" __table_args__ = (UniqueConstraint('name', 'notebook_id'),) #: Unique id of the grade cell (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique human-readable name of the grade cell. This need only be unique #: within the notebook, not across notebooks. name = Column(String(128), nullable=False) #: Maximum score that can be assigned to this grade cell max_score = Column(Float(), nullable=False) #: The cell type, either "code" or "markdown" cell_type = Column(Enum("code", "markdown", name="grade_cell_type"), nullable=False) #: The :class:`~nbgrader.api.Notebook` that this grade cell is contained in notebook = None #: Unique id of the :attr:`~nbgrader.api.GradeCell.notebook` notebook_id = Column(String(32), ForeignKey('notebook.id')) #: The assignment that this cell is contained within, represented by a #: :class:`~nbgrader.api.Assignment` object assignment = association_proxy("notebook", "assignment") #: A collection of grades assigned to submitted versions of this grade cell, #: represented by :class:`~nbgrader.api.Grade` objects grades = relationship("Grade", backref="cell")
[docs] def to_dict(self): """Convert the grade cell object to a JSON-friendly dictionary representation. Note that this includes keys for ``notebook`` and ``assignment`` which correspond to the names of the notebook and assignment, not the objects themselves. """ return { "id": self.id, "name": self.name, "max_score": self.max_score, "cell_type": self.cell_type, "notebook": self.notebook.name, "assignment": self.assignment.name }
def __repr__(self): return "GradeCell<{}/{}/{}>".format( self.assignment.name, self.notebook.name, self.name)
[docs]class SolutionCell(Base): __tablename__ = "solution_cell" __table_args__ = (UniqueConstraint('name', 'notebook_id'),) #: Unique id of the solution cell (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique human-readable name of the solution cell. This need only be unique #: within the notebook, not across notebooks. name = Column(String(128), nullable=False) #: The :class:`~nbgrader.api.Notebook` that this solution cell is contained in notebook = None #: Unique id of the :attr:`~nbgrader.api.SolutionCell.notebook` notebook_id = Column(String(32), ForeignKey('notebook.id')) #: The assignment that this cell is contained within, represented by a #: :class:`~nbgrader.api.Assignment` object assignment = association_proxy("notebook", "assignment") #: A collection of comments assigned to submitted versions of this grade cell, #: represented by :class:`~nbgrader.api.Comment` objects comments = relationship("Comment", backref="cell")
[docs] def to_dict(self): """Convert the solution cell object to a JSON-friendly dictionary representation. Note that this includes keys for ``notebook`` and ``assignment`` which correspond to the names of the notebook and assignment, not the objects themselves. """ return { "id": self.id, "name": self.name, "notebook": self.notebook.name, "assignment": self.assignment.name }
def __repr__(self): return "{}/{}".format(self.notebook, self.name)
[docs]class SourceCell(Base): __tablename__ = "source_cell" __table_args__ = (UniqueConstraint('name', 'notebook_id'),) #: Unique id of the source cell (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique human-readable name of the source cell. This need only be unique #: within the notebook, not across notebooks. name = Column(String(128), nullable=False) #: The cell type, either "code" or "markdown" cell_type = Column(Enum("code", "markdown", name="source_cell_type"), nullable=False) #: Whether the cell is locked (e.g. the source saved in the database should #: be used to overwrite the source of students' cells) locked = Column(Boolean, default=False, nullable=False) #: The source code or text of the cell source = Column(Text()) #: A checksum of the cell contents. This should usually be computed #: using :func:`nbgrader.utils.compute_checksum` checksum = Column(String(128)) #: The :class:`~nbgrader.api.Notebook` that this source cell is contained in notebook = None #: Unique id of the :attr:`~nbgrader.api.SourceCell.notebook` notebook_id = Column(String(32), ForeignKey('notebook.id')) #: The assignment that this cell is contained within, represented by a #: :class:`~nbgrader.api.Assignment` object assignment = association_proxy("notebook", "assignment")
[docs] def to_dict(self): """Convert the source cell object to a JSON-friendly dictionary representation. Note that this includes keys for ``notebook`` and ``assignment`` which correspond to the names of the notebook and assignment, not the objects themselves. """ return { "id": self.id, "name": self.name, "cell_type": self.cell_type, "locked": self.locked, "source": self.source, "checksum": self.checksum, "notebook": self.notebook.name, "assignment": self.assignment.name }
def __repr__(self): return "SolutionCell<{}/{}/{}>".format( self.assignment.name, self.notebook.name, self.name)
[docs]class Student(Base): """Database representation of a student.""" __tablename__ = "student" #: Unique id of the student. This could be a student ID, a username, an #: email address, etc., so long as it is unique. id = Column(String(128), primary_key=True, nullable=False) #: (Optional) The first name of the student first_name = Column(String(128)) #: (Optional) The last name of the student last_name = Column(String(128)) #: (Optional) The student's email address, if the :attr:`~nbgrader.api.Student.id` #: does not correspond to an email address email = Column(String(128)) #: A collection of assignments submitted by the student, represented as #: :class:`~nbgrader.api.SubmittedAssignment` objects submissions = relationship("SubmittedAssignment", backref="student") #: The overall score of the student across all assignments, computed #: automatically from the :attr:`~nbgrader.api.SubmittedAssignment.score` #: of each submitted assignment. score = None #: The maximum possible score the student could achieve across all assignments, #: computed automatically from the :attr:`~nbgrader.api.Assignment.max_score` #: of each assignment. max_score = None
[docs] def to_dict(self): """Convert the student object to a JSON-friendly dictionary representation. """ return { "id": self.id, "first_name": self.first_name, "last_name": self.last_name, "email": self.email, "score": self.score, "max_score": self.max_score }
def __repr__(self): return "Student<{}>".format(self.id)
[docs]class SubmittedAssignment(Base): """Database representation of an assignment submitted by a student.""" __tablename__ = "submitted_assignment" __table_args__ = (UniqueConstraint('assignment_id', 'student_id'),) #: Unique id of the submitted assignment (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Name of the assignment, inherited from :class:`~nbgrader.api.Assignment` name = association_proxy("assignment", "name") #: The master version of this assignment, represented by a #: :class:`~nbgrader.api.Assignment` object assignment = None #: Unique id of :attr:`~nbgrader.api.SubmittedAssignment.assignment` assignment_id = Column(String(32), ForeignKey('assignment.id')) #: The student who submitted this assignment, represented by a #: :class:`~nbgrader.api.Student` object student = None #: Unique id of :attr:`~nbgrader.api.SubmittedAssignment.student` student_id = Column(String(128), ForeignKey('student.id')) #: (Optional) The date and time that the assignment was submitted, in date #: time format with a UTC timezone timestamp = Column(DateTime()) #: (Optional) An extension given to the student for this assignment, in #: time delta format extension = Column(Interval()) #: A collection of notebooks contained within this submitted assignment, #: represented by :class:`~nbgrader.api.SubmittedNotebook` objects notebooks = relationship("SubmittedNotebook", backref="assignment") #: The score assigned to this assignment, automatically calculated from the #: :attr:`~nbgrader.api.SubmittedNotebook.score` of each notebook within #: this submitted assignment. score = None #: The maximum possible score of this assignment, inherited from #: :class:`~nbgrader.api.Assignment` max_score = None #: The code score assigned to this assignment, automatically calculated from #: the :attr:`~nbgrader.api.SubmittedNotebook.code_score` of each notebook #: within this submitted assignment. code_score = None #: The maximum possible code score of this assignment, inherited from #: :class:`~nbgrader.api.Assignment` max_code_score = None #: The written score assigned to this assignment, automatically calculated #: from the :attr:`~nbgrader.api.SubmittedNotebook.written_score` of each #: notebook within this submitted assignment. written_score = None #: The maximum possible written score of this assignment, inherited from #: :class:`~nbgrader.api.Assignment` max_written_score = None #: Whether this assignment has parts that need to be manually graded, #: automatically determined from the :attr:`~nbgrader.api.SubmittedNotebook.needs_manual_grade` #: attribute of each notebook. needs_manual_grade = None @property def duedate(self): """The duedate of this student's assignment, which includes any extension given, if applicable, and which is just the regular assignment duedate otherwise. """ orig_duedate = self.assignment.duedate if self.extension is not None: return orig_duedate + self.extension else: return orig_duedate @property def total_seconds_late(self): """The number of seconds that this assignment was turned in past the duedate (including extensions, if any). If the assignment was turned in before the deadline, this value will just be zero. """ if self.timestamp is None or self.duedate is None: return 0 else: return max(0, (self.timestamp - self.duedate).total_seconds())
[docs] def to_dict(self): """Convert the submitted assignment object to a JSON-friendly dictionary representation. Note that this includes a ``student`` key which is the unique id of the student, not the object itself. """ return { "id": self.id, "name": self.name, "student": self.student.id, "timestamp": self.timestamp.isoformat() if self.timestamp is not None else None, "extension": self.extension.total_seconds() if self.extension is not None else None, "duedate": self.duedate.isoformat() if self.duedate is not None else None, "total_seconds_late": self.total_seconds_late, "score": self.score, "max_score": self.max_score, "code_score": self.code_score, "max_code_score": self.max_code_score, "written_score": self.written_score, "max_written_score": self.max_written_score, "needs_manual_grade": self.needs_manual_grade }
def __repr__(self): return "SubmittedAssignment<{} for {}>".format(self.name, self.student.id)
[docs]class SubmittedNotebook(Base): """Database representation of a notebook submitted by a student.""" __tablename__ = "submitted_notebook" __table_args__ = (UniqueConstraint('notebook_id', 'assignment_id'),) #: Unique id of the submitted notebook (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Name of the notebook, inherited from :class:`~nbgrader.api.Notebook` name = association_proxy("notebook", "name") #: The submitted assignment this notebook is a part of, represented by a #: :class:`~nbgrader.api.SubmittedAssignment` object assignment = None #: Unique id of :attr:`~nbgrader.api.SubmittedNotebook.assignment` assignment_id = Column(String(32), ForeignKey('submitted_assignment.id')) #: The master version of this notebook, represesnted by a #: :class:`~nbgrader.api.Notebook` object notebook = None #: Unique id of :attr:`~nbgrader.api.SubmittedNotebook.notebook` notebook_id = Column(String(32), ForeignKey('notebook.id')) #: Collection of grades associated with this submitted notebook, represented #: by :class:`~nbgrader.api.Grade` objects grades = relationship("Grade", backref="notebook") #: Collection of comments associated with this submitted notebook, represented #: by :class:`~nbgrader.api.Comment` objects comments = relationship("Comment", backref="notebook") #: The student who submitted this notebook, represented by a #: :class:`~nbgrader.api.Student` object student = association_proxy('assignment', 'student') #: Whether this assignment has been flagged by a human grader flagged = Column(Boolean, default=False) #: The score assigned to this notebook, automatically calculated from the #: :attr:`~nbgrader.api.Grade.score` of each grade cell within #: this submitted notebook. score = None #: The maximum possible score of this notebook, inherited from #: :class:`~nbgrader.api.Notebook` max_score = None #: The code score assigned to this notebook, automatically calculated from #: the :attr:`~nbgrader.api.Grade.score` and :attr:`~nbgrader.api.GradeCell.cell_type` #: of each grade within this submitted notebook. code_score = None #: The maximum possible code score of this notebook, inherited from #: :class:`~nbgrader.api.Notebook` max_code_score = None #: The written score assigned to this notebook, automatically calculated from #: the :attr:`~nbgrader.api.Grade.score` and :attr:`~nbgrader.api.GradeCell.cell_type` #: of each grade within this submitted notebook. written_score = None #: The maximum possible written score of this notebook, inherited from #: :class:`~nbgrader.api.Notebook` max_written_score = None #: Whether this notebook has parts that need to be manually graded, #: automatically determined from the :attr:`~nbgrader.api.Grade.needs_manual_grade` #: attribute of each grade. needs_manual_grade = None #: Whether this notebook contains autograder tests that failed to pass, #: automatically determined from the :attr:`~nbgrader.api.Grade.failed_tests` #: attribute of each grade. failed_tests = None def to_dict(self): """Convert the submitted notebook object to a JSON-friendly dictionary representation. Note that this includes a key for ``student`` which is the unique id of the student, not the actual student object. """ return { "id": self.id, "name": self.name, "student": self.student.id, "score": self.score, "max_score": self.max_score, "code_score": self.code_score, "max_code_score": self.max_code_score, "written_score": self.written_score, "max_written_score": self.max_written_score, "needs_manual_grade": self.needs_manual_grade, "failed_tests": self.failed_tests, "flagged": self.flagged } def __repr__(self): return "SubmittedNotebook<{}/{} for {}>".format( self.assignment.name, self.name, self.student.id)
[docs]class Grade(Base): """Database representation of a grade assigned to the submitted version of a grade cell. """ __tablename__ = "grade" __table_args__ = (UniqueConstraint('cell_id', 'notebook_id'),) #: Unique id of the grade (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique name of the grade cell, inherited from :class:`~nbgrader.api.GradeCell` name = association_proxy('cell', 'name') #: The submitted assignment that this grade is contained in, represented by #: a :class:`~nbgrader.api.SubmittedAssignment` object assignment = association_proxy('notebook', 'assignment') #: The submitted notebook that this grade is assigned to, represented by a #: :class:`~nbgrader.api.SubmittedNotebook` object notebook = None #: Unique id of :attr:`~nbgrader.api.Grade.notebook` notebook_id = Column(String(32), ForeignKey('submitted_notebook.id')) #: The master version of the cell this grade is assigned to, represented by #: a :class:`~nbgrader.api.GradeCell` object. cell = None #: Unique id of :attr:`~nbgrader.api.Grade.cell` cell_id = Column(String(32), ForeignKey('grade_cell.id')) #: The type of cell this grade corresponds to, inherited from #: :class:`~nbgrader.api.GradeCell` cell_type = None #: The student who this grade is assigned to, represented by a #: :class:`~nbgrader.api.Student` object student = association_proxy('notebook', 'student') #: Score assigned by the autograder auto_score = Column(Float()) #: Score assigned by a human grader manual_score = Column(Float()) #: Whether a score needs to be assigned manually. This is True by default. needs_manual_grade = Column(Boolean, default=True, nullable=False) #: The overall score, computed automatically from the #: :attr:`~nbgrader.api.Grade.auto_score` and :attr:`~nbgrader.api.Grade.manual_score` #: values. If neither are set, the score is zero. If both are set, then the #: manual score takes precedence. If only one is set, then that value is used #: for the score. score = column_property(case( [ (manual_score != None, manual_score), (auto_score != None, auto_score) ], else_=literal_column("0.0") )) #: The maximum possible score that can be assigned, inherited from #: :class:`~nbgrader.api.GradeCell` max_score = None #: Whether the autograded score is a result of failed autograder tests. This #: is True if the autograder score is zero and the cell type is "code", and #: otherwise False. failed_tests = None
[docs] def to_dict(self): """Convert the grade object to a JSON-friendly dictionary representation. Note that this includes keys for ``notebook`` and ``assignment`` which correspond to the name of the notebook and assignment, not the actual objects. It also includes a key for ``student`` which corresponds to the unique id of the student, not the actual student object. """ return { "id": self.id, "name": self.name, "notebook": self.notebook.name, "assignment": self.assignment.name, "student": self.student.id, "auto_score": self.auto_score, "manual_score": self.manual_score, "max_score": self.max_score, "needs_manual_grade": self.needs_manual_grade, "failed_tests": self.failed_tests, "cell_type": self.cell_type }
def __repr__(self): return "Grade<{}/{}/{} for {}>".format( self.assignment.name, self.notebook.name, self.name, self.student.id)
[docs]class Comment(Base): """Database representation of a comment on a cell in a submitted notebook.""" __tablename__ = "comment" __table_args__ = (UniqueConstraint('cell_id', 'notebook_id'),) #: Unique id of the comment (automatically generated) id = Column(String(32), primary_key=True, default=new_uuid) #: Unique name of the solution cell, inherited from :class:`~nbgrader.api.SolutionCell` name = association_proxy('cell', 'name') #: The submitted assignment that this comment is contained in, represented by #: a :class:`~nbgrader.api.SubmittedAssignment` object assignment = association_proxy('notebook', 'assignment') #: The submitted notebook that this comment is assigned to, represented by a #: :class:`~nbgrader.api.SubmittedNotebook` object notebook = None #: Unique id of :attr:`~nbgrader.api.Comment.notebook` notebook_id = Column(String(32), ForeignKey('submitted_notebook.id')) #: The master version of the cell this comment is assigned to, represented by #: a :class:`~nbgrader.api.SolutionCell` object. cell = None #: Unique id of :attr:`~nbgrader.api.Comment.cell` cell_id = Column(String(32), ForeignKey('solution_cell.id')) #: The student who this comment is assigned to, represented by a #: :class:`~nbgrader.api.Student` object student = association_proxy('notebook', 'student') #: A comment which is automatically assigned by the autograder auto_comment = Column(Text()) #: A commment which is assigned manually manual_comment = Column(Text()) #: The overall comment, computed automatically from the #: :attr:`~nbgrader.api.Comment.auto_comment` and #: :attr:`~nbgrader.api.Comment.manual_comment` values. If neither are set, #: the comment is None. If both are set, then the manual comment #: takes precedence. If only one is set, then that value is used for the #: comment. comment = column_property(case( [ (manual_comment != None, manual_comment), (auto_comment != None, auto_comment) ], else_=None ))
[docs] def to_dict(self): """Convert the comment object to a JSON-friendly dictionary representation. Note that this includes keys for ``notebook`` and ``assignment`` which correspond to the name of the notebook and assignment, not the actual objects. It also includes a key for ``student`` which corresponds to the unique id of the student, not the actual student object. """ return { "id": self.id, "name": self.name, "notebook": self.notebook.name, "assignment": self.assignment.name, "student": self.student.id, "auto_comment": self.auto_comment, "manual_comment": self.manual_comment }
def __repr__(self): return "Comment<{}/{}/{} for {}>".format( self.assignment.name, self.notebook.name, self.name, self.student.id)
## Needs manual grade SubmittedNotebook.needs_manual_grade = column_property( exists().where(and_( Grade.notebook_id == SubmittedNotebook.id, Grade.needs_manual_grade))\ .correlate_except(Grade), deferred=True) SubmittedAssignment.needs_manual_grade = column_property( exists().where(and_( SubmittedNotebook.assignment_id == SubmittedAssignment.id, Grade.notebook_id == SubmittedNotebook.id, Grade.needs_manual_grade))\ .correlate_except(Grade), deferred=True) Notebook.needs_manual_grade = column_property( exists().where(and_( Notebook.id == SubmittedNotebook.notebook_id, Grade.notebook_id == SubmittedNotebook.id, Grade.needs_manual_grade))\ .correlate_except(Grade), deferred=True) ## Overall scores SubmittedNotebook.score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(Grade.notebook_id == SubmittedNotebook.id)\ .correlate_except(Grade), deferred=True) SubmittedAssignment.score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( SubmittedNotebook.assignment_id == SubmittedAssignment.id, Grade.notebook_id == SubmittedNotebook.id))\ .correlate_except(Grade), deferred=True) Student.score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( SubmittedAssignment.student_id == Student.id, SubmittedNotebook.assignment_id == SubmittedAssignment.id, Grade.notebook_id == SubmittedNotebook.id))\ .correlate_except(Grade), deferred=True) ## Overall max scores Grade.max_score = column_property( select([GradeCell.max_score])\ .where(Grade.cell_id == GradeCell.id)\ .correlate_except(GradeCell), deferred=True) Notebook.max_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(GradeCell.notebook_id == Notebook.id)\ .correlate_except(GradeCell), deferred=True) SubmittedNotebook.max_score = column_property( select([Notebook.max_score])\ .where(SubmittedNotebook.notebook_id == Notebook.id)\ .correlate_except(Notebook), deferred=True) Assignment.max_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(and_( Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id))\ .correlate_except(GradeCell), deferred=True) SubmittedAssignment.max_score = column_property( select([Assignment.max_score])\ .where(SubmittedAssignment.assignment_id == Assignment.id)\ .correlate_except(Assignment), deferred=True) Student.max_score = column_property( select([func.coalesce(func.sum(Assignment.max_score), 0.0)])\ .correlate_except(Assignment), deferred=True) ## Written scores SubmittedNotebook.written_score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( Grade.notebook_id == SubmittedNotebook.id, GradeCell.id == Grade.cell_id, GradeCell.cell_type == "markdown"))\ .correlate_except(Grade), deferred=True) SubmittedAssignment.written_score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( SubmittedNotebook.assignment_id == SubmittedAssignment.id, Grade.notebook_id == SubmittedNotebook.id, GradeCell.id == Grade.cell_id, GradeCell.cell_type == "markdown"))\ .correlate_except(Grade), deferred=True) ## Written max scores Notebook.max_written_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(and_( GradeCell.notebook_id == Notebook.id, GradeCell.cell_type == "markdown"))\ .correlate_except(GradeCell), deferred=True) SubmittedNotebook.max_written_score = column_property( select([Notebook.max_written_score])\ .where(Notebook.id == SubmittedNotebook.notebook_id)\ .correlate_except(Notebook), deferred=True) Assignment.max_written_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(and_( Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, GradeCell.cell_type == "markdown"))\ .correlate_except(GradeCell), deferred=True) SubmittedAssignment.max_written_score = column_property( select([Assignment.max_written_score])\ .where(Assignment.id == SubmittedAssignment.assignment_id)\ .correlate_except(Assignment), deferred=True) ## Code scores SubmittedNotebook.code_score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( Grade.notebook_id == SubmittedNotebook.id, GradeCell.id == Grade.cell_id, GradeCell.cell_type == "code"))\ .correlate_except(Grade), deferred=True) SubmittedAssignment.code_score = column_property( select([func.coalesce(func.sum(Grade.score), 0.0)])\ .where(and_( SubmittedNotebook.assignment_id == SubmittedAssignment.id, Grade.notebook_id == SubmittedNotebook.id, GradeCell.id == Grade.cell_id, GradeCell.cell_type == "code"))\ .correlate_except(Grade), deferred=True) ## Code max scores Notebook.max_code_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(and_( GradeCell.notebook_id == Notebook.id, GradeCell.cell_type == "code"))\ .correlate_except(GradeCell), deferred=True) SubmittedNotebook.max_code_score = column_property( select([Notebook.max_code_score])\ .where(Notebook.id == SubmittedNotebook.notebook_id)\ .correlate_except(Notebook), deferred=True) Assignment.max_code_score = column_property( select([func.coalesce(func.sum(GradeCell.max_score), 0.0)])\ .where(and_( Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, GradeCell.cell_type == "code"))\ .correlate_except(GradeCell), deferred=True) SubmittedAssignment.max_code_score = column_property( select([Assignment.max_code_score])\ .where(Assignment.id == SubmittedAssignment.assignment_id)\ .correlate_except(Assignment), deferred=True) ## Number of submissions Assignment.num_submissions = column_property( select([func.count(SubmittedAssignment.id)])\ .where(SubmittedAssignment.assignment_id == Assignment.id)\ .correlate_except(SubmittedAssignment), deferred=True) Notebook.num_submissions = column_property( select([func.count(SubmittedNotebook.id)])\ .where(SubmittedNotebook.notebook_id == Notebook.id)\ .correlate_except(SubmittedNotebook), deferred=True) ## Cell type Grade.cell_type = column_property( select([GradeCell.cell_type])\ .where(Grade.cell_id == GradeCell.id)\ .correlate_except(GradeCell), deferred=True) ## Failed tests Grade.failed_tests = column_property( (Grade.auto_score < Grade.max_score) & (Grade.cell_type == "code")) SubmittedNotebook.failed_tests = column_property( exists().where(and_( Grade.notebook_id == SubmittedNotebook.id, Grade.failed_tests))\ .correlate_except(Grade), deferred=True)
[docs]class Gradebook(object): """The gradebook object to interface with the database holding nbgrader grades. """
[docs] def __init__(self, db_url): """Initialize the connection to the database. Parameters ---------- db_url : string The URL to the database, e.g. ``sqlite:///grades.db`` """ # create the connection to the database engine = create_engine(db_url) self.db = scoped_session(sessionmaker(autoflush=True, bind=engine)) # this creates all the tables in the database if they don't already exist Base.metadata.create_all(bind=engine)
#### Students @property def students(self): """A list of all students in the database.""" return self.db.query(Student)\ .order_by(Student.last_name, Student.first_name)\ .all()
[docs] def add_student(self, student_id, **kwargs): """Add a new student to the database. Parameters ---------- student_id : string The unique id of the student `**kwargs` : dict other keyword arguments to the :class:`~nbgrader.api.Student` object Returns ------- student : :class:`~nbgrader.api.Student` """ student = Student(id=student_id, **kwargs) self.db.add(student) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return student
[docs] def find_student(self, student_id): """Find a student. Parameters ---------- student_id : string The unique id of the student Returns ------- student : :class:`~nbgrader.api.Student` """ try: student = self.db.query(Student)\ .filter(Student.id == student_id)\ .one() except NoResultFound: raise MissingEntry("No such student: {}".format(student_id)) return student
[docs] def update_or_create_student(self, name, **kwargs): """Update an existing student, or create it if it doesn't exist. Parameters ---------- name : string the name of the student `**kwargs` additional keyword arguments for the :class:`~nbgrader.api.Student` object Returns ------- student : :class:`~nbgrader.api.Student` """ try: student = self.find_student(name) except MissingEntry: student = self.add_student(name, **kwargs) else: for attr in kwargs: setattr(student, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return student
[docs] def remove_student(self, name): """Deletes an existing student from the gradebook, including any submissions the might be associated with that student. Parameters ---------- name : string the name of the student to delete """ student = self.find_student(name) for submission in student.submissions: self.remove_submission(submission.assignment.name, name) self.db.delete(student) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args)
#### Assignments @property def assignments(self): """A list of all assignments in the gradebook.""" return self.db.query(Assignment)\ .order_by(Assignment.duedate, Assignment.name)\ .all()
[docs] def add_assignment(self, name, **kwargs): """Add a new assignment to the gradebook. Parameters ---------- name : string the unique name of the new assignment `**kwargs` additional keyword arguments for the :class:`~nbgrader.api.Assignment` object Returns ------- assignment : :class:`~nbgrader.api.Assignment` """ if 'duedate' in kwargs: kwargs['duedate'] = utils.parse_utc(kwargs['duedate']) assignment = Assignment(name=name, **kwargs) self.db.add(assignment) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return assignment
[docs] def find_assignment(self, name): """Find an assignment in the gradebook. Parameters ---------- name : string the unique name of the assignment Returns ------- assignment : :class:`~nbgrader.api.Assignment` """ try: assignment = self.db.query(Assignment)\ .filter(Assignment.name == name)\ .one() except NoResultFound: raise MissingEntry("No such assignment: {}".format(name)) return assignment
[docs] def update_or_create_assignment(self, name, **kwargs): """Update an existing assignment, or create it if it doesn't exist. Parameters ---------- name : string the name of the assignment `**kwargs` additional keyword arguments for the :class:`~nbgrader.api.Assignment` object Returns ------- assignment : :class:`~nbgrader.api.Assignment` """ try: assignment = self.find_assignment(name) except MissingEntry: assignment = self.add_assignment(name, **kwargs) else: for attr in kwargs: if attr == 'duedate': setattr(assignment, attr, utils.parse_utc(kwargs[attr])) else: setattr(assignment, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return assignment
[docs] def remove_assignment(self, name): """Deletes an existing assignment from the gradebook, including any submissions the might be associated with that assignment. Parameters ---------- name : string the name of the assignment to delete """ assignment = self.find_assignment(name) for submission in assignment.submissions: self.remove_submission(name, submission.student.id) for notebook in assignment.notebooks: self.remove_notebook(notebook.name, name) self.db.delete(assignment) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args)
#### Notebooks
[docs] def add_notebook(self, name, assignment, **kwargs): """Add a new notebook to an assignment. Parameters ---------- name : string the name of the new notebook assignment : string the name of an existing assignment `**kwargs` additional keyword arguments for the :class:`~nbgrader.api.Notebook` object Returns ------- notebook : :class:`~nbgrader.api.Notebook` """ notebook = Notebook( name=name, assignment=self.find_assignment(assignment), **kwargs) self.db.add(notebook) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return notebook
[docs] def find_notebook(self, name, assignment): """Find a particular notebook in an assignment. Parameters ---------- name : string the name of the notebook assignment : string the name of the assignment Returns ------- notebook : :class:`~nbgrader.api.Notebook` """ try: notebook = self.db.query(Notebook)\ .join(Assignment, Assignment.id == Notebook.assignment_id)\ .filter(Notebook.name == name, Assignment.name == assignment)\ .one() except NoResultFound: raise MissingEntry("No such notebook: {}/{}".format(assignment, name)) return notebook
[docs] def update_or_create_notebook(self, name, assignment, **kwargs): """Update an existing notebook, or create it if it doesn't exist. Parameters ---------- name : string the name of the notebook assignment : string the name of the assignment `**kwargs` additional keyword arguments for the :class:`~nbgrader.api.Notebook` object Returns ------- notebook : :class:`~nbgrader.api.Notebook` """ try: notebook = self.find_notebook(name, assignment) except MissingEntry: notebook = self.add_notebook(name, assignment, **kwargs) else: for attr in kwargs: setattr(notebook, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return notebook
[docs] def remove_notebook(self, name, assignment): """Deletes an existing notebook from the gradebook, including any submissions the might be associated with that notebook. Parameters ---------- name : string the name of the notebook to delete assignment : string the name of an existing assignment """ notebook = self.find_notebook(name, assignment) for submission in notebook.submissions: self.remove_submission_notebook(name, assignment, submission.student.id) for grade_cell in notebook.grade_cells: self.db.delete(grade_cell) for solution_cell in notebook.solution_cells: self.db.delete(solution_cell) for source_cell in notebook.source_cells: self.db.delete(source_cell) self.db.delete(notebook) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args)
#### Grade cells
[docs] def add_grade_cell(self, name, notebook, assignment, **kwargs): """Add a new grade cell to an existing notebook of an existing assignment. Parameters ---------- name : string the name of the new grade cell notebook : string the name of an existing notebook assignment : string the name of an existing assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.GradeCell` Returns ------- grade_cell : :class:`~nbgrader.api.GradeCell` """ notebook = self.find_notebook(notebook, assignment) grade_cell = GradeCell(name=name, notebook=notebook, **kwargs) self.db.add(grade_cell) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return grade_cell
[docs] def find_grade_cell(self, name, notebook, assignment): """Find a grade cell in a particular notebook of an assignment. Parameters ---------- name : string the name of the grade cell notebook : string the name of the notebook assignment : string the name of the assignment Returns ------- grade_cell : :class:`~nbgrader.api.GradeCell` """ try: grade_cell = self.db.query(GradeCell)\ .join(Notebook, Notebook.id == GradeCell.notebook_id)\ .join(Assignment, Assignment.id == Notebook.assignment_id)\ .filter( GradeCell.name == name, Notebook.name == notebook, Assignment.name == assignment)\ .one() except NoResultFound: raise MissingEntry("No such grade cell: {}/{}/{}".format(assignment, notebook, name)) return grade_cell
[docs] def update_or_create_grade_cell(self, name, notebook, assignment, **kwargs): """Update an existing grade cell in a notebook of an assignment, or create the grade cell if it does not exist. Parameters ---------- name : string the name of the grade cell notebook : string the name of the notebook assignment : string the name of the assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.GradeCell` Returns ------- grade_cell : :class:`~nbgrader.api.GradeCell` """ try: grade_cell = self.find_grade_cell(name, notebook, assignment) except MissingEntry: grade_cell = self.add_grade_cell(name, notebook, assignment, **kwargs) else: for attr in kwargs: setattr(grade_cell, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return grade_cell
#### Solution cells
[docs] def add_solution_cell(self, name, notebook, assignment, **kwargs): """Add a new solution cell to an existing notebook of an existing assignment. Parameters ---------- name : string the name of the new solution cell notebook : string the name of an existing notebook assignment : string the name of an existing assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SolutionCell` Returns ------- solution_cell : :class:`~nbgrader.api.SolutionCell` """ notebook = self.find_notebook(notebook, assignment) solution_cell = SolutionCell(name=name, notebook=notebook, **kwargs) self.db.add(solution_cell) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return solution_cell
[docs] def find_solution_cell(self, name, notebook, assignment): """Find a solution cell in a particular notebook of an assignment. Parameters ---------- name : string the name of the solution cell notebook : string the name of the notebook assignment : string the name of the assignment Returns ------- solution_cell : :class:`~nbgrader.api.SolutionCell` """ try: solution_cell = self.db.query(SolutionCell)\ .join(Notebook, Notebook.id == SolutionCell.notebook_id)\ .join(Assignment, Assignment.id == Notebook.assignment_id)\ .filter(SolutionCell.name == name, Notebook.name == notebook, Assignment.name == assignment)\ .one() except NoResultFound: raise MissingEntry("No such solution cell: {}/{}/{}".format(assignment, notebook, name)) return solution_cell
[docs] def update_or_create_solution_cell(self, name, notebook, assignment, **kwargs): """Update an existing solution cell in a notebook of an assignment, or create the solution cell if it does not exist. Parameters ---------- name : string the name of the solution cell notebook : string the name of the notebook assignment : string the name of the assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SolutionCell` Returns ------- solution_cell : :class:`~nbgrader.api.SolutionCell` """ try: solution_cell = self.find_solution_cell(name, notebook, assignment) except MissingEntry: solution_cell = self.add_solution_cell(name, notebook, assignment, **kwargs) else: for attr in kwargs: setattr(solution_cell, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: raise InvalidEntry(*e.args) return solution_cell
#### Source cells
[docs] def add_source_cell(self, name, notebook, assignment, **kwargs): """Add a new source cell to an existing notebook of an existing assignment. Parameters ---------- name : string the name of the new source cell notebook : string the name of an existing notebook assignment : string the name of an existing assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SourceCell` Returns ------- source_cell : :class:`~nbgrader.api.SourceCell` """ notebook = self.find_notebook(notebook, assignment) source_cell = SourceCell(name=name, notebook=notebook, **kwargs) self.db.add(source_cell) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return source_cell
[docs] def find_source_cell(self, name, notebook, assignment): """Find a source cell in a particular notebook of an assignment. Parameters ---------- name : string the name of the source cell notebook : string the name of the notebook assignment : string the name of the assignment Returns ------- source_cell : :class:`~nbgrader.api.SourceCell` """ try: source_cell = self.db.query(SourceCell)\ .join(Notebook, Notebook.id == SourceCell.notebook_id)\ .join(Assignment, Assignment.id == Notebook.assignment_id)\ .filter(SourceCell.name == name, Notebook.name == notebook, Assignment.name == assignment)\ .one() except NoResultFound: raise MissingEntry("No such source cell: {}/{}/{}".format(assignment, notebook, name)) return source_cell
[docs] def update_or_create_source_cell(self, name, notebook, assignment, **kwargs): """Update an existing source cell in a notebook of an assignment, or create the source cell if it does not exist. Parameters ---------- name : string the name of the source cell notebook : string the name of the notebook assignment : string the name of the assignment `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SourceCell` Returns ------- source_cell : :class:`~nbgrader.api.SourceCell` """ try: source_cell = self.find_source_cell(name, notebook, assignment) except MissingEntry: source_cell = self.add_source_cell(name, notebook, assignment, **kwargs) else: for attr in kwargs: setattr(source_cell, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: raise InvalidEntry(*e.args) return source_cell
#### Submissions
[docs] def add_submission(self, assignment, student, **kwargs): """Add a new submission of an assignment by a student. This method not only creates the high-level submission object, but also mirrors the entire structure of the existing assignment. Thus, once this method has been called, the new submission exists and is completely ready to be filled in with grades and comments. Parameters ---------- assignment : string the name of an existing assignment student : string the name of an existing student `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SubmittedAssignment` Returns ------- submission : :class:`~nbgrader.api.SubmittedAssignment` """ if 'timestamp' in kwargs: kwargs['timestamp'] = utils.parse_utc(kwargs['timestamp']) try: submission = SubmittedAssignment( assignment=self.find_assignment(assignment), student=self.find_student(student), **kwargs) for notebook in submission.assignment.notebooks: nb = SubmittedNotebook(notebook=notebook, assignment=submission) for grade_cell in notebook.grade_cells: Grade(cell=grade_cell, notebook=nb) for solution_cell in notebook.solution_cells: Comment(cell=solution_cell, notebook=nb) self.db.add(submission) self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return submission
[docs] def find_submission(self, assignment, student): """Find a student's submission for a given assignment. Parameters ---------- assignment : string the name of an assignment student : string the unique id of a student Returns ------- submission : :class:`~nbgrader.api.SubmittedAssignment` """ try: submission = self.db.query(SubmittedAssignment)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .join(Student, Student.id == SubmittedAssignment.student_id)\ .filter(Assignment.name == assignment, Student.id == student)\ .one() except NoResultFound: raise MissingEntry("No such submission: {} for {}".format( assignment, student)) return submission
[docs] def update_or_create_submission(self, assignment, student, **kwargs): """Update an existing submission of an assignment by a given student, or create a new submission if it doesn't exist. See :func:`~nbgrader.api.Gradebook.add_submission` for additional details. Parameters ---------- assignment : string the name of an existing assignment student : string the name of an existing student `**kwargs` additional keyword arguments for :class:`~nbgrader.api.SubmittedAssignment` Returns ------- submission : :class:`~nbgrader.api.SubmittedAssignment` """ try: submission = self.find_submission(assignment, student) except MissingEntry: submission = self.add_submission(assignment, student, **kwargs) else: for attr in kwargs: if attr == 'timestamp': setattr(submission, attr, utils.parse_utc(kwargs[attr])) else: setattr(submission, attr, kwargs[attr]) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args) return submission
[docs] def remove_submission(self, assignment, student): """Removes a submission from the database. Parameters ---------- assignment : string the name of an assignment student : string the name of a student """ submission = self.find_submission(assignment, student) for notebook in submission.notebooks: self.remove_submission_notebook(notebook.name, assignment, student) self.db.delete(submission) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args)
[docs] def remove_submission_notebook(self, notebook, assignment, student): """Removes a submitted notebook from the database. Parameters ---------- notebook : string the name of a notebook assignment : string the name of an assignment student : string the name of a student """ submission = self.find_submission_notebook(notebook, assignment, student) for grade in submission.grades: self.db.delete(grade) for comment in submission.comments: self.db.delete(comment) self.db.delete(submission) try: self.db.commit() except (IntegrityError, FlushError) as e: self.db.rollback() raise InvalidEntry(*e.args)
[docs] def assignment_submissions(self, assignment): """Find all submissions of a given assignment. Parameters ---------- assignment : string the name of an assignment Returns ------- submissions : list A list of :class:`~nbgrader.api.SubmittedAssignment` objects """ return self.db.query(SubmittedAssignment)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .filter(Assignment.name == assignment)\ .all()
[docs] def notebook_submissions(self, notebook, assignment): """Find all submissions of a given notebook in a given assignment. Parameters ---------- notebook : string the name of an assignment assignment : string the name of an assignment Returns ------- submissions : list A list of :class:`~nbgrader.api.SubmittedNotebook` objects """ return self.db.query(SubmittedNotebook)\ .join(Notebook, Notebook.id == SubmittedNotebook.notebook_id)\ .join(SubmittedAssignment, SubmittedAssignment.id == SubmittedNotebook.assignment_id)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .filter(Notebook.name == notebook, Assignment.name == assignment)\ .all()
[docs] def student_submissions(self, student): """Find all submissions by a given student. Parameters ---------- student : string the student's unique id Returns ------- submissions : list A list of :class:`~nbgrader.api.SubmittedAssignment` objects """ return self.db.query(SubmittedAssignment)\ .join(Student, Student.id == SubmittedAssignment.student_id)\ .filter(Student.id == student)\ .all()
[docs] def find_submission_notebook(self, notebook, assignment, student): """Find a particular notebook in a student's submission for a given assignment. Parameters ---------- notebook : string the name of a notebook assignment : string the name of an assignment student : string the unique id of a student Returns ------- notebook : :class:`~nbgrader.api.SubmittedNotebook` """ try: notebook = self.db.query(SubmittedNotebook)\ .join(Notebook, Notebook.id == SubmittedNotebook.notebook_id)\ .join(SubmittedAssignment, SubmittedAssignment.id == SubmittedNotebook.assignment_id)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .join(Student, Student.id == SubmittedAssignment.student_id)\ .filter( Notebook.name == notebook, Assignment.name == assignment, Student.id == student)\ .one() except NoResultFound: raise MissingEntry("No such submitted notebook: {}/{} for {}".format( assignment, notebook, student)) return notebook
[docs] def find_submission_notebook_by_id(self, notebook_id): """Find a submitted notebook by its unique id. Parameters ---------- notebook_id : string the unique id of the submitted notebook Returns ------- notebook : :class:`~nbgrader.api.SubmittedNotebook` """ try: notebook = self.db.query(SubmittedNotebook)\ .filter(SubmittedNotebook.id == notebook_id)\ .one() except NoResultFound: raise MissingEntry("No such submitted notebook: {}".format(notebook_id)) return notebook
[docs] def find_grade(self, grade_cell, notebook, assignment, student): """Find a particular grade in a notebook in a student's submission for a given assignment. Parameters ---------- grade_cell : string the name of a grade cell notebook : string the name of a notebook assignment : string the name of an assignment student : string the unique id of a student Returns ------- grade : :class:`~nbgrader.api.Grade` """ try: grade = self.db.query(Grade)\ .join(GradeCell, GradeCell.id == Grade.cell_id)\ .join(SubmittedNotebook, SubmittedNotebook.id == Grade.notebook_id)\ .join(Notebook, Notebook.id == SubmittedNotebook.notebook_id)\ .join(SubmittedAssignment, SubmittedAssignment.id == SubmittedNotebook.assignment_id)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .join(Student, Student.id == SubmittedAssignment.student_id)\ .filter( GradeCell.name == grade_cell, Notebook.name == notebook, Assignment.name == assignment, Student.id == student)\ .one() except NoResultFound: raise MissingEntry("No such grade: {}/{}/{} for {}".format( assignment, notebook, grade_cell, student)) return grade
[docs] def find_grade_by_id(self, grade_id): """Find a grade by its unique id. Parameters ---------- grade_id : string the unique id of the grade Returns ------- grade : :class:`~nbgrader.api.Grade` """ try: grade = self.db.query(Grade).filter(Grade.id == grade_id).one() except NoResultFound: raise MissingEntry("No such grade: {}".format(grade_id)) return grade
[docs] def find_comment(self, solution_cell, notebook, assignment, student): """Find a particular comment in a notebook in a student's submission for a given assignment. Parameters ---------- solution_cell : string the name of a solution cell notebook : string the name of a notebook assignment : string the name of an assignment student : string the unique id of a student Returns ------- comment : :class:`~nbgrader.api.Comment` """ try: comment = self.db.query(Comment)\ .join(SolutionCell, SolutionCell.id == Comment.cell_id)\ .join(SubmittedNotebook, SubmittedNotebook.id == Comment.notebook_id)\ .join(Notebook, Notebook.id == SubmittedNotebook.notebook_id)\ .join(SubmittedAssignment, SubmittedAssignment.id == SubmittedNotebook.assignment_id)\ .join(Assignment, Assignment.id == SubmittedAssignment.assignment_id)\ .join(Student, Student.id == SubmittedAssignment.student_id)\ .filter( SolutionCell.name == solution_cell, Notebook.name == notebook, Assignment.name == assignment, Student.id == student)\ .one() except NoResultFound: raise MissingEntry("No such comment: {}/{}/{} for {}".format( assignment, notebook, solution_cell, student)) return comment
[docs] def find_comment_by_id(self, comment_id): """Find a comment by its unique id. Parameters ---------- comment_id : string the unique id of the comment Returns ------- comment : :class:`~nbgrader.api.Comment` """ try: comment = self.db.query(Comment).filter(Comment.id == comment_id).one() except NoResultFound: raise MissingEntry("No such comment: {}".format(comment_id)) return comment
[docs] def average_assignment_score(self, assignment_id): """Compute the average score for an assignment. Parameters ---------- assignment_id : string the name of the assignment Returns ------- score : float The average score """ assignment = self.find_assignment(assignment_id) if assignment.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(GradeCell, Notebook, Assignment)\ .filter(Assignment.name == assignment_id).scalar() return score_sum / assignment.num_submissions
[docs] def average_assignment_code_score(self, assignment_id): """Compute the average code score for an assignment. Parameters ---------- assignment_id : string the name of the assignment Returns ------- score : float The average code score """ assignment = self.find_assignment(assignment_id) if assignment.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(GradeCell, Notebook, Assignment)\ .filter(and_( Assignment.name == assignment_id, Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, Grade.cell_id == GradeCell.id, GradeCell.cell_type == "code")).scalar() return score_sum / assignment.num_submissions
[docs] def average_assignment_written_score(self, assignment_id): """Compute the average written score for an assignment. Parameters ---------- assignment_id : string the name of the assignment Returns ------- score : float The average written score """ assignment = self.find_assignment(assignment_id) if assignment.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(GradeCell, Notebook, Assignment)\ .filter(and_( Assignment.name == assignment_id, Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, Grade.cell_id == GradeCell.id, GradeCell.cell_type == "markdown")).scalar() return score_sum / assignment.num_submissions
[docs] def average_notebook_score(self, notebook_id, assignment_id): """Compute the average score for a particular notebook in an assignment. Parameters ---------- notebook_id : string the name of the notebook assignment_id : string the name of the assignment Returns ------- score : float The average notebook score """ notebook = self.find_notebook(notebook_id, assignment_id) if notebook.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(SubmittedNotebook, Notebook, Assignment)\ .filter(and_( Notebook.name == notebook_id, Assignment.name == assignment_id)).scalar() return score_sum / notebook.num_submissions
[docs] def average_notebook_code_score(self, notebook_id, assignment_id): """Compute the average code score for a particular notebook in an assignment. Parameters ---------- notebook_id : string the name of the notebook assignment_id : string the name of the assignment Returns ------- score : float The average notebook code score """ notebook = self.find_notebook(notebook_id, assignment_id) if notebook.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(GradeCell, Notebook, Assignment)\ .filter(and_( Notebook.name == notebook_id, Assignment.name == assignment_id, Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, Grade.cell_id == GradeCell.id, GradeCell.cell_type == "code")).scalar() return score_sum / notebook.num_submissions
[docs] def average_notebook_written_score(self, notebook_id, assignment_id): """Compute the average written score for a particular notebook in an assignment. Parameters ---------- notebook_id : string the name of the notebook assignment_id : string the name of the assignment Returns ------- score : float The average notebook written score """ notebook = self.find_notebook(notebook_id, assignment_id) if notebook.num_submissions == 0: return 0.0 score_sum = self.db.query(func.coalesce(func.sum(Grade.score), 0.0))\ .join(GradeCell, Notebook, Assignment)\ .filter(and_( Notebook.name == notebook_id, Assignment.name == assignment_id, Notebook.assignment_id == Assignment.id, GradeCell.notebook_id == Notebook.id, Grade.cell_id == GradeCell.id, GradeCell.cell_type == "markdown")).scalar() return score_sum / notebook.num_submissions
[docs] def student_dicts(self): """Returns a list of dictionaries containing student data. Equivalent to calling :func:`~nbgrader.api.Student.to_dict` for each student, except that this method is implemented using proper SQL joins and is much faster. Returns ------- students : list A list of dictionaries, one per student """ # subquery the scores scores = self.db.query( Student.id, func.sum(Grade.score).label("score") ).join(SubmittedAssignment, SubmittedNotebook, Grade)\ .group_by(Student.id)\ .subquery() # full query _scores = func.coalesce(scores.c.score, 0.0) students = self.db.query( Student.id, Student.first_name, Student.last_name, Student.email, _scores, func.sum(GradeCell.max_score) ).outerjoin(scores, Student.id == scores.c.id)\ .group_by( Student.id, Student.first_name, Student.last_name, Student.email, _scores)\ .all() keys = ["id", "first_name", "last_name", "email", "score", "max_score"] return [dict(zip(keys, x)) for x in students]
[docs] def notebook_submission_dicts(self, notebook_id, assignment_id): """Returns a list of dictionaries containing submission data. Equivalent to calling :func:`~nbgrader.api.SubmittedNotebook.to_dict` for each submission, except that this method is implemented using proper SQL joins and is much faster. Parameters ---------- notebook_id : string the name of the notebook assignment_id : string the name of the assignment Returns ------- submissions : list A list of dictionaries, one per submitted notebook """ # subquery the code scores code_scores = self.db.query( SubmittedNotebook.id, func.sum(Grade.score).label("code_score"), func.sum(GradeCell.max_score).label("max_code_score"), ).join(SubmittedAssignment, Notebook, Assignment, Student, Grade, GradeCell)\ .filter(GradeCell.cell_type == "code")\ .group_by(SubmittedNotebook.id)\ .subquery() # subquery for the written scores written_scores = self.db.query( SubmittedNotebook.id, func.sum(Grade.score).label("written_score"), func.sum(GradeCell.max_score).label("max_written_score"), ).join(SubmittedAssignment, Notebook, Assignment, Student, Grade, GradeCell)\ .filter(GradeCell.cell_type == "markdown")\ .group_by(SubmittedNotebook.id)\ .subquery() # subquery for needing manual grading manual_grade = self.db.query( SubmittedNotebook.id, exists().where(Grade.needs_manual_grade).label("needs_manual_grade") ).join(SubmittedAssignment, Assignment, Notebook)\ .filter( Grade.notebook_id == SubmittedNotebook.id, Grade.needs_manual_grade)\ .group_by(SubmittedNotebook.id)\ .subquery() # subquery for failed tests failed_tests = self.db.query( SubmittedNotebook.id, exists().where(Grade.failed_tests).label("failed_tests") ).join(SubmittedAssignment, Assignment, Notebook)\ .filter( Grade.notebook_id == SubmittedNotebook.id, Grade.failed_tests)\ .group_by(SubmittedNotebook.id)\ .subquery() # full query _manual_grade = func.coalesce(manual_grade.c.needs_manual_grade, False) _failed_tests = func.coalesce(failed_tests.c.failed_tests, False) submissions = self.db.query( SubmittedNotebook.id, Notebook.name, Student.id, func.sum(Grade.score), func.sum(GradeCell.max_score), code_scores.c.code_score, code_scores.c.max_code_score, written_scores.c.written_score, written_scores.c.max_written_score, _manual_grade, _failed_tests, SubmittedNotebook.flagged ).join(SubmittedAssignment, Notebook, Assignment, Student, Grade, GradeCell)\ .outerjoin(code_scores, SubmittedNotebook.id == code_scores.c.id)\ .outerjoin(written_scores, SubmittedNotebook.id == written_scores.c.id)\ .outerjoin(manual_grade, SubmittedNotebook.id == manual_grade.c.id)\ .outerjoin(failed_tests, SubmittedNotebook.id == failed_tests.c.id)\ .filter(and_( Notebook.name == notebook_id, Assignment.name == assignment_id, Student.id == SubmittedAssignment.student_id, SubmittedAssignment.id == SubmittedNotebook.assignment_id, SubmittedNotebook.id == Grade.notebook_id, GradeCell.id == Grade.cell_id))\ .group_by( SubmittedNotebook.id, Notebook.name, Student.id, code_scores.c.code_score, code_scores.c.max_code_score, written_scores.c.written_score, written_scores.c.max_written_score, _manual_grade, _failed_tests, SubmittedNotebook.flagged)\ .all() keys = [ "id", "name", "student", "score", "max_score", "code_score", "max_code_score", "written_score", "max_written_score", "needs_manual_grade", "failed_tests", "flagged" ] return [dict(zip(keys, x)) for x in submissions]