/*
* Firetweet - Twitter client for Android
*
* Copyright (C) 2012-2014 Mariotaku Lee <mariotaku.lee@gmail.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package org.getlantern.firetweet.util;
import android.graphics.BitmapFactory;
import android.net.Uri;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.util.Arrays;
import static org.getlantern.firetweet.util.Utils.closeSilently;
public class ImageValidator {
public static int INVALID = 0;
public static int VALID_FOR_BITMAP_FACTORY = 0x1;
public static int VALID_FOR_REGION_DECODER = 0x2;
public static int VALID_FOR_ALL = VALID_FOR_BITMAP_FACTORY | VALID_FOR_REGION_DECODER;
private static final byte[] PNG_HEAD = {(byte) 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A};
private static final byte[] PNG_TAIL = {0x49, 0x45, 0x4E, 0x44, (byte) 0xAE, 0x42, 0x60, (byte) 0x82};
private static final byte[] JPEG_HEAD = {(byte) 0xFF, (byte) 0xD8};
private static final byte[] JPEG_TAIL = {(byte) 0xFF, (byte) 0xD9};
public static boolean isValidForRegionDecoder(int validity) {
return (validity & VALID_FOR_REGION_DECODER) != 0;
}
public static boolean isValid(int validity) {
return validity != 0;
}
public static int checkImageValidity(final File file) {
if (file == null) return INVALID;
return checkImageValidity(file.getPath());
}
public static int checkImageValidity(final String file) {
final BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true;
BitmapFactory.decodeFile(file, opts);
final String type = opts.outMimeType;
if (type == null) return INVALID;
if ("image/jpeg".equalsIgnoreCase(type))
return checkJPEGValidity(file);
else if ("image/png".equalsIgnoreCase(type)) return checkPNGValidity(file);
return opts.outWidth > 0 && opts.outHeight > 0 ? VALID_FOR_BITMAP_FACTORY : INVALID;
}
public static int checkImageValidity(final Uri uri) {
if (uri == null) return INVALID;
return checkImageValidity(uri.getPath());
}
public static int checkJPEGValidity(final String file) {
return checkHeadTailValidity(file, JPEG_HEAD, JPEG_TAIL);
}
public static int checkPNGValidity(final String file) {
return checkHeadTailValidity(file, PNG_HEAD, PNG_TAIL);
}
private static int checkHeadTailValidity(final RandomAccessFile raf, final byte[] head, final byte[] tail) {
if (raf == null) return INVALID;
try {
final long length = raf.length();
// The file has 0-length, so it can't be a PNG file.
if (length == 0) return INVALID;
byte[] buffer;
// Read head.
buffer = new byte[head.length];
raf.seek(0);
if (raf.read(buffer) != buffer.length || !Arrays.equals(buffer, head)) return INVALID;
// Read tail.
buffer = new byte[tail.length];
raf.seek(length - buffer.length);
if (raf.read(buffer) != buffer.length || !Arrays.equals(buffer, tail))
return VALID_FOR_BITMAP_FACTORY;
} catch (final IOException e) {
return INVALID;
} finally {
closeSilently(raf);
}
return VALID_FOR_ALL;
}
private static int checkHeadTailValidity(final String file, final byte[] head, final byte[] tail) {
try {
return checkHeadTailValidity(new RandomAccessFile(file, "r"), head, tail);
} catch (final FileNotFoundException e) {
return INVALID;
}
}
}