import os import sys sys.path.insert( 0, os.path.abspath("../..") ) # Adds the parent directory to the system path from litellm_proxy_extras.utils import ProxyExtrasDBManager def test_custom_prisma_dir(monkeypatch): import tempfile # create a temp directory temp_dir = tempfile.mkdtemp() monkeypatch.setenv("LITELLM_MIGRATION_DIR", temp_dir) ## Check if the prisma dir is the temp directory assert ProxyExtrasDBManager._get_prisma_dir() == temp_dir ## Check if the schema.prisma file is in the temp directory schema_path = os.path.join(temp_dir, "schema.prisma") assert os.path.exists(schema_path) ## Check if the migrations dir is in the temp directory migrations_dir = os.path.join(temp_dir, "migrations") assert os.path.exists(migrations_dir) class TestPermissionErrorDetection: """Test cases for permission error detection in Prisma migrations""" def test_is_permission_error_postgres_42501(self): """Test detection of PostgreSQL 42501 error code (insufficient privilege)""" error_message = "Database error code: 42501 - permission denied for table users" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_must_be_owner(self): """Test detection of 'must be owner of table' error""" error_message = "ERROR: must be owner of table my_table" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_permission_denied_schema(self): """Test detection of 'permission denied for schema' error""" error_message = "permission denied for schema public" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_permission_denied_table(self): """Test detection of 'permission denied for table' error""" error_message = "permission denied for table my_table" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_must_be_owner_schema(self): """Test detection of 'must be owner of schema' error""" error_message = "must be owner of schema public" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_case_insensitive(self): """Test that permission error detection is case insensitive""" error_message = "PERMISSION DENIED FOR TABLE my_table" assert ProxyExtrasDBManager._is_permission_error(error_message) is True def test_is_permission_error_negative(self): """Test that non-permission errors are not detected as permission errors""" error_message = "column 'id' already exists" assert ProxyExtrasDBManager._is_permission_error(error_message) is False class TestIdempotentErrorDetection: """Test cases for idempotent error detection in Prisma migrations""" def test_is_idempotent_error_already_exists(self): """Test detection of generic 'already exists' error""" error_message = "object already exists" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_column_already_exists(self): """Test detection of 'column already exists' error""" error_message = "column 'email' already exists" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_duplicate_key(self): """Test detection of duplicate key violation error""" error_message = "duplicate key value violates unique constraint" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_relation_already_exists(self): """Test detection of 'relation already exists' error""" error_message = "relation 'users_pkey' already exists" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_constraint_already_exists(self): """Test detection of 'constraint already exists' error""" error_message = "constraint 'fk_user_id' already exists" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_case_insensitive(self): """Test that idempotent error detection is case insensitive""" error_message = "COLUMN 'ID' ALREADY EXISTS" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True def test_is_idempotent_error_negative(self): """Test that non-idempotent errors are not detected as idempotent errors""" error_message = "Database error code: 42501 - permission denied" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is False class TestErrorClassificationPriority: """Test cases to ensure errors are correctly classified""" def test_permission_error_not_classified_as_idempotent(self): """Ensure permission errors are not mistakenly classified as idempotent""" error_message = "Database error code: 42501 - must be owner of table users" assert ProxyExtrasDBManager._is_permission_error(error_message) is True assert ProxyExtrasDBManager._is_idempotent_error(error_message) is False def test_idempotent_error_not_classified_as_permission(self): """Ensure idempotent errors are not mistakenly classified as permission errors""" error_message = "column 'created_at' already exists" assert ProxyExtrasDBManager._is_idempotent_error(error_message) is True assert ProxyExtrasDBManager._is_permission_error(error_message) is False def test_unknown_error_classified_as_neither(self): """Ensure unknown errors are classified as neither permission nor idempotent""" error_message = "connection timeout" assert ProxyExtrasDBManager._is_permission_error(error_message) is False assert ProxyExtrasDBManager._is_idempotent_error(error_message) is False