| 1 | package net.oni2.httpfiledownloader; | 
 
 
 
 
 | 2 |  | 
 
 
 
 
 | 3 | import java.io.BufferedInputStream; | 
 
 
 
 
 | 4 | import java.io.File; | 
 
 
 
 
 | 5 | import java.io.FileNotFoundException; | 
 
 
 
 
 | 6 | import java.io.IOException; | 
 
 
 
 
 | 7 | import java.io.RandomAccessFile; | 
 
 
 
 
 | 8 | import java.net.URL; | 
 
 
 
 
 | 9 | import java.net.URLConnection; | 
 
 
 
 
 | 10 | import java.util.HashSet; | 
 
 
 
 
 | 11 |  | 
 
 
 
 
 | 12 | import net.oni2.ProxySettings; | 
 
 
 
 
 | 13 |  | 
 
 
 
 
 | 14 | /** | 
 
 
 
 
 | 15 | * @author Christian Illy | 
 
 
 
 
 | 16 | */ | 
 
 
 
 
 | 17 | public class FileDownloader implements Runnable { | 
 
 
 
 
 | 18 | /** | 
 
 
 
 
 | 19 | * @author Christian Illy | 
 
 
 
 
 | 20 | */ | 
 
 
 
 
 | 21 | public enum EState { | 
 
 
 
 
 | 22 | /** | 
 
 
 
 
 | 23 | * Downloader initialized but not started | 
 
 
 
 
 | 24 | */ | 
 
 
 
 
 | 25 | INIT, | 
 
 
 
 
 | 26 | /** | 
 
 
 
 
 | 27 | * Download running | 
 
 
 
 
 | 28 | */ | 
 
 
 
 
 | 29 | RUNNING, | 
 
 
 
 
 | 30 | /** | 
 
 
 
 
 | 31 | * Download suspended | 
 
 
 
 
 | 32 | */ | 
 
 
 
 
 | 33 | PAUSED, | 
 
 
 
 
 | 34 | /** | 
 
 
 
 
 | 35 | * Download interrupted | 
 
 
 
 
 | 36 | */ | 
 
 
 
 
 | 37 | INTERRUPTED, | 
 
 
 
 
 | 38 | /** | 
 
 
 
 
 | 39 | * Download finished successfully | 
 
 
 
 
 | 40 | */ | 
 
 
 
 
 | 41 | FINISHED, | 
 
 
 
 
 | 42 | /** | 
 
 
 
 
 | 43 | * Aborted because of an error | 
 
 
 
 
 | 44 | */ | 
 
 
 
 
 | 45 | ERROR | 
 
 
 
 
 | 46 | }; | 
 
 
 
 
 | 47 |  | 
 
 
 
 
 | 48 | private HashSet<FileDownloadListener> listeners = new HashSet<FileDownloadListener>(); | 
 
 
 
 
 | 49 | private Thread t = null; | 
 
 
 
 
 | 50 | private URL url = null; | 
 
 
 
 
 | 51 | private File target = null; | 
 
 
 
 
 | 52 | private int fileLength = Integer.MAX_VALUE; | 
 
 
 
 
 | 53 | private int downloaded = 0; | 
 
 
 
 
 | 54 |  | 
 
 
 
 
 | 55 | private EState state = EState.INIT; | 
 
 
 
 
 | 56 |  | 
 
 
 
 
 | 57 | /** | 
 
 
 
 
 | 58 | * @param url | 
 
 
 
 
 | 59 | *            URL of file to download | 
 
 
 
 
 | 60 | * @param target | 
 
 
 
 
 | 61 | *            Path of target file to save to | 
 
 
 
 
 | 62 | * @throws IOException | 
 
 
 
 
 | 63 | *             If url could not be opened | 
 
 
 
 
 | 64 | */ | 
 
 
 
 
 | 65 | public FileDownloader(String url, File target) throws IOException { | 
 
 
 
 
 | 66 | this.url = new URL(url); | 
 
 
 
 
 | 67 | this.target = target; | 
 
 
 
 
 | 68 | } | 
 
 
 
 
 | 69 |  | 
 
 
 
 
 | 70 | /** | 
 
 
 
 
 | 71 | * @param listener | 
 
 
 
 
 | 72 | *            Listener to add | 
 
 
 
 
 | 73 | */ | 
 
 
 
 
 | 74 | public void addListener(FileDownloadListener listener) { | 
 
 
 
 
 | 75 | listeners.add(listener); | 
 
 
 
 
 | 76 | } | 
 
 
 
 
 | 77 |  | 
 
 
 
 
 | 78 | /** | 
 
 
 
 
 | 79 | * @param listener | 
 
 
 
 
 | 80 | *            Listener to remove | 
 
 
 
 
 | 81 | */ | 
 
 
 
 
 | 82 | public void removeListener(FileDownloadListener listener) { | 
 
 
 
 
 | 83 | listeners.remove(listener); | 
 
 
 
 
 | 84 | } | 
 
 
 
 
 | 85 |  | 
 
 
 
 
 | 86 | /** | 
 
 
 
 
 | 87 | * Start the download process | 
 
 
 
 
 | 88 | */ | 
 
 
 
 
 | 89 | public synchronized void start() { | 
 
 
 
 
 | 90 | if (t == null) { | 
 
 
 
 
 | 91 | state = EState.RUNNING; | 
 
 
 
 
 | 92 | t = new Thread(this); | 
 
 
 
 
 | 93 | t.start(); | 
 
 
 
 
 | 94 | } | 
 
 
 
 
 | 95 | } | 
 
 
 
 
 | 96 |  | 
 
 
 
 
 | 97 | /** | 
 
 
 
 
 | 98 | * @param suspend | 
 
 
 
 
 | 99 | *            Suspend or resume | 
 
 
 
 
 | 100 | */ | 
 
 
 
 
 | 101 | public synchronized void suspend(boolean suspend) { | 
 
 
 
 
 | 102 | if ((state == EState.RUNNING) || (state == EState.PAUSED)) { | 
 
 
 
 
 | 103 | if (suspend) | 
 
 
 
 
 | 104 | state = EState.PAUSED; | 
 
 
 
 
 | 105 | else | 
 
 
 
 
 | 106 | state = EState.RUNNING; | 
 
 
 
 
 | 107 | updateStatus(downloaded, fileLength); | 
 
 
 
 
 | 108 | } | 
 
 
 
 
 | 109 | } | 
 
 
 
 
 | 110 |  | 
 
 
 
 
 | 111 | /** | 
 
 
 
 
 | 112 | * Stop (abort) download | 
 
 
 
 
 | 113 | */ | 
 
 
 
 
 | 114 | public synchronized void stop() { | 
 
 
 
 
 | 115 | if (state != EState.FINISHED) { | 
 
 
 
 
 | 116 | state = EState.INTERRUPTED; | 
 
 
 
 
 | 117 | if (t != null) { | 
 
 
 
 
 | 118 | try { | 
 
 
 
 
 | 119 | t.join(); | 
 
 
 
 
 | 120 | } catch (InterruptedException e) { | 
 
 
 
 
 | 121 | e.printStackTrace(); | 
 
 
 
 
 | 122 | } | 
 
 
 
 
 | 123 | t = null; | 
 
 
 
 
 | 124 | } | 
 
 
 
 
 | 125 | updateStatus(0, 1); | 
 
 
 
 
 | 126 | if (target.exists()) | 
 
 
 
 
 | 127 | target.delete(); | 
 
 
 
 
 | 128 | } | 
 
 
 
 
 | 129 | } | 
 
 
 
 
 | 130 |  | 
 
 
 
 
 | 131 | /** | 
 
 
 
 
 | 132 | * @return current state | 
 
 
 
 
 | 133 | */ | 
 
 
 
 
 | 134 | public EState getState() { | 
 
 
 
 
 | 135 | return state; | 
 
 
 
 
 | 136 | } | 
 
 
 
 
 | 137 |  | 
 
 
 
 
 | 138 | private synchronized void updateStatus(int current, int total) { | 
 
 
 
 
 | 139 | downloaded = current; | 
 
 
 
 
 | 140 | for (FileDownloadListener l : listeners) { | 
 
 
 
 
 | 141 | l.statusUpdate(this, state, current, total); | 
 
 
 
 
 | 142 | } | 
 
 
 
 
 | 143 | } | 
 
 
 
 
 | 144 |  | 
 
 
 
 
 | 145 | @Override | 
 
 
 
 
 | 146 | public void run() { | 
 
 
 
 
 | 147 | int downloaded = 0; | 
 
 
 
 
 | 148 | String strLastModified = null; | 
 
 
 
 
 | 149 | String strEtag = null; | 
 
 
 
 
 | 150 | RandomAccessFile outFile = null; | 
 
 
 
 
 | 151 |  | 
 
 
 
 
 | 152 | try { | 
 
 
 
 
 | 153 | outFile = new RandomAccessFile(target, "rw"); | 
 
 
 
 
 | 154 | } catch (FileNotFoundException e1) { | 
 
 
 
 
 | 155 | e1.printStackTrace(); | 
 
 
 
 
 | 156 | state = EState.ERROR; | 
 
 
 
 
 | 157 | updateStatus(downloaded, fileLength); | 
 
 
 
 
 | 158 | return; | 
 
 
 
 
 | 159 | } | 
 
 
 
 
 | 160 |  | 
 
 
 
 
 | 161 | try { | 
 
 
 
 
 | 162 | while (downloaded < fileLength) { | 
 
 
 
 
 | 163 | switch (state) { | 
 
 
 
 
 | 164 | case ERROR: | 
 
 
 
 
 | 165 | updateStatus(downloaded, fileLength); | 
 
 
 
 
 | 166 | return; | 
 
 
 
 
 | 167 | case PAUSED: | 
 
 
 
 
 | 168 | try { | 
 
 
 
 
 | 169 | Thread.sleep(100); | 
 
 
 
 
 | 170 | } catch (InterruptedException e) { | 
 
 
 
 
 | 171 | e.printStackTrace(); | 
 
 
 
 
 | 172 | } | 
 
 
 
 
 | 173 | break; | 
 
 
 
 
 | 174 | case INTERRUPTED: | 
 
 
 
 
 | 175 | return; | 
 
 
 
 
 | 176 | case RUNNING: | 
 
 
 
 
 | 177 | BufferedInputStream input = null; | 
 
 
 
 
 | 178 | try { | 
 
 
 
 
 | 179 | URLConnection connection = url | 
 
 
 
 
 | 180 | .openConnection(ProxySettings.getInstance() | 
 
 
 
 
 | 181 | .getProxy()); | 
 
 
 
 
 | 182 | connection.setRequestProperty("Cache-Control", | 
 
 
 
 
 | 183 | "no-cache"); | 
 
 
 
 
 | 184 | if (downloaded == 0) { | 
 
 
 
 
 | 185 | connection.connect(); | 
 
 
 
 
 | 186 | strLastModified = connection | 
 
 
 
 
 | 187 | .getHeaderField("Last-Modified"); | 
 
 
 
 
 | 188 | strEtag = connection.getHeaderField("ETag"); | 
 
 
 
 
 | 189 | fileLength = connection.getContentLength(); | 
 
 
 
 
 | 190 | } else { | 
 
 
 
 
 | 191 | connection.setRequestProperty("Range", "bytes=" | 
 
 
 
 
 | 192 | + downloaded + "-"); | 
 
 
 
 
 | 193 | if (strEtag != null) | 
 
 
 
 
 | 194 | connection.setRequestProperty("If-Range", | 
 
 
 
 
 | 195 | strEtag); | 
 
 
 
 
 | 196 | else | 
 
 
 
 
 | 197 | connection.setRequestProperty("If-Range", | 
 
 
 
 
 | 198 | strLastModified); | 
 
 
 
 
 | 199 | connection.connect(); | 
 
 
 
 
 | 200 | } | 
 
 
 
 
 | 201 |  | 
 
 
 
 
 | 202 | // Setup streams and buffers. | 
 
 
 
 
 | 203 | input = new BufferedInputStream( | 
 
 
 
 
 | 204 | connection.getInputStream(), 8192); | 
 
 
 
 
 | 205 | if (downloaded > 0) | 
 
 
 
 
 | 206 | outFile.seek(downloaded); | 
 
 
 
 
 | 207 | byte data[] = new byte[1024]; | 
 
 
 
 
 | 208 |  | 
 
 
 
 
 | 209 | // Download file. | 
 
 
 
 
 | 210 | int dataRead = 0; | 
 
 
 
 
 | 211 | int i = 0; | 
 
 
 
 
 | 212 | while (((dataRead = input.read(data, 0, 1024)) != -1) | 
 
 
 
 
 | 213 | && (state == EState.RUNNING)) { | 
 
 
 
 
 | 214 | outFile.write(data, 0, dataRead); | 
 
 
 
 
 | 215 | downloaded += dataRead; | 
 
 
 
 
 | 216 | if (downloaded >= fileLength) | 
 
 
 
 
 | 217 | break; | 
 
 
 
 
 | 218 |  | 
 
 
 
 
 | 219 | i++; | 
 
 
 
 
 | 220 | if ((i % 50) == 0) | 
 
 
 
 
 | 221 | updateStatus(downloaded, fileLength); | 
 
 
 
 
 | 222 | } | 
 
 
 
 
 | 223 | input.close(); | 
 
 
 
 
 | 224 | } catch (IOException e) { | 
 
 
 
 
 | 225 | // TODO Auto-generated catch block | 
 
 
 
 
 | 226 | e.printStackTrace(); | 
 
 
 
 
 | 227 | try { | 
 
 
 
 
 | 228 | if (input != null) | 
 
 
 
 
 | 229 | input.close(); | 
 
 
 
 
 | 230 | } catch (IOException e2) { | 
 
 
 
 
 | 231 | e2.printStackTrace(); | 
 
 
 
 
 | 232 | } | 
 
 
 
 
 | 233 | } | 
 
 
 
 
 | 234 | break; | 
 
 
 
 
 | 235 | default: | 
 
 
 
 
 | 236 | break; | 
 
 
 
 
 | 237 | } | 
 
 
 
 
 | 238 | } | 
 
 
 
 
 | 239 | } finally { | 
 
 
 
 
 | 240 | try { | 
 
 
 
 
 | 241 | // Close streams. | 
 
 
 
 
 | 242 | outFile.close(); | 
 
 
 
 
 | 243 | } catch (IOException e) { | 
 
 
 
 
 | 244 | e.printStackTrace(); | 
 
 
 
 
 | 245 | } | 
 
 
 
 
 | 246 | } | 
 
 
 
 
 | 247 |  | 
 
 
 
 
 | 248 | state = EState.FINISHED; | 
 
 
 
 
 | 249 | updateStatus(downloaded, fileLength); | 
 
 
 
 
 | 250 | } | 
 
 
 
 
 | 251 |  | 
 
 
 
 
 | 252 | /** | 
 
 
 
 
 | 253 | * @return the target | 
 
 
 
 
 | 254 | */ | 
 
 
 
 
 | 255 | public File getTarget() { | 
 
 
 
 
 | 256 | return target; | 
 
 
 
 
 | 257 | } | 
 
 
 
 
 | 258 |  | 
 
 
 
 
 | 259 | /** | 
 
 
 
 
 | 260 | * @return the downloaded size | 
 
 
 
 
 | 261 | */ | 
 
 
 
 
 | 262 | public int getDownloaded() { | 
 
 
 
 
 | 263 | return downloaded; | 
 
 
 
 
 | 264 | } | 
 
 
 
 
 | 265 |  | 
 
 
 
 
 | 266 | } |