forked from hedyorg/hedy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
courses.py
140 lines (107 loc) · 4.19 KB
/
courses.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
import json
import copy
import attr
from ruamel import yaml
import docs
class LevelDefaults:
def __init__(self, language):
self.language = language
self.levels = load_yaml(f'coursedata/level-defaults/{language}.yaml')
def get_defaults(self, level):
"""Return the level defaults for a given level number.
Returns: Default
"""
return copy.deepcopy(self.levels.get(int(level), {}))
class NoSuchDefaults:
def get_defaults(self, level):
return {}
class Course:
def __init__(self, course_name, language, defaults):
self.course_name = course_name
self.language = language
self.defaults = defaults
self.course = load_yaml(f'coursedata/course/{course_name}/{language}.yaml').get('course')
if not self.course:
raise RuntimeError(f'File should have top-level "course" field: coursedata/course/{course_name}/{language}.yaml')
self.validate_course()
self.docs = docs.DocCollection(keys=['level', 'slug'], synth={
'slug': lambda d: docs.slugify(d.front_matter.get('title', None))
})
self.docs.load_dir(f'coursedata/course/{course_name}/docs-{language}')
def max_level(self):
return len(self.course)
def max_step(self, level):
level_ix = int(level) - 1
if level_ix >= len(self.course): return 0
return len(self.course[level_ix].get('assignments', []))
def get_assignment(self, level, number):
"""Return the 1-based Assignment from this course."""
level_ix = int(level) - 1
if level_ix >= len(self.course): return None
assignments = self.course[level_ix].get('assignments')
assignment_values = {
"level": str(level),
}
assignment_values.update(**self.defaults.get_defaults(int(level)))
# If we don't have any "assignments", return a default Assignment object
# based off the level and the level defaults. This is used in the Hedy main
# course.
#
# Otherwise, if this is a course that DOES have assignments, validate and
# load the data into the accumulator object.
if assignments:
step_ix = int(number) - 1
if step_ix >= len(assignments): return None
assignment_values.update(**assignments[step_ix])
assignment_values['commands'] = [Command(**c) for c in assignment_values.get('commands', [])]
assignment_values['docs'] = [Doc(slug=slug, title=doc.front_matter.get('title'))
for slug, doc in self.docs.get(level).items()]
return Assignment(**assignment_values)
def validate_course(self):
"""Check that the 'level' and 'step' fields have the right number in the entire course.
This information is redundant, but it helps the human locate their way in the YAML file.
"""
for level_i, level in enumerate(self.course):
expected_value = str(level_i + 1)
actual_value = level.get('level')
if expected_value != actual_value:
raise RuntimeError(f'Expected \'level: "{expected_value}"\' but found "{actual_value}" in {self.course_name}-{self.language}')
assignments = level.get('assignments')
if assignments:
for ass_i, ass in enumerate(assignments):
expected_ass = str(ass_i + 1)
actual_ass = ass.get('step')
if expected_value != actual_value:
raise RuntimeError(f'Expected \'step: "{expected_ass}"\' but got "{actual_ass}" in {self.course_name}-{self.language}')
class NoSuchCourse:
def get_assignment(self, level, number):
return None
@attr.s(slots=True, frozen=True)
class Command:
name = attr.ib(default='')
explanation = attr.ib(default='')
example = attr.ib(default='')
demo_code = attr.ib(default='')
@attr.s(slots=True, frozen=True)
class Assignment:
"""A single assignment.
Either a concrete assignment written in the YAML, or a default-generated
Assignment.
"""
level = attr.ib()
prompt = attr.ib(default='')
intro_text = attr.ib(default=None)
start_code = attr.ib(default=None)
commands = attr.ib(default=None)
step = attr.ib(default=None)
docs = attr.ib(default=list)
@attr.s(slots=True, frozen=True)
class Doc:
slug = attr.ib()
title = attr.ib()
def load_yaml(filename):
try:
with open(filename, 'r', encoding='utf-8') as f:
return yaml.safe_load(f)
except IOError:
return {}