imghdr.py 3.46 KB
Newer Older
1
"""Recognize image file formats based on their first few bytes."""
Guido van Rossum's avatar
Guido van Rossum committed
2

3
__all__ = ["what"]
Guido van Rossum's avatar
Guido van Rossum committed
4 5

#-------------------------#
6
# Recognize image headers #
Guido van Rossum's avatar
Guido van Rossum committed
7 8
#-------------------------#

9
def what(file, h=None):
10 11 12 13 14 15 16 17
    if h is None:
        if type(file) == type(''):
            f = open(file, 'rb')
            h = f.read(32)
        else:
            location = file.tell()
            h = file.read(32)
            file.seek(location)
Tim Peters's avatar
Tim Peters committed
18
            f = None
19 20 21 22 23 24 25 26 27 28
    else:
        f = None
    try:
        for tf in tests:
            res = tf(h, f)
            if res:
                return res
    finally:
        if f: f.close()
    return None
Guido van Rossum's avatar
Guido van Rossum committed
29 30 31 32 33 34 35 36 37


#---------------------------------#
# Subroutines per image file type #
#---------------------------------#

tests = []

def test_rgb(h, f):
38 39 40
    """SGI image library"""
    if h[:2] == '\001\332':
        return 'rgb'
Guido van Rossum's avatar
Guido van Rossum committed
41 42 43 44

tests.append(test_rgb)

def test_gif(h, f):
45 46 47
    """GIF ('87 and '89 variants)"""
    if h[:6] in ('GIF87a', 'GIF89a'):
        return 'gif'
Guido van Rossum's avatar
Guido van Rossum committed
48 49 50

tests.append(test_gif)

51
def test_pbm(h, f):
52 53 54 55
    """PBM (portable bitmap)"""
    if len(h) >= 3 and \
        h[0] == 'P' and h[1] in '14' and h[2] in ' \t\n\r':
        return 'pbm'
Guido van Rossum's avatar
Guido van Rossum committed
56

57 58 59
tests.append(test_pbm)

def test_pgm(h, f):
60 61 62 63
    """PGM (portable graymap)"""
    if len(h) >= 3 and \
        h[0] == 'P' and h[1] in '25' and h[2] in ' \t\n\r':
        return 'pgm'
64 65 66 67

tests.append(test_pgm)

def test_ppm(h, f):
68 69 70 71
    """PPM (portable pixmap)"""
    if len(h) >= 3 and \
        h[0] == 'P' and h[1] in '36' and h[2] in ' \t\n\r':
        return 'ppm'
72 73

tests.append(test_ppm)
Guido van Rossum's avatar
Guido van Rossum committed
74 75

def test_tiff(h, f):
76 77 78
    """TIFF (can be in Motorola or Intel byte order)"""
    if h[:2] in ('MM', 'II'):
        return 'tiff'
Guido van Rossum's avatar
Guido van Rossum committed
79 80 81 82

tests.append(test_tiff)

def test_rast(h, f):
83 84 85
    """Sun raster file"""
    if h[:4] == '\x59\xA6\x6A\x95':
        return 'rast'
Guido van Rossum's avatar
Guido van Rossum committed
86 87 88

tests.append(test_rast)

89
def test_xbm(h, f):
90 91 92 93
    """X bitmap (X10 or X11)"""
    s = '#define '
    if h[:len(s)] == s:
        return 'xbm'
94 95 96

tests.append(test_xbm)

97
def test_jpeg(h, f):
98 99 100
    """JPEG data in JFIF format"""
    if h[6:10] == 'JFIF':
        return 'jpeg'
101 102

tests.append(test_jpeg)
Guido van Rossum's avatar
Guido van Rossum committed
103

104 105 106 107 108 109 110
def test_exif(h, f):
    """JPEG data in Exif format"""
    if h[6:10] == 'Exif':
        return 'jpeg'

tests.append(test_exif)

111
def test_bmp(h, f):
112 113
    if h[:2] == 'BM':
        return 'bmp'
Tim Peters's avatar
Tim Peters committed
114

115 116 117
tests.append(test_bmp)

def test_png(h, f):
118 119
    if h[:8] == "\211PNG\r\n\032\n":
        return 'png'
120 121 122

tests.append(test_png)

Guido van Rossum's avatar
Guido van Rossum committed
123 124 125 126 127
#--------------------#
# Small test program #
#--------------------#

def test():
128 129 130 131 132 133 134 135 136 137 138 139 140
    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)
Guido van Rossum's avatar
Guido van Rossum committed
141 142

def testall(list, recursive, toplevel):
143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
    import sys
    import os
    for filename in list:
        if os.path.isdir(filename):
            print filename + '/:',
            if recursive or toplevel:
                print 'recursing down:'
                import glob
                names = glob.glob(os.path.join(filename, '*'))
                testall(names, recursive, 0)
            else:
                print '*** directory (use -r) ***'
        else:
            print filename + ':',
            sys.stdout.flush()
            try:
                print what(filename)
            except IOError:
                print '*** not found ***'