ref:Cleaned the Structure

This commit is contained in:
2025-11-07 14:20:22 +05:30
parent d7dbcff4b6
commit 1b622b6a19
42 changed files with 2164 additions and 806 deletions

142
api/routers/clean.py Normal file
View File

@@ -0,0 +1,142 @@
"""
Data Cleaning Router
Handles PII detection and anonymization endpoints
"""
from fastapi import APIRouter, File, UploadFile, HTTPException
from fastapi.responses import JSONResponse
import pandas as pd
import numpy as np
import io
import os
from datetime import datetime
from typing import Dict, Any
# Import cleaning module
import sys
sys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))
from data_cleaning import DataCleaner
router = APIRouter()
def convert_to_serializable(obj):
"""Convert numpy/pandas types to native Python types for JSON serialization"""
if isinstance(obj, (np.integer, np.int64, np.int32)):
return int(obj)
elif isinstance(obj, (np.floating, np.float64, np.float32)):
return float(obj)
elif isinstance(obj, np.ndarray):
return obj.tolist()
elif isinstance(obj, dict):
return {key: convert_to_serializable(value) for key, value in obj.items()}
elif isinstance(obj, list):
return [convert_to_serializable(item) for item in obj]
return obj
@router.post("/clean")
async def clean_dataset(file: UploadFile = File(...)):
"""
Clean uploaded dataset - detect and anonymize PII
- **file**: CSV file to clean
Returns:
- Cleaned dataset statistics
- PII detections and anonymization actions
- Report file path for download
- Cleaned CSV file path for download
"""
# Validate file type
if not file.filename.endswith('.csv'):
raise HTTPException(status_code=400, detail="Only CSV files are supported")
try:
# Read uploaded file
contents = await file.read()
df = pd.read_csv(io.BytesIO(contents))
if df.empty:
raise HTTPException(status_code=400, detail="Uploaded file is empty")
# Initialize Data Cleaner (with GPU if available)
print(f"Cleaning dataset: {file.filename} ({len(df)} rows, {len(df.columns)} columns)")
cleaner = DataCleaner(df, use_gpu=True)
# Run cleaning (non-interactive mode for API)
cleaned_df, audit_report = cleaner.clean(
risky_features=None, # Auto-detect
interactive=False, # No user prompts in API mode
scan_all_cells=True
)
# Generate filenames
timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
safe_filename = file.filename.replace('.csv', '')
# Save cleaned CSV
cleaned_csv_filename = f"cleaned_{safe_filename}_{timestamp}.csv"
cleaned_csv_path = os.path.join("reports", cleaned_csv_filename)
full_cleaned_csv_path = os.path.join(
os.path.dirname(os.path.dirname(os.path.dirname(__file__))),
cleaned_csv_path
)
cleaner.save_cleaned_data(cleaned_df, full_cleaned_csv_path)
# Save audit report
audit_report_filename = f"cleaning_audit_{safe_filename}_{timestamp}.json"
audit_report_path = os.path.join("reports", audit_report_filename)
full_audit_report_path = os.path.join(
os.path.dirname(os.path.dirname(os.path.dirname(__file__))),
audit_report_path
)
cleaner.save_audit_report(audit_report, full_audit_report_path)
# Prepare response
response_data = {
"status": "success",
"filename": file.filename,
"dataset_info": {
"original_rows": int(audit_report["metadata"]["original_rows"]),
"original_columns": int(audit_report["metadata"]["original_columns"]),
"cleaned_rows": int(audit_report["metadata"]["cleaned_rows"]),
"cleaned_columns": int(audit_report["metadata"]["cleaned_columns"])
},
"gpu_acceleration": audit_report["metadata"].get("gpu_acceleration", {
"enabled": False,
"device": "CPU"
}),
"summary": {
"columns_removed": audit_report["summary"]["columns_removed"],
"columns_anonymized": audit_report["summary"]["columns_anonymized"],
"total_cells_affected": int(audit_report["summary"]["total_cells_affected"])
},
"pii_detections": {
col: {
"action": details["action"],
"entity_types": details["entity_types_found"],
"num_affected_rows": int(details.get("num_affected_rows", 0)),
"examples": details.get("examples", [])[:2] # Show 2 examples
}
for col, details in audit_report["details"].items()
},
"gdpr_compliance": audit_report["compliance"]["gdpr_articles_applied"],
"files": {
"cleaned_csv": f"/{cleaned_csv_path}",
"audit_report": f"/{audit_report_path}"
},
"timestamp": datetime.now().isoformat()
}
# Convert all numpy/pandas types to native Python types
response_data = convert_to_serializable(response_data)
return JSONResponse(content=response_data)
except pd.errors.EmptyDataError:
raise HTTPException(status_code=400, detail="File is empty or invalid CSV format")
except Exception as e:
print(f"Error during cleaning: {str(e)}")
raise HTTPException(status_code=500, detail=f"Cleaning failed: {str(e)}")