itests/examples: integrate imghdr code directly (fix #3570)
Python 3.13 removes it, and I would rather not add another dependency only for tests or examples…
This commit is contained in:
		
							
								
								
									
										184
									
								
								examples/imghdr.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								examples/imghdr.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,184 @@
 | 
			
		||||
"""
 | 
			
		||||
Recognize image file formats based on their first few bytes.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Taken from cpython 3.11 source code before the removal in 3.13.
 | 
			
		||||
 | 
			
		||||
Licensed under Zero-Clause BSD
 | 
			
		||||
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
from os import PathLike
 | 
			
		||||
import warnings
 | 
			
		||||
 | 
			
		||||
__all__ = ["what"]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
warnings._deprecated(__name__, remove=(3, 13))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#-------------------------#
 | 
			
		||||
# Recognize image headers #
 | 
			
		||||
#-------------------------#
 | 
			
		||||
 | 
			
		||||
def what(file, h=None):
 | 
			
		||||
    f = None
 | 
			
		||||
    try:
 | 
			
		||||
        if h is None:
 | 
			
		||||
            if isinstance(file, (str, PathLike)):
 | 
			
		||||
                f = open(file, 'rb')
 | 
			
		||||
                h = f.read(32)
 | 
			
		||||
            else:
 | 
			
		||||
                location = file.tell()
 | 
			
		||||
                h = file.read(32)
 | 
			
		||||
                file.seek(location)
 | 
			
		||||
        for tf in tests:
 | 
			
		||||
            res = tf(h, f)
 | 
			
		||||
            if res:
 | 
			
		||||
                return res
 | 
			
		||||
    finally:
 | 
			
		||||
        if f: f.close()
 | 
			
		||||
    return None
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#---------------------------------#
 | 
			
		||||
# Subroutines per image file type #
 | 
			
		||||
#---------------------------------#
 | 
			
		||||
 | 
			
		||||
tests = []
 | 
			
		||||
 | 
			
		||||
def test_jpeg(h, f):
 | 
			
		||||
    """JPEG data with JFIF or Exif markers; and raw JPEG"""
 | 
			
		||||
    if h[6:10] in (b'JFIF', b'Exif'):
 | 
			
		||||
        return 'jpeg'
 | 
			
		||||
    elif h[:4] == b'\xff\xd8\xff\xdb':
 | 
			
		||||
        return 'jpeg'
 | 
			
		||||
 | 
			
		||||
tests.append(test_jpeg)
 | 
			
		||||
 | 
			
		||||
def test_png(h, f):
 | 
			
		||||
    if h.startswith(b'\211PNG\r\n\032\n'):
 | 
			
		||||
        return 'png'
 | 
			
		||||
 | 
			
		||||
tests.append(test_png)
 | 
			
		||||
 | 
			
		||||
def test_gif(h, f):
 | 
			
		||||
    """GIF ('87 and '89 variants)"""
 | 
			
		||||
    if h[:6] in (b'GIF87a', b'GIF89a'):
 | 
			
		||||
        return 'gif'
 | 
			
		||||
 | 
			
		||||
tests.append(test_gif)
 | 
			
		||||
 | 
			
		||||
def test_tiff(h, f):
 | 
			
		||||
    """TIFF (can be in Motorola or Intel byte order)"""
 | 
			
		||||
    if h[:2] in (b'MM', b'II'):
 | 
			
		||||
        return 'tiff'
 | 
			
		||||
 | 
			
		||||
tests.append(test_tiff)
 | 
			
		||||
 | 
			
		||||
def test_rgb(h, f):
 | 
			
		||||
    """SGI image library"""
 | 
			
		||||
    if h.startswith(b'\001\332'):
 | 
			
		||||
        return 'rgb'
 | 
			
		||||
 | 
			
		||||
tests.append(test_rgb)
 | 
			
		||||
 | 
			
		||||
def test_pbm(h, f):
 | 
			
		||||
    """PBM (portable bitmap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'14' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'pbm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_pbm)
 | 
			
		||||
 | 
			
		||||
def test_pgm(h, f):
 | 
			
		||||
    """PGM (portable graymap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'25' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'pgm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_pgm)
 | 
			
		||||
 | 
			
		||||
def test_ppm(h, f):
 | 
			
		||||
    """PPM (portable pixmap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'36' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'ppm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_ppm)
 | 
			
		||||
 | 
			
		||||
def test_rast(h, f):
 | 
			
		||||
    """Sun raster file"""
 | 
			
		||||
    if h.startswith(b'\x59\xA6\x6A\x95'):
 | 
			
		||||
        return 'rast'
 | 
			
		||||
 | 
			
		||||
tests.append(test_rast)
 | 
			
		||||
 | 
			
		||||
def test_xbm(h, f):
 | 
			
		||||
    """X bitmap (X10 or X11)"""
 | 
			
		||||
    if h.startswith(b'#define '):
 | 
			
		||||
        return 'xbm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_xbm)
 | 
			
		||||
 | 
			
		||||
def test_bmp(h, f):
 | 
			
		||||
    if h.startswith(b'BM'):
 | 
			
		||||
        return 'bmp'
 | 
			
		||||
 | 
			
		||||
tests.append(test_bmp)
 | 
			
		||||
 | 
			
		||||
def test_webp(h, f):
 | 
			
		||||
    if h.startswith(b'RIFF') and h[8:12] == b'WEBP':
 | 
			
		||||
        return 'webp'
 | 
			
		||||
 | 
			
		||||
tests.append(test_webp)
 | 
			
		||||
 | 
			
		||||
def test_exr(h, f):
 | 
			
		||||
    if h.startswith(b'\x76\x2f\x31\x01'):
 | 
			
		||||
        return 'exr'
 | 
			
		||||
 | 
			
		||||
tests.append(test_exr)
 | 
			
		||||
 | 
			
		||||
#--------------------#
 | 
			
		||||
# Small test program #
 | 
			
		||||
#--------------------#
 | 
			
		||||
 | 
			
		||||
def test():
 | 
			
		||||
    import sys
 | 
			
		||||
    recursive = 0
 | 
			
		||||
    if sys.argv[1:] and sys.argv[1] == '-r':
 | 
			
		||||
        del sys.argv[1:2]
 | 
			
		||||
        recursive = 1
 | 
			
		||||
    try:
 | 
			
		||||
        if sys.argv[1:]:
 | 
			
		||||
            testall(sys.argv[1:], recursive, 1)
 | 
			
		||||
        else:
 | 
			
		||||
            testall(['.'], recursive, 1)
 | 
			
		||||
    except KeyboardInterrupt:
 | 
			
		||||
        sys.stderr.write('\n[Interrupted]\n')
 | 
			
		||||
        sys.exit(1)
 | 
			
		||||
 | 
			
		||||
def testall(list, recursive, toplevel):
 | 
			
		||||
    import sys
 | 
			
		||||
    import os
 | 
			
		||||
    for filename in list:
 | 
			
		||||
        if os.path.isdir(filename):
 | 
			
		||||
            print(filename + '/:', end=' ')
 | 
			
		||||
            if recursive or toplevel:
 | 
			
		||||
                print('recursing down:')
 | 
			
		||||
                import glob
 | 
			
		||||
                names = glob.glob(os.path.join(glob.escape(filename), '*'))
 | 
			
		||||
                testall(names, recursive, 0)
 | 
			
		||||
            else:
 | 
			
		||||
                print('*** directory (use -r) ***')
 | 
			
		||||
        else:
 | 
			
		||||
            print(filename + ':', end=' ')
 | 
			
		||||
            sys.stdout.flush()
 | 
			
		||||
            try:
 | 
			
		||||
                print(what(filename))
 | 
			
		||||
            except OSError:
 | 
			
		||||
                print('*** not found ***')
 | 
			
		||||
 | 
			
		||||
if __name__ == '__main__':
 | 
			
		||||
    test()
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										184
									
								
								itests/imghdr.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								itests/imghdr.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,184 @@
 | 
			
		||||
"""
 | 
			
		||||
Recognize image file formats based on their first few bytes.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Taken from cpython 3.11 source code before the removal in 3.13.
 | 
			
		||||
 | 
			
		||||
Licensed under Zero-Clause BSD
 | 
			
		||||
 | 
			
		||||
"""
 | 
			
		||||
 | 
			
		||||
from os import PathLike
 | 
			
		||||
import warnings
 | 
			
		||||
 | 
			
		||||
__all__ = ["what"]
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
warnings._deprecated(__name__, remove=(3, 13))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#-------------------------#
 | 
			
		||||
# Recognize image headers #
 | 
			
		||||
#-------------------------#
 | 
			
		||||
 | 
			
		||||
def what(file, h=None):
 | 
			
		||||
    f = None
 | 
			
		||||
    try:
 | 
			
		||||
        if h is None:
 | 
			
		||||
            if isinstance(file, (str, PathLike)):
 | 
			
		||||
                f = open(file, 'rb')
 | 
			
		||||
                h = f.read(32)
 | 
			
		||||
            else:
 | 
			
		||||
                location = file.tell()
 | 
			
		||||
                h = file.read(32)
 | 
			
		||||
                file.seek(location)
 | 
			
		||||
        for tf in tests:
 | 
			
		||||
            res = tf(h, f)
 | 
			
		||||
            if res:
 | 
			
		||||
                return res
 | 
			
		||||
    finally:
 | 
			
		||||
        if f: f.close()
 | 
			
		||||
    return None
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#---------------------------------#
 | 
			
		||||
# Subroutines per image file type #
 | 
			
		||||
#---------------------------------#
 | 
			
		||||
 | 
			
		||||
tests = []
 | 
			
		||||
 | 
			
		||||
def test_jpeg(h, f):
 | 
			
		||||
    """JPEG data with JFIF or Exif markers; and raw JPEG"""
 | 
			
		||||
    if h[6:10] in (b'JFIF', b'Exif'):
 | 
			
		||||
        return 'jpeg'
 | 
			
		||||
    elif h[:4] == b'\xff\xd8\xff\xdb':
 | 
			
		||||
        return 'jpeg'
 | 
			
		||||
 | 
			
		||||
tests.append(test_jpeg)
 | 
			
		||||
 | 
			
		||||
def test_png(h, f):
 | 
			
		||||
    if h.startswith(b'\211PNG\r\n\032\n'):
 | 
			
		||||
        return 'png'
 | 
			
		||||
 | 
			
		||||
tests.append(test_png)
 | 
			
		||||
 | 
			
		||||
def test_gif(h, f):
 | 
			
		||||
    """GIF ('87 and '89 variants)"""
 | 
			
		||||
    if h[:6] in (b'GIF87a', b'GIF89a'):
 | 
			
		||||
        return 'gif'
 | 
			
		||||
 | 
			
		||||
tests.append(test_gif)
 | 
			
		||||
 | 
			
		||||
def test_tiff(h, f):
 | 
			
		||||
    """TIFF (can be in Motorola or Intel byte order)"""
 | 
			
		||||
    if h[:2] in (b'MM', b'II'):
 | 
			
		||||
        return 'tiff'
 | 
			
		||||
 | 
			
		||||
tests.append(test_tiff)
 | 
			
		||||
 | 
			
		||||
def test_rgb(h, f):
 | 
			
		||||
    """SGI image library"""
 | 
			
		||||
    if h.startswith(b'\001\332'):
 | 
			
		||||
        return 'rgb'
 | 
			
		||||
 | 
			
		||||
tests.append(test_rgb)
 | 
			
		||||
 | 
			
		||||
def test_pbm(h, f):
 | 
			
		||||
    """PBM (portable bitmap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'14' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'pbm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_pbm)
 | 
			
		||||
 | 
			
		||||
def test_pgm(h, f):
 | 
			
		||||
    """PGM (portable graymap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'25' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'pgm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_pgm)
 | 
			
		||||
 | 
			
		||||
def test_ppm(h, f):
 | 
			
		||||
    """PPM (portable pixmap)"""
 | 
			
		||||
    if len(h) >= 3 and \
 | 
			
		||||
        h[0] == ord(b'P') and h[1] in b'36' and h[2] in b' \t\n\r':
 | 
			
		||||
        return 'ppm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_ppm)
 | 
			
		||||
 | 
			
		||||
def test_rast(h, f):
 | 
			
		||||
    """Sun raster file"""
 | 
			
		||||
    if h.startswith(b'\x59\xA6\x6A\x95'):
 | 
			
		||||
        return 'rast'
 | 
			
		||||
 | 
			
		||||
tests.append(test_rast)
 | 
			
		||||
 | 
			
		||||
def test_xbm(h, f):
 | 
			
		||||
    """X bitmap (X10 or X11)"""
 | 
			
		||||
    if h.startswith(b'#define '):
 | 
			
		||||
        return 'xbm'
 | 
			
		||||
 | 
			
		||||
tests.append(test_xbm)
 | 
			
		||||
 | 
			
		||||
def test_bmp(h, f):
 | 
			
		||||
    if h.startswith(b'BM'):
 | 
			
		||||
        return 'bmp'
 | 
			
		||||
 | 
			
		||||
tests.append(test_bmp)
 | 
			
		||||
 | 
			
		||||
def test_webp(h, f):
 | 
			
		||||
    if h.startswith(b'RIFF') and h[8:12] == b'WEBP':
 | 
			
		||||
        return 'webp'
 | 
			
		||||
 | 
			
		||||
tests.append(test_webp)
 | 
			
		||||
 | 
			
		||||
def test_exr(h, f):
 | 
			
		||||
    if h.startswith(b'\x76\x2f\x31\x01'):
 | 
			
		||||
        return 'exr'
 | 
			
		||||
 | 
			
		||||
tests.append(test_exr)
 | 
			
		||||
 | 
			
		||||
#--------------------#
 | 
			
		||||
# Small test program #
 | 
			
		||||
#--------------------#
 | 
			
		||||
 | 
			
		||||
def test():
 | 
			
		||||
    import sys
 | 
			
		||||
    recursive = 0
 | 
			
		||||
    if sys.argv[1:] and sys.argv[1] == '-r':
 | 
			
		||||
        del sys.argv[1:2]
 | 
			
		||||
        recursive = 1
 | 
			
		||||
    try:
 | 
			
		||||
        if sys.argv[1:]:
 | 
			
		||||
            testall(sys.argv[1:], recursive, 1)
 | 
			
		||||
        else:
 | 
			
		||||
            testall(['.'], recursive, 1)
 | 
			
		||||
    except KeyboardInterrupt:
 | 
			
		||||
        sys.stderr.write('\n[Interrupted]\n')
 | 
			
		||||
        sys.exit(1)
 | 
			
		||||
 | 
			
		||||
def testall(list, recursive, toplevel):
 | 
			
		||||
    import sys
 | 
			
		||||
    import os
 | 
			
		||||
    for filename in list:
 | 
			
		||||
        if os.path.isdir(filename):
 | 
			
		||||
            print(filename + '/:', end=' ')
 | 
			
		||||
            if recursive or toplevel:
 | 
			
		||||
                print('recursing down:')
 | 
			
		||||
                import glob
 | 
			
		||||
                names = glob.glob(os.path.join(glob.escape(filename), '*'))
 | 
			
		||||
                testall(names, recursive, 0)
 | 
			
		||||
            else:
 | 
			
		||||
                print('*** directory (use -r) ***')
 | 
			
		||||
        else:
 | 
			
		||||
            print(filename + ':', end=' ')
 | 
			
		||||
            sys.stdout.flush()
 | 
			
		||||
            try:
 | 
			
		||||
                print(what(filename))
 | 
			
		||||
            except OSError:
 | 
			
		||||
                print('*** not found ***')
 | 
			
		||||
 | 
			
		||||
if __name__ == '__main__':
 | 
			
		||||
    test()
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user