-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcanvas_mock_data.py
246 lines (194 loc) · 8.09 KB
/
canvas_mock_data.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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
from __future__ import print_function, unicode_literals
from faker import Faker
from canvasapi import Canvas
from config import API_URL, API_KEY
canvas = Canvas(API_URL, API_KEY)
fake = Faker()
def generate_users(account_id, num_users):
"""
Create realistic-looking mock users.
:param account_id: The Canvas Account to create users in.
:type account_id: int
:param num_users: The number of users to create.
:type num_users: int
"""
account = canvas.get_account(account_id)
for i in range(1, num_users + 1):
first_name = fake.first_name()
last_name = fake.last_name()
user = {
'name': '{} {}'.format(first_name, last_name),
'short_name': first_name,
'sortable_name': '{}, {}'.format(last_name, first_name)
}
pseudonym = {
'unique_id': '{}.{}{}@example.com'.format(first_name[0], last_name, i),
'password': 'password'
}
account.create_user(pseudonym, user=user)
def generate_courses(account_id, num_courses):
"""
Create realistic-looking mock courses.
:param account_id: The Canvas Account to create courses in.
:type account_id: int
:param num_courses: The number of courses to create.
:type num_courses: int
"""
account = canvas.get_account(account_id)
for i in range(1, num_courses + 1):
course_dict = {
'name': fake.catch_phrase().title(),
'course_code': fake.bothify(text='???-####').upper(),
'offer': True
}
account.create_course(course=course_dict)
def generate_enrollments(account_id, min_students=1, max_students=5):
"""
Enroll random users into each course.
:param account_id: The Canvas Account to create enrollments in.
:type account_id: int
:param min_students: The minimum number of student enrollments to add per course.
:type min_students: int
:param max_students: The maximum number of student enrollments to add per course.
:type max_students: int
"""
account = canvas.get_account(account_id)
user_list = [user for user in account.get_users()]
courses = account.get_courses()
already_enrolled = []
for course in courses:
num_students = fake.random_int(min=min_students, max=max_students)
for i in range(0, num_students):
enroll_user = fake.random_element(user_list)
if enroll_user.id in already_enrolled:
continue
enroll_type = 'StudentEnrollment'
enrollment = {
'notify': False,
'enrollment_state': 'active'
}
course.enroll_user(enroll_user, enroll_type, enrollment=enrollment)
already_enrolled.append(enroll_user.id)
# TODO: Add support for enrolling Teachers/ TAs
def generate_groups(course_id, num_groups=3):
"""
Sort users into groups.
:param course_id: The Canvas Course to create groups in.
:type course_id: int
:param num_groups: The number of groups to create
:type num_groups: int
"""
course = canvas.get_course(course_id)
group_category = course.create_group_category(
name='User Groups',
create_group_count=num_groups
)
group_category.assign_members()
def generate_quizzes(course_id, min_quizzes=1, max_quizzes=5):
"""
Create randomized quizzes in a course.
:param course_id: The Canvas Course to create quizzes in.
:type course_id: int
:param min_quizzes: The minimum number of quizzes to create.
:type min_quizzes: int
:param max_quizzes: The maximum number of quizzes to create.
:type max_quizzes: int
"""
course = canvas.get_course(course_id)
num_quizzes = fake.random_int(min_quizzes, max_quizzes)
QUIZ_TYPES = ['practice_quiz', 'assignment', 'graded_survey', 'survey']
for i in range(num_quizzes):
course.create_quiz(quiz={
'title': fake.bs().title(),
'description': fake.text(max_nb_chars=fake.random_int(100, 500)),
'quiz_type': fake.random_element(QUIZ_TYPES),
'time_limit': fake.random_int(1, 30) * 5 if fake.boolean(50) else None,
'published': fake.boolean(75),
'allowed_attempts': fake.random_int(1, 3) if fake.boolean(75) else -1
})
# TODO: add quiz questions
def generate_assignments(course_id, min_assignments=5, max_assignments=10):
"""
Create randomized assignments in a course.
:param course_id: The Canvas Course to create assignments in.
:type course_id: int
:param min_assignments: The minimum number of assignments to create.
:type min_assignments: int
:param max_assignments: The maximum number of assignments to create.
:type max_assignments: int
"""
def get_sub_types():
"""
Return a list of valid sub types.
:rtype: list
"""
ASSIGN_SUB_TYPES = {
'online': ['online_upload', 'online_text_entry', 'online_url'],
'offline': ['online_quiz', 'none', 'on_paper', 'discussion_topic', 'external_tool']
}
if fake.boolean(25): # 25% chance of being offline only
# if not online, allow only one type of submission from the offline group
return [fake.random_element(ASSIGN_SUB_TYPES['offline'])]
# if online, pick at least one online (and possibly some offline) sub types
online = fake.random_sample(ASSIGN_SUB_TYPES['online'], fake.random_int(1, 3))
offline = fake.random_sample(ASSIGN_SUB_TYPES['offline'], fake.random_int(0, 3))
return list(online | offline)
course = canvas.get_course(course_id)
num_assignments = fake.random_int(min_assignments, max_assignments)
for i in range(num_assignments):
submission_types = get_sub_types()
course.create_assignment(assignment={
'name': fake.bs().title(),
'submission_types': submission_types,
'description': fake.text(max_nb_chars=fake.random_int(100, 500)),
'published': fake.boolean(75)
})
def generate_discussions(course_id, min_discussions=3, max_discussions=7):
"""
Create discussion topics in a course.
:param course_id: The Canvas Course to create discussions in.
:type course_id: int
:param min_discussions: The minimum number of discussions to create.
:type min_discussions: int
:param max_discussions: The maximum number of discussions to create.
:type max_discussions: int
"""
course = canvas.get_course(course_id)
num_discussions = fake.random_int(min_discussions, max_discussions)
for i in range(num_discussions):
course.create_discussion_topic(
title=fake.bs().title(),
message=fake.text(max_nb_chars=fake.random_int(100, 500)),
published=True
)
def generate_discussion_entries(course_id, chance_to_skip=25, students_only=True):
"""
Create several entries in all Discussion Topics by various users in
the course.
:param course_id: The Canvas Course to add entries to
:type course_id: int
:param chance_to_skip: Percent chance (between 0 and 100, inclusive)
to not create a particular entry for a user. Makes response
patterns look more realistic.
:type chance_to_skip: int
:param students_only: Only create entries from students. Otherwise,
allow non-student replies. Defaults to "True".
:type students_only: bool
"""
course = canvas.get_course(course_id)
if students_only:
users = course.get_users(enrollment_type=['student'])
else:
users = course.get_users()
# Have each user make one reply
for user in users:
# Reply to each discussion
for discussion in course.get_discussion_topics():
# Sometimes don't post based on chance to skip
if fake.boolean(100 - chance_to_skip):
paragraphs = fake.paragraphs(nb=fake.random_int(2, 5))
paragraphs = ['<p>{}</p>'.format(p) for p in paragraphs]
discussion.post_entry(
message='\n'.join(paragraphs),
as_user_id=user.id
)