Fixed EXIF longitude bug
[mediagoblin.git] / mediagoblin / media_types / image / processing.py
1 # GNU MediaGoblin -- federated, autonomous media hosting
2 # Copyright (C) 2011, 2012 MediaGoblin contributors. See AUTHORS.
3 #
4 # This program is free software: you can redistribute it and/or modify
5 # it under the terms of the GNU Affero General Public License as published by
6 # the Free Software Foundation, either version 3 of the License, or
7 # (at your option) any later version.
8 #
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU Affero General Public License for more details.
13 #
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16
17 import Image
18 import os
19 import logging
20
21 from mediagoblin import mg_globals as mgg
22 from mediagoblin.processing import BadMediaFail, \
23 create_pub_filepath, FilenameBuilder
24 from mediagoblin.tools.exif import exif_fix_image_orientation, \
25 extract_exif, clean_exif, get_gps_data, get_useful, \
26 exif_image_needs_rotation
27
28 _log = logging.getLogger(__name__)
29
30
31 def resize_image(entry, filename, new_path, exif_tags, workdir, new_size,
32 size_limits=(0, 0)):
33 """
34 Store a resized version of an image and return its pathname.
35
36 Arguments:
37 entry -- the entry for the image to resize
38 filename -- the filename of the original image being resized
39 new_path -- public file path for the new resized image
40 exif_tags -- EXIF data for the original image
41 workdir -- directory path for storing converted image files
42 new_size -- 2-tuple size for the resized image
43 """
44 try:
45 resized = Image.open(filename)
46 except IOError:
47 raise BadMediaFail()
48 resized = exif_fix_image_orientation(resized, exif_tags) # Fix orientation
49 resized.thumbnail(new_size, Image.ANTIALIAS)
50
51 # Copy the new file to the conversion subdir, then remotely.
52 tmp_resized_filename = os.path.join(workdir, new_path[-1])
53 with file(tmp_resized_filename, 'w') as resized_file:
54 resized.save(resized_file)
55 mgg.public_store.copy_local_to_storage(tmp_resized_filename, new_path)
56
57
58 SUPPORTED_FILETYPES = ['png', 'gif', 'jpg', 'jpeg']
59
60
61 def sniff_handler(media_file, **kw):
62 if kw.get('media') is not None: # That's a double negative!
63 name, ext = os.path.splitext(kw['media'].filename)
64 clean_ext = ext[1:].lower() # Strip the . from ext and make lowercase
65
66 _log.debug('name: {0}\next: {1}\nlower_ext: {2}'.format(
67 name,
68 ext,
69 clean_ext))
70
71 if clean_ext in SUPPORTED_FILETYPES:
72 _log.info('Found file extension in supported filetypes')
73 return True
74 else:
75 _log.debug('Media present, extension not found in {0}'.format(
76 SUPPORTED_FILETYPES))
77 else:
78 _log.warning('Need additional information (keyword argument \'media\')'
79 ' to be able to handle sniffing')
80
81 return False
82
83
84 def process_image(entry):
85 """
86 Code to process an image
87 """
88 workbench = mgg.workbench_manager.create_workbench()
89 # Conversions subdirectory to avoid collisions
90 conversions_subdir = os.path.join(
91 workbench.dir, 'conversions')
92 os.mkdir(conversions_subdir)
93 queued_filepath = entry.queued_media_file
94 queued_filename = workbench.localized_file(
95 mgg.queue_store, queued_filepath,
96 'source')
97 name_builder = FilenameBuilder(queued_filename)
98
99 # EXIF extraction
100 exif_tags = extract_exif(queued_filename)
101 gps_data = get_gps_data(exif_tags)
102
103 # Always create a small thumbnail
104 thumb_filepath = create_pub_filepath(
105 entry, name_builder.fill('{basename}.thumbnail{ext}'))
106 resize_image(entry, queued_filename, thumb_filepath,
107 exif_tags, conversions_subdir,
108 (mgg.global_config['media:thumb']['max_width'],
109 mgg.global_config['media:thumb']['max_height']))
110
111 # If the size of the original file exceeds the specified size of a `medium`
112 # file, a `.medium.jpg` files is created and later associated with the media
113 # entry.
114 medium = Image.open(queued_filename)
115 if medium.size[0] > mgg.global_config['media:medium']['max_width'] \
116 or medium.size[1] > mgg.global_config['media:medium']['max_height'] \
117 or exif_image_needs_rotation(exif_tags):
118 medium_filepath = create_pub_filepath(
119 entry, name_builder.fill('{basename}.medium{ext}'))
120 resize_image(
121 entry, queued_filename, medium_filepath,
122 exif_tags, conversions_subdir,
123 (mgg.global_config['media:medium']['max_width'],
124 mgg.global_config['media:medium']['max_height']))
125 else:
126 medium_filepath = None
127
128 # we have to re-read because unlike PIL, not everything reads
129 # things in string representation :)
130 queued_file = file(queued_filename, 'rb')
131
132 with queued_file:
133 original_filepath = create_pub_filepath(
134 entry, name_builder.fill('{basename}{ext}'))
135
136 with mgg.public_store.get_file(original_filepath, 'wb') \
137 as original_file:
138 original_file.write(queued_file.read())
139
140 # Remove queued media file from storage and database
141 mgg.queue_store.delete_file(queued_filepath)
142 entry.queued_media_file = []
143
144 # Insert media file information into database
145 media_files_dict = entry.setdefault('media_files', {})
146 media_files_dict['thumb'] = thumb_filepath
147 media_files_dict['original'] = original_filepath
148 if medium_filepath:
149 media_files_dict['medium'] = medium_filepath
150
151 # Insert exif data into database
152 exif_all = clean_exif(exif_tags)
153
154 if len(exif_all):
155 entry.media_data_init(exif_all=exif_all)
156
157 if len(gps_data):
158 for key in list(gps_data.keys()):
159 gps_data['gps_' + key] = gps_data.pop(key)
160 entry.media_data_init(**gps_data)
161
162 # clean up workbench
163 workbench.destroy_self()
164
165 if __name__ == '__main__':
166 import sys
167 import pprint
168
169 pp = pprint.PrettyPrinter()
170
171 result = extract_exif(sys.argv[1])
172 gps = get_gps_data(result)
173 clean = clean_exif(result)
174 useful = get_useful(clean)
175
176 print pp.pprint(
177 clean)