mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-10-25 21:21:01 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			61 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			61 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from __future__ import annotations
 | |
| 
 | |
| import shutil
 | |
| from io import BytesIO
 | |
| from pathlib import Path
 | |
| from typing import IO, Callable
 | |
| 
 | |
| import pytest
 | |
| 
 | |
| from PIL import GifImagePlugin, Image, JpegImagePlugin
 | |
| 
 | |
| from .helper import cjpeg_available, djpeg_available, is_win32, netpbm_available
 | |
| 
 | |
| TEST_JPG = "Tests/images/hopper.jpg"
 | |
| TEST_GIF = "Tests/images/hopper.gif"
 | |
| 
 | |
| test_filenames = ("temp_';", 'temp_";', "temp_'\"|", "temp_'\"||", "temp_'\"&&")
 | |
| 
 | |
| 
 | |
| @pytest.mark.skipif(is_win32(), reason="Requires Unix or macOS")
 | |
| class TestShellInjection:
 | |
|     def assert_save_filename_check(
 | |
|         self,
 | |
|         tmp_path: Path,
 | |
|         src_img: Image.Image,
 | |
|         save_func: Callable[[Image.Image, IO[bytes], str | bytes], None],
 | |
|     ) -> None:
 | |
|         for filename in test_filenames:
 | |
|             dest_file = str(tmp_path / filename)
 | |
|             save_func(src_img, BytesIO(), dest_file)
 | |
|             # If file can't be opened, shell injection probably occurred
 | |
|             with Image.open(dest_file) as im:
 | |
|                 im.load()
 | |
| 
 | |
|     @pytest.mark.skipif(not djpeg_available(), reason="djpeg not available")
 | |
|     def test_load_djpeg_filename(self, tmp_path: Path) -> None:
 | |
|         for filename in test_filenames:
 | |
|             src_file = tmp_path / filename
 | |
|             shutil.copy(TEST_JPG, src_file)
 | |
| 
 | |
|             with Image.open(src_file) as im:
 | |
|                 assert isinstance(im, JpegImagePlugin.JpegImageFile)
 | |
|                 im.load_djpeg()
 | |
| 
 | |
|     @pytest.mark.skipif(not cjpeg_available(), reason="cjpeg not available")
 | |
|     def test_save_cjpeg_filename(self, tmp_path: Path) -> None:
 | |
|         with Image.open(TEST_JPG) as im:
 | |
|             self.assert_save_filename_check(tmp_path, im, JpegImagePlugin._save_cjpeg)
 | |
| 
 | |
|     @pytest.mark.skipif(not netpbm_available(), reason="Netpbm not available")
 | |
|     def test_save_netpbm_filename_bmp_mode(self, tmp_path: Path) -> None:
 | |
|         with Image.open(TEST_GIF) as im:
 | |
|             im = im.convert("RGB")
 | |
|             self.assert_save_filename_check(tmp_path, im, GifImagePlugin._save_netpbm)
 | |
| 
 | |
|     @pytest.mark.skipif(not netpbm_available(), reason="Netpbm not available")
 | |
|     def test_save_netpbm_filename_l_mode(self, tmp_path: Path) -> None:
 | |
|         with Image.open(TEST_GIF) as im:
 | |
|             im = im.convert("L")
 | |
|             self.assert_save_filename_check(tmp_path, im, GifImagePlugin._save_netpbm)
 |