tommymarto commited on
Commit
7832cd9
Β·
1 Parent(s): 7fcb63d

wip update

Browse files
Files changed (1) hide show
  1. demo.py +373 -444
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
- creds = credentials.Certificate(json.loads(FIREBASE_API_KEY))
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
- categories = ["Factual Knowledge", "Understanding of Concepts", "Application of Skills", "Analysys And Evaluation"]
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
- print(input_json)
89
-
90
- output = ""
91
- try:
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
- print(input_json)
151
-
152
- output = ""
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
- print(input_json)
222
-
223
- output = ""
224
- try:
225
- response = requests.post(
226
- base_url + "QuizExercise/generateexercise",
227
- headers={"ApiKey": API_KEY},
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
- raise gr.Error("No generated text to comment.")
281
-
282
- def flag(flag_btn):
283
- global firebase_current_ref
284
- if firebase_current_ref is not None:
285
- firebase_current_ref.update({"flagged": not firebase_current_ref.get()["flagged"]})
286
- if firebase_current_ref.get()["flagged"]:
287
- gr.Info("Generated text flagged.")
288
- value = "⚠️ Unflag"
289
- else:
290
- gr.Info("Generated text unflagged.")
291
- value = "⚠️ Flag"
292
- return gr.Button(value=value)
293
- else:
294
- gr.Warning("No generated text to flag.")
295
-
296
- return flag_btn
297
-
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
298
 
299
- #################################################################################################################################################
300
- # Interface building
301
- #################################################################################################################################################
302
 
303
- def build_fill_gaps_interface():
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)