import gradio as gr from groq import Groq import os from PIL import Image, ImageDraw, ImageFont from datetime import datetime import json import tempfile # Initialize Groq client client = Groq( api_key=os.getenv("GROQ_API_KEY") ) class QuizApp: def __init__(self): self.current_questions = [] def generate_questions(self, text, num_questions): prompt = f"""Create exactly {num_questions} multiple choice questions based on this text: {text} For each question: 1. Create a clear, concise question 2. Provide exactly 4 options 3. Mark the correct answer with the index (0-3) 4. Ensure options are concise and clear Return ONLY a JSON array with this EXACT format - no other text: [ {{ "question": "Question text here?", "options": [ "Brief option 1", "Brief option 2", "Brief option 3", "Brief option 4" ], "correct_answer": 0 }} ] Keep all options concise (10 words or less each). """ try: response = client.chat.completions.create( messages=[ { "role": "system", "content": "You are a quiz generator. Create clear questions with concise answer options." }, { "role": "user", "content": prompt } ], model="llama-3.2-3b-preview", temperature=0.3, max_tokens=2048 ) response_text = response.choices[0].message.content.strip() response_text = response_text.replace("```json", "").replace("```", "").strip() start_idx = response_text.find("[") end_idx = response_text.rfind("]") if start_idx == -1 or end_idx == -1: raise ValueError("No valid JSON array found in response") response_text = response_text[start_idx:end_idx + 1] questions = json.loads(response_text) validated_questions = [] for q in questions: if not all(key in q for key in ["question", "options", "correct_answer"]): continue clean_options = [opt.strip()[:100] for opt in q["options"] if isinstance(opt, str)] if len(clean_options) != 4: continue clean_q = { "question": q["question"].strip(), "options": clean_options, "correct_answer": int(q["correct_answer"]) % 4 } validated_questions.append(clean_q) if not validated_questions: raise ValueError("No valid questions after validation") self.current_questions = validated_questions[:num_questions] return True, self.current_questions except Exception as e: print(f"Error in question generation: {str(e)}") return False, [] def calculate_score(self, answers): if not answers or not self.current_questions: return 0, None, [] total = len(self.current_questions) correct = 0 feedback = [] for i, (q, a) in enumerate(zip(self.current_questions, answers)): try: if a is not None: selected_index = q["options"].index(a) is_correct = selected_index == q["correct_answer"] if is_correct: correct += 1 feedback.append({ "is_correct": is_correct, "selected": a, "correct_answer": q["options"][q["correct_answer"]] }) else: feedback.append({ "is_correct": False, "selected": None, "correct_answer": q["options"][q["correct_answer"]] }) except (ValueError, TypeError) as e: print(f"Error processing answer {i}: {e}") feedback.append({ "is_correct": False, "selected": None, "correct_answer": q["options"][q["correct_answer"]] }) score = (correct / total) * 100 passed = score >= 80 return score, passed, feedback # Update the interface functions def create_quiz_interface(): quiz_app = QuizApp() with gr.Blocks(title="CertifyMe AI", theme=gr.themes.Soft()) as demo: current_questions = gr.State([]) # Header gr.Markdown(""" # 🎓 CertifyMe AI ### Transform Your Knowledge into Recognized Achievements """) # Tabs with gr.Tabs() as tabs: # Step 1: Profile Setup with gr.Tab("📋 Step 1: Profile Setup") as tab1: with gr.Row(): name = gr.Textbox(label="Full Name", placeholder="Enter your full name") email = gr.Textbox(label="Email", placeholder="Enter your email") text_input = gr.Textbox( label="Learning Content", placeholder="Enter the text content you want to be assessed on", lines=10 ) num_questions = gr.Slider( minimum=1, maximum=5, value=3, step=1, label="Number of Questions" ) with gr.Row(): company_logo = gr.Image(label="Company Logo (Optional)", type="filepath") participant_photo = gr.Image(label="Your Photo (Optional)", type="filepath") generate_btn = gr.Button("Generate Assessment", variant="primary", size="lg") # Step 2: Take Assessment with gr.Tab("📝 Step 2: Take Assessment") as tab2: feedback_box = gr.Markdown("") question_box = gr.Markdown("") answers = [] for i in range(5): with gr.Group(): radio = gr.Radio( choices=[], label=f"Question {i+1}", visible=False, interactive=True ) answers.append(radio) submit_btn = gr.Button("Submit Assessment", variant="primary", size="lg") # Step 3: Get Certified with gr.Tab("🎓 Step 3: Get Certified") as tab3: score_display = gr.Number(label="Your Score") result_message = gr.Markdown("") course_name = gr.Textbox( label="Certification Title", value="Professional Assessment Certification" ) certificate_display = gr.Image(label="Your Certificate") def update_questions(text, num_questions): if not text.strip(): return ( gr.update(value=""), gr.update(value="⚠️ Please enter some text content to generate questions."), *[gr.update(visible=False, choices=[]) for _ in range(5)], [], gr.update(selected=1) ) success, questions = quiz_app.generate_questions(text, num_questions) if not success or not questions: return ( gr.update(value=""), gr.update(value="❌ Failed to generate questions. Please try again."), *[gr.update(visible=False, choices=[]) for _ in range(5)], [], gr.update(selected=1) ) # Create question display questions_html = "# 📝 Assessment Questions\n\n" questions_html += "> Please select one answer for each question.\n\n" # Update radio buttons updates = [] for i, q in enumerate(questions): questions_html += f"### Question {i+1}\n{q['question']}\n\n" updates.append(gr.update( visible=True, choices=q["options"], value=None, label=f"Select your answer:" )) # Hide unused radio buttons for i in range(len(questions), 5): updates.append(gr.update(visible=False, choices=[])) return ( gr.update(value=questions_html), gr.update(value=""), *updates, questions, gr.update(selected=1) ) def submit_quiz(q1, q2, q3, q4, q5, questions): answers = [q1, q2, q3, q4, q5][:len(questions)] if not all(a is not None for a in answers): return ( gr.update(value="⚠️ Please answer all questions before submitting."), *[gr.update() for _ in range(5)], 0, "", gr.update(selected=1) ) score, passed, feedback = quiz_app.calculate_score(answers) # Create feedback HTML feedback_html = "# Assessment Results\n\n" for i, (q, f) in enumerate(zip(quiz_app.current_questions, feedback)): color = "green" if f["is_correct"] else "red" symbol = "✅" if f["is_correct"] else "❌" feedback_html += f""" ### Question {i+1} {q["question"]}
You passed the assessment with a score of {score:.1f}%
Your certificate has been generated.
Your score: {score:.1f}%
You need 80% or higher to pass and receive a certificate.