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 */ 16 17 package com.android.email.mail.store.imap; 18 19 import com.android.email.FixedLengthInputStream; 20 import com.android.emailcommon.Logging; 21 import com.android.emailcommon.TempDirectory; 22 import com.android.emailcommon.utility.Utility; 23 import com.android.mail.utils.LogUtils; 24 25 import org.apache.commons.io.IOUtils; 26 27 import java.io.ByteArrayInputStream; 28 import java.io.File; 29 import java.io.FileInputStream; 30 import java.io.FileNotFoundException; 31 import java.io.FileOutputStream; 32 import java.io.IOException; 33 import java.io.InputStream; 34 import java.io.OutputStream; 35 36 /** 37 * Subclass of {@link ImapString} used for literals backed by a temp file. 38 */ 39 public class ImapTempFileLiteral extends ImapString { 40 /* package for test */ final File mFile; 41 42 /** Size is purely for toString() */ 43 private final int mSize; 44 45 /* package */ ImapTempFileLiteral(FixedLengthInputStream stream) throws IOException { 46 mSize = stream.getLength(); 47 mFile = File.createTempFile("imap", ".tmp", TempDirectory.getTempDirectory()); 48 49 // Unfortunately, we can't really use deleteOnExit(), because temp filenames are random 50 // so it'd simply cause a memory leak. 51 // deleteOnExit() simply adds filenames to a static list and the list will never shrink. 52 // mFile.deleteOnExit(); 53 OutputStream out = new FileOutputStream(mFile); 54 IOUtils.copy(stream, out); 55 out.close(); 56 } 57 58 /** 59 * Make sure we delete the temp file. 60 * 61 * We should always be calling {@link ImapResponse#destroy()}, but it's here as a last resort. 62 */ 63 @Override 64 protected void finalize() throws Throwable { 65 try { 66 destroy(); 67 } finally { 68 super.finalize(); 69 } 70 } 71 72 @Override 73 public InputStream getAsStream() { 74 checkNotDestroyed(); 75 try { 76 return new FileInputStream(mFile); 77 } catch (FileNotFoundException e) { 78 // It's probably possible if we're low on storage and the system clears the cache dir. 79 LogUtils.w(Logging.LOG_TAG, "ImapTempFileLiteral: Temp file not found"); 80 81 // Return 0 byte stream as a dummy... 82 return new ByteArrayInputStream(new byte[0]); 83 } 84 } 85 86 @Override 87 public String getString() { 88 checkNotDestroyed(); 89 try { 90 byte[] bytes = IOUtils.toByteArray(getAsStream()); 91 // Prevent crash from OOM; we've seen this, but only rarely and not reproducibly 92 if (bytes.length > ImapResponseParser.LITERAL_KEEP_IN_MEMORY_THRESHOLD) { 93 throw new IOException(); 94 } 95 return Utility.fromAscii(bytes); 96 } catch (IOException e) { 97 LogUtils.w(Logging.LOG_TAG, "ImapTempFileLiteral: Error while reading temp file", e); 98 return ""; 99 } 100 } 101 102 @Override 103 public void destroy() { 104 try { 105 if (!isDestroyed() && mFile.exists()) { 106 mFile.delete(); 107 } 108 } catch (RuntimeException re) { 109 // Just log and ignore. 110 LogUtils.w(Logging.LOG_TAG, "Failed to remove temp file: " + re.getMessage()); 111 } 112 super.destroy(); 113 } 114 115 @Override 116 public String toString() { 117 return String.format("{%d byte literal(file)}", mSize); 118 } 119 120 public boolean tempFileExistsForTest() { 121 return mFile.exists(); 122 } 123 } 124