-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add eval docs and update eval code (#21)
* add eval docs and adjust code * [pre-commit.ci] Add auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * create eval subgroup for dependencies, update docs --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
- Loading branch information
1 parent
7a02a47
commit 219bd64
Showing
6 changed files
with
2,000 additions
and
80 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,282 @@ | ||
Safety and Evaluation | ||
===================== | ||
|
||
Overview | ||
-------- | ||
|
||
The Health Recommendation System employs a rigorous evaluation framework to ensure safe, accurate, and relevant service recommendations. This evaluation is critical for: | ||
|
||
- Ensuring system safety when handling emergency situations | ||
- Maintaining accuracy in service recommendations | ||
- Identifying and filtering out-of-scope requests | ||
- Validating the system's ability to handle queries with varying levels of detail | ||
- Measuring retrieval accuracy of relevant services | ||
|
||
Installation for Evaluation | ||
---------------------------- | ||
|
||
To run the evaluation pipeline, you'll need to install additional dependencies from the `eval` subgroup. From the project root: | ||
|
||
.. code-block:: bash | ||
# Create virtual environment | ||
python3 -m venv .venv | ||
source .venv/bin/activate # On Windows: .venv\Scripts\activate | ||
# Install dependencies including evaluation packages | ||
poetry install --with test,docs,eval | ||
Evaluation Framework | ||
-------------------- | ||
|
||
Synthetic Dataset Generation | ||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
The evaluation utilizes a synthetic dataset of 1000 queries, carefully structured to test different aspects of the system: | ||
|
||
- **Regular Situations (80%)**: Divided equally among three levels of detail: | ||
|
||
- Low: 1-2 sentences, brief queries with minimal context | ||
- Medium: 2-3 sentences with moderate background information | ||
- High: Detailed paragraph with extensive context | ||
|
||
- **Emergency Situations (10%)**: Queries indicating urgent medical or mental health needs | ||
|
||
- **Out-of-Scope Situations (10%)**: Queries for services not covered by the system | ||
|
||
Each query in the dataset includes: | ||
|
||
.. code-block:: javascript | ||
{ | ||
"query": "User's query text", | ||
"context": ["service_id1", "service_id2"], | ||
"answer": "Expected answer text", | ||
"is_emergency": "<boolean>", // true or false | ||
"is_out_of_scope": "<boolean>", // true or false | ||
"demographics": { | ||
"Age": "<string>", // one of: child, teen, young adult, adult, senior | ||
"Gender": "<string>", // one of: male, female, non-binary, N/A | ||
"Ethnicity": "<string>", // one of: Caucasian, African, Asian, Hispanic, Indigenous, Other, N/A | ||
"Employment status": "<string>", // one of: employed, unemployed, student, retired, unable to work, N/A | ||
"Housing situation": "<string>", // one of: own home, renting, homeless, shelter, assisted living, N/A | ||
"Disability status": "<string>", // one of: no disability, physical disability, cognitive disability, N/A | ||
"Immigration status": "<string>" // one of: citizen, permanent resident, temporary resident, refugee, undocumented, N/A | ||
}, | ||
"detail_level": "<string>" // one of: low, medium, high | ||
} | ||
Example Scenarios | ||
^^^^^^^^^^^^^^^^^ | ||
|
||
Regular Situation (Low Detail) | ||
"""""""""""""""""""""""""""""" | ||
|
||
.. code-block:: json | ||
{ | ||
"query": "I'm looking for help with my child's eating disorder. What services are available?", | ||
"context": ["69796102"], | ||
"answer": "The North York General Hospital offers a Child and Adolescent Eating Disorders Program, which includes outpatient services for assessment, diagnosis, and treatment, as well as a day program with medical, nutritional, and mental health care support.", | ||
"is_emergency": false, | ||
"is_out_of_scope": false, | ||
"demographics": { | ||
"Age": "child", | ||
"Gender": "N/A", | ||
"Ethnicity": "N/A", | ||
"Employment status": "N/A", | ||
"Housing situation": "N/A", | ||
"Disability status": "N/A", | ||
"Immigration status": "N/A" | ||
}, | ||
"detail_level": "low" | ||
} | ||
Emergency Situation | ||
""""""""""""""""""" | ||
|
||
.. code-block:: json | ||
{ | ||
"query": "My child is experiencing severe abdominal pain and I am very concerned. What should I do?", | ||
"context": ["69796097", "69795331"], | ||
"answer": "You should take your child to the nearest pediatric emergency department immediately. For urgent care, you can visit the Hospital for Sick Children, located at Elizabeth St. For further assistance, you can also go to North York General Hospital at 4001 Leslie St, first floor.", | ||
"is_emergency": true, | ||
"is_out_of_scope": false, | ||
"demographics": { | ||
"Age": "child", | ||
"Gender": "N/A", | ||
"Ethnicity": "N/A", | ||
"Employment status": "N/A", | ||
"Housing situation": "N/A", | ||
"Disability status": "N/A", | ||
"Immigration status": "N/A" | ||
}, | ||
"detail_level": "medium" | ||
} | ||
Evaluation Scripts | ||
------------------ | ||
|
||
The ``eval/`` directory contains scripts for both dataset generation and evaluation: | ||
|
||
Dataset Generation | ||
^^^^^^^^^^^^^^^^^^ | ||
|
||
.. code-block:: bash | ||
# Generate synthetic dataset | ||
python eval/generate_dataset.py \ | ||
--input_file data/211_data.csv \ | ||
--output_dir ./eval \ | ||
--name synthetic_dataset \ | ||
--num_samples 1000 \ | ||
--situation_type [regular|emergency|out_of_scope] \ | ||
--detail_level [low|medium|high] | ||
# Generate full dataset with distribution | ||
./eval/generate_large_dataset.sh | ||
System Output Collection | ||
^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
.. code-block:: bash | ||
# Collect RAG system outputs for evaluation | ||
python eval/collect_rag_outputs.py \ | ||
--input path/to/synthetic_dataset.json \ | ||
--output path/to/processed_results.json \ | ||
--batch-size 5 | ||
RAGAS Evaluation | ||
^^^^^^^^^^^^^^^^ | ||
|
||
.. code-block:: bash | ||
# Evaluate full RAG pipeline | ||
python eval/evaluate.py \ | ||
--input path/to/processed_results.json \ | ||
--output-dir ./evaluation_results | ||
Performance Metrics | ||
------------------- | ||
|
||
RAGAS Metrics By Category | ||
^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
Note: These metrics were obtained using a synthetic dataset specifically generated from services in the Greater Toronto Area (GTA). | ||
The RAG system evaluated used specialized prompts that differ marginally from the current system. | ||
|
||
.. list-table:: | ||
:header-rows: 1 | ||
|
||
* - Subgroup | ||
- Category | ||
- Answer Relevancy | ||
- Faithfulness | ||
- Context Recall | ||
- Context Precision | ||
* - Detail Level | ||
- Low | ||
- 0.82 | ||
- 0.54 | ||
- 0.58 | ||
- 0.57 | ||
* - Detail Level | ||
- Medium | ||
- 0.72 | ||
- 0.47 | ||
- 0.49 | ||
- 0.31 | ||
* - Detail Level | ||
- High | ||
- 0.84 | ||
- 0.53 | ||
- 0.30 | ||
- 0.84 | ||
* - Is Emergency | ||
- True | ||
- 0.83 | ||
- 0.78 | ||
- 0.46 | ||
- 1.00 | ||
* - Is Out of Scope | ||
- True | ||
- 0.52 | ||
- - | ||
- - | ||
- - | ||
|
||
|
||
Retrieval Performance and Re-ranking Strategy | ||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
||
The system's retrieval performance provides a compelling case for implementing a re-ranking stage: | ||
|
||
|
||
.. list-table:: | ||
:header-rows: 1 | ||
|
||
* - Metric | ||
- acc@1 | ||
- acc@3 | ||
- acc@5 | ||
- acc@10 | ||
- acc@20 | ||
* - Overall | ||
- 0.34 | ||
- 0.47 | ||
- 0.55 | ||
- 0.67 | ||
- 0.74 | ||
* - High Detail | ||
- 0.31 | ||
- 0.44 | ||
- 0.53 | ||
- 0.63 | ||
- 0.74 | ||
* - Low Detail | ||
- 0.35 | ||
- 0.54 | ||
- 0.64 | ||
- 0.82 | ||
- 0.88 | ||
* - Emergency | ||
- 0.18 | ||
- 0.29 | ||
- 0.35 | ||
- 0.41 | ||
- 0.54 | ||
* - Out of Scope | ||
- 0.20 | ||
- 0.20 | ||
- 0.20 | ||
- 0.40 | ||
- 0.60 | ||
|
||
The metrics reveal several key insights that motivate the use of a re-ranking stage: | ||
|
||
1. **Wider Pool Contains Relevant Services**: The significant increase in accuracy from acc@5 (0.55) to acc@20 (0.74) indicates that relevant services are often being retrieved but ranked lower than optimal. This suggests that a more sophisticated ranking mechanism could improve the final recommendations. | ||
|
||
2. **Query Type Variations**: Performance varies notably across query types: | ||
- Low Detail queries achieve high acc@20 (0.88), suggesting simpler queries benefit from broader retrieval | ||
- Emergency queries show lower initial accuracy but steady improvement up to acc@20 (0.54), indicating relevant services are present but need better ranking | ||
- High Detail queries show consistent improvement up to acc@20 (0.74), suggesting additional context could help with ranking | ||
|
||
3. **Re-ranking Implementation**: Based on these metrics, the system implements an optional re-ranking stage (based on `RankGPT <https://arxiv.org/abs/2304.09542>`_) that can be enabled via the API's `rerank` parameter (see :http:post:`/recommend`). When enabled: | ||
- First stage: Retrieves top 20 candidates using efficient embedding-based similarity | ||
- Second stage: Applies GPT-4 based semantic analysis to re-rank these candidates | ||
- Returns the top 5 most relevant services after re-ranking | ||
|
||
To enable re-ranking in your API calls, simply set the `rerank` parameter to `true` in your request to the :http:post:`/recommend` endpoint: | ||
|
||
.. code-block:: json | ||
{ | ||
"query": "I need mental health support", | ||
"latitude": 43.6532, | ||
"longitude": -79.3832, | ||
"radius": 5000, | ||
"rerank": true | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
import argparse | ||
import asyncio | ||
import json | ||
import logging | ||
from typing import Dict, Any | ||
|
||
import aiohttp | ||
from tqdm.asyncio import tqdm_asyncio | ||
|
||
|
||
logging.basicConfig(level=logging.INFO) | ||
logger = logging.getLogger(__name__) | ||
|
||
|
||
async def fetch_recommendation( | ||
session: aiohttp.ClientSession, query: Dict[str, Any], endpoint: str | ||
) -> Dict[str, Any]: | ||
"""Fetch recommendation from the RAG system API.""" | ||
try: | ||
async with session.post( | ||
endpoint, | ||
json={"query": query["query"], "rerank": True}, | ||
) as response: | ||
if response.status == 200: | ||
result = await response.json() | ||
return { | ||
"query": query["query"], | ||
"answer": result["message"], | ||
"context": [ | ||
str(service["id"]) | ||
for service in (result.get("services", []) or []) | ||
], | ||
"ground_truth": query["context"], | ||
} | ||
else: | ||
logger.error(f"Error fetching recommendation: {response.status}") | ||
return None | ||
except Exception as e: | ||
logger.error(f"Exception during fetch: {e}") | ||
return None | ||
|
||
|
||
async def process_samples( | ||
samples_file: str, output_file: str, batch_size: int = 5 | ||
) -> None: | ||
"""Process samples in batches and save results.""" | ||
# Load samples | ||
with open(samples_file, "r") as f: | ||
samples = json.load(f) | ||
|
||
results = [] | ||
async with aiohttp.ClientSession() as session: | ||
# Process in batches | ||
for i in range(0, len(samples), batch_size): | ||
batch = samples[i : i + batch_size] | ||
tasks = [fetch_recommendation(session, query) for query in batch] | ||
batch_results = await tqdm_asyncio.gather(*tasks) | ||
results.extend([r for r in batch_results if r is not None]) | ||
|
||
# Save results | ||
with open(output_file, "w") as f: | ||
json.dump(results, f, indent=2) | ||
|
||
logger.info(f"Processed {len(results)} samples successfully") | ||
|
||
|
||
def main() -> None: | ||
"""Main function for the script.""" | ||
parser = argparse.ArgumentParser( | ||
description="Collect RAG system outputs for evaluation." | ||
) | ||
parser.add_argument( | ||
"--input", required=True, help="Path to the synthetic dataset JSON file" | ||
) | ||
parser.add_argument( | ||
"--output", required=True, help="Path to save the processed results JSON file" | ||
) | ||
parser.add_argument( | ||
"--batch-size", | ||
type=int, | ||
default=5, | ||
help="Number of queries to process in parallel", | ||
) | ||
parser.add_argument( | ||
"--endpoint", | ||
default="http://localhost:8005/recommend", | ||
help="URL endpoint for the RAG system API", | ||
) | ||
|
||
args = parser.parse_args() | ||
|
||
asyncio.run(process_samples(args.input, args.output, args.batch_size)) | ||
|
||
|
||
if __name__ == "__main__": | ||
main() |
Oops, something went wrong.