<!-- CURSOR_SUMMARY --> > [!NOTE] > **Medium Risk** > Touches authentication/login flow, credentials-file permissions, and subprocess lifecycle behavior across platforms; while mostly defensive, regressions could impact login or process management on Windows/macOS runners. > > **Overview** > Improves Windows/cross-platform reliability across the CLI and MCP server: OAuth login now binds the callback server to `127.0.0.1`, avoids slow loopback reverse-DNS, adds a configurable callback timeout (`--timeout` + env default), and opens URLs via a Windows-friendly `_open_browser` to avoid flashing console windows. > > Centralizes CLI output via a shared `console` that forces UTF-8 on Windows, standardizes UTF-8 file reads/writes throughout, tightens credentials-file permissions on Windows using `icacls`, and adds shared Windows subprocess helpers for **no-window** process creation and graceful termination (used by `deploy`, MCP reload, and usage-tracking worker). > > Updates client configuration UX/robustness (Windows AppData resolution via `platformdirs`, Cursor config path fallbacks + compatibility writes, overwrite warnings, absolute `uv` path for GUI clients, safer path display) and improves `deploy` child-process handling to avoid pipe-buffer deadlocks while giving better debug-aware error messages. > > Expands CI to run tests on Linux/Windows/macOS, adds a no-auth CLI integration workflow, disables usage tracking in toolkits CI, and adds extensive regression tests for Windows signals, subprocess cleanup, UTF-8, and config-path edge cases; bumps `arcade-core` to `4.4.2` and `arcade-mcp-server` to `1.17.2` (with updated dependency pin). > > <sup>Written by [Cursor Bugbot](https://cursor.com/dashboard?tab=bugbot) for commit 0fabd8ca1cd647039ba6ddbdf3f7809c330bab9e. This will update automatically on new commits. Configure [here](https://cursor.com/dashboard?tab=bugbot).</sup> <!-- /CURSOR_SUMMARY -->
620 lines
19 KiB
Python
620 lines
19 KiB
Python
import tempfile
|
|
from pathlib import Path
|
|
from unittest.mock import Mock
|
|
|
|
import pytest
|
|
from arcade_cli.display import display_eval_results
|
|
|
|
# test_display exercises eval-formatting paths that depend on optional extras.
|
|
# Skip cleanly when those extras are not installed in the active environment.
|
|
pytest.importorskip("openai")
|
|
pytest.importorskip("pytz")
|
|
pytest.importorskip("numpy")
|
|
pytest.importorskip("scipy")
|
|
from arcade_evals.eval import EvaluationResult
|
|
|
|
# Mark all tests in this module as requiring evals dependencies
|
|
pytestmark = pytest.mark.evals
|
|
|
|
|
|
def create_mock_evaluation_result(passed: bool, warning: bool, score: float) -> Mock:
|
|
"""Create a mock EvaluationResult with the specified properties."""
|
|
evaluation = Mock(spec=EvaluationResult)
|
|
evaluation.passed = passed
|
|
evaluation.warning = warning
|
|
evaluation.score = score
|
|
evaluation.failure_reason = None
|
|
evaluation.results = []
|
|
return evaluation
|
|
|
|
|
|
def test_display_eval_results_normal() -> None:
|
|
"""Test normal display without filtering."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case 1",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.95
|
|
),
|
|
},
|
|
{
|
|
"name": "Test Case 2",
|
|
"input": "Test input 2",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.5
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Should not raise any exceptions
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_with_failed_only() -> None:
|
|
"""Test display with failed_only flag and original counts."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Original counts: 3 total, 1 passed, 1 failed, 1 warned
|
|
original_counts = (3, 1, 1, 1)
|
|
|
|
# Should not raise any exceptions
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
failed_only=True,
|
|
original_counts=original_counts,
|
|
)
|
|
|
|
|
|
def test_display_eval_results_with_output_file() -> None:
|
|
"""Test display with output file."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.9
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "test_output.txt"
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
output_file=str(output_file),
|
|
output_formats=["txt"],
|
|
)
|
|
|
|
# Verify file was created
|
|
assert output_file.exists()
|
|
|
|
# Verify file contains some expected content
|
|
content = output_file.read_text(encoding="utf-8")
|
|
assert "Model:" in content or "gpt-4o" in content
|
|
|
|
|
|
def test_display_eval_results_with_output_file_and_failed_only() -> None:
|
|
"""Test display with both output file and failed_only flag."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.2
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
original_counts = (5, 3, 1, 1)
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "test_output.txt"
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
output_file=str(output_file),
|
|
failed_only=True,
|
|
original_counts=original_counts,
|
|
output_formats=["txt"],
|
|
)
|
|
|
|
# Verify file was created
|
|
assert output_file.exists()
|
|
|
|
# Verify file contains disclaimer and summary
|
|
content = output_file.read_text(encoding="utf-8")
|
|
assert "failed-only" in content.lower() or "failed evaluation" in content.lower()
|
|
assert "Total: 5" in content # Should show original total
|
|
|
|
|
|
def test_display_eval_results_creates_parent_directories() -> None:
|
|
"""Test that output file creates parent directories if they don't exist."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.9
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "nested" / "path" / "test_output.txt"
|
|
|
|
# Parent directories don't exist yet
|
|
assert not output_file.parent.exists()
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
output_file=str(output_file),
|
|
output_formats=["txt"],
|
|
)
|
|
|
|
# Parent directories should be created
|
|
assert output_file.parent.exists()
|
|
assert output_file.exists()
|
|
|
|
|
|
def test_display_eval_results_with_warnings() -> None:
|
|
"""Test display with cases that have warnings."""
|
|
results: list = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Warning Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=True, score=0.85
|
|
),
|
|
},
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Should not raise any exceptions
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_empty_results() -> None:
|
|
"""Test display with empty results."""
|
|
results: list = []
|
|
|
|
# Should not raise any exceptions
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_with_details() -> None:
|
|
"""Test display with show_details=True."""
|
|
evaluation = create_mock_evaluation_result(passed=True, warning=False, score=0.95)
|
|
evaluation.results = [
|
|
{
|
|
"field": "test_field",
|
|
"match": True,
|
|
"score": 1.0,
|
|
"weight": 1.0,
|
|
"expected": "expected_value",
|
|
"actual": "actual_value",
|
|
"is_criticized": True,
|
|
}
|
|
]
|
|
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case",
|
|
"input": "Test input",
|
|
"evaluation": evaluation,
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Should not raise any exceptions
|
|
display_eval_results(results, show_details=True)
|
|
|
|
|
|
def test_display_eval_results_with_failed_only_no_warnings() -> None:
|
|
"""Test display with failed_only but original counts have no warnings."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Original counts: 10 total, 8 passed, 2 failed, 0 warned
|
|
original_counts = (10, 8, 2, 0)
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
failed_only=True,
|
|
original_counts=original_counts,
|
|
)
|
|
|
|
|
|
def test_display_eval_results_with_failed_only_no_failed() -> None:
|
|
"""Test display with failed_only but original counts have no failed."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Original counts: 5 total, 5 passed, 0 failed, 0 warned (edge case)
|
|
original_counts = (5, 5, 0, 0)
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
failed_only=True,
|
|
original_counts=original_counts,
|
|
)
|
|
|
|
|
|
def test_display_eval_results_multiple_suites() -> None:
|
|
"""Test display with multiple eval suites."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric 1",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case 1",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.95
|
|
),
|
|
},
|
|
],
|
|
}
|
|
],
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric 2",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case 2",
|
|
"input": "Test input 2",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.5
|
|
),
|
|
},
|
|
],
|
|
}
|
|
],
|
|
]
|
|
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_multiple_models() -> None:
|
|
"""Test display with multiple models in same suite."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case 1",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.95
|
|
),
|
|
},
|
|
],
|
|
},
|
|
{
|
|
"model": "gpt-3.5-turbo",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case 2",
|
|
"input": "Test input 2",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.5
|
|
),
|
|
},
|
|
],
|
|
},
|
|
]
|
|
]
|
|
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_summary_with_warnings() -> None:
|
|
"""Test summary display when warnings are present."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Passed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.95
|
|
),
|
|
},
|
|
{
|
|
"name": "Warning Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=True, score=0.85
|
|
),
|
|
},
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_summary_only_passed() -> None:
|
|
"""Test summary when all cases passed."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Passed Case 1",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.95
|
|
),
|
|
},
|
|
{
|
|
"name": "Passed Case 2",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.98
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
display_eval_results(results, show_details=False)
|
|
|
|
|
|
def test_display_eval_results_failed_only_with_warnings_in_summary() -> None:
|
|
"""Test failed_only display when original counts include warnings."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Failed Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=False, warning=False, score=0.3
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
# Original counts: 10 total, 7 passed, 2 failed, 1 warned
|
|
original_counts = (10, 7, 2, 1)
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "test_output.txt"
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
output_file=str(output_file),
|
|
failed_only=True,
|
|
original_counts=original_counts,
|
|
output_formats=["txt"],
|
|
)
|
|
|
|
content = output_file.read_text(encoding="utf-8")
|
|
# Should show warnings in summary
|
|
assert "Warnings: 1" in content or "Warnings" in content
|
|
|
|
|
|
def test_display_eval_results_with_details_and_output() -> None:
|
|
"""Test display with details and output file."""
|
|
evaluation = create_mock_evaluation_result(passed=True, warning=False, score=0.95)
|
|
evaluation.results = [
|
|
{
|
|
"field": "test_field",
|
|
"match": True,
|
|
"score": 1.0,
|
|
"weight": 1.0,
|
|
"expected": "expected_value",
|
|
"actual": "actual_value",
|
|
"is_criticized": True,
|
|
}
|
|
]
|
|
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case",
|
|
"input": "Test input",
|
|
"evaluation": evaluation,
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "test_output.txt"
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=True,
|
|
output_file=str(output_file),
|
|
output_formats=["txt"],
|
|
)
|
|
|
|
assert output_file.exists()
|
|
content = output_file.read_text(encoding="utf-8")
|
|
assert "User Input:" in content
|
|
assert "Details:" in content
|
|
|
|
|
|
def test_display_eval_results_multi_format_output() -> None:
|
|
"""Test display with multiple output formats."""
|
|
results = [
|
|
[
|
|
{
|
|
"model": "gpt-4o",
|
|
"rubric": "Test Rubric",
|
|
"cases": [
|
|
{
|
|
"name": "Test Case",
|
|
"input": "Test input",
|
|
"evaluation": create_mock_evaluation_result(
|
|
passed=True, warning=False, score=0.9
|
|
),
|
|
},
|
|
],
|
|
}
|
|
]
|
|
]
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
output_file = Path(tmpdir) / "results"
|
|
|
|
display_eval_results(
|
|
results,
|
|
show_details=False,
|
|
output_file=str(output_file),
|
|
output_formats=["txt", "md", "html"],
|
|
)
|
|
|
|
# Verify all three files were created
|
|
assert (Path(tmpdir) / "results.txt").exists()
|
|
assert (Path(tmpdir) / "results.md").exists()
|
|
assert (Path(tmpdir) / "results.html").exists()
|
|
|
|
# Verify each file has appropriate content
|
|
txt_content = (Path(tmpdir) / "results.txt").read_text(encoding="utf-8")
|
|
assert "Test Case" in txt_content
|
|
|
|
md_content = (Path(tmpdir) / "results.md").read_text(encoding="utf-8")
|
|
assert "# " in md_content # Markdown header
|
|
|
|
html_content = (Path(tmpdir) / "results.html").read_text(encoding="utf-8")
|
|
assert "<html" in html_content
|