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