-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapp.py
280 lines (205 loc) · 8.62 KB
/
app.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
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
# Standard imports
import os
from requests import post
# Flask imports
from flask import render_template, Flask, request, redirect, session
# Local imports
from helpers.credentials import create_authorization_url, create_credentials
from helpers.database import get_credentials, remove_credentials, store_credentials
from connections.classroom_connection import classroom_connection
from connections.google_sheets_connection import google_sheets_connection
# Configure application
app = Flask(__name__)
# Auto reload the templates
app.config["TEMPLATES_AUTO_RELOAD"] = True
# Set the secret key
app.secret_key = "secret"
# OAuth 2 client id for google classroom api
CLASSROOM_CLIENT_SECRETS = {
"web": {
"client_id": os.environ["CLASSROOM_CLIENT_ID"],
"project_id": "classroom-data-308900",
"auth_uri": "https://accounts.google.com/o/oauth2/auth",
"token_uri": "https://oauth2.googleapis.com/token",
"auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
"client_secret": os.environ["CLASSROOM_CLIENT_SECRET"]
}
}
# The user need to grant us authorization to see
# the user courses,
# the topics from the courses,
# the students from the courses,
# and the students submissions.
CLASSROOM_SCOPES = [
"https://www.googleapis.com/auth/classroom.courses.readonly",
"https://www.googleapis.com/auth/classroom.topics.readonly",
"https://www.googleapis.com/auth/classroom.student-submissions.students.readonly",
"https://www.googleapis.com/auth/classroom.rosters.readonly"
]
# OAuth 2 client id for google google sheets api
GOOGLE_SHEETS_CLIENT_SECRETS = {
"web": {
"client_id": os.environ["GOOGLE_SHEETS_CLIENT_ID"],
"project_id": "classroom-data-308900",
"auth_uri": "https://accounts.google.com/o/oauth2/auth",
"token_uri": "https://oauth2.googleapis.com/token",
"auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
"client_secret": os.environ["GOOGLE_SHEETS_CLIENT_SECRET"]
}
}
# The user need to grant us authorization to create and modify spread sheets
GOOGLE_SHEETS_SCOPES = ["https://www.googleapis.com/auth/spreadsheets"]
@app.route("/")
def index():
""" Main route redirect to /select """
return redirect("/select")
@app.route("/select")
def select():
""" Load the select template so the user can select the course topic and activity to get the data from """
courses = None
# If the use have already granted authorization get the user's courses
if "user_id" in session:
with classroom_connection(get_credentials("classroom")) as classroom:
courses = classroom.get_courses()
# render select.html
return render_template("select.html", courses=courses)
@app.route("/request_api", methods=["POST"])
def request_api():
""" Use AJAX. Get the requested info by requesting the classroom API """
# Get from the request the course id and the topic id, they might not exist
course_id = request.form.get("course_id")
topic_id = request.form.get("topic_id")
if course_id:
with classroom_connection(get_credentials("classroom")) as classroom:
# If the topic id was given in the request return the activities from this topic
if topic_id:
return classroom.get_activities_from_topic(course_id, topic_id)
# Else return the course's topics
return classroom.get_course_topics(course_id)
else:
# Return 406 if wasn't course id given in the request
return "bad request", 406
@app.route("/submission_data", methods=["POST"])
def submission_data():
"""
Load the submission data template so the user can see the submission state from the students
and export the data to csv or to google sheets
"""
# Get the course id from the request
course_id = request.form.get("course_id")
# Get the activity id and title
activity_id, activity_title = request.form.get("activity").split("---")
# Get the students submission state
with classroom_connection(get_credentials("classroom")) as classroom:
students = classroom.get_students(course_id)
students_submission_states = classroom.submission_data(course_id, activity_id, students)
# Render submission_data.html gives it the students submission state and title from the activity
return render_template(
"submission_data.html",
students_submission_states=students_submission_states,
activity_title=activity_title
)
@app.route("/export_google_sheets", methods=["POST"])
def export_google_sheets():
""" Use AJAX. Export the students submission state data to google sheets """
# Get google sheets credentials from the database
credentials = get_credentials("google_sheets")
# If the google sheets credentials exist in the database
if credentials:
with google_sheets_connection(credentials) as google_sheets:
# Create a new spread sheet
spread_sheet_id, spread_sheet_url = google_sheets.create_spread_sheet(
request.form.get("activity_title")
)
# Append the data to the recently created spread sheet
google_sheets.append(request.form, spread_sheet_id=spread_sheet_id)
# Return the url for the recently created spread sheet
return spread_sheet_url
# Else ask the user for authorization
else:
authorization_url, state = create_authorization_url(
"google_sheets",
GOOGLE_SHEETS_CLIENT_SECRETS,
GOOGLE_SHEETS_SCOPES
)
session["google_sheets_state"] = state
return authorization_url
@app.route("/google_sheets_oauth2callback")
def google_sheets_oauth2callback():
"""
The route that the user will be send after grant permission to
create and modify his/her spread sheets
"""
# Create the credentials for the google sheets
credentials = create_credentials(
"google_sheets",
GOOGLE_SHEETS_CLIENT_SECRETS,
GOOGLE_SHEETS_SCOPES,
session["google_sheets_state"],
request.url
)
# Store the user google sheets credentials in the database
store_credentials("google_sheets", credentials)
# Go back to the /select route
return redirect("/select")
@app.route("/login")
def login():
""" Ask the user for authorization """
# Create the authorization url
authorization_url, state = create_authorization_url(
"classroom",
CLASSROOM_CLIENT_SECRETS,
CLASSROOM_SCOPES
)
session["classroom_state"] = state
return redirect(authorization_url)
@app.route("/classroom_oauth2callback")
def classroom_oauth2callback():
""" The route that the user will be send after grant permission to see his/her classroom info """
# Create the credentials for the google classroom
credentials = create_credentials(
"classroom",
CLASSROOM_CLIENT_SECRETS,
CLASSROOM_SCOPES,
session["classroom_state"],
request.url
)
# Store the user id
with classroom_connection(credentials) as classroom:
session["user_id"] = classroom.get_user_id()
# Store the credentials in the database
store_credentials("classroom", credentials)
# Go back to the /select route
return redirect("/select")
@app.route("/logout")
def logout():
""" Logout """
# Revoke the credentials
if "user_id" in session:
classroom_credentials = get_credentials("classroom")
if classroom_credentials:
# Revoke the classroom token
post(
"https://oauth2.googleapis.com/revoke",
params={"token": classroom_credentials.token},
headers={"content-type": "application/x-www-form-urlencoded"}
)
# Remove the classroom credentials to from the database
remove_credentials("classroom")
google_sheets_credentials = get_credentials("google_sheets")
if google_sheets_credentials:
# Revoke the google sheets token
post(
"https://oauth2.googleapis.com/revoke",
params={"token": google_sheets_credentials.token},
headers={"content-type": "application/x-www-form-urlencoded"}
)
# Remove the google sheets credentials to from the database
remove_credentials("google_sheets")
# Remove the user id from the session cookies
del session["user_id"]
# Go back to the main route
return redirect("/")
if __name__ == "__main__":
os.environ["OAUTHLIB_INSECURE_TRANSPORT"] = "1"
app.run("localhost", 5000)