Spaces:
Runtime error
Runtime error
Commit
Β·
7832cd9
1
Parent(s):
7fcb63d
wip update
Browse files
demo.py
CHANGED
@@ -4,6 +4,7 @@ import json
|
|
4 |
import gradio as gr
|
5 |
import requests
|
6 |
import firebase_admin
|
|
|
7 |
from firebase_admin import db, credentials
|
8 |
|
9 |
def clamp(x, minimum, maximum):
|
@@ -13,463 +14,391 @@ def clamp(x, minimum, maximum):
|
|
13 |
# API calls
|
14 |
#################################################################################################################################################
|
15 |
|
|
|
|
|
16 |
# read secret api key
|
17 |
API_KEY = os.environ['ApiKey']
|
18 |
FIREBASE_API_KEY = os.environ['FirebaseSecret']
|
19 |
FIREBASE_URL = os.environ['FirebaseURL']
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
20 |
|
21 |
-
|
22 |
-
firebase_app = firebase_admin.initialize_app(creds, {'databaseURL': FIREBASE_URL})
|
23 |
-
firebase_data_ref = db.reference("data")
|
24 |
-
firebase_current_ref = None
|
25 |
|
|
|
|
|
|
|
26 |
|
27 |
-
base_url = "https://skapi.polyglot-edu.com/"
|
28 |
levels = ["Primary School", "Middle School", "High School", "College", "Academy"]
|
29 |
languages = ["English", "Italian", "French", "German", "Spanish"]
|
30 |
-
|
31 |
-
|
32 |
-
def get_level_mapping(level):
|
33 |
-
if level is None:
|
34 |
-
raise gr.Error("Please select a level.")
|
35 |
-
|
36 |
-
return levels.index(level)
|
37 |
-
|
38 |
-
def get_category_mapping(category):
|
39 |
-
if category is None:
|
40 |
-
raise gr.Error("Please select a level.")
|
41 |
-
|
42 |
-
return categories.index(category)
|
43 |
-
|
44 |
-
def generate_fill_gaps(original_text, level, number_of_words, number_of_gaps, number_of_distractors, temperature, language):
|
45 |
-
"""
|
46 |
-
Generate a fill-gaps question from a given text.
|
47 |
-
|
48 |
-
|
49 |
-
Parameters
|
50 |
-
----------
|
51 |
-
original_text : str
|
52 |
-
The original text from which to generate the fill-gaps question.
|
53 |
-
number_of_words : int
|
54 |
-
The number of words of the generated text.
|
55 |
-
number_of_gaps : int
|
56 |
-
The number of gaps to generate.
|
57 |
-
number_of_distractors : int
|
58 |
-
The number of distractors to generate for each gap.
|
59 |
-
temperature : float
|
60 |
-
The temperature for the generation.
|
61 |
-
language : str
|
62 |
-
The language of the generated text.
|
63 |
-
|
64 |
-
Returns
|
65 |
-
-------
|
66 |
-
str
|
67 |
-
The fill-gaps question.
|
68 |
-
"""
|
69 |
-
|
70 |
-
match number_of_words:
|
71 |
-
case "β 150 Words":
|
72 |
-
number_of_words = 150
|
73 |
-
case "β 250 Words":
|
74 |
-
number_of_words = 250
|
75 |
-
case "β 350 Words":
|
76 |
-
number_of_words = 350
|
77 |
-
|
78 |
-
input_json = {
|
79 |
-
"text": original_text,
|
80 |
-
"level": get_level_mapping(level),
|
81 |
-
"n_o_w": int(number_of_words),
|
82 |
-
"n_o_g": int(number_of_gaps),
|
83 |
-
"n_o_d": int(number_of_distractors),
|
84 |
-
"temperature": int(temperature) * 0.2,
|
85 |
-
"language": language
|
86 |
-
}
|
87 |
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
response = requests.post(
|
93 |
-
base_url + "FillTheGaps/generateexercise",
|
94 |
-
headers={"ApiKey": API_KEY},
|
95 |
-
json=input_json,
|
96 |
-
timeout=20
|
97 |
-
)
|
98 |
-
|
99 |
-
if response.status_code != 200:
|
100 |
-
output = f"API call failed with status code {response.status_code} and message '{response.text}'"
|
101 |
-
raise Exception(f"API call failed with status code {response.status_code} and message '{response.text}'")
|
102 |
-
|
103 |
-
output = response.text
|
104 |
-
return output
|
105 |
-
finally:
|
106 |
-
global firebase_current_ref
|
107 |
-
firebase_current_ref = firebase_data_ref.push({
|
108 |
-
"type": "fill_the_gaps",
|
109 |
-
**input_json,
|
110 |
-
"output": output,
|
111 |
-
"datetime": str(datetime.datetime.now()),
|
112 |
-
"like": 0,
|
113 |
-
"comment_text": "",
|
114 |
-
"flagged": False,
|
115 |
-
})
|
116 |
-
|
117 |
-
def generate_open_question(original_text, level, temperature, language, question_type, question_category):
|
118 |
-
"""
|
119 |
-
Generate an open question from a given text.
|
120 |
-
|
121 |
-
|
122 |
-
Parameters
|
123 |
-
----------
|
124 |
-
original_text : str
|
125 |
-
The original text from which to generate the open question.
|
126 |
-
temperature : float
|
127 |
-
The temperature for the generation.
|
128 |
-
language : str
|
129 |
-
The language of the generated text.
|
130 |
-
question_type : str
|
131 |
-
The type of the question.
|
132 |
-
question_category : str
|
133 |
-
The category of the question.
|
134 |
-
|
135 |
-
Returns
|
136 |
-
-------
|
137 |
-
str
|
138 |
-
The open question.
|
139 |
-
"""
|
140 |
-
|
141 |
-
input_json = {
|
142 |
-
"text": original_text,
|
143 |
-
"level": get_level_mapping(level),
|
144 |
-
"temperature": int(temperature) * 0.2,
|
145 |
-
"language": language,
|
146 |
-
"type": question_type,
|
147 |
-
"category": get_category_mapping(question_category),
|
148 |
-
}
|
149 |
|
150 |
-
|
151 |
-
|
152 |
-
|
153 |
-
try:
|
154 |
-
response = requests.post(
|
155 |
-
base_url + "QuestionExercise/generateexercise",
|
156 |
-
headers={"ApiKey": API_KEY},
|
157 |
-
json=input_json,
|
158 |
-
timeout=20
|
159 |
-
)
|
160 |
-
|
161 |
-
if response.status_code != 200:
|
162 |
-
output = f"API call failed with status code {response.status_code} and message '{response.text}'"
|
163 |
-
raise Exception(f"API call failed with status code {response.status_code} and message '{response.text}'")
|
164 |
-
|
165 |
-
output = response.text
|
166 |
-
return output
|
167 |
-
finally:
|
168 |
-
global firebase_current_ref
|
169 |
-
firebase_current_ref = firebase_data_ref.push({
|
170 |
-
"type": "open_question",
|
171 |
-
**input_json,
|
172 |
-
"output": output,
|
173 |
-
"datetime": str(datetime.datetime.now()),
|
174 |
-
"like": 0,
|
175 |
-
"comment_text": "",
|
176 |
-
"flagged": False,
|
177 |
-
})
|
178 |
-
|
179 |
-
def generate_multiplechoice(original_text, level, number_of_easy_distractors, number_of_distractors, temperature, language, question_category, correct_answers, exercise_type):
|
180 |
-
"""
|
181 |
-
Generate a multiple-choice question from a given text.
|
182 |
-
|
183 |
-
|
184 |
-
Parameters
|
185 |
-
----------
|
186 |
-
original_text : str
|
187 |
-
The original text from which to generate the multiple-choice question.
|
188 |
-
number_of_easy_distractors : int
|
189 |
-
The number of easy distractors to generate for each option.
|
190 |
-
number_of_distractors : int
|
191 |
-
The number of distractors to generate for each option.
|
192 |
-
temperature : float
|
193 |
-
The temperature for the generation.
|
194 |
-
language : str
|
195 |
-
The language of the generated text.
|
196 |
-
question_category : str
|
197 |
-
The category of the question.
|
198 |
-
correct_answers : int
|
199 |
-
The number of correct answers.
|
200 |
-
exercise_type : bool
|
201 |
-
The type of the exercise (theory or practice).
|
202 |
-
|
203 |
-
Returns
|
204 |
-
-------
|
205 |
-
str
|
206 |
-
The multiple-choice question.
|
207 |
-
"""
|
208 |
-
|
209 |
-
input_json = {
|
210 |
-
"text": original_text,
|
211 |
-
"type": bool(exercise_type),
|
212 |
-
"level": get_level_mapping(level),
|
213 |
-
"category": get_category_mapping(question_category),
|
214 |
-
"n_o_ca": int(correct_answers),
|
215 |
-
"n_o_d": int(number_of_distractors),
|
216 |
-
"nedd": int(number_of_easy_distractors),
|
217 |
-
"temperature": int(temperature) * 0.2,
|
218 |
-
"language": language
|
219 |
}
|
220 |
|
221 |
-
|
222 |
-
|
223 |
-
|
224 |
-
|
225 |
-
|
226 |
-
|
227 |
-
|
228 |
-
json=input_json,
|
229 |
-
timeout=20
|
230 |
-
)
|
231 |
-
|
232 |
-
if response.status_code != 200:
|
233 |
-
output = f"API call failed with status code {response.status_code} and message '{response.text}'"
|
234 |
-
raise Exception(f"API call failed with status code {response.status_code} and message '{response.text}'")
|
235 |
-
|
236 |
-
output = response.text
|
237 |
-
return output
|
238 |
-
finally:
|
239 |
-
global firebase_current_ref
|
240 |
-
firebase_current_ref = firebase_data_ref.push({
|
241 |
-
"type": "open_question",
|
242 |
-
**input_json,
|
243 |
-
"output": output,
|
244 |
-
"datetime": str(datetime.datetime.now()),
|
245 |
-
"like": 0,
|
246 |
-
"comment_text": "",
|
247 |
-
"flagged": False,
|
248 |
-
})
|
249 |
-
|
250 |
-
def like():
|
251 |
-
global firebase_current_ref
|
252 |
-
if firebase_current_ref is not None:
|
253 |
-
firebase_current_ref.update({"like": 1})
|
254 |
-
gr.Info("Generated text liked.")
|
255 |
-
else:
|
256 |
-
gr.Warning("No generated text to vote.")
|
257 |
-
|
258 |
-
def neutral_like():
|
259 |
-
global firebase_current_ref
|
260 |
-
if firebase_current_ref is not None:
|
261 |
-
firebase_current_ref.update({"like": 0})
|
262 |
-
gr.Info("Generated text preference removed.")
|
263 |
-
else:
|
264 |
-
gr.Warning("No generated text to vote.")
|
265 |
-
|
266 |
-
def dislike():
|
267 |
-
global firebase_current_ref
|
268 |
-
if firebase_current_ref is not None:
|
269 |
-
firebase_current_ref.update({"like": -1})
|
270 |
-
gr.Info("Generated text disliked.")
|
271 |
-
else:
|
272 |
-
gr.Warning("No generated text to vote.")
|
273 |
-
|
274 |
-
def comment(comment_text):
|
275 |
-
global firebase_current_ref
|
276 |
-
if firebase_current_ref is not None:
|
277 |
-
firebase_current_ref.update({"comment": comment_text})
|
278 |
-
gr.Info("Comment added.")
|
279 |
else:
|
280 |
-
|
281 |
-
|
282 |
-
|
283 |
-
|
284 |
-
|
285 |
-
|
286 |
-
|
287 |
-
|
288 |
-
|
289 |
-
|
290 |
-
|
291 |
-
|
292 |
-
|
293 |
-
|
294 |
-
|
295 |
-
|
296 |
-
|
297 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
298 |
|
299 |
-
|
300 |
-
|
301 |
-
|
302 |
|
303 |
-
|
304 |
-
"""
|
305 |
-
Build the fill-gaps interface.
|
306 |
-
"""
|
307 |
-
with gr.Blocks(title="Fill-gaps") as demo:
|
308 |
-
with gr.Row():
|
309 |
-
with gr.Column(scale=5):
|
310 |
-
input_field = gr.TextArea(lines=10, max_lines=10, label="Input Educational resource text or URL",
|
311 |
-
info="Note: If the input resource is too long, the API will return an error. If the input is an URL, it must be a valid URL to webpage or a file.")
|
312 |
-
submit_btn = gr.Button(value="Submit")
|
313 |
-
with gr.Column(scale=4):
|
314 |
-
level = gr.Radio(levels, label="Level")
|
315 |
-
language = gr.Dropdown(languages, label="Language", value="English")
|
316 |
-
output_text_length = gr.Radio(["β 150 Words", "β 250 Words", "β 350 Words"], label="Output text length", value="β 150 Words")
|
317 |
-
with gr.Row():
|
318 |
-
blanks = gr.Number(value=5, minimum=4, maximum=8, step=1, label="Number of blanks")
|
319 |
-
distractors = gr.Number(value=2, minimum=0, maximum=5, step=1, label="Number of distractors")
|
320 |
-
temperature = gr.Checkbox(value=False, label="Increase creativity (decreases preciseness)", visible=False)
|
321 |
-
|
322 |
-
def update_numeric(output_text_length, blanks, distractors):
|
323 |
-
if output_text_length == "β 150 Words":
|
324 |
-
min_, max_ = 4, 8
|
325 |
-
elif output_text_length == "β 250 Words":
|
326 |
-
min_, max_ = 6, 10
|
327 |
-
elif output_text_length == "β 350 Words":
|
328 |
-
min_, max_ = 8, 12
|
329 |
-
|
330 |
-
return (
|
331 |
-
gr.Number(value=clamp(blanks, min_, max_), minimum=min_, maximum=max_, label="Number of blanks"),
|
332 |
-
gr.Number(value=clamp(distractors, 0, blanks), minimum=0, maximum=blanks, label="Number of distractors")
|
333 |
-
)
|
334 |
-
|
335 |
-
def update_blanks(blanks, distractors):
|
336 |
-
return gr.Number(value=clamp(distractors, 0, blanks), minimum=0, maximum=blanks, label="Number of distractors")
|
337 |
-
|
338 |
-
blanks.change(update_blanks, [blanks, distractors], [distractors])
|
339 |
-
output_text_length.change(update_numeric, [output_text_length, blanks, distractors], [blanks, distractors])
|
340 |
-
|
341 |
-
with gr.Row():
|
342 |
-
output = gr.TextArea(placeholder="Generated text", label="Output")
|
343 |
-
with gr.Row() as button_row:
|
344 |
-
upvote_btn = gr.Button(value="π Upvote")
|
345 |
-
remove_preference_btn = gr.Button(value="Remove Preference")
|
346 |
-
downvote_btn = gr.Button(value="π Downvote")
|
347 |
-
flag_btn = gr.Button(value="β οΈ Flag")
|
348 |
-
|
349 |
-
submit_btn.click(generate_fill_gaps, [input_field, level, output_text_length, blanks, distractors, temperature, language], [output])
|
350 |
-
upvote_btn.click(like)
|
351 |
-
remove_preference_btn.click(neutral_like)
|
352 |
-
downvote_btn.click(dislike)
|
353 |
-
flag_btn.click(flag, [flag_btn], [flag_btn])
|
354 |
-
|
355 |
-
return demo
|
356 |
-
|
357 |
-
def build_multiplechoice_interface():
|
358 |
-
"""
|
359 |
-
Build the open question interface.
|
360 |
-
"""
|
361 |
-
with gr.Blocks(title="Open Question") as demo:
|
362 |
-
with gr.Row():
|
363 |
-
with gr.Column(scale=5):
|
364 |
-
input_field = gr.TextArea(lines=10, max_lines=10, label="Input Educational resource text or URL",
|
365 |
-
info="Note: If the input resource is too long, the API will return an error. If the input is an URL, it must be a valid URL to webpage or a file.")
|
366 |
-
submit_btn = gr.Button(value="Submit")
|
367 |
-
with gr.Column(scale=4):
|
368 |
-
level = gr.Radio(levels, label="Level")
|
369 |
-
language = gr.Dropdown(languages, label="Language", value="English")
|
370 |
-
with gr.Row():
|
371 |
-
exercise_type = gr.Radio(["Theoretical", "Practical"], label="Exercise type", value="Theoretical", type="index")
|
372 |
-
question_category = gr.Dropdown(categories, label="Question Category", value="Factual Knowledge")
|
373 |
-
with gr.Row():
|
374 |
-
correct_answers = gr.Number(value=1, minimum=1, maximum=3, step=1, label="Number of correct answers")
|
375 |
-
easy_distractors = gr.Number(value=1, minimum=0, maximum=8, step=1, label="Number of easy distractors")
|
376 |
-
distractors = gr.Number(value=1, minimum=0, maximum=8, step=1, label="Number of distractors")
|
377 |
-
temperature = gr.Checkbox(value=False, label="Increase creativity (decreases preciseness)", visible=False)
|
378 |
-
|
379 |
-
def update_distractors(easy_distractors, distractors):
|
380 |
-
easy_distractors = clamp(easy_distractors, 0, distractors)
|
381 |
-
|
382 |
-
return gr.Number(value=easy_distractors, minimum=0, maximum=distractors, label="Number of easy distractors")
|
383 |
-
|
384 |
-
def update_exercise_type(exercise_type, question_category, number_of_correct_answers):
|
385 |
-
if exercise_type == 0:
|
386 |
-
return (
|
387 |
-
gr.Dropdown(categories, label="Question Category", value="Analysys And Evaluation"),
|
388 |
-
gr.Number(value=number_of_correct_answers, minimum=1, maximum=3, step=1, label="Number of correct answers")
|
389 |
-
)
|
390 |
-
elif exercise_type == 1:
|
391 |
-
return (
|
392 |
-
gr.Dropdown(["Analysys And Evaluation"], label="Question Category", value="Analysys And Evaluation"),
|
393 |
-
gr.Number(value=1, minimum=1, maximum=1, step=1, label="Number of correct answers")
|
394 |
-
)
|
395 |
-
|
396 |
-
distractors.change(update_distractors, [easy_distractors, distractors], [easy_distractors])
|
397 |
-
exercise_type.change(update_exercise_type, [exercise_type, question_category, correct_answers], [question_category, correct_answers])
|
398 |
-
|
399 |
-
with gr.Row():
|
400 |
-
output = gr.TextArea(placeholder="Generated text", label="Output")
|
401 |
-
with gr.Row() as button_row:
|
402 |
-
upvote_btn = gr.Button(value="π Upvote")
|
403 |
-
remove_preference_btn = gr.Button(value="Remove Preference")
|
404 |
-
downvote_btn = gr.Button(value="π Downvote")
|
405 |
-
flag_btn = gr.Button(value="β οΈ Flag")
|
406 |
-
|
407 |
-
submit_btn.click(generate_multiplechoice, [input_field, level, easy_distractors, distractors, temperature, language, question_category, correct_answers, exercise_type], [output])
|
408 |
-
upvote_btn.click(like)
|
409 |
-
remove_preference_btn.click(neutral_like)
|
410 |
-
downvote_btn.click(dislike)
|
411 |
-
flag_btn.click(flag, [flag_btn], [flag_btn])
|
412 |
-
|
413 |
-
return demo
|
414 |
-
|
415 |
-
def build_open_question_interface():
|
416 |
-
"""
|
417 |
-
Build the multiple-choice interface.
|
418 |
-
"""
|
419 |
-
with gr.Blocks(title="Multiple choice") as demo:
|
420 |
-
with gr.Row():
|
421 |
-
with gr.Column(scale=5):
|
422 |
-
input_field = gr.TextArea(lines=10, max_lines=10, label="Input Educational resource text or URL",
|
423 |
-
info="Note: If the input resource is too long, the API will return an error. If the input is an URL, it must be a valid URL to webpage or a file.")
|
424 |
-
submit_btn = gr.Button(value="Submit")
|
425 |
-
with gr.Column(scale=4):
|
426 |
-
level = gr.Radio(levels, label="Level")
|
427 |
-
language = gr.Dropdown(languages, label="Language", value="English")
|
428 |
-
with gr.Row():
|
429 |
-
question_type = gr.Dropdown(["Open", "ShortAnswer", "TrueFalse"], label="Question Type", value="Open", type="index")
|
430 |
-
question_category = gr.Dropdown(categories, label="Question Category", value="Factual Knowledge")
|
431 |
-
temperature = gr.Checkbox(value=False, label="Increase creativity (decreases preciseness)", visible=False)
|
432 |
-
with gr.Row():
|
433 |
-
output = gr.TextArea(placeholder="Generated text", label="Output")
|
434 |
-
with gr.Row() as button_row:
|
435 |
-
upvote_btn = gr.Button(value="π Upvote")
|
436 |
-
remove_preference_btn = gr.Button(value="Remove Preference")
|
437 |
-
downvote_btn = gr.Button(value="π Downvote")
|
438 |
-
flag_btn = gr.Button(value="β οΈ Flag")
|
439 |
-
|
440 |
-
submit_btn.click(generate_open_question, [input_field, level, temperature, language, question_type, question_category], [output])
|
441 |
-
upvote_btn.click(like)
|
442 |
-
remove_preference_btn.click(neutral_like)
|
443 |
-
downvote_btn.click(dislike)
|
444 |
-
flag_btn.click(flag, [flag_btn], [flag_btn])
|
445 |
-
|
446 |
-
return demo
|
447 |
-
|
448 |
-
def build_demo():
|
449 |
-
with gr.Blocks(title="Educational AI") as demo:
|
450 |
-
gr.Markdown("<h1 style='text-align: center; margin-bottom: 1rem'>Educational AI</h1>")
|
451 |
-
|
452 |
-
|
453 |
-
with gr.Row():
|
454 |
-
gr.Markdown("<h4>Click on the button on the right to fill up our questionnaire!</h4>")
|
455 |
-
gr.Button(value="π Questionnaire", link="https://forms.gle/T8CS5CiQgPbKUdeM9", scale=0.5, interactive=True)
|
456 |
-
|
457 |
-
with gr.Tab("Fill-gaps"):
|
458 |
-
build_fill_gaps_interface()
|
459 |
-
|
460 |
-
with gr.Tab("Open question"):
|
461 |
-
build_open_question_interface()
|
462 |
-
|
463 |
-
with gr.Tab("Multiple-choice"):
|
464 |
-
build_multiplechoice_interface()
|
465 |
-
|
466 |
-
with gr.Blocks():
|
467 |
-
with gr.Row():
|
468 |
-
comment_text = gr.TextArea(placeholder="Comment", label="Comment", scale=6)
|
469 |
-
comment_btn = gr.Button(value="π¬ Comment", scale=2)
|
470 |
-
comment_btn.click(comment, [comment_text])
|
471 |
-
|
472 |
-
return demo
|
473 |
-
|
474 |
-
if __name__ == "__main__":
|
475 |
-
build_demo().launch(share=False)
|
|
|
4 |
import gradio as gr
|
5 |
import requests
|
6 |
import firebase_admin
|
7 |
+
from itertools import chain
|
8 |
from firebase_admin import db, credentials
|
9 |
|
10 |
def clamp(x, minimum, maximum):
|
|
|
14 |
# API calls
|
15 |
#################################################################################################################################################
|
16 |
|
17 |
+
|
18 |
+
|
19 |
# read secret api key
|
20 |
API_KEY = os.environ['ApiKey']
|
21 |
FIREBASE_API_KEY = os.environ['FirebaseSecret']
|
22 |
FIREBASE_URL = os.environ['FirebaseURL']
|
23 |
+
SETUP_MODEL = os.environ['SETUP_MODEL']
|
24 |
+
|
25 |
+
|
26 |
+
# creds = credentials.Certificate(json.loads(FIREBASE_API_KEY))
|
27 |
+
# firebase_app = firebase_admin.initialize_app(creds, {'databaseURL': FIREBASE_URL})
|
28 |
+
# firebase_data_ref = db.reference("data")
|
29 |
+
# firebase_current_ref = None
|
30 |
|
31 |
+
BASE_URL = "https://skapi.polyglot-edu.com/"
|
|
|
|
|
|
|
32 |
|
33 |
+
##################################################################
|
34 |
+
# Data Layer
|
35 |
+
##################################################################
|
36 |
|
|
|
37 |
levels = ["Primary School", "Middle School", "High School", "College", "Academy"]
|
38 |
languages = ["English", "Italian", "French", "German", "Spanish"]
|
39 |
+
type_of_exercise = ["Open Question", "Short Answer Question", "True or False", "Fill in the Blanks", "Single Choice", "Multiple Choice", "Debate", "Essay", "Brainstorming", "Knoledge Exposition"]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
40 |
|
41 |
+
def generation_to_dict(experiment, skip=False):
|
42 |
+
info = {
|
43 |
+
# experiment info
|
44 |
+
**experiment,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
45 |
|
46 |
+
# chosen image set info
|
47 |
+
"corrupted_filename": experiment["corrupted"]["name"],
|
48 |
+
"options": [img["name"] for img in experiment["options"]],
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
49 |
}
|
50 |
|
51 |
+
if skip:
|
52 |
+
info = {
|
53 |
+
**info,
|
54 |
+
# selected image info
|
55 |
+
"selected_image": "None",
|
56 |
+
"selected_algo": "None",
|
57 |
+
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
58 |
else:
|
59 |
+
info = {
|
60 |
+
**info,
|
61 |
+
# selected image info
|
62 |
+
"selected_image": experiment["options"][experiment["selected_image"]]["name"],
|
63 |
+
"selected_algo": experiment["options"][experiment["selected_image"]]["algo"],
|
64 |
+
}
|
65 |
+
|
66 |
+
return info
|
67 |
+
|
68 |
+
def generate_new_experiment():
|
69 |
+
pass
|
70 |
+
# wanted_corruptions = ["spatter", "impulse_noise", "speckle_noise", "gaussian_noise", "pixelate", "jpeg_compression", "elastic_transform"]
|
71 |
+
# corruption = random.choice([f for f in list(Path(f"./images/{DATASET}").glob("*/*")) if f.is_dir() and f.name in wanted_corruptions])
|
72 |
+
# image_id = random.choice(list(corruption.glob("*")))
|
73 |
+
# imgs_to_sample = (NUMBER_OF_IMAGES_PER_ROW * NUMBER_OF_ROWS) // 2
|
74 |
+
|
75 |
+
# corrupted_image = {"name": str(random.choice(list(image_id.glob("*corrupted*"))))}
|
76 |
+
# sdedit_images = [
|
77 |
+
# {"name": str(img), "algo": "SDEdit"}
|
78 |
+
# for img in random.sample(list((image_id / "sde").glob(f"*")), imgs_to_sample)
|
79 |
+
# ]
|
80 |
+
# odedit_images = [
|
81 |
+
# {"name": str(img), "algo": "ODEdit"}
|
82 |
+
# for img in random.sample(list((image_id / "ode").glob(f"*")), imgs_to_sample)
|
83 |
+
# ]
|
84 |
+
# total_images = sdedit_images + odedit_images
|
85 |
+
# random.shuffle(total_images)
|
86 |
+
|
87 |
+
# return Experiment(
|
88 |
+
# DATASET,
|
89 |
+
# corruption.name,
|
90 |
+
# image_id.name,
|
91 |
+
# corrupted_image,
|
92 |
+
# total_images,
|
93 |
+
# )
|
94 |
+
|
95 |
+
def save(experiment, corrupted_component, *img_components, mode):
|
96 |
+
if mode == "save" and (experiment is None or experiment["selected_image"] is None):
|
97 |
+
gr.Warning("You must select an image before submitting")
|
98 |
+
return [experiment, corrupted_component, *img_components]
|
99 |
+
if mode == "skip":
|
100 |
+
experiment["selected_image"] = None
|
101 |
+
|
102 |
+
dict_to_save = {
|
103 |
+
**generation_to_dict(experiment, skip=(mode=="skip")),
|
104 |
+
"timestamp": datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),
|
105 |
+
}
|
106 |
+
# firebase_data_ref.push(dict_to_save)
|
107 |
+
|
108 |
+
print("=====================")
|
109 |
+
print(dict_to_save)
|
110 |
+
print("=====================")
|
111 |
+
|
112 |
+
gr.Info("Your choice has been saved to Firebase")
|
113 |
+
return next()
|
114 |
+
|
115 |
+
def analyze_resource(url):
|
116 |
+
response = requests.post(
|
117 |
+
BASE_URL + "Analyser/analyseMaterial",
|
118 |
+
headers={"ApiKey": API_KEY, "SetupModel": str(SETUP_MODEL)},
|
119 |
+
json={
|
120 |
+
"material": url
|
121 |
+
},
|
122 |
+
timeout=20
|
123 |
+
)
|
124 |
+
|
125 |
+
if response.status_code != 200:
|
126 |
+
raise gr.Error(f"Failed to analyze resource: {response.text}")
|
127 |
+
|
128 |
+
return response.json()
|
129 |
+
|
130 |
+
def generate_learning_objective(topic, context, level):
|
131 |
+
response = requests.post(
|
132 |
+
BASE_URL + "LOGenerator/generatelearningobjective",
|
133 |
+
headers={"ApiKey": API_KEY, "SetupModel": str(SETUP_MODEL)},
|
134 |
+
json={
|
135 |
+
"topic": topic,
|
136 |
+
"context": context,
|
137 |
+
"level": levels.index(level)
|
138 |
+
},
|
139 |
+
timeout=20
|
140 |
+
)
|
141 |
+
|
142 |
+
print(topic, context, level)
|
143 |
+
|
144 |
+
if response.status_code != 200:
|
145 |
+
raise gr.Error(f"Failed to generate learning objective: {response.text}")
|
146 |
+
|
147 |
+
return response.json()
|
148 |
+
|
149 |
+
def generate_exercise(state):
|
150 |
+
def find_key(d, item):
|
151 |
+
for key, value in d.items():
|
152 |
+
if item in value:
|
153 |
+
return key
|
154 |
+
return None
|
155 |
+
|
156 |
+
step3 = requests.post(
|
157 |
+
BASE_URL + "Exercises/GenerateExercise",
|
158 |
+
headers={"ApiKey": API_KEY, "SetupModel": str(SETUP_MODEL)},
|
159 |
+
json={
|
160 |
+
# filled in with the data from the previous steps
|
161 |
+
"macroSubject": state['MacroSubject'],
|
162 |
+
"title": state['Title'],
|
163 |
+
"level": state['level'],
|
164 |
+
"learningObjective": state['learningObjective'],
|
165 |
+
"bloomLevel": find_key(state["learningObjectiveList"], state['learningObjective']),
|
166 |
+
"language": state['Language'],
|
167 |
+
"material": state['material_url'],
|
168 |
+
"assignmentType": [topic['Type'] for topic in state['MainTopics'] if topic['Topic'] == state['topic']][0],
|
169 |
+
"topic": state['topic'],
|
170 |
+
"temperature": 0,
|
171 |
+
|
172 |
+
# to be filled in manually
|
173 |
+
"correctAnswersNumber": 0,
|
174 |
+
"distractorsNumber": 0,
|
175 |
+
"easilyDiscardableDistractorsNumber": 0,
|
176 |
+
"typeOfExercise": 0
|
177 |
+
},
|
178 |
+
timeout=20
|
179 |
+
)
|
180 |
+
|
181 |
+
##################################################################
|
182 |
+
# UI Layer
|
183 |
+
##################################################################
|
184 |
+
|
185 |
+
def format_output(output, exercise_type):
|
186 |
+
if type_of_exercise[exercise_type] in ["Open Question", "Short Answer Question"]:
|
187 |
+
return f"<div class='markdown-body'><h3>{output['Assignment']}</h3><p>{output['Solutions'][0]}</p></div>"
|
188 |
+
|
189 |
+
return f"<div class='markdown-body'><h3>Ouput (unknown type)</h3><p>{output['Solutions'][0]}</p></div>"
|
190 |
+
|
191 |
+
|
192 |
+
def reset_UI():
|
193 |
+
return [
|
194 |
+
{"level": levels[-1], "language": "English"},
|
195 |
+
]
|
196 |
+
|
197 |
+
def next():
|
198 |
+
new_experiment = generate_new_experiment()
|
199 |
+
|
200 |
+
new_img_components = [
|
201 |
+
gr.Image(value=img["name"], label=f"{i}", elem_id="unsel", show_label=False, show_download_button=False, show_share_button=False, interactive=False)
|
202 |
+
for i, img in enumerate(new_experiment["options"])
|
203 |
+
]
|
204 |
+
new_corrupted_component = gr.Image(value=new_experiment["corrupted"]["name"], label="corr", elem_id="corrupted", show_label=False, show_download_button=False, show_share_button=False, interactive=False)
|
205 |
+
|
206 |
+
return [new_experiment, new_corrupted_component, *new_img_components]
|
207 |
+
|
208 |
+
def on_url_change(url, state):
|
209 |
+
material = analyze_resource(url)
|
210 |
+
|
211 |
+
topics = [topic['Topic'] for topic in material['MainTopics']]
|
212 |
+
|
213 |
+
components = reset_UI()
|
214 |
+
state = components[0]
|
215 |
+
state = state | material
|
216 |
+
state['material_url'] = url
|
217 |
+
|
218 |
+
return [gr.Radio(label="Topic", choices=topics, interactive=True), state]
|
219 |
+
|
220 |
+
def on_topic_change(topic, old_state):
|
221 |
+
old_state['topic'] = topic
|
222 |
+
|
223 |
+
learning_objective = generate_learning_objective(topic, f"A {old_state['level']} class", old_state["level"])
|
224 |
+
old_state['learningObjectiveList'] = learning_objective
|
225 |
+
|
226 |
+
possible_objectives = list(chain.from_iterable(learning_objective.values()))
|
227 |
+
|
228 |
+
return [gr.Dropdown(label="Learning Objective", choices=possible_objectives, value=possible_objectives[0], interactive=True), old_state]
|
229 |
+
|
230 |
+
css = """
|
231 |
+
body, html {
|
232 |
+
margin: 0;
|
233 |
+
height: 100%; /* Full height */
|
234 |
+
width: 100%; /* Full width */
|
235 |
+
overflow: hidden; /* Prevent scrolling */
|
236 |
+
}
|
237 |
+
.interface, .block-container {
|
238 |
+
display: flex;
|
239 |
+
flex-direction: column;
|
240 |
+
height: 100%; /* Full height */
|
241 |
+
width: 100%; /* Full width */
|
242 |
+
}
|
243 |
+
.row-content {
|
244 |
+
height: 90vh; /* Full height */
|
245 |
+
}
|
246 |
+
.column-content {
|
247 |
+
display: flex;
|
248 |
+
flex-direction: column;
|
249 |
+
flex: 1; /* Flexibly take up available space */
|
250 |
+
height: 100%; /* Full height */
|
251 |
+
}
|
252 |
+
iframe.second-row {
|
253 |
+
width: 100%; /* Full width */
|
254 |
+
height: 50vh; /* Full height */
|
255 |
+
border: none; /* No border */
|
256 |
+
background-color: #f9f9f9; /* Light background */
|
257 |
+
}
|
258 |
+
|
259 |
+
/* Base style for Markdown content */
|
260 |
+
.markdown-body {
|
261 |
+
font-family: 'Helvetica Neue', Arial, sans-serif; /* Clean and modern font */
|
262 |
+
line-height: 1.6; /* Ample line height for readability */
|
263 |
+
font-size: 16px; /* Standard font size for readability */
|
264 |
+
color: #333; /* Dark grey color for text for less strain */
|
265 |
+
background-color: #f9f9f9; /* Light background to reduce glare */
|
266 |
+
padding: 20px; /* Padding around text */
|
267 |
+
border-radius: 8px; /* Slightly rounded corners for a softer look */
|
268 |
+
box-shadow: 0 2px 4px rgba(0,0,0,0.1); /* Subtle shadow for depth */
|
269 |
+
max-width: 800px; /* Max width to maintain optimal line length */
|
270 |
+
margin: 20px auto; /* Center align the Markdown content */
|
271 |
+
}
|
272 |
+
|
273 |
+
/* Headings with increased weight and spacing for clear hierarchy */
|
274 |
+
.markdown-body h1,
|
275 |
+
.markdown-body h2,
|
276 |
+
.markdown-body h3,
|
277 |
+
.markdown-body h4,
|
278 |
+
.markdown-body h5,
|
279 |
+
.markdown-body h6 {
|
280 |
+
color: #2a2a2a; /* Slightly darker than the text color */
|
281 |
+
margin-top: 24px;
|
282 |
+
margin-bottom: 16px;
|
283 |
+
font-weight: bold;
|
284 |
+
}
|
285 |
+
|
286 |
+
.markdown-body h1 {
|
287 |
+
font-size: 2em; /* Larger size for main titles */
|
288 |
+
}
|
289 |
+
|
290 |
+
.markdown-body h2 {
|
291 |
+
font-size: 1.5em;
|
292 |
+
}
|
293 |
+
|
294 |
+
.markdown-body h3 {
|
295 |
+
font-size: 1.17em;
|
296 |
+
}
|
297 |
+
|
298 |
+
/* Paragraphs with bottom margin for better separation */
|
299 |
+
.markdown-body p {
|
300 |
+
margin-bottom: 16px;
|
301 |
+
}
|
302 |
+
|
303 |
+
/* Links with a subtle color to stand out */
|
304 |
+
.markdown-body a {
|
305 |
+
color: #0656b5;
|
306 |
+
text-decoration: none; /* No underline */
|
307 |
+
}
|
308 |
+
|
309 |
+
.markdown-body a:hover,
|
310 |
+
.markdown-body a:focus {
|
311 |
+
text-decoration: underline; /* Underline on hover/focus for visibility */
|
312 |
+
}
|
313 |
+
|
314 |
+
/* Lists styled with padding and margin for clarity */
|
315 |
+
.markdown-body ul,
|
316 |
+
.markdown-body ol {
|
317 |
+
padding-left: 20px;
|
318 |
+
margin-top: 0;
|
319 |
+
margin-bottom: 16px;
|
320 |
+
}
|
321 |
+
|
322 |
+
.markdown-body li {
|
323 |
+
margin-bottom: 8px; /* Space between list items */
|
324 |
+
}
|
325 |
+
|
326 |
+
/* Blockquotes with a left border and padding for emphasis */
|
327 |
+
.markdown-body blockquote {
|
328 |
+
padding: 10px 20px;
|
329 |
+
margin: 0;
|
330 |
+
border-left: 5px solid #ccc; /* Subtle grey line to indicate quotes */
|
331 |
+
background-color: #f0f0f0; /* Very light background for contrast */
|
332 |
+
font-style: italic;
|
333 |
+
}
|
334 |
+
|
335 |
+
/* Code styling for inline and blocks */
|
336 |
+
.markdown-body code {
|
337 |
+
font-family: monospace;
|
338 |
+
background-color: #eee; /* Light grey background */
|
339 |
+
padding: 2px 4px;
|
340 |
+
border-radius: 3px; /* Rounded corners for code blocks */
|
341 |
+
font-size: 90%;
|
342 |
+
}
|
343 |
+
|
344 |
+
.markdown-body pre {
|
345 |
+
background-color: #f4f4f4; /* Slightly different background for distinction */
|
346 |
+
border: 1px solid #ddd; /* Border for definition */
|
347 |
+
padding: 10px; /* Padding inside code blocks */
|
348 |
+
overflow: auto; /* Auto-scroll for overflow */
|
349 |
+
line-height: 1.45;
|
350 |
+
border-radius: 5px;
|
351 |
+
}
|
352 |
+
"""
|
353 |
+
|
354 |
+
|
355 |
+
with gr.Blocks(title="Educational AI", css=css) as demo:
|
356 |
+
state = gr.State({"level": levels[-1], "language": "English"})
|
357 |
+
|
358 |
+
with gr.Row(elem_classes=["row-content"]):
|
359 |
+
with gr.Column(scale=3, elem_classes=["column-content"]):
|
360 |
+
language_component = gr.Dropdown(languages, label="Exercise Language", value="English")
|
361 |
+
level_component = gr.Dropdown(label="Level", choices=levels, value=levels[-1])
|
362 |
+
url_component = gr.Textbox(label="Input URL", placeholder="Enter URL here...")
|
363 |
+
iframe_component = gr.HTML("<iframe class='second-row' src='' allowfullscreen></iframe>")
|
364 |
+
|
365 |
+
with gr.Column(scale=3):
|
366 |
+
topic_component = gr.Radio(label="Topic", choices=["placeholder"], interactive=False)
|
367 |
+
lo_component = gr.Dropdown(label="Learning Objective", choices=[], value="placeholder", interactive=False)
|
368 |
+
question_type_component = gr.Dropdown(label="Question Type", choices=type_of_exercise, type="index")
|
369 |
+
|
370 |
+
correct_answers_component = gr.Number(value=1, minimum=1, maximum=3, step=1, label="Number of correct answers", interactive=False)
|
371 |
+
easy_distractors_component = gr.Number(value=1, minimum=0, maximum=8, step=1, label="Number of easy distractors", interactive=False)
|
372 |
+
distractors_component = gr.Number(value=1, minimum=0, maximum=8, step=1, label="Number of distractors", interactive=False)
|
373 |
+
|
374 |
+
with gr.Column(scale=3):
|
375 |
+
generate_btn = gr.Button("Generate Question")
|
376 |
+
output_component = gr.HTML("<div class='markdown-body'><h3>Output</h3>Placeholder for output</div>")
|
377 |
+
|
378 |
+
# on language change
|
379 |
+
language_component.change(lambda x, old_state: old_state | {"language": x}, [language_component, state], [state])
|
380 |
+
|
381 |
+
# on level change
|
382 |
+
level_component.change(lambda x, old_state: old_state | {"level": x}, [level_component, state], [state])
|
383 |
+
|
384 |
+
# on url change
|
385 |
+
url_component.change(lambda x: gr.HTML(f"<iframe class='second-row' src='{x}' allowfullscreen></iframe>"), [url_component], [iframe_component])
|
386 |
+
url_component.change(lambda x: gr.Info(f"Analyzing resource at {x}..."), [url_component], [])
|
387 |
+
url_component.change(on_url_change, [url_component, state], [topic_component, state])
|
388 |
+
|
389 |
+
# on topic change
|
390 |
+
topic_component.change(lambda x: gr.Info(f"Generating learning objective for {x}..."), [topic_component], [])
|
391 |
+
topic_component.change(on_topic_change, [topic_component, state], [lo_component, state])
|
392 |
+
|
393 |
+
# on lo change
|
394 |
+
lo_component.change(lambda x, old_state: old_state | {"learningObjective": x}, [lo_component, state], [state])
|
395 |
+
|
396 |
+
# on question type change
|
397 |
+
question_type_component.change(lambda x, old_state: old_state | {"typeOfExercise": x}, [question_type_component, state], [state])
|
398 |
+
|
399 |
|
400 |
+
# on generate question
|
401 |
+
generate_btn.click(lambda: gr.Info("Generating question..."))
|
402 |
+
generate_btn.click(generate_exercise, [state], [output_component])
|
403 |
|
404 |
+
demo.launch(show_api=False)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|