| 1 | """Recognize image file formats based on their first few bytes."""
|
|---|
| 2 |
|
|---|
| 3 | __all__ = ["what"]
|
|---|
| 4 |
|
|---|
| 5 | #-------------------------#
|
|---|
| 6 | # Recognize image headers #
|
|---|
| 7 | #-------------------------#
|
|---|
| 8 |
|
|---|
| 9 | def what(file, h=None):
|
|---|
| 10 | if h is None:
|
|---|
| 11 | if isinstance(file, basestring):
|
|---|
| 12 | f = open(file, 'rb')
|
|---|
| 13 | h = f.read(32)
|
|---|
| 14 | else:
|
|---|
| 15 | location = file.tell()
|
|---|
| 16 | h = file.read(32)
|
|---|
| 17 | file.seek(location)
|
|---|
| 18 | f = None
|
|---|
| 19 | else:
|
|---|
| 20 | f = None
|
|---|
| 21 | try:
|
|---|
| 22 | for tf in tests:
|
|---|
| 23 | res = tf(h, f)
|
|---|
| 24 | if res:
|
|---|
| 25 | return res
|
|---|
| 26 | finally:
|
|---|
| 27 | if f: f.close()
|
|---|
| 28 | return None
|
|---|
| 29 |
|
|---|
| 30 |
|
|---|
| 31 | #---------------------------------#
|
|---|
| 32 | # Subroutines per image file type #
|
|---|
| 33 | #---------------------------------#
|
|---|
| 34 |
|
|---|
| 35 | tests = []
|
|---|
| 36 |
|
|---|
| 37 | def test_jpeg(h, f):
|
|---|
| 38 | """JPEG data in JFIF format"""
|
|---|
| 39 | if h[6:10] == 'JFIF':
|
|---|
| 40 | return 'jpeg'
|
|---|
| 41 |
|
|---|
| 42 | tests.append(test_jpeg)
|
|---|
| 43 |
|
|---|
| 44 | def test_exif(h, f):
|
|---|
| 45 | """JPEG data in Exif format"""
|
|---|
| 46 | if h[6:10] == 'Exif':
|
|---|
| 47 | return 'jpeg'
|
|---|
| 48 |
|
|---|
| 49 | tests.append(test_exif)
|
|---|
| 50 |
|
|---|
| 51 | def test_png(h, f):
|
|---|
| 52 | if h[:8] == "\211PNG\r\n\032\n":
|
|---|
| 53 | return 'png'
|
|---|
| 54 |
|
|---|
| 55 | tests.append(test_png)
|
|---|
| 56 |
|
|---|
| 57 | def test_gif(h, f):
|
|---|
| 58 | """GIF ('87 and '89 variants)"""
|
|---|
| 59 | if h[:6] in ('GIF87a', 'GIF89a'):
|
|---|
| 60 | return 'gif'
|
|---|
| 61 |
|
|---|
| 62 | tests.append(test_gif)
|
|---|
| 63 |
|
|---|
| 64 | def test_tiff(h, f):
|
|---|
| 65 | """TIFF (can be in Motorola or Intel byte order)"""
|
|---|
| 66 | if h[:2] in ('MM', 'II'):
|
|---|
| 67 | return 'tiff'
|
|---|
| 68 |
|
|---|
| 69 | tests.append(test_tiff)
|
|---|
| 70 |
|
|---|
| 71 | def test_rgb(h, f):
|
|---|
| 72 | """SGI image library"""
|
|---|
| 73 | if h[:2] == '\001\332':
|
|---|
| 74 | return 'rgb'
|
|---|
| 75 |
|
|---|
| 76 | tests.append(test_rgb)
|
|---|
| 77 |
|
|---|
| 78 | def test_pbm(h, f):
|
|---|
| 79 | """PBM (portable bitmap)"""
|
|---|
| 80 | if len(h) >= 3 and \
|
|---|
| 81 | h[0] == 'P' and h[1] in '14' and h[2] in ' \t\n\r':
|
|---|
| 82 | return 'pbm'
|
|---|
| 83 |
|
|---|
| 84 | tests.append(test_pbm)
|
|---|
| 85 |
|
|---|
| 86 | def test_pgm(h, f):
|
|---|
| 87 | """PGM (portable graymap)"""
|
|---|
| 88 | if len(h) >= 3 and \
|
|---|
| 89 | h[0] == 'P' and h[1] in '25' and h[2] in ' \t\n\r':
|
|---|
| 90 | return 'pgm'
|
|---|
| 91 |
|
|---|
| 92 | tests.append(test_pgm)
|
|---|
| 93 |
|
|---|
| 94 | def test_ppm(h, f):
|
|---|
| 95 | """PPM (portable pixmap)"""
|
|---|
| 96 | if len(h) >= 3 and \
|
|---|
| 97 | h[0] == 'P' and h[1] in '36' and h[2] in ' \t\n\r':
|
|---|
| 98 | return 'ppm'
|
|---|
| 99 |
|
|---|
| 100 | tests.append(test_ppm)
|
|---|
| 101 |
|
|---|
| 102 | def test_rast(h, f):
|
|---|
| 103 | """Sun raster file"""
|
|---|
| 104 | if h[:4] == '\x59\xA6\x6A\x95':
|
|---|
| 105 | return 'rast'
|
|---|
| 106 |
|
|---|
| 107 | tests.append(test_rast)
|
|---|
| 108 |
|
|---|
| 109 | def test_xbm(h, f):
|
|---|
| 110 | """X bitmap (X10 or X11)"""
|
|---|
| 111 | s = '#define '
|
|---|
| 112 | if h[:len(s)] == s:
|
|---|
| 113 | return 'xbm'
|
|---|
| 114 |
|
|---|
| 115 | tests.append(test_xbm)
|
|---|
| 116 |
|
|---|
| 117 | def test_bmp(h, f):
|
|---|
| 118 | if h[:2] == 'BM':
|
|---|
| 119 | return 'bmp'
|
|---|
| 120 |
|
|---|
| 121 | tests.append(test_bmp)
|
|---|
| 122 |
|
|---|
| 123 | #--------------------#
|
|---|
| 124 | # Small test program #
|
|---|
| 125 | #--------------------#
|
|---|
| 126 |
|
|---|
| 127 | def test():
|
|---|
| 128 | import sys
|
|---|
| 129 | recursive = 0
|
|---|
| 130 | if sys.argv[1:] and sys.argv[1] == '-r':
|
|---|
| 131 | del sys.argv[1:2]
|
|---|
| 132 | recursive = 1
|
|---|
| 133 | try:
|
|---|
| 134 | if sys.argv[1:]:
|
|---|
| 135 | testall(sys.argv[1:], recursive, 1)
|
|---|
| 136 | else:
|
|---|
| 137 | testall(['.'], recursive, 1)
|
|---|
| 138 | except KeyboardInterrupt:
|
|---|
| 139 | sys.stderr.write('\n[Interrupted]\n')
|
|---|
| 140 | sys.exit(1)
|
|---|
| 141 |
|
|---|
| 142 | def testall(list, recursive, toplevel):
|
|---|
| 143 | import sys
|
|---|
| 144 | import os
|
|---|
| 145 | for filename in list:
|
|---|
| 146 | if os.path.isdir(filename):
|
|---|
| 147 | print filename + '/:',
|
|---|
| 148 | if recursive or toplevel:
|
|---|
| 149 | print 'recursing down:'
|
|---|
| 150 | import glob
|
|---|
| 151 | names = glob.glob(os.path.join(filename, '*'))
|
|---|
| 152 | testall(names, recursive, 0)
|
|---|
| 153 | else:
|
|---|
| 154 | print '*** directory (use -r) ***'
|
|---|
| 155 | else:
|
|---|
| 156 | print filename + ':',
|
|---|
| 157 | sys.stdout.flush()
|
|---|
| 158 | try:
|
|---|
| 159 | print what(filename)
|
|---|
| 160 | except IOError:
|
|---|
| 161 | print '*** not found ***'
|
|---|