import gradio as gr import uuid import os import tempfile import hashlib from reportlab.lib.pagesizes import A5, landscape from reportlab.pdfgen import canvas from reportlab.lib.units import mm from reportlab.lib.colors import HexColor from datetime import datetime # --- Certificate Generator (This function is part of the generated code) --- # Note: This function is defined here for completeness but is also embedded # in the generated string. def generate_certificate(name, score, total, instructor="Instructor"): """ Generates a PDF certificate of completion. This entire function will be part of the output code string. """ unique_id = str(uuid.uuid4()) filename = f"cert_{unique_id}.pdf" filepath = os.path.join(tempfile.gettempdir(), filename) c = canvas.Canvas(filepath, pagesize=landscape(A5)) width, height = landscape(A5) # Set background and border c.setFillColor(HexColor("#fffdf6")) # Creamy background c.rect(0, 0, width, height, stroke=0, fill=1) c.setStrokeColor(HexColor("#001858")) # Dark blue border c.setLineWidth(3) margin = 10 * mm c.rect(margin, margin, width - 2 * margin, height - 2 * margin) # Add text content c.setFillColor(HexColor("#001858")) # Dark blue text c.setFont("Helvetica-Bold", 24) c.drawCentredString(width / 2, height - 60, "Certificate of Completion") c.setFont("Helvetica", 14) c.drawCentredString(width / 2, height - 100, "This is awarded to") c.setFont("Helvetica-Bold", 18) c.drawCentredString(width / 2, height - 130, name) c.setFont("Helvetica", 14) c.drawCentredString(width / 2, height - 160, "For successfully completing the quiz") c.setFont("Helvetica", 12) c.drawCentredString(width / 2, height - 185, f"Score: {score} / {total}") # Footer c.setFont("Helvetica-Oblique", 10) c.drawString(margin + 10, margin + 20, f"Instructor: {instructor}") date_str = datetime.now().strftime("%d %B %Y") c.setFont("Helvetica", 10) c.drawRightString(width - margin - 10, margin + 20, f"Issued on: {date_str}") c.save() return filepath # --- Main Quiz Code Generator --- def generate_python_code(title, instructor, quiz_type, questions_text): """ Parses instructor input and generates a self-contained Python script for a student-facing Gradio quiz app. """ # Parse questions and hash the answers for security parsed_questions = [] for line in questions_text.strip().split("\n"): if not line.strip(): continue parts = [p.strip() for p in line.split(",")] if quiz_type == "Multiple Choice": if len(parts) < 3: continue # Skip malformed lines q_text = parts[0] options = parts[1:-1] answer = parts[-1] parsed_questions.append({ "question": q_text, "options": options, "answer_hash": hashlib.sha256(answer.lower().encode()).hexdigest() }) else: # Text Answer if len(parts) < 2: continue # Skip malformed lines q_text = parts[0] answer = parts[1] parsed_questions.append({ "question": q_text, "answer_hash": hashlib.sha256(answer.lower().encode()).hexdigest() }) # Generate the complete Python code string for the student quiz app # f-strings with {{ and }} are used to escape braces for the final code. python_code = f'''!pip install reportlab # --- Generated Quiz App --- # Copy and paste this entire code block into a single Google Colab cell and run it. import gradio as gr import uuid, os, tempfile, hashlib from reportlab.lib.pagesizes import A5, landscape from reportlab.pdfgen import canvas from reportlab.lib.units import mm from reportlab.lib.colors import HexColor from datetime import datetime # Certificate generation function (included for a self-contained script) def generate_certificate(name, score, total, instructor="{instructor}"): unique_id = str(uuid.uuid4()) filename = f"cert_{{unique_id}}.pdf" filepath = os.path.join(tempfile.gettempdir(), filename) c = canvas.Canvas(filepath, pagesize=landscape(A5)) width, height = landscape(A5) c.setFillColor(HexColor("#fffdf6")) c.rect(0, 0, width, height, stroke=0, fill=1) c.setStrokeColor(HexColor("#001858")) c.setLineWidth(3) margin = 10 * mm c.rect(margin, margin, width - 2 * margin, height - 2 * margin) c.setFillColor(HexColor("#001858")) c.setFont("Helvetica-Bold", 24) c.drawCentredString(width / 2, height - 60, "Certificate of Completion") c.setFont("Helvetica", 14) c.drawCentredString(width / 2, height - 100, "This is awarded to") c.setFont("Helvetica-Bold", 18) c.drawCentredString(width / 2, height - 130, name) c.setFont("Helvetica", 14) c.drawCentredString(width / 2, height - 160, "For successfully completing the quiz") c.setFont("Helvetica", 12) c.drawCentredString(width / 2, height - 185, f"Score: {{score}} / {{total}}") c.setFont("Helvetica-Oblique", 10) c.drawString(margin + 10, margin + 20, f"Instructor: {{instructor}}") date_str = datetime.now().strftime("%d %B %Y") c.setFont("Helvetica", 10) c.drawRightString(width - margin - 10, margin + 20, f"Issued on: {{date_str}}") c.save() return filepath # Quiz data (answers are hashed) quiz_type = "{quiz_type}" questions = {parsed_questions} def eval_quiz(name, *answers): """ Evaluates the student's answers and generates a certificate only if the score is 80% or higher. """ if not name.strip(): name = "Anonymous" score = 0 for i, ans in enumerate(answers): if ans and hashlib.sha256(str(ans).lower().strip().encode()).hexdigest() == questions[i]["answer_hash"]: score += 1 total_questions = len(questions) passing_threshold = 0.8 result_message = f"Hi {{name}}, your score is: {{score}} / {{total_questions}}." cert_path = None # Default to no certificate # Check if the score meets the passing threshold if total_questions > 0 and (score / total_questions) >= passing_threshold: cert_path = generate_certificate(name, score, total_questions, instructor="{instructor}") result_message += " Congratulations, you passed and earned a certificate!" else: result_message += " A score of 80% is required to receive a certificate." return result_message, cert_path # Gradio interface for the student with gr.Blocks(theme=gr.themes.Soft()) as app: gr.Markdown("## {title}") with gr.Row(): name = gr.Textbox(label="Enter Your Full Name to Generate Certificate", placeholder="e.g., Ada Lovelace") answer_inputs = [] for q in questions: gr.Markdown("**Question:** " + q['question']) if quiz_type == "Multiple Choice": answer_inputs.append(gr.Radio(choices=q["options"], label="Select your answer")) else: answer_inputs.append(gr.Textbox(label="Type your answer")) submit_btn = gr.Button("Submit Quiz") with gr.Row(): result_output = gr.Textbox(label="Your Result") certificate_output = gr.File(label="Download Your Certificate") submit_btn.click( fn=eval_quiz, inputs=[name] + answer_inputs, outputs=[result_output, certificate_output] ) app.launch(debug=True) ''' return python_code # --- Instructor Interface --- with gr.Blocks(theme=gr.themes.Base()) as interface: gr.Markdown("# Instructor Quiz Generator") gr.Markdown("Create a secure, interactive quiz for your Google Colab notebooks.") with gr.Row(): # Left column for inputs with gr.Column(scale=2): title = gr.Textbox(label="Quiz Title", placeholder="e.g. Python Basics Quiz") instructor = gr.Textbox(label="Instructor Name", placeholder="e.g. Dr. Ada Lovelace") quiz_type = gr.Dropdown( choices=["Multiple Choice", "Text Answer"], label="Quiz Type", value="Multiple Choice" ) questions = gr.Textbox( lines=10, label="Questions & Answers", placeholder=( "One question per line. Separate parts with commas.\\n\\n" "MCQ Format: Question,Option1,Option2,CorrectOption\\n" "Text Format: Question,CorrectAnswer" ) ) generate_btn = gr.Button("🚀 Generate Python Quiz Code", variant="primary") # Right column for the generated code output with gr.Column(scale=1): output = gr.Code(label="Generated Python Code for Colab", language="python", lines=22) # Link the button to the generation function generate_btn.click( fn=generate_python_code, inputs=[title, instructor, quiz_type, questions], outputs=output ) interface.launch()