|
|
| import streamlit as st |
| import requests |
| from fpdf import FPDF |
| import os |
| import time |
| from datetime import datetime |
| import groq |
|
|
| |
| mistral_api_key = os.getenv("MISTRAL_API_KEY", "gz6lDXokxgR6cLY72oomALWcm7vhjRzQ") |
| groq_api_key = os.getenv("GROQ_API_KEY", "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") |
|
|
| |
| groq_client = groq.Client(api_key=groq_api_key) |
|
|
| |
| def call_mistral_api(prompt): |
| url = "https://api.mistral.ai/v1/chat/completions" |
| headers = { |
| "Authorization": f"Bearer {mistral_api_key}", |
| "Content-Type": "application/json" |
| } |
| payload = { |
| "model": "mistral-medium", |
| "messages": [ |
| {"role": "user", "content": prompt} |
| ] |
| } |
| try: |
| response = requests.post(url, headers=headers, json=payload) |
| response.raise_for_status() |
| return response.json()['choices'][0]['message']['content'] |
| except requests.exceptions.HTTPError as err: |
| if response.status_code == 429: |
| st.warning("Rate limit exceeded. Please wait a few seconds and try again.") |
| time.sleep(5) |
| return call_mistral_api(prompt) |
| return f"HTTP Error: {err}" |
| except Exception as err: |
| return f"Error: {err}" |
|
|
| |
| def call_groq_api(prompt): |
| try: |
| response = groq_client.chat.completions.create( |
| model="llama-3.3-70b-versatile", |
| messages=[ |
| {"role": "user", "content": prompt} |
| ] |
| ) |
| return response.choices[0].message.content |
| except Exception as err: |
| st.error(f"Error: {err}") |
| return f"Error: {err}" |
|
|
| |
| def analyze_requirement(requirement): |
| |
| type_prompt = f"Classify the following requirement as Functional or Non-Functional in one word:\n\n{requirement}\n\nType:" |
| req_type = call_mistral_api(type_prompt).strip() |
|
|
| domain_prompt = f"Classify the domain for the following requirement in one word (e.g., E-commerce, Education, etc.):\n\n{requirement}\n\nDomain:" |
| domain = call_mistral_api(domain_prompt).strip() |
|
|
| |
| defects_prompt = f"""List ONLY the major defects in the following requirement (e.g., Ambiguity, Incompleteness, etc.) in 1-2 words each:\n\n{requirement}\n\nDefects:""" |
| defects = call_groq_api(defects_prompt).strip() |
|
|
| rewritten_prompt = f"""Rewrite the following requirement in 1-2 sentences to address the defects:\n\n{requirement}\n\nRewritten:""" |
| rewritten = call_groq_api(rewritten_prompt).strip() |
|
|
| return { |
| "Requirement": requirement, |
| "Type": req_type, |
| "Domain": domain, |
| "Defects": defects, |
| "Rewritten": rewritten |
| } |
|
|
| |
| def generate_pdf_report(results): |
| pdf = FPDF() |
| pdf.add_page() |
| pdf.set_font("Arial", size=12) |
|
|
| |
| pdf.set_font("Arial", 'B', 50) |
| pdf.set_text_color(230, 230, 230) |
| pdf.rotate(45) |
| pdf.text(60, 150, "MSSE31 Student's Project") |
| pdf.rotate(0) |
|
|
|
|
| |
| pdf.set_font("Arial", 'B', 16) |
| pdf.set_text_color(0, 0, 0) |
| pdf.cell(200, 10, txt="AI-Based Requirement Defect Detection Using Large Language Models (LLMs)s", ln=True, align='C') |
| pdf.set_font("Arial", size=12) |
| pdf.cell(200, 10, txt=f"Report Generated on: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}", ln=True, align='C') |
| pdf.ln(10) |
|
|
| |
| pdf.set_font("Arial", size=12) |
| for i, result in enumerate(results, start=1): |
| if pdf.get_y() > 250: |
| pdf.add_page() |
| pdf.set_font("Arial", 'B', 16) |
| pdf.cell(200, 10, txt="AI Powered Requirement Analysis and Defect Detection", ln=True, align='C') |
| pdf.set_font("Arial", size=12) |
| pdf.cell(200, 10, txt=f"Report Generated on: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}", ln=True, align='C') |
| pdf.ln(10) |
|
|
| |
| pdf.set_font("Arial", 'B', 14) |
| pdf.multi_cell(200, 10, txt=f"Requirement R{i}: {result['Requirement']}", align='L') |
| pdf.set_font("Arial", size=12) |
| pdf.multi_cell(200, 10, txt=f"Type: {result['Type']}", align='L') |
| pdf.multi_cell(200, 10, txt=f"Domain: {result['Domain']}", align='L') |
| pdf.multi_cell(200, 10, txt=f"Defects: {result['Defects']}", align='L') |
| pdf.multi_cell(200, 10, txt=f"Rewritten: {result['Rewritten']}", align='L') |
| pdf.multi_cell(200, 10, txt="-" * 50, align='L') |
| pdf.ln(5) |
|
|
| pdf_output = "requirements_report.pdf" |
| pdf.output(pdf_output) |
| return pdf_output |
|
|
| |
| def main(): |
| st.title("AI-Based Requirement Defect Detection Using Large Language Models (LLMs)") |
| st.markdown("By: Sadia Iqbal") |
| st.markdown("**Models:** Mistral (Classification & Domain) + Groq (Defects & Rewriting)") |
|
|
| |
| input_text = st.text_area("Enter your requirements (one per line or separated by periods):") |
| requirements = [] |
| if input_text: |
| requirements = [req.strip() for req in input_text.replace("\n", ".").split(".") if req.strip()] |
|
|
| |
| if st.button("Analyze Requirements"): |
| if not requirements: |
| st.warning("Please enter requirements.") |
| else: |
| results = [] |
| for req in requirements: |
| if req.strip(): |
| results.append(analyze_requirement(req.strip())) |
|
|
| |
| st.subheader("Analysis Results") |
| for i, result in enumerate(results, start=1): |
| st.write(f"### Requirement R{i}: {result['Requirement']}") |
| st.write(f"**Type:** {result['Type']}") |
| st.write(f"**Domain:** {result['Domain']}") |
| st.write(f"**Defects:** {result['Defects']}") |
| st.write(f"**Rewritten:** {result['Rewritten']}") |
| st.write("---") |
|
|
| |
| pdf_report = generate_pdf_report(results) |
| with open(pdf_report, "rb") as f: |
| st.download_button( |
| label="Download PDF Report", |
| data=f, |
| file_name="requirements_report.pdf", |
| mime="application/pdf" |
| ) |
|
|
| |
| if __name__ == "__main__": |
| main() |
|
|