""" Unit tests for dataset.py module. Tests functions for downloading and extracting the SkillScope dataset. """ import pytest from pathlib import Path import tempfile import zipfile import sqlite3 from unittest.mock import patch, MagicMock from hopcroft_skill_classification_tool_competition.dataset import ( download_skillscope_dataset, ) @pytest.mark.unit class TestDatasetDownload: """Unit tests for dataset download functionality.""" def test_download_returns_path(self): """Test that download function returns a Path object.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) # Mock the actual download to avoid network calls with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create a mock zip file zip_path = output_dir / "skillscope_data.zip" db_path = output_dir / "skillscope_data.db" # Create a dummy database conn = sqlite3.connect(db_path) conn.execute("CREATE TABLE test (id INTEGER)") conn.close() # Create a zip of the database with zipfile.ZipFile(zip_path, 'w') as zf: zf.write(db_path, arcname='skillscope_data.db') # Remove the original database db_path.unlink() # Mock download to return the zip path mock_download.return_value = str(zip_path) result = download_skillscope_dataset(output_dir) assert isinstance(result, Path) assert result.exists() assert result.name == "skillscope_data.db" def test_download_creates_directory(self): """Test that download creates output directory if it doesn't exist.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) / "nonexistent" / "nested" / "dir" assert not output_dir.exists() with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create mock database temp_db = Path(tmpdir) / "skillscope_data.db" conn = sqlite3.connect(temp_db) conn.execute("CREATE TABLE test (id INTEGER)") conn.close() # Create zip zip_path = Path(tmpdir) / "skillscope_data.zip" with zipfile.ZipFile(zip_path, 'w') as zf: zf.write(temp_db, arcname='skillscope_data.db') mock_download.return_value = str(zip_path) download_skillscope_dataset(output_dir) assert output_dir.exists() def test_download_skips_if_exists(self): """Test that download is skipped if database already exists.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) db_path = output_dir / "skillscope_data.db" # Create existing database conn = sqlite3.connect(db_path) conn.execute("CREATE TABLE test (id INTEGER)") conn.close() with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: result = download_skillscope_dataset(output_dir) # Should not call download if file exists mock_download.assert_not_called() assert result == db_path def test_download_extracts_zip(self): """Test that zip file is properly extracted.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create database and zip it temp_db = Path(tmpdir) / "temp_skillscope_data.db" conn = sqlite3.connect(temp_db) conn.execute("CREATE TABLE nlbse_tool_competition_data_by_issue (id INTEGER)") conn.close() zip_path = output_dir / "skillscope_data.zip" with zipfile.ZipFile(zip_path, 'w') as zf: zf.write(temp_db, arcname='skillscope_data.db') temp_db.unlink() mock_download.return_value = str(zip_path) result = download_skillscope_dataset(output_dir) # Check database was extracted assert result.exists() # Verify it's a valid SQLite database conn = sqlite3.connect(result) cursor = conn.cursor() cursor.execute("SELECT name FROM sqlite_master WHERE type='table'") tables = cursor.fetchall() conn.close() assert len(tables) > 0 def test_download_cleans_up_zip(self): """Test that zip file is deleted after extraction.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create database and zip temp_db = Path(tmpdir) / "temp_db.db" conn = sqlite3.connect(temp_db) conn.execute("CREATE TABLE test (id INTEGER)") conn.close() zip_path = output_dir / "skillscope_data.zip" with zipfile.ZipFile(zip_path, 'w') as zf: zf.write(temp_db, arcname='skillscope_data.db') temp_db.unlink() mock_download.return_value = str(zip_path) download_skillscope_dataset(output_dir) # Zip should be deleted assert not zip_path.exists() def test_download_raises_on_missing_database(self): """Test that error is raised if database not in zip.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create zip without database file zip_path = output_dir / "skillscope_data.zip" with zipfile.ZipFile(zip_path, 'w') as zf: zf.writestr('dummy.txt', 'dummy content') mock_download.return_value = str(zip_path) with pytest.raises(FileNotFoundError): download_skillscope_dataset(output_dir) @pytest.mark.unit class TestDatasetEdgeCases: """Unit tests for edge cases in dataset handling.""" def test_download_with_none_output_dir(self): """Test download with None as output directory (should use default).""" with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: with patch('hopcroft_skill_classification_tool_competition.dataset.RAW_DATA_DIR') as mock_raw_dir: # Mock the default directory with tempfile.TemporaryDirectory() as tmpdir: mock_raw_dir.__truediv__ = MagicMock(return_value=Path(tmpdir) / "skillscope_data.db") # Create existing database to skip download db_path = Path(tmpdir) / "skillscope_data.db" db_path.parent.mkdir(parents=True, exist_ok=True) conn = sqlite3.connect(db_path) conn.execute("CREATE TABLE test (id INTEGER)") conn.close() # This should use default RAW_DATA_DIR result = download_skillscope_dataset(None) assert isinstance(result, Path) def test_download_handles_permission_error(self): """Test handling of permission errors during file operations.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # This test is platform-dependent, so we'll just verify it doesn't crash # with invalid permissions pass # Placeholder for permission tests @pytest.mark.unit class TestDatasetIntegration: """Integration-like tests for dataset module (still unit-scoped).""" def test_download_produces_valid_sqlite_database(self): """Test that downloaded file is a valid SQLite database.""" with tempfile.TemporaryDirectory() as tmpdir: output_dir = Path(tmpdir) with patch('hopcroft_skill_classification_tool_competition.dataset.hf_hub_download') as mock_download: # Create a proper database structure temp_db = Path(tmpdir) / "temp.db" conn = sqlite3.connect(temp_db) conn.execute(""" CREATE TABLE nlbse_tool_competition_data_by_issue ( id INTEGER PRIMARY KEY, repo_name TEXT, pr_number INTEGER ) """) conn.execute(""" INSERT INTO nlbse_tool_competition_data_by_issue VALUES (1, 'test_repo', 123) """) conn.commit() conn.close() # Create zip zip_path = output_dir / "skillscope_data.zip" with zipfile.ZipFile(zip_path, 'w') as zf: zf.write(temp_db, arcname='skillscope_data.db') temp_db.unlink() mock_download.return_value = str(zip_path) result = download_skillscope_dataset(output_dir) # Verify database is valid and queryable conn = sqlite3.connect(result) cursor = conn.cursor() cursor.execute("SELECT * FROM nlbse_tool_competition_data_by_issue") rows = cursor.fetchall() conn.close() assert len(rows) == 1 assert rows[0][1] == 'test_repo' assert rows[0][2] == 123