scmmishra | 1778b59 | 2018-10-30 18:25:49 +0530 | [diff] [blame] | 1 | from __future__ import unicode_literals |
| 2 | import frappe |
scmmishra | 1778b59 | 2018-10-30 18:25:49 +0530 | [diff] [blame] | 3 | |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 4 | # Academy Utils |
scmmishra | bcafe84 | 2018-11-03 14:48:42 +0530 | [diff] [blame] | 5 | @frappe.whitelist(allow_guest=True) |
scmmishra | 1778b59 | 2018-10-30 18:25:49 +0530 | [diff] [blame] | 6 | def get_portal_details(): |
| 7 | settings = frappe.get_doc("Education Settings") |
| 8 | title = settings.portal_title |
| 9 | description = settings.description |
| 10 | return dict(title=title, description=description) |
| 11 | |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 12 | def check_program_enrollment(program_name): |
| 13 | if frappe.session.user in ("Guest", "Administrator"): |
| 14 | return False |
| 15 | student = get_student_id(frappe.session.user) |
| 16 | enrollment = frappe.get_list("Program Enrollment", filters={'student':student, 'program': program_name}) |
| 17 | if enrollment: |
| 18 | return True |
| 19 | else: |
| 20 | return False |
| 21 | |
scmmishra | bcafe84 | 2018-11-03 14:48:42 +0530 | [diff] [blame] | 22 | @frappe.whitelist(allow_guest=True) |
scmmishra | 1778b59 | 2018-10-30 18:25:49 +0530 | [diff] [blame] | 23 | def get_featured_programs(): |
scmmishra | bcafe84 | 2018-11-03 14:48:42 +0530 | [diff] [blame] | 24 | featured_program_names = frappe.get_all("Program", filters={"is_published": True, "is_featured": True}) |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 25 | if featured_program_names: |
| 26 | featured_list = [get_program(program['name']) for program in featured_program_names] |
scmmishra | 1778b59 | 2018-10-30 18:25:49 +0530 | [diff] [blame] | 27 | return featured_list |
| 28 | else: |
| 29 | return None |
| 30 | |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 31 | def get_program(program_name): |
| 32 | program = frappe.get_doc('Program', program_name) |
| 33 | is_enrolled = check_program_enrollment(program_name) |
| 34 | return {'program': program, 'is_enrolled': is_enrolled} |
| 35 | |
scmmishra | f9b2fb6 | 2018-11-12 12:50:54 +0530 | [diff] [blame] | 36 | @frappe.whitelist(allow_guest=True) |
| 37 | def get_program_details(program_name): |
| 38 | try: |
| 39 | program = frappe.get_doc('Program', program_name) |
| 40 | return program |
| 41 | except: |
| 42 | return None |
| 43 | |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 44 | |
| 45 | def get_enrollment(course_name): |
| 46 | student = get_student_id(frappe.session.user) |
| 47 | enrollment_name = frappe.get_all("Course Enrollment", filters={'student': student, 'course':course_name}) |
| 48 | try: |
| 49 | name = enrollment_name[0].name |
| 50 | enrollment = frappe.get_doc("Course Enrollment", name) |
| 51 | return enrollment |
| 52 | except: |
| 53 | return None |
| 54 | |
| 55 | @frappe.whitelist() |
| 56 | def get_student_id(email=None): |
| 57 | """Returns student user name, example EDU-STU-2018-00001 (Based on the naming series). |
| 58 | |
| 59 | :param user: a user email address |
| 60 | """ |
| 61 | try: |
| 62 | student_id = frappe.db.get_all("Student", {"student_email_id": email}, ["name"])[0].name |
| 63 | return student_id |
| 64 | except IndexError: |
| 65 | return None |
| 66 | |
| 67 | def create_student(student_name=frappe.session.user): |
| 68 | student = frappe.get_doc({ |
| 69 | "doctype": "Student", |
| 70 | "first_name": student_name, |
| 71 | "student_email_id": student_name, |
| 72 | }) |
| 73 | student.save(ignore_permissions=True) |
| 74 | frappe.db.commit() |
| 75 | return student_name |
| 76 | |
scmmishra | 9c642ac | 2018-11-01 17:17:49 +0530 | [diff] [blame] | 77 | # Functions to get program & course details |
scmmishra | bcafe84 | 2018-11-03 14:48:42 +0530 | [diff] [blame] | 78 | @frappe.whitelist(allow_guest=True) |
scmmishra | 9c642ac | 2018-11-01 17:17:49 +0530 | [diff] [blame] | 79 | def get_courses(program_name): |
| 80 | program = frappe.get_doc('Program', program_name) |
| 81 | courses = program.get_course_list() |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 82 | course_data = [{'meta':get_continue_content(item.name), 'course':item} for item in courses] |
| 83 | return course_data |
| 84 | |
| 85 | @frappe.whitelist() |
| 86 | def get_continue_content(course_name): |
scmmishra | d7c993f | 2018-11-13 12:22:49 +0530 | [diff] [blame^] | 87 | if frappe.session.user == "Guest": |
| 88 | return dict(content=None, content_type=None, flag=None) |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 89 | enrollment = get_enrollment(course_name) |
| 90 | course = frappe.get_doc("Course", enrollment.course) |
| 91 | last_activity = enrollment.get_last_activity() |
| 92 | |
| 93 | if last_activity == None: |
| 94 | next_content = course.get_first_content() |
| 95 | return dict(content=next_content.name, content_type=next_content.doctype, flag="Start") |
| 96 | |
| 97 | if last_activity.doctype == "Quiz Activity": |
| 98 | next_content = get_next_content(last_activity.quiz, "Quiz", course.name) |
| 99 | else: |
| 100 | next_content = get_next_content(last_activity.content, last_activity.content_type, course.name) |
| 101 | |
| 102 | if next_content == None: |
| 103 | next_content = course.get_first_content() |
| 104 | return dict(content=next_content.name, content_type=next_content.doctype, flag="Complete") |
| 105 | else: |
| 106 | next_content['flag'] = "Continue" |
| 107 | return next_content |
| 108 | |
scmmishra | 9c642ac | 2018-11-01 17:17:49 +0530 | [diff] [blame] | 109 | |
| 110 | @frappe.whitelist() |
| 111 | def get_starting_content(course_name): |
| 112 | course = frappe.get_doc('Course', course_name) |
| 113 | content = course.course_content[0].content |
| 114 | content_type = course.course_content[0].content_type |
| 115 | return dict(content=content, content_type=content_type) |
| 116 | |
scmmishra | 9c642ac | 2018-11-01 17:17:49 +0530 | [diff] [blame] | 117 | # Functions to get content details |
| 118 | @frappe.whitelist() |
| 119 | def get_content(content_name, content_type): |
| 120 | try: |
| 121 | content = frappe.get_doc(content_type, content_name) |
| 122 | return content |
| 123 | except: |
| 124 | frappe.throw("{0} with name {1} does not exist".format(content_type, content_name)) |
| 125 | return None |
| 126 | |
| 127 | @frappe.whitelist() |
| 128 | def get_next_content(content, content_type, course): |
scmmishra | d7c993f | 2018-11-13 12:22:49 +0530 | [diff] [blame^] | 129 | if frappe.session.user == "Guest": |
| 130 | return None |
scmmishra | 9c642ac | 2018-11-01 17:17:49 +0530 | [diff] [blame] | 131 | course_doc = frappe.get_doc("Course", course) |
| 132 | content_list = [{'content_type':item.content_type, 'content':item.content} for item in course_doc.get_all_children()] |
| 133 | current_index = content_list.index({'content': content, 'content_type': content_type}) |
| 134 | try: |
| 135 | return content_list[current_index + 1] |
| 136 | except IndexError: |
scmmishra | 3741844 | 2018-11-01 19:50:47 +0530 | [diff] [blame] | 137 | return None |
| 138 | |
| 139 | def get_quiz_with_answers(quiz_name): |
| 140 | try: |
| 141 | quiz = frappe.get_doc("Quiz", quiz_name).get_questions() |
| 142 | quiz_output = [{'name':question.name, 'question':question.question, 'options':[{'name': option.name, 'option':option.option, 'is_correct':option.is_correct} for option in question.options]} for question in quiz] |
| 143 | return quiz_output |
| 144 | except: |
| 145 | frappe.throw("Quiz {0} does not exist".format(quiz_name)) |
| 146 | return None |
| 147 | |
| 148 | @frappe.whitelist() |
| 149 | def get_quiz_without_answers(quiz_name): |
| 150 | try: |
| 151 | quiz = frappe.get_doc("Quiz", quiz_name).get_questions() |
| 152 | quiz_output = [{'name':question.name, 'question':question.question, 'options':[{'name': option.name, 'option':option.option} for option in question.options]} for question in quiz] |
| 153 | return quiz_output |
| 154 | except: |
| 155 | frappe.throw("Quiz {0} does not exist".format(quiz_name)) |
scmmishra | f7029f0 | 2018-11-02 13:15:49 +0530 | [diff] [blame] | 156 | return None |
| 157 | |
| 158 | @frappe.whitelist() |
scmmishra | 0187b18 | 2018-11-05 12:16:02 +0530 | [diff] [blame] | 159 | def evaluate_quiz(enrollment, quiz_response, quiz_name): |
scmmishra | ef9a75e | 2018-11-02 16:35:46 +0530 | [diff] [blame] | 160 | """LMS Function: Evaluates a simple multiple choice quiz. |
scmmishra | f7029f0 | 2018-11-02 13:15:49 +0530 | [diff] [blame] | 161 | |
| 162 | |
| 163 | :param quiz_response: contains user selected choices for a quiz in the form of a string formatted as a dictionary. The function uses `json.loads()` to convert it to a python dictionary. |
scmmishra | f7029f0 | 2018-11-02 13:15:49 +0530 | [diff] [blame] | 164 | """ |
| 165 | import json |
scmmishra | f7029f0 | 2018-11-02 13:15:49 +0530 | [diff] [blame] | 166 | quiz_response = json.loads(quiz_response) |
scmmishra | ef9a75e | 2018-11-02 16:35:46 +0530 | [diff] [blame] | 167 | quiz = frappe.get_doc("Quiz", quiz_name) |
scmmishra | 0187b18 | 2018-11-05 12:16:02 +0530 | [diff] [blame] | 168 | answers, score, status = quiz.evaluate(quiz_response, quiz_name) |
scmmishra | 0187b18 | 2018-11-05 12:16:02 +0530 | [diff] [blame] | 169 | |
| 170 | result = {k: ('Correct' if v else 'Wrong') for k,v in answers.items()} |
| 171 | result_data = [] |
| 172 | for key in answers: |
| 173 | item = {} |
| 174 | item['question'] = key |
| 175 | item['quiz_result'] = result[key] |
| 176 | try: |
| 177 | item['selected_option'] = frappe.get_value('Options', quiz_response[key], 'option') |
| 178 | except: |
| 179 | item['selected_option'] = "Unattempted" |
| 180 | result_data.append(item) |
| 181 | # result_data = [{'question': key, 'selected_option': frappe.get_value('Options', quiz_response[key], 'option'), 'quiz_result': result[key]} for key in answers] |
| 182 | |
| 183 | add_quiz_activity(enrollment, quiz_name, result_data, score, status) |
scmmishra | ef9a75e | 2018-11-02 16:35:46 +0530 | [diff] [blame] | 184 | return(score) |
scmmishra | bcafe84 | 2018-11-03 14:48:42 +0530 | [diff] [blame] | 185 | |
| 186 | @frappe.whitelist() |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 187 | def get_completed_courses(): |
| 188 | student = get_student_id(frappe.session.user) |
| 189 | if student == None: |
scmmishra | b409f7a | 2018-11-07 22:12:55 +0530 | [diff] [blame] | 190 | return None |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 191 | try: |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 192 | student = frappe.get_doc("Student", student) |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 193 | return student.get_completed_courses() |
| 194 | except: |
| 195 | return None |
scmmishra | 8b8db54 | 2018-11-03 18:03:24 +0530 | [diff] [blame] | 196 | |
| 197 | @frappe.whitelist() |
| 198 | def get_continue_data(program_name): |
| 199 | program = frappe.get_doc("Program", program_name) |
| 200 | courses = program.get_all_children() |
scmmishra | 8fa9ade | 2018-11-08 13:01:55 +0530 | [diff] [blame] | 201 | try: |
| 202 | continue_data = get_starting_content(courses[0].course) |
| 203 | continue_data['course'] = courses[0].course |
| 204 | return continue_data |
| 205 | except: |
| 206 | return None |
scmmishra | 8b8db54 | 2018-11-03 18:03:24 +0530 | [diff] [blame] | 207 | |
scmmishra | 8b8db54 | 2018-11-03 18:03:24 +0530 | [diff] [blame] | 208 | @frappe.whitelist() |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 209 | def enroll_all_courses_in_program(program_enrollment, student): |
scmmishra | f9b2fb6 | 2018-11-12 12:50:54 +0530 | [diff] [blame] | 210 | program = frappe.get_doc("Program", program_enrollment.program) |
| 211 | course_list = [course.course for course in program.get_all_children()] |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 212 | for course_name in course_list: |
| 213 | student.enroll_in_course(course_name=course_name, program_enrollment=program_enrollment.name) |
| 214 | |
| 215 | @frappe.whitelist() |
scmmishra | f9b2fb6 | 2018-11-12 12:50:54 +0530 | [diff] [blame] | 216 | def enroll_in_program(program_name): |
| 217 | if(not get_student_id(frappe.session.user)): |
| 218 | create_student(frappe.session.user) |
| 219 | student = frappe.get_doc("Student", get_student_id(frappe.session.user)) |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 220 | program_enrollment = student.enroll_in_program(program_name) |
| 221 | enroll_all_courses_in_program(program_enrollment, student) |
scmmishra | 8b8db54 | 2018-11-03 18:03:24 +0530 | [diff] [blame] | 222 | |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 223 | @frappe.whitelist() |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 224 | def get_program_enrollments(email=frappe.session.user): |
scmmishra | b409f7a | 2018-11-07 22:12:55 +0530 | [diff] [blame] | 225 | if get_student_id(email) == None: |
| 226 | return None |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 227 | try: |
| 228 | student = frappe.get_doc("Student", get_student_id(email)) |
| 229 | return student.get_program_enrollments() |
| 230 | except: |
| 231 | return None |
| 232 | |
| 233 | @frappe.whitelist() |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 234 | def get_course_enrollments(): |
| 235 | student = get_student_id(frappe.session.user) |
| 236 | if student == None: |
scmmishra | b409f7a | 2018-11-07 22:12:55 +0530 | [diff] [blame] | 237 | return None |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 238 | try: |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 239 | student = frappe.get_doc("Student", student) |
scmmishra | 73a31ec | 2018-11-03 19:31:53 +0530 | [diff] [blame] | 240 | return student.get_course_enrollments() |
| 241 | except: |
scmmishra | 1a04f77 | 2018-11-03 20:43:59 +0530 | [diff] [blame] | 242 | return None |
| 243 | |
scmmishra | db9c8b7 | 2018-11-09 18:25:44 +0530 | [diff] [blame] | 244 | |
| 245 | # Academty Activity |
scmmishra | 1a04f77 | 2018-11-03 20:43:59 +0530 | [diff] [blame] | 246 | @frappe.whitelist() |
| 247 | def add_activity(enrollment, content_type, content): |
scmmishra | 76222f7 | 2018-11-05 11:02:14 +0530 | [diff] [blame] | 248 | if(check_activity_exists(enrollment, content_type, content)): |
| 249 | pass |
| 250 | else: |
| 251 | activity = frappe.get_doc({ |
| 252 | "doctype": "Course Activity", |
| 253 | "enrollment": enrollment, |
| 254 | "content_type": content_type, |
| 255 | "content": content, |
| 256 | "activity_date": frappe.utils.datetime.datetime.now() |
| 257 | }) |
| 258 | activity.save() |
| 259 | frappe.db.commit() |
| 260 | |
| 261 | def check_activity_exists(enrollment, content_type, content): |
| 262 | activity = frappe.get_all("Course Activity", filters={'enrollment': enrollment, 'content_type': content_type, 'content': content}) |
scmmishra | 0187b18 | 2018-11-05 12:16:02 +0530 | [diff] [blame] | 263 | return bool(activity) |
scmmishra | 76222f7 | 2018-11-05 11:02:14 +0530 | [diff] [blame] | 264 | |
scmmishra | 0187b18 | 2018-11-05 12:16:02 +0530 | [diff] [blame] | 265 | def add_quiz_activity(enrollment, quiz_name, result_data, score, status): |
| 266 | quiz_activity = frappe.get_doc({ |
| 267 | "doctype": "Quiz Activity", |
| 268 | "enrollment": enrollment, |
| 269 | "quiz": quiz_name, |
| 270 | "result": result_data, |
| 271 | "score": score, |
| 272 | "status": status |
| 273 | }) |
| 274 | quiz_activity.save() |
scmmishra | 25e42ba | 2018-11-05 13:11:37 +0530 | [diff] [blame] | 275 | frappe.db.commit() |
| 276 | |
| 277 | @frappe.whitelist() |
| 278 | def mark_course_complete(enrollment): |
| 279 | course_enrollment = frappe.get_doc("Course Enrollment", enrollment) |
| 280 | course_enrollment.completed = True |
| 281 | course_enrollment.save() |
| 282 | frappe.db.commit() |