123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- """Recognize image file formats based on their first few bytes."""
- from os import PathLike
- __all__ = ["what"]
- #-------------------------#
- # 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 in JFIF or Exif format"""
- if h[6:10] in (b'JFIF', b'Exif'):
- 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()
|