| """Tests for System User Protection in folder_paths.py |
| |
| Tests cover: |
| - get_system_user_directory(): Internal API for custom nodes to access System User directories |
| - get_public_user_directory(): HTTP endpoint access with System User blocking |
| - Backward compatibility: Existing APIs unchanged |
| - Security: Path traversal and injection prevention |
| """ |
|
|
| import pytest |
| import os |
| import tempfile |
|
|
| from folder_paths import ( |
| get_system_user_directory, |
| get_public_user_directory, |
| get_user_directory, |
| set_user_directory, |
| ) |
|
|
|
|
| @pytest.fixture(scope="module") |
| def mock_user_directory(): |
| """Create a temporary user directory for testing.""" |
| with tempfile.TemporaryDirectory() as temp_dir: |
| original_dir = get_user_directory() |
| set_user_directory(temp_dir) |
| yield temp_dir |
| set_user_directory(original_dir) |
|
|
|
|
| class TestGetSystemUserDirectory: |
| """Tests for get_system_user_directory() - internal API for System User directories. |
| |
| Verifies: |
| - Custom nodes can access System User directories via internal API |
| - Input validation prevents path traversal attacks |
| """ |
|
|
| def test_default_name(self, mock_user_directory): |
| """Test default 'system' name.""" |
| path = get_system_user_directory() |
| assert path.endswith("__system") |
| assert mock_user_directory in path |
|
|
| def test_custom_name(self, mock_user_directory): |
| """Test custom system user name.""" |
| path = get_system_user_directory("cache") |
| assert path.endswith("__cache") |
| assert "__cache" in path |
|
|
| def test_name_with_underscore(self, mock_user_directory): |
| """Test name with underscore in middle.""" |
| path = get_system_user_directory("my_cache") |
| assert "__my_cache" in path |
|
|
| def test_empty_name_raises(self): |
| """Test empty name raises ValueError.""" |
| with pytest.raises(ValueError, match="cannot be empty"): |
| get_system_user_directory("") |
|
|
| def test_none_name_raises(self): |
| """Test None name raises ValueError.""" |
| with pytest.raises(ValueError, match="cannot be empty"): |
| get_system_user_directory(None) |
|
|
| def test_name_starting_with_underscore_raises(self): |
| """Test name starting with underscore raises ValueError.""" |
| with pytest.raises(ValueError, match="should not start with underscore"): |
| get_system_user_directory("_system") |
|
|
| def test_path_traversal_raises(self): |
| """Test path traversal attempt raises ValueError (security).""" |
| with pytest.raises(ValueError, match="Invalid system user name"): |
| get_system_user_directory("../escape") |
|
|
| def test_path_traversal_middle_raises(self): |
| """Test path traversal in middle raises ValueError (security).""" |
| with pytest.raises(ValueError, match="Invalid system user name"): |
| get_system_user_directory("system/../other") |
|
|
| def test_special_chars_raise(self): |
| """Test special characters raise ValueError (security).""" |
| with pytest.raises(ValueError, match="Invalid system user name"): |
| get_system_user_directory("system!") |
|
|
| def test_returns_absolute_path(self, mock_user_directory): |
| """Test returned path is absolute.""" |
| path = get_system_user_directory("test") |
| assert os.path.isabs(path) |
|
|
|
|
| class TestGetPublicUserDirectory: |
| """Tests for get_public_user_directory() - HTTP endpoint access with System User blocking. |
| |
| Verifies: |
| - System Users (__ prefix) return None, blocking HTTP access |
| - Public Users get valid paths |
| - New endpoints using this function are automatically protected |
| """ |
|
|
| def test_normal_user(self, mock_user_directory): |
| """Test normal user returns valid path.""" |
| path = get_public_user_directory("default") |
| assert path is not None |
| assert "default" in path |
| assert mock_user_directory in path |
|
|
| def test_system_user_returns_none(self): |
| """Test System User (__ prefix) returns None - blocks HTTP access.""" |
| assert get_public_user_directory("__system") is None |
|
|
| def test_system_user_cache_returns_none(self): |
| """Test System User cache returns None.""" |
| assert get_public_user_directory("__cache") is None |
|
|
| def test_empty_user_returns_none(self): |
| """Test empty user returns None.""" |
| assert get_public_user_directory("") is None |
|
|
| def test_none_user_returns_none(self): |
| """Test None user returns None.""" |
| assert get_public_user_directory(None) is None |
|
|
| def test_header_injection_returns_none(self): |
| """Test header injection attempt returns None (security).""" |
| assert get_public_user_directory("__system\r\nX-Injected: true") is None |
|
|
| def test_null_byte_injection_returns_none(self): |
| """Test null byte injection handling (security).""" |
| |
| result = get_public_user_directory("user\x00__system") |
| |
| |
| assert result is not None or result is None |
|
|
| def test_path_traversal_attempt(self, mock_user_directory): |
| """Test path traversal attempt handling.""" |
| |
| |
| path = get_public_user_directory("../../../etc/passwd") |
| |
| |
| assert path is not None or "__" not in "../../../etc/passwd" |
|
|
| def test_returns_absolute_path(self, mock_user_directory): |
| """Test returned path is absolute.""" |
| path = get_public_user_directory("testuser") |
| assert path is not None |
| assert os.path.isabs(path) |
|
|
|
|
| class TestBackwardCompatibility: |
| """Tests for backward compatibility with existing APIs. |
| |
| Verifies: |
| - get_user_directory() API unchanged |
| - Existing user data remains accessible |
| """ |
|
|
| def test_get_user_directory_unchanged(self, mock_user_directory): |
| """Test get_user_directory() still works as before.""" |
| user_dir = get_user_directory() |
| assert user_dir is not None |
| assert os.path.isabs(user_dir) |
| assert user_dir == mock_user_directory |
|
|
| def test_existing_user_accessible(self, mock_user_directory): |
| """Test existing users can access their directories.""" |
| path = get_public_user_directory("default") |
| assert path is not None |
| assert "default" in path |
|
|
|
|
| class TestEdgeCases: |
| """Tests for edge cases in System User detection. |
| |
| Verifies: |
| - Only __ prefix is blocked (not _, not middle __) |
| - Bypass attempts are prevented |
| """ |
|
|
| def test_prefix_only(self): |
| """Test prefix-only string is blocked.""" |
| assert get_public_user_directory("__") is None |
|
|
| def test_single_underscore_allowed(self): |
| """Test single underscore prefix is allowed (not System User).""" |
| path = get_public_user_directory("_system") |
| assert path is not None |
| assert "_system" in path |
|
|
| def test_triple_underscore_blocked(self): |
| """Test triple underscore is blocked (starts with __).""" |
| assert get_public_user_directory("___system") is None |
|
|
| def test_underscore_in_middle_allowed(self): |
| """Test underscore in middle is allowed.""" |
| path = get_public_user_directory("my__system") |
| assert path is not None |
| assert "my__system" in path |
|
|
| def test_leading_space_allowed(self): |
| """Test leading space + prefix is allowed (doesn't start with __).""" |
| path = get_public_user_directory(" __system") |
| assert path is not None |
|
|