{
  "root": {
    "name": "Footer Addition and Formatting on All Slides",
    "description": "Evaluates whether the agent successfully added a center-aligned footer with the text 'Fundamental Accounting Concepts' to every slide in the PowerPoint presentation.",
    "is_critical": true,
    "metadata": {},
    "children": [
      {
        "name": "Footer Addition Coverage",
        "description": "Check that every slide in the presentation has a footer added.",
        "is_critical": true,
        "metadata": {},
        "children": [
          {
            "name": "Footer Present on All Slides",
            "description": "Verifies every slide has a footer element added.",
            "is_critical": true,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score() -> tuple[str, float]:\n    \"\"\"\n    Code-based verification that footer exists on all slides by parsing PowerPoint XML.\n    Proves footer existence through direct examination of slide XML structure.\n    \"\"\"\n    try:\n        from pptx import Presentation\n        import zipfile\n        import xml.etree.ElementTree as ET\n        from pathlib import Path\n        \n        # Open the modified PowerPoint file\n        pres = Presentation(modified_ppt_path)\n        \n        # Also need to check raw XML for footer elements\n        total_slides = len(pres.slides)\n        slides_with_footer = 0\n        \n        # Method 1: Check through python-pptx API for footer shapes\n        for slide in pres.slides:\n            has_footer = False\n            \n            # Check all shapes on the slide for footer-like elements\n            for shape in slide.shapes:\n                if hasattr(shape, 'text'):\n                    # Check if this shape contains footer text\n                    if 'Fundamental Accounting Concepts' in shape.text:\n                        has_footer = True\n                        break\n            \n            if has_footer:\n                slides_with_footer += 1\n        \n        # Method 2: Also check raw XML for more thorough verification\n        # Extract and examine slide XML files directly\n        try:\n            with zipfile.ZipFile(modified_ppt_path, 'r') as pptx_zip:\n                xml_slides_with_footer = 0\n                \n                # Check each slide's XML\n                for i in range(1, total_slides + 1):\n                    slide_xml_path = f'ppt/slides/slide{i}.xml'\n                    try:\n                        with pptx_zip.open(slide_xml_path) as slide_file:\n                            slide_content = slide_file.read().decode('utf-8')\n                            \n                            # Check if the slide XML contains footer text\n                            if 'Fundamental Accounting Concepts' in slide_content:\n                                xml_slides_with_footer += 1\n                    except KeyError:\n                        continue  # Slide file doesn't exist\n                \n                # Use the higher count (more reliable detection)\n                slides_with_footer = max(slides_with_footer, xml_slides_with_footer)\n        \n        except Exception as xml_error:\n            # Fall back to python-pptx method only\n            pass\n        \n        # Score based on how many slides have footers\n        if slides_with_footer == total_slides:\n            return f\"Footer verified present on all {total_slides} slides through XML parsing.\", 1.0\n        elif slides_with_footer > 0:\n            score = slides_with_footer / total_slides\n            return f\"Footer found on {slides_with_footer}/{total_slides} slides through XML analysis.\", score\n        else:\n            return \"No footer containing 'Fundamental Accounting Concepts' found on any slide through XML verification.\", 0.0\n            \n    except Exception as e:\n        return f\"Error during footer existence verification: {str(e)[:100]}\", 0.0\n"
            }
          }
        ]
      },
      {
        "name": "Footer Text Accuracy",
        "description": "Checks that the footer text on each slide matches exactly 'Fundamental Accounting Concepts'.",
        "is_critical": true,
        "metadata": {},
        "children": [
          {
            "name": "Footer Text Matches Specification",
            "description": "All slide footers must contain exactly the required text.",
            "is_critical": true,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score() -> tuple[str, float]:\n    \"\"\"\n    Code-based verification that footer contains the exact required text.\n    Proves text accuracy through direct XML parsing and text extraction.\n    \"\"\"\n    try:\n        from pptx import Presentation\n        import zipfile\n        \n        required_text = \"Fundamental Accounting Concepts\"\n        pres = Presentation(modified_ppt_path)\n        total_slides = len(pres.slides)\n        slides_with_correct_text = 0\n        \n        # Method 1: Check through python-pptx API\n        for slide in pres.slides:\n            has_correct_text = False\n            exact_text_found = False\n            \n            # Check all text shapes on the slide\n            for shape in slide.shapes:\n                if hasattr(shape, 'text'):\n                    # Check for exact text match (case sensitive)\n                    if required_text in shape.text:\n                        has_correct_text = True\n                        if shape.text == required_text:\n                            exact_text_found = True\n                            break\n            \n            if has_correct_text and exact_text_found:\n                slides_with_correct_text += 1\n        \n        # Method 2: Verify through raw XML parsing for more thorough check\n        try:\n            with zipfile.ZipFile(modified_ppt_path, 'r') as pptx_zip:\n                xml_slides_with_correct_text = 0\n                \n                # Check each slide's XML for exact text\n                for i in range(1, total_slides + 1):\n                    slide_xml_path = f'ppt/slides/slide{i}.xml'\n                    try:\n                        with pptx_zip.open(slide_xml_path) as slide_file:\n                            slide_content = slide_file.read().decode('utf-8')\n                            \n                            # Check for exact text in XML\n                            if required_text in slide_content:\n                                xml_slides_with_correct_text += 1\n                    except KeyError:\n                        continue\n\n                # Use the min for rigorous verification\n                slides_with_correct_text = min(slides_with_correct_text, xml_slides_with_correct_text)\n\n        except Exception:\n            # Fall back to python-pptx method\n            pass\n        \n        # Score based on text accuracy\n        if slides_with_correct_text == total_slides:\n            return f\"Footer text '{required_text}' verified correct on all {total_slides} slides.\", 1.0\n        elif slides_with_correct_text > 0:\n            score = slides_with_correct_text / total_slides\n            return f\"Correct footer text found on {slides_with_correct_text}/{total_slides} slides.\", score\n        else:\n            return f\"Required footer text '{required_text}' not found on any slide through XML verification.\", 0.0\n            \n    except Exception as e:\n        return f\"Error during footer text verification: {str(e)[:100]}\", 0.0\n"
            }
          }
        ]
      },
      {
        "name": "Footer Center Alignment",
        "description": "Checks that the footer on each slide is center-aligned.",
        "is_critical": true,
        "metadata": {},
        "children": [
          {
            "name": "Footer is Center-Aligned on All Slides",
            "description": "Verifies the footer text on each slide is horizontally center-aligned.",
            "is_critical": true,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score() -> tuple[str, float]:\n    \"\"\"\n    Code-based verification that footer is center-aligned on all slides.\n    Proves center alignment through XML parsing of positioning and alignment attributes.\n    \"\"\"\n    try:\n        from pptx import Presentation\n        from pptx.enum.text import PP_ALIGN\n        import zipfile\n        import xml.etree.ElementTree as ET\n        \n        required_text = \"Fundamental Accounting Concepts\"\n        pres = Presentation(modified_ppt_path)\n        total_slides = len(pres.slides)\n        slides_with_centered_footer = 0\n        \n        # Method 1: Check through python-pptx API for alignment\n        for slide in pres.slides:\n            has_centered_footer = False\n            \n            # Check all text shapes on the slide\n            for shape in slide.shapes:\n                if hasattr(shape, 'text') and required_text in shape.text:\n                    # Found the footer text, now check alignment\n                    if hasattr(shape, 'text_frame'):\n                        # Check paragraph alignment\n                        for paragraph in shape.text_frame.paragraphs:\n                            if paragraph.alignment == PP_ALIGN.CENTER:\n                                has_centered_footer = True\n                                break\n                        \n                        # Also check if the shape itself is centered horizontally on slide\n                        slide_width = pres.slide_width\n                        shape_center_x = shape.left + (shape.width / 2)\n                        slide_center_x = slide_width / 2\n                        \n                        # Allow some tolerance for \"center\" (within 5% of slide width)\n                        center_tolerance = slide_width * 0.05\n                        if abs(shape_center_x - slide_center_x) <= center_tolerance:\n                            has_centered_footer = True\n                    break  # Found the footer, no need to check other shapes\n            \n            if has_centered_footer:\n                slides_with_centered_footer += 1\n        \n        # Method 2: Check XML for alignment attributes for more thorough verification\n        try:\n            with zipfile.ZipFile(modified_ppt_path, 'r') as pptx_zip:\n                xml_slides_with_centered_footer = 0\n                \n                for i in range(1, total_slides + 1):\n                    slide_xml_path = f'ppt/slides/slide{i}.xml'\n                    try:\n                        with pptx_zip.open(slide_xml_path) as slide_file:\n                            slide_content = slide_file.read().decode('utf-8')\n                            \n                            # Check if this slide has the footer text\n                            if required_text in slide_content:\n                                # Parse XML to check alignment\n                                root = ET.fromstring(slide_content)\n                                \n                                # Look for text elements containing our footer text\n                                # and check for alignment attributes\n                                for text_elem in root.iter():\n                                    if text_elem.text and required_text in text_elem.text:\n                                        # Found footer text, look for parent alignment\n                                        parent = text_elem.getparent()\n                                        while parent is not None:\n                                            # Check for alignment attributes\n                                            if 'algn' in parent.attrib:\n                                                if parent.attrib['algn'] == 'ctr':  # center alignment\n                                                    xml_slides_with_centered_footer += 1\n                                                    break\n                                            parent = parent.getparent()\n                                        break\n                                \n                                # Also check for center alignment in paragraph properties\n                                if 'a:pPr' in slide_content and 'algn=\"ctr\"' in slide_content:\n                                    xml_slides_with_centered_footer += 1\n                    \n                    except (KeyError, ET.ParseError):\n                        continue\n                \n                # Use the higher count for verification\n                slides_with_centered_footer = max(slides_with_centered_footer, xml_slides_with_centered_footer)\n        \n        except Exception:\n            # Fall back to python-pptx method\n            pass\n        \n        # Score based on center alignment\n        if slides_with_centered_footer == total_slides:\n            return f\"Footer verified center-aligned on all {total_slides} slides through XML analysis.\", 1.0\n        elif slides_with_centered_footer > 0:\n            score = slides_with_centered_footer / total_slides\n            return f\"Center-aligned footer found on {slides_with_centered_footer}/{total_slides} slides.\", score\n        else:\n            return \"No center-aligned footer found on any slide through XML verification.\", 0.0\n            \n    except Exception as e:\n        return f\"Error during footer alignment verification: {str(e)[:100]}\", 0.0\n"
            }
          }
        ]
      },
      {
        "name": "No Extraneous Additions",
        "description": "Ensures that only the required footer was added and no extra unrelated footers or text were inserted.",
        "is_critical": false,
        "metadata": {},
        "children": [
          {
            "name": "No Extra Changes Added",
            "description": "Checks for unwarranted additions to the footer area.",
            "is_critical": false,
            "metadata": {},
            "scorer": {
              "type": "function",
              "function_code": "def compute_score() -> tuple[str, float]:\n    \"\"\"\n    Code-based verification that no extraneous footer or text was added.\n    Proves that only the required footer text was added, nothing extra.\n    \"\"\"\n    if ppt_diff.added_animations: \n        return (\"Animations were added, which is not allowed.\", 0.0)\n    if ppt_diff.added_transitions:\n        return (\"Transitions were added, which is not allowed.\", 0.0)\n    try:\n        from pptx import Presentation\n        import zipfile\n        from collections import Counter\n        import re\n        \n        required_text = \"Fundamental Accounting Concepts\"\n        # Resolve expected globals injected by environment (robust fallback)\n        global original_ppt_path, modified_ppt_path  # type: ignore\n        try:\n            orig_path = original_ppt_path  # type: ignore[name-defined]\n            mod_path = modified_ppt_path  # type: ignore[name-defined]\n        except NameError:\n            # Attempt to pull from a context dict if provided (common harness pattern)\n            ctx = globals()\n            orig_path = ctx.get('original_ppt_path')\n            mod_path = ctx.get('modified_ppt_path')\n            if not orig_path or not mod_path:\n                return \"Original or modified PPT path not provided to scorer.\", 0.0\n        orig = Presentation(orig_path)\n        pres = Presentation(mod_path)\n        total_slides = len(pres.slides)\n        if len(orig.slides) != total_slides:\n            return \"Slide count changed (not allowed).\", 0.0\n        slides_with_only_required_footer = 0\n        structural_integrity_ok = True  # tracks non-footer change introduction\n        reason_details: list[str] = []\n        \n        # Helper: extract texts of shapes for comparison\n        def shape_texts(slide):\n            texts = []\n            for shp in slide.shapes:\n                if hasattr(shp, 'text'):\n                    txt = (shp.text or '').strip()\n                    texts.append(txt)\n            return texts\n        \n        # Compare each slide's shapes & texts allowing only footer insertion\n        for idx, (orig_slide, mod_slide) in enumerate(zip(orig.slides, pres.slides), start=1):\n            orig_shapes = list(orig_slide.shapes)\n            mod_shapes = list(mod_slide.shapes)\n            orig_count = len(orig_shapes)\n            mod_count = len(mod_shapes)\n            \n            # Allow at most +1 shape for the footer\n            if mod_count not in (orig_count, orig_count + 1):\n                structural_integrity_ok = False\n                reason_details.append(f\"Slide {idx}: shape count changed from {orig_count} to {mod_count} (>+1).\")\n                continue\n            \n            # Gather texts\n            orig_texts = shape_texts(orig_slide)\n            mod_texts = shape_texts(mod_slide)\n            \n            # Count occurrences of required text in modified slide\n            req_in_mod = [t for t in mod_texts if required_text == t]\n            \n            # Determine extra texts (multiset diff)\n            orig_counter = Counter([t for t in orig_texts if t])\n            mod_counter = Counter([t for t in mod_texts if t])\n            # Remove one occurrence of required footer from comparison if it wasn't in original\n            if required_text not in orig_counter and required_text in mod_counter:\n                mod_counter[required_text] -= 1\n                if mod_counter[required_text] <= 0:\n                    del mod_counter[required_text]\n            # After neutralizing allowed footer, counters must match\n            if orig_counter != mod_counter:\n                structural_integrity_ok = False\n                # Identify first difference for message brevity\n                diff_parts = []\n                for k in set(orig_counter) | set(mod_counter):\n                    if orig_counter.get(k, 0) != mod_counter.get(k, 0):\n                        if k == required_text:\n                            continue\n                        diff_parts.append(k[:30])\n                        if len(diff_parts) >= 3:\n                            break\n                sample = \", \".join(diff_parts) if diff_parts else \"unknown change\"\n                reason_details.append(f\"Slide {idx}: text content changed ({sample}).\")\n                continue\n            \n            # If footer inserted, ensure exactly one new occurrence and it's at bottom 20% or replaces empty placeholder\n            if required_text not in orig_texts:\n                if len(req_in_mod) != 1:\n                    structural_integrity_ok = False\n                    reason_details.append(f\"Slide {idx}: footer occurrence count invalid ({len(req_in_mod)}).\")\n                    continue\n                # Check location of the shape containing the footer\n                footer_shape = None\n                for shp in mod_slide.shapes:\n                    if hasattr(shp, 'text') and shp.text.strip() == required_text:\n                        footer_shape = shp\n                        break\n                if footer_shape is None:\n                    structural_integrity_ok = False\n                    reason_details.append(f\"Slide {idx}: footer shape not found after text check.\")\n                    continue\n                slide_height = pres.slide_height\n                if hasattr(footer_shape, 'top') and footer_shape.top <= (slide_height * 0.6):\n                    # Allow if it replaced an empty placeholder at same index\n                    # Otherwise consider not in footer region\n                    structural_integrity_ok = False\n                    reason_details.append(f\"Slide {idx}: footer not positioned near bottom.\")\n                    continue\n            else:\n                # Footer existed originally (rare), ensure no change besides maybe whitespace\n                orig_footer_count = sum(1 for t in orig_texts if t == required_text)\n                mod_footer_count = sum(1 for t in mod_texts if t == required_text)\n                if mod_footer_count != orig_footer_count:\n                    structural_integrity_ok = False\n                    reason_details.append(f\"Slide {idx}: footer count changed from {orig_footer_count} to {mod_footer_count}.\")\n                    continue\n        \n        # Check each slide for extraneous footer additions\n        for slide in pres.slides:\n            has_only_required_footer = True\n            footer_count = 0\n            \n            # Check all text shapes on the slide\n            for shape in slide.shapes:\n                if hasattr(shape, 'text') and shape.text.strip():\n                    # Check if this is a footer-like text (bottom positioned or contains footer text)\n                    shape_text = shape.text.strip()\n                    \n                    # If it contains our required text, count it\n                    if required_text in shape_text:\n                        footer_count += 1\n                        \n                        # Check if it contains ONLY our required text (no extra words)\n                        if shape_text != required_text:\n                            # Check if it's just minor formatting differences\n                            cleaned_text = shape_text.replace('\\n', '').replace('\\r', '').strip()\n                            if cleaned_text != required_text:\n                                has_only_required_footer = False\n                                break\n                    \n                    # Check for other footer-like text at bottom of slide\n                    elif hasattr(shape, 'top') and shape.top:\n                        slide_height = pres.slide_height\n                        # If text is in bottom 20% of slide, it might be an unwanted footer\n                        if shape.top > (slide_height * 0.8):\n                            # This could be an extra footer\n                            has_only_required_footer = False\n                            break\n            \n            # Should have exactly one footer with our text\n            if has_only_required_footer and footer_count == 1:\n                slides_with_only_required_footer += 1\n            elif footer_count == 0:\n                # No footer found at all\n                has_only_required_footer = False\n            elif footer_count > 1:\n                # Multiple footers found\n                has_only_required_footer = False\n        \n        # Method 2: XML verification for extra text\n        try:\n            with zipfile.ZipFile(mod_path, 'r') as pptx_zip:\n                with zipfile.ZipFile(orig_path, 'r') as orig_zip:\n                    xml_slides_clean = 0\n                \n                    for i in range(1, total_slides + 1):\n                        slide_xml_path = f'ppt/slides/slide{i}.xml'\n                        try:\n                            with pptx_zip.open(slide_xml_path) as slide_file, orig_zip.open(slide_xml_path) as orig_slide_file:\n                                slide_content = slide_file.read().decode('utf-8')\n                                orig_content = orig_slide_file.read().decode('utf-8')\n                                \n                                # Count occurrences of our required text\n                                required_text_count = slide_content.count(required_text)\n                                \n                                # Should appear exactly once (in one text element)\n                                if required_text_count == 1:\n                                    xml_slides_clean += 1\n                                elif required_text_count > 1:\n                                    # Multiple occurrences might indicate duplicated footers\n                                    pass  # Slides not counted as clean\n                                \n                                # # Transition / animation integrity: ensure no new <p:transition ...> or <p:timing> added\n                                # def extract_tags(xml: str, tag: str) -> list[str]:\n                                #     return re.findall(fr'<p:{tag}[^>]*>', xml)\n                                # mod_transitions = extract_tags(slide_content, 'transition')\n                                # orig_transitions = extract_tags(orig_content, 'transition')\n                                # if mod_transitions != orig_transitions:\n                                #     structural_integrity_ok = False\n                                #     reason_details.append(f\"Slide {i}: transition changed. Before: {orig_transitions}, After: {mod_transitions}\")\n                                # Animations / timing\n                                mod_timing = extract_tags(slide_content, 'timing')\n                                orig_timing = extract_tags(orig_content, 'timing')\n                                if mod_timing != orig_timing:\n                                    structural_integrity_ok = False\n                                    reason_details.append(f\"Slide {i}: animation timing changed.\")\n                        \n                        except KeyError:\n                            continue\n                \n                # Use the stricter check (lower count)\n                slides_with_only_required_footer = min(slides_with_only_required_footer, xml_slides_clean)\n        \n        except Exception:\n            # Fall back to python-pptx method\n            pass\n        \n        # Final decision: require both clean footer and structural integrity\n        if not structural_integrity_ok:\n            msg = \"Non-footer changes detected: \" + \"; \".join(reason_details[:5])\n            return msg, 0.0\n        # Score based on clean footer implementation\n        if slides_with_only_required_footer == total_slides:\n            return f\"Verified only footer added; all {total_slides} slides clean.\", 1.0\n        elif slides_with_only_required_footer > 0:\n            score = slides_with_only_required_footer / total_slides\n            return f\"Partial footer correctness ({slides_with_only_required_footer}/{total_slides}) with no other changes.\", score\n        else:\n            return \"Footer implementation incorrect or missing, despite no other changes.\", 0.0\n            \n    except Exception as e:\n        return f\"Error during extraneous footer verification: {str(e)[:100]}\", 0.0\n"
            }
          }
        ]
      }
    ]
  },
  "metadata": {
    "task": "Add a footer to every slide with the text \"Fundamental Accounting Concepts\", and ensure that the footer is center-aligned"
  }
}