MimeUtils.java

  1/*
  2 * Copyright (C) 2010 The Android Open Source Project
  3 *
  4 * Licensed under the Apache License, Version 2.0 (the "License");
  5 * you may not use this file except in compliance with the License.
  6 * You may obtain a copy of the License at
  7 *
  8 *      http://www.apache.org/licenses/LICENSE-2.0
  9 *
 10 * Unless required by applicable law or agreed to in writing, software
 11 * distributed under the License is distributed on an "AS IS" BASIS,
 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13 * See the License for the specific language governing permissions and
 14 * limitations under the License.
 15 */
 16package eu.siacs.conversations.utils;
 17
 18import android.content.Context;
 19import android.database.Cursor;
 20import android.net.Uri;
 21import android.provider.OpenableColumns;
 22import android.util.Log;
 23import com.google.common.base.Strings;
 24import com.google.common.collect.ImmutableList;
 25import eu.siacs.conversations.Config;
 26import eu.siacs.conversations.entities.Transferable;
 27import eu.siacs.conversations.worker.ExportBackupWorker;
 28import java.io.File;
 29import java.io.FileInputStream;
 30import java.io.IOException;
 31import java.io.InputStream;
 32import java.util.Arrays;
 33import java.util.Collection;
 34import java.util.HashMap;
 35import java.util.List;
 36import java.util.Map;
 37import java.util.Properties;
 38
 39/**
 40 * Utilities for dealing with MIME types. Used to implement java.net.URLConnection and
 41 * android.webkit.MimeTypeMap.
 42 */
 43public final class MimeUtils {
 44
 45    public static final List<String> AMBIGUOUS_CONTAINER_FORMATS =
 46            ImmutableList.of(
 47                    "application/ogg",
 48                    "video/3gpp", // .3gp files can contain audio, video or both
 49                    "video/3gpp2");
 50
 51    public static final List<String> WORD_DOCUMENT_MIMES =
 52            Arrays.asList(
 53                    "application/vnd.oasis.opendocument.text",
 54                    "application/msword",
 55                    "application/vnd.openxmlformats-officedocument.wordprocessingml.document");
 56
 57    private static final Map<String, String> mimeTypeToExtensionMap = new HashMap<>();
 58    private static final Map<String, String> extensionToMimeTypeMap = new HashMap<>();
 59
 60    static {
 61        // The following table is based on /etc/mime.types data minus
 62        // chemical/* MIME types and MIME types that don't map to any
 63        // file extensions. We also exclude top-level domain names to
 64        // deal with cases like:
 65        //
 66        // mail.google.com/a/google.com
 67        //
 68        // and "active" MIME types (due to potential security issues).
 69        // Note that this list is _not_ in alphabetical order and must not be sorted.
 70        // The "most popular" extension must come first, so that it's the one returned
 71        // by guessExtensionFromMimeType.
 72        add("application/andrew-inset", "ez");
 73        add("application/dsptype", "tsp");
 74        add("application/json", "json");
 75        add("application/epub+zip", "epub");
 76        add("application/gpx+xml", "gpx");
 77        add("application/hta", "hta");
 78        add("application/mac-binhex40", "hqx");
 79        add("application/mathematica", "nb");
 80        add("application/msaccess", "mdb");
 81        add("application/oda", "oda");
 82        add("application/ogg", "ogg");
 83        add("application/pdf", "pdf");
 84        add("application/pgp-keys", "key");
 85        add("application/pgp-signature", "pgp");
 86        add("application/pics-rules", "prf");
 87        add("application/pkix-cert", "cer");
 88        add("application/rar", "rar");
 89        add("application/rdf+xml", "rdf");
 90        add("application/rss+xml", "rss");
 91        add("application/zip", "zip");
 92        add("application/vnd.amazon.mobi8-ebook", "azw3");
 93        add("application/vnd.amazon.mobi8-ebook", "azw");
 94        add("application/vnd.amazon.mobi8-ebook", "kfx");
 95        add("application/vnd.android.package-archive", "apk");
 96        add("application/vnd.cinderella", "cdy");
 97        add(ExportBackupWorker.MIME_TYPE, "ceb");
 98        add("application/vnd.ms-pki.stl", "stl");
 99        add("application/vnd.oasis.opendocument.database", "odb");
100        add("application/vnd.oasis.opendocument.formula", "odf");
101        add("application/vnd.oasis.opendocument.graphics", "odg");
102        add("application/vnd.oasis.opendocument.graphics-template", "otg");
103        add("application/vnd.oasis.opendocument.image", "odi");
104        add("application/vnd.oasis.opendocument.spreadsheet", "ods");
105        add("application/vnd.oasis.opendocument.spreadsheet-template", "ots");
106        add("application/vnd.oasis.opendocument.text", "odt");
107        add("application/vnd.oasis.opendocument.text-master", "odm");
108        add("application/vnd.oasis.opendocument.text-template", "ott");
109        add("application/vnd.oasis.opendocument.text-web", "oth");
110        add("application/vnd.oasis.opendocument.presentation", "odp");
111        add("application/vnd.google-earth.kml+xml", "kml");
112        add("application/vnd.google-earth.kmz", "kmz");
113        add("application/msword", "doc");
114        add("application/msword", "dot");
115        add("application/vnd.openxmlformats-officedocument.wordprocessingml.document", "docx");
116        add("application/vnd.openxmlformats-officedocument.wordprocessingml.template", "dotx");
117        add("application/vnd.ms-excel", "xls");
118        add("application/vnd.ms-excel", "xlt");
119        add("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", "xlsx");
120        add("application/vnd.openxmlformats-officedocument.spreadsheetml.template", "xltx");
121        add("application/vnd.ms-powerpoint", "ppt");
122        add("application/vnd.ms-powerpoint", "pot");
123        add("application/vnd.ms-powerpoint", "pps");
124        add("application/vnd.openxmlformats-officedocument.presentationml.presentation", "pptx");
125        add("application/vnd.openxmlformats-officedocument.presentationml.template", "potx");
126        add("application/vnd.openxmlformats-officedocument.presentationml.slideshow", "ppsx");
127        add("application/vnd.rim.cod", "cod");
128        add("application/vnd.smaf", "mmf");
129        add("application/vnd.stardivision.calc", "sdc");
130        add("application/vnd.stardivision.draw", "sda");
131        add("application/vnd.stardivision.impress", "sdd");
132        add("application/vnd.stardivision.impress", "sdp");
133        add("application/vnd.stardivision.math", "smf");
134        add("application/vnd.stardivision.writer", "sdw");
135        add("application/vnd.stardivision.writer", "vor");
136        add("application/vnd.stardivision.writer-global", "sgl");
137        add("application/vnd.sun.xml.calc", "sxc");
138        add("application/vnd.sun.xml.calc.template", "stc");
139        add("application/vnd.sun.xml.draw", "sxd");
140        add("application/vnd.sun.xml.draw.template", "std");
141        add("application/vnd.sun.xml.impress", "sxi");
142        add("application/vnd.sun.xml.impress.template", "sti");
143        add("application/vnd.sun.xml.math", "sxm");
144        add("application/vnd.sun.xml.writer", "sxw");
145        add("application/vnd.sun.xml.writer.global", "sxg");
146        add("application/vnd.sun.xml.writer.template", "stw");
147        add("application/vnd.visio", "vsd");
148        // https://www.iana.org/assignments/media-types/application/vnd.tcpdump.pcap
149        add("application/vnd.tcpdump.pcap", "pcap");
150        add("application/vnd.tcpdump.pcap", "cap");
151        add("application/vnd.tcpdump.pcap", "dmp");
152        add("application/x-pcapng", "pcapng");
153        add("application/wasm", "wasm");
154        add("application/x-7z-compressed", "7z");
155        add("application/x-abiword", "abw");
156        add("application/x-apple-diskimage", "dmg");
157        add("application/x-bcpio", "bcpio");
158        add("application/x-bittorrent", "torrent");
159        add("application/x-cdf", "cdf");
160        add("application/x-cdlink", "vcd");
161        add("application/x-chess-pgn", "pgn");
162        add("application/x-cpio", "cpio");
163        add("application/x-debian-package", "deb");
164        add("application/x-debian-package", "udeb");
165        add("application/x-director", "dcr");
166        add("application/x-director", "dir");
167        add("application/x-director", "dxr");
168        add("application/x-dms", "dms");
169        add("application/x-doom", "wad");
170        add("application/x-dvi", "dvi");
171        add("application/x-font", "pfa");
172        add("application/x-font", "pfb");
173        add("application/x-font", "gsf");
174        add("application/x-font", "pcf");
175        add("application/x-font", "pcf.Z");
176        add("application/x-freemind", "mm");
177        // application/futuresplash isn't IANA, so application/x-futuresplash should come first.
178        add("application/x-futuresplash", "spl");
179        add("application/futuresplash", "spl");
180        add("application/x-gnumeric", "gnumeric");
181        add("application/x-go-sgf", "sgf");
182        add("application/x-graphing-calculator", "gcf");
183        add("application/x-gtar", "tgz");
184        add("application/x-gtar", "gtar");
185        add("application/x-gtar", "taz");
186        add("application/x-hdf", "hdf");
187        add("application/x-ica", "ica");
188        add("application/x-internet-signup", "ins");
189        add("application/x-internet-signup", "isp");
190        add("application/x-iphone", "iii");
191        add("application/x-iso9660-image", "iso");
192        add("application/x-jmol", "jmz");
193        add("application/x-kchart", "chrt");
194        add("application/x-killustrator", "kil");
195        add("application/x-koan", "skp");
196        add("application/x-koan", "skd");
197        add("application/x-koan", "skt");
198        add("application/x-koan", "skm");
199        add("application/x-kpresenter", "kpr");
200        add("application/x-kpresenter", "kpt");
201        add("application/x-kspread", "ksp");
202        add("application/x-kword", "kwd");
203        add("application/x-kword", "kwt");
204        add("application/x-latex", "latex");
205        add("application/x-lha", "lha");
206        add("application/x-lzh", "lzh");
207        add("application/x-lzx", "lzx");
208        add("application/x-maker", "frm");
209        add("application/x-maker", "maker");
210        add("application/x-maker", "frame");
211        add("application/x-maker", "fb");
212        add("application/x-maker", "book");
213        add("application/x-maker", "fbdoc");
214        add("application/x-mif", "mif");
215        add("application/x-mobipocket-ebook", "mobi");
216        add("application/x-ms-wmd", "wmd");
217        add("application/x-ms-wmz", "wmz");
218        add("application/x-msi", "msi");
219        add("application/x-ns-proxy-autoconfig", "pac");
220        add("application/x-nwc", "nwc");
221        add("application/x-object", "o");
222        add("application/x-oz-application", "oza");
223        add("application/x-pem-file", "pem");
224        add("application/x-pkcs12", "p12");
225        add("application/x-pkcs12", "pfx");
226        add("application/x-pkcs7-certreqresp", "p7r");
227        add("application/x-pkcs7-crl", "crl");
228        add("application/x-quicktimeplayer", "qtl");
229        add("application/x-shar", "shar");
230        add("application/x-shockwave-flash", "swf");
231        add("application/x-stuffit", "sit");
232        add("application/x-sv4cpio", "sv4cpio");
233        add("application/x-sv4crc", "sv4crc");
234        add("application/x-tar", "tar");
235        add("application/x-texinfo", "texinfo");
236        add("application/x-texinfo", "texi");
237        add("application/x-troff", "t");
238        add("application/x-troff", "roff");
239        add("application/x-troff-man", "man");
240        add("application/x-ustar", "ustar");
241        add("application/x-wais-source", "src");
242        add("application/x-wingz", "wz");
243        add("application/x-webarchive", "webarchive");
244        add("application/x-webarchive-xml", "webarchivexml");
245        add("application/x-x509-ca-cert", "crt");
246        add("application/x-x509-user-cert", "crt");
247        add("application/x-x509-server-cert", "crt");
248        add("application/x-xcf", "xcf");
249        add("application/x-xfig", "fig");
250        add("application/webxdc+zip", "xdc");
251        add("application/xhtml+xml", "xhtml");
252        add("video/3gpp", "3gpp");
253        add("video/3gpp", "3gp");
254        add("video/3gpp2", "3gpp2");
255        add("video/3gpp2", "3g2");
256        add("audio/3gpp", "3gpp");
257        add("audio/3gpp", "3gp");
258        add("audio/aac", "aac");
259        add("audio/aac-adts", "aac");
260        add("audio/amr", "amr");
261        add("audio/amr-wb", "awb");
262        add("audio/basic", "snd");
263        add("audio/flac", "flac");
264        add("application/x-flac", "flac");
265        add("audio/imelody", "imy");
266        add("audio/midi", "mid");
267        add("audio/midi", "midi");
268        add("audio/midi", "ota");
269        add("audio/midi", "kar");
270        add("audio/midi", "rtttl");
271        add("audio/midi", "xmf");
272        add("audio/mobile-xmf", "mxmf");
273        // add ".mp3" first so it will be the default for guessExtensionFromMimeType
274        add("audio/mpeg", "mp3");
275        add("audio/mpeg", "mpga");
276        add("audio/mpeg", "mpega");
277        add("audio/mpeg", "mp2");
278        add("audio/mp4", "m4a");
279        add("audio/x-m4b", "m4b");
280        add("audio/mpegurl", "m3u");
281        add("audio/ogg", "oga");
282        add("audio/ogg; codecs=opus", "opus"); // opus in ogg container
283        add("audio/opus", "opus"); // audio/opus for containerless opus
284        add("audio/prs.sid", "sid");
285        add("audio/x-aiff", "aif");
286        add("audio/x-aiff", "aiff");
287        add("audio/x-aiff", "aifc");
288        add("audio/x-gsm", "gsm");
289        add("audio/x-matroska", "mka");
290        add("audio/x-mpegurl", "m3u");
291        add("audio/x-ms-wma", "wma");
292        add("audio/x-ms-wax", "wax");
293        add("audio/x-pn-realaudio", "ra");
294        add("audio/x-pn-realaudio", "rm");
295        add("audio/x-pn-realaudio", "ram");
296        add("audio/x-realaudio", "ra");
297        add("audio/x-scpls", "pls");
298        add("audio/x-sd2", "sd2");
299        add("audio/x-wav", "wav");
300        // image/bmp isn't IANA, so image/x-ms-bmp should come first.
301        add("image/x-ms-bmp", "bmp");
302        add("image/bmp", "bmp");
303        add("image/gif", "gif");
304        // image/ico isn't IANA, so image/x-icon should come first.
305        add("image/x-icon", "ico");
306        add("image/ico", "cur");
307        add("image/ico", "ico");
308        add("image/ief", "ief");
309        add("image/heic", "heic");
310        add("image/heif", "heif");
311        add("image/avif", "avif");
312        // add ".jpg" first so it will be the default for guessExtensionFromMimeType
313        add("image/jpeg", "jpg");
314        add("image/jpeg", "jpeg");
315        add("image/jpeg", "jpe");
316        add("image/jpeg", "jfif");
317        add("image/jpeg", "jif");
318        add("image/pcx", "pcx");
319        add("image/png", "png");
320        add("image/svg+xml", "svg");
321        add("image/svg+xml", "svgz");
322        add("image/tiff", "tiff");
323        add("image/tiff", "tif");
324        add("image/vnd.djvu", "djvu");
325        add("image/vnd.djvu", "djv");
326        add("image/vnd.wap.wbmp", "wbmp");
327        add("image/webp", "webp");
328        add("image/x-cmu-raster", "ras");
329        add("image/x-coreldraw", "cdr");
330        add("image/x-coreldrawpattern", "pat");
331        add("image/x-coreldrawtemplate", "cdt");
332        add("image/x-corelphotopaint", "cpt");
333        add("image/x-jg", "art");
334        add("image/x-jng", "jng");
335        add("image/x-photoshop", "psd");
336        add("image/x-portable-anymap", "pnm");
337        add("image/x-portable-bitmap", "pbm");
338        add("image/x-portable-graymap", "pgm");
339        add("image/x-portable-pixmap", "ppm");
340        add("image/x-rgb", "rgb");
341        add("image/x-xbitmap", "xbm");
342        add("image/x-xpixmap", "xpm");
343        add("image/x-xwindowdump", "xwd");
344        add("message/rfc822", "eml");
345        add("message/rfc822", "mime");
346        add("model/iges", "igs");
347        add("model/iges", "iges");
348        add("model/mesh", "msh");
349        add("model/mesh", "mesh");
350        add("model/mesh", "silo");
351        add("text/calendar", "ics");
352        add("text/calendar", "icz");
353        add("text/comma-separated-values", "csv");
354        add("text/css", "css");
355        add("text/html", "htm");
356        add("text/html", "html");
357        add("text/h323", "323");
358        add("text/iuls", "uls");
359        add("text/javascript", "js");
360        add("text/mathml", "mml");
361        // add ".txt" first so it will be the default for guessExtensionFromMimeType
362        add("text/plain", "txt");
363        add("text/plain", "asc");
364        add("text/plain", "text");
365        add("text/plain", "diff");
366        add("text/plain", "po"); // reserve "pot" for vnd.ms-powerpoint
367        add("text/richtext", "rtx");
368        add("text/rtf", "rtf");
369        add("text/text", "phps");
370        add("text/tab-separated-values", "tsv");
371        add("text/xml", "xml");
372        add("text/x-bibtex", "bib");
373        add("text/x-boo", "boo");
374        add("text/x-c++hdr", "hpp");
375        add("text/x-c++hdr", "h++");
376        add("text/x-c++hdr", "hxx");
377        add("text/x-c++hdr", "hh");
378        add("text/x-c++src", "cpp");
379        add("text/x-c++src", "c++");
380        add("text/x-c++src", "cc");
381        add("text/x-c++src", "cxx");
382        add("text/x-chdr", "h");
383        add("text/x-component", "htc");
384        add("text/x-csh", "csh");
385        add("text/x-csrc", "c");
386        add("text/x-dsrc", "d");
387        add("text/x-haskell", "hs");
388        add("text/x-java", "java");
389        add("text/x-literate-haskell", "lhs");
390        add("text/x-moc", "moc");
391        add("text/x-pascal", "p");
392        add("text/x-pascal", "pas");
393        add("text/x-pcs-gcd", "gcd");
394        add("text/x-setext", "etx");
395        add("text/x-tcl", "tcl");
396        add("text/x-tex", "tex");
397        add("text/x-tex", "ltx");
398        add("text/x-tex", "sty");
399        add("text/x-tex", "cls");
400        add("text/x-vcalendar", "vcs");
401        add("text/x-vcard", "vcf");
402        add("video/avi", "avi");
403        add("video/dl", "dl");
404        add("video/dv", "dif");
405        add("video/dv", "dv");
406        add("video/fli", "fli");
407        add("video/m4v", "m4v");
408        add("video/mp2ts", "ts");
409        add("video/ogg", "ogv");
410        add("video/mpeg", "mpeg");
411        add("video/mpeg", "mpg");
412        add("video/mpeg", "mpe");
413        add("video/mp4", "mp4");
414        add("video/mpeg", "VOB");
415        add("video/quicktime", "qt");
416        add("video/quicktime", "mov");
417        add("video/vnd.mpegurl", "mxu");
418        add("video/webm", "webm");
419        add("video/x-la-asf", "lsf");
420        add("video/x-la-asf", "lsx");
421        add("video/x-matroska", "mkv");
422        add("video/x-mng", "mng");
423        add("video/x-ms-asf", "asf");
424        add("video/x-ms-asf", "asx");
425        add("video/x-ms-wm", "wm");
426        add("video/x-ms-wmv", "wmv");
427        add("video/x-ms-wmx", "wmx");
428        add("video/x-ms-wvx", "wvx");
429        add("video/x-sgi-movie", "movie");
430        add("video/x-webex", "wrf");
431        add("x-conference/x-cooltalk", "ice");
432        add("x-epoc/x-sisx-app", "sisx");
433        applyOverrides();
434    }
435
436    // mime types that are more reliant by path
437    private static final Collection<String> PATH_PRECEDENCE_MIME_TYPE = List.of("audio/x-m4b");
438
439    private static void add(String mimeType, String extension) {
440        // If we have an existing x -> y mapping, we do not want to
441        // override it with another mapping x -> y2.
442        // If a mime type maps to several extensions
443        // the first extension added is considered the most popular
444        // so we do not want to overwrite it later.
445        if (!mimeTypeToExtensionMap.containsKey(mimeType)) {
446            mimeTypeToExtensionMap.put(mimeType, extension);
447        }
448        if (!extensionToMimeTypeMap.containsKey(extension)) {
449            extensionToMimeTypeMap.put(extension, mimeType);
450        }
451    }
452
453    private static InputStream getContentTypesPropertiesStream() {
454        // User override?
455        String userTable = System.getProperty("content.types.user.table");
456        if (userTable != null) {
457            File f = new File(userTable);
458            if (f.exists()) {
459                try {
460                    return new FileInputStream(f);
461                } catch (IOException ignored) {
462                }
463            }
464        }
465        // Standard location?
466        File f =
467                new File(
468                        System.getProperty("java.home"),
469                        "lib" + File.separator + "content-types.properties");
470        if (f.exists()) {
471            try {
472                return new FileInputStream(f);
473            } catch (IOException ignored) {
474            }
475        }
476        return null;
477    }
478
479    /**
480     * This isn't what the RI does. The RI doesn't have hard-coded defaults, so supplying your own
481     * "content.types.user.table" means you don't get any of the built-ins, and the built-ins come
482     * from "$JAVA_HOME/lib/content-types.properties".
483     */
484    private static void applyOverrides() {
485        // Get the appropriate InputStream to read overrides from, if any.
486        InputStream stream = getContentTypesPropertiesStream();
487        if (stream == null) {
488            return;
489        }
490        try {
491            try {
492                // Read the properties file...
493                Properties overrides = new Properties();
494                overrides.load(stream);
495                // And translate its mapping to ours...
496                for (Map.Entry<Object, Object> entry : overrides.entrySet()) {
497                    String extension = (String) entry.getKey();
498                    String mimeType = (String) entry.getValue();
499                    add(mimeType, extension);
500                }
501            } finally {
502                stream.close();
503            }
504        } catch (IOException ignored) {
505        }
506    }
507
508    private MimeUtils() {}
509
510    /**
511     * Returns true if the given MIME type has an entry in the map.
512     *
513     * @param mimeType A MIME type (i.e. text/plain)
514     * @return True iff there is a mimeType entry in the map.
515     */
516    public static boolean hasMimeType(String mimeType) {
517        if (mimeType == null || mimeType.isEmpty()) {
518            return false;
519        }
520        return mimeTypeToExtensionMap.containsKey(mimeType);
521    }
522
523    /**
524     * Returns the MIME type for the given extension.
525     *
526     * @param extension A file extension without the leading '.'
527     * @return The MIME type for the given extension or null iff there is none.
528     */
529    public static String guessMimeTypeFromExtension(String extension) {
530        if (extension == null || extension.isEmpty()) {
531            return null;
532        }
533        return extensionToMimeTypeMap.get(extension.toLowerCase());
534    }
535
536    /**
537     * Returns true if the given extension has a registered MIME type.
538     *
539     * @param extension A file extension without the leading '.'
540     * @return True iff there is an extension entry in the map.
541     */
542    public static boolean hasExtension(String extension) {
543        if (extension == null || extension.isEmpty()) {
544            return false;
545        }
546        return extensionToMimeTypeMap.containsKey(extension);
547    }
548
549    /**
550     * Returns the registered extension for the given MIME type. Note that some MIME types map to
551     * multiple extensions. This call will return the most common extension for the given MIME type.
552     *
553     * @param mimeType A MIME type (i.e. text/plain)
554     * @return The extension for the given MIME type or null iff there is none.
555     */
556    public static String guessExtensionFromMimeType(String mimeType) {
557        if (mimeType == null || mimeType.isEmpty()) {
558            return null;
559        }
560        return mimeTypeToExtensionMap.get(mimeType.split(";")[0]);
561    }
562
563    public static String guessMimeTypeFromUriAndMime(
564            final Context context, final Uri uri, final String mime) {
565        Log.d(Config.LOGTAG, "guessMimeTypeFromUriAndMime(" + uri + "," + mime + ")");
566        final String mimeFromUri = guessMimeTypeFromUri(context, uri);
567        Log.d(Config.LOGTAG, "mimeFromUri:" + mimeFromUri);
568        if (PATH_PRECEDENCE_MIME_TYPE.contains(mimeFromUri)) {
569            return mimeFromUri;
570        } else if (mime == null || mime.equals("application/octet-stream")) {
571            return mimeFromUri;
572        } else {
573            return mime;
574        }
575    }
576
577    public static String guessMimeTypeFromUri(final Context context, final Uri uri) {
578        if ("data".equals(uri.getScheme())) {
579            String[] parts = uri.getSchemeSpecificPart().split(",", 2);
580            String[] parts2 = parts[0].split(";", 2);
581            if (parts2.length > 0) return parts2[0];
582        }
583        final String mimeTypeContentResolver = guessFromContentResolver(context, uri);
584        final String mimeTypeFromQueryParameter = uri.isHierarchical() ? uri.getQueryParameter("mimeType") : null;
585        final String name = "content".equals(uri.getScheme()) ? getDisplayName(context, uri) : null;
586        final String mimeTypeFromName = Strings.isNullOrEmpty(name) ? null : guessFromPath(name);
587        final String path = uri.getPath();
588        final String mimeTypeFromPath = Strings.isNullOrEmpty(path) ? null : guessFromPath(path);
589        if (PATH_PRECEDENCE_MIME_TYPE.contains(mimeTypeFromName)) {
590            return mimeTypeFromName;
591        }
592        if (PATH_PRECEDENCE_MIME_TYPE.contains(mimeTypeFromPath)) {
593            return mimeTypeFromPath;
594        }
595        if (mimeTypeContentResolver != null
596                && !"application/octet-stream".equals(mimeTypeContentResolver)) {
597            return mimeTypeContentResolver;
598        }
599        if (mimeTypeFromName != null) {
600            return mimeTypeFromName;
601        }
602        if (mimeTypeFromQueryParameter != null) {
603            return mimeTypeFromQueryParameter;
604        }
605        return mimeTypeFromPath;
606    }
607
608    private static String guessFromContentResolver(final Context context, final Uri uri) {
609        try {
610            return context.getContentResolver().getType(uri);
611        } catch (final Throwable e) {
612            return null;
613        }
614    }
615
616    private static String getDisplayName(final Context context, final Uri uri) {
617        try (final Cursor cursor =
618                context.getContentResolver().query(uri, null, null, null, null)) {
619            if (cursor != null && cursor.moveToFirst()) {
620                final int index = cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);
621                if (index == -1) {
622                    return null;
623                }
624                return cursor.getString(index);
625            }
626        } catch (final Exception e) {
627            return null;
628        }
629        return null;
630    }
631
632    public static String guessFromPath(final String path) {
633        final int start = path.lastIndexOf('.') + 1;
634        if (start < path.length()) {
635            return MimeUtils.guessMimeTypeFromExtension(path.substring(start));
636        }
637        return null;
638    }
639
640    public static String extractRelevantExtension(final String path) {
641        return extractRelevantExtension(path, false);
642    }
643
644    public static String extractRelevantExtension(
645            final String path, final boolean ignoreCryptoExtension) {
646        if (Strings.isNullOrEmpty(path)) {
647            return null;
648        }
649
650        final String filenameQueryAnchor = path.substring(path.lastIndexOf('/') + 1);
651        final String filenameQuery = cutBefore(filenameQueryAnchor, '#');
652        final String filename = cutBefore(filenameQuery, '?');
653        final int dotPosition = filename.lastIndexOf('.');
654
655        if (dotPosition == -1) {
656            return null;
657        }
658        final String extension = filename.substring(dotPosition + 1);
659        // we want the real file extension, not the crypto one
660        if (ignoreCryptoExtension && Transferable.VALID_CRYPTO_EXTENSIONS.contains(extension)) {
661            return extractRelevantExtension(filename.substring(0, dotPosition));
662        } else {
663            return extension;
664        }
665    }
666
667    private static String cutBefore(final String input, final char c) {
668        final int position = input.indexOf(c);
669        if (position > 0) {
670            return input.substring(0, position);
671        } else {
672            return input;
673        }
674    }
675}