blob: 9a01433772d4da105bb0e24dcda58571af94533f [file] [log] [blame]
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +00001//
2// Copyright (C) 2008 Wimba, Inc. All Rights Reserved.
3//
4// This is free software; you can redistribute it and/or modify
5// it under the terms of the GNU General Public License as published by
6// the Free Software Foundation; either version 2 of the License, or
7// (at your option) any later version.
8//
9// This software is distributed in the hope that it will be useful,
10// but WITHOUT ANY WARRANTY; without even the implied warranty of
11// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12// GNU General Public License for more details.
13//
14// You should have received a copy of the GNU General Public License
15// along with this software; if not, write to the Free Software
16// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
17// USA.
18//
19
20//
21// FbsConnection.java
22//
23
24package com.tightvnc.rfbplayer;
25
26import java.io.*;
27import java.net.*;
28import java.applet.Applet;
29
30public class FbsConnection {
31
32 URL fbsURL;
33 URL fbiURL;
34 URL fbkURL;
35
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +000036 /** Index data loaded from the .fbi file. */
Constantin Kaplinsky5ba167a2008-06-20 12:24:52 +000037 FbsEntryPoint[] indexData;
Constantin Kaplinskybe68e7f2008-06-20 12:21:57 +000038 int numIndexRecords;
Constantin Kaplinsky73540122008-06-23 12:30:41 +000039
Constantin Kaplinsky9e501642008-06-23 01:57:55 +000040 /** RFB initialization data loaded from the .fbi file. */
41 byte[] rfbInitData;
Constantin Kaplinskyade425a2008-06-20 06:44:03 +000042
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000043 FbsConnection(String fbsLocation, String indexLocationPrefix, Applet applet)
44 throws MalformedURLException {
45
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +000046 // Construct URLs from strings.
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000047 URL base = null;
48 if (applet != null) {
49 base = applet.getCodeBase();
50 }
51 fbsURL = new URL(base, fbsLocation);
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +000052 fbiURL = fbkURL = null;
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000053 if (indexLocationPrefix != null) {
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +000054 try {
55 fbiURL = new URL(base, indexLocationPrefix + ".fbi");
56 fbkURL = new URL(base, indexLocationPrefix + ".fbk");
57 } catch (MalformedURLException e) {
58 fbiURL = fbkURL = null;
59 }
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000060 }
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +000061
62 // Try to load the .fbi index file.
Constantin Kaplinsky5ba167a2008-06-20 12:24:52 +000063 indexData = null;
Constantin Kaplinskybe68e7f2008-06-20 12:21:57 +000064 numIndexRecords = 0;
Constantin Kaplinsky9e501642008-06-23 01:57:55 +000065 rfbInitData = null;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +000066 loadIndex();
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000067 }
68
69 FbsInputStream connect(long timeOffset) throws IOException {
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +000070 FbsInputStream fbs = null;
71
Constantin Kaplinsky38f044f2008-06-20 12:57:37 +000072 // Try efficient seeking first.
73 if (timeOffset > 0 && indexData != null && numIndexRecords > 0) {
74 int i = 0;
75 while (i < numIndexRecords && indexData[i].timestamp <= timeOffset) {
76 i++;
77 }
78 if (i > 0) {
79 FbsEntryPoint entryPoint = indexData[i - 1];
80 if (entryPoint.key_size < entryPoint.fbs_fpos) {
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +000081 try {
82 fbs = openFbsFile(entryPoint);
83 } catch (IOException e) {
84 System.err.println(e);
85 }
86 if (fbs == null) {
87 System.err.println("Could not open FBS file at entry point " +
88 entryPoint.timestamp + " ms");
89 }
Constantin Kaplinsky38f044f2008-06-20 12:57:37 +000090 }
91 }
92 }
93
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +000094 // Fallback to the dumb version of openFbsFile().
95 if (fbs == null) {
96 fbs = openFbsFile();
97 }
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +000098
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +000099 // Seek to the specified position.
100 fbs.setTimeOffset(timeOffset);
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +0000101 return fbs;
102 }
103
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000104 /**
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +0000105 * Load index data from .fbi file to {@link #indexData}.
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000106 */
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000107 private void loadIndex() {
108 // Loading .fbi makes sense only if both .fbi and .fbk files are available.
109 if (fbiURL != null && fbkURL != null) {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000110 FbsEntryPoint[] newIndex;
111 int numRecordsRead = 0;
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000112 byte[] newInitData;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000113 try {
114 // Connect.
115 URLConnection connection = fbiURL.openConnection();
116 connection.connect();
117 DataInputStream is = new DataInputStream(connection.getInputStream());
118
119 // Check file signature.
120 byte[] b = new byte[12];
121 is.readFully(b);
122 if (b[0] != 'F' || b[1] != 'B' || b[2] != 'I' || b[3] != ' ' ||
123 b[4] != '0' || b[5] != '0' || b[6] != '1' || b[7] != '.' ||
124 b[8] < '0' || b[8] > '9' || b[9] < '0' || b[9] > '9' ||
125 b[10] < '0' || b[10] > '9' || b[11] != '\n') {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000126 System.err.println("Could not load index: bad .fbi file signature");
127 return;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000128 }
129
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000130 // Read the record counter and allocate index array.
131 int numRecords = is.readInt();
132 if (numRecords <= 0) {
133 System.err.println("Could not load index: bad .fbi record counter");
134 return;
135 }
136 newIndex = new FbsEntryPoint[numRecords];
137
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000138 // Read byte counter and allocate byte array for RFB initialization.
139 int initSize = is.readInt();
140 if (initSize <= 0) {
141 System.err.println("Could not load index: bad RFB init data size");
142 return;
143 }
144 newInitData = new byte[initSize];
145
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000146 // Load index from the .fbi file.
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000147 try {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000148 for (int i = 0; i < numRecords; i++) {
149 FbsEntryPoint record = new FbsEntryPoint();
150 record.timestamp = (long)is.readInt() & 0xFFFFFFFFL;
151 record.key_fpos = (long)is.readInt() & 0xFFFFFFFFL;
152 record.key_size = (long)is.readInt() & 0xFFFFFFFFL;
153 record.fbs_fpos = (long)is.readInt() & 0xFFFFFFFFL;
154 record.fbs_skip = (long)is.readInt() & 0xFFFFFFFFL;
155 newIndex[i] = record;
156 numRecordsRead++;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000157 }
158 } catch (EOFException e) {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000159 System.err.println("Preliminary end of .fbi file");
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000160 } catch (IOException e) {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000161 System.err.println("Ignored exception: " + e);
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000162 }
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000163 if (numRecordsRead == 0) {
164 System.err.println("Could not load index: failed to read .fbi data");
165 return;
166 } else if (numRecordsRead != numRecords) {
167 System.err.println("Warning: read not as much .fbi data as expected");
168 }
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000169 is.readFully(newInitData);
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000170 } catch (FileNotFoundException e) {
171 System.err.println("Could not load index: .fbi file not found: " +
172 e.getMessage());
173 return;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000174 } catch (IOException e) {
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000175 System.err.println(e);
176 System.err.println("Could not load index: failed to load .fbi file");
177 return;
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000178 }
Constantin Kaplinsky4f39f8e2008-06-23 01:32:53 +0000179 // Check correctness of the data read.
180 for (int i = 1; i < numRecordsRead; i++) {
Constantin Kaplinsky73540122008-06-23 12:30:41 +0000181 if (newIndex[i].timestamp <= newIndex[i - 1].timestamp) {
Constantin Kaplinsky4f39f8e2008-06-23 01:32:53 +0000182 System.err.println("Could not load index: wrong .fbi file contents");
183 return;
184 }
185 }
186 // Loaded successfully.
Constantin Kaplinsky5ba167a2008-06-20 12:24:52 +0000187 indexData = newIndex;
Constantin Kaplinskybe68e7f2008-06-20 12:21:57 +0000188 numIndexRecords = numRecordsRead;
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000189 rfbInitData = newInitData;
Constantin Kaplinskyf338d2c2008-06-20 11:53:19 +0000190 System.err.println("Loaded index data, " + numRecordsRead + " records");
Constantin Kaplinsky5f1f8862008-06-20 05:17:39 +0000191 }
192 }
193
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +0000194 /**
195 * Open FBS file identified by {@link #fbsURL}. The file is open at its very
196 * beginning, no seek is performed.
197 *
198 * @return a newly created FBS input stream.
199 * @throws java.io.IOException if an I/O exception occurs.
200 */
201 private FbsInputStream openFbsFile() throws IOException {
202 return new FbsInputStream(fbsURL.openStream());
203 }
204
205 /**
206 * Open FBS file identified by {@link #fbsURL}. The stream is
207 * positioned at the entry point described by <code>entryPoint</code>.
208 *
209 * @param entryPoint entry point information.
210 *
211 * @return a newly created FBS input stream on success, <code>null</code> if
212 * any error occured and the FBS stream is not opened.
213 * @throws java.io.IOException if an I/O exception occurs.
214 */
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000215 private FbsInputStream openFbsFile(FbsEntryPoint entry)
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +0000216 throws IOException {
Constantin Kaplinskyd3a2df12008-06-20 20:12:22 +0000217
218 // Make sure the protocol is HTTP.
219 if (!fbkURL.getProtocol().equalsIgnoreCase("http") ||
220 !fbsURL.getProtocol().equalsIgnoreCase("http")) {
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000221 System.err.println("Indexed access requires HTTP protocol in URLs");
Constantin Kaplinskyd3a2df12008-06-20 20:12:22 +0000222 return null;
223 }
224
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000225 // Seek to the keyframe.
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000226 InputStream is = openHttpByteRange(fbkURL, entry.key_fpos, entry.key_size);
Constantin Kaplinskyf5821e52008-06-23 11:38:51 +0000227 if (is == null) {
228 return null;
229 }
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000230
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000231 // Load keyframe data from the .fbk file, prepend RFB initialization data.
Constantin Kaplinskyf5821e52008-06-23 11:38:51 +0000232 DataInputStream data = new DataInputStream(is);
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000233 byte[] keyData = new byte[rfbInitData.length + (int)entry.key_size];
Constantin Kaplinsky9e501642008-06-23 01:57:55 +0000234 System.arraycopy(rfbInitData, 0, keyData, 0, rfbInitData.length);
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000235 data.readFully(keyData, rfbInitData.length, (int)entry.key_size);
236 data.close();
Constantin Kaplinskyd3a2df12008-06-20 20:12:22 +0000237
238 // Open the FBS stream.
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000239 is = openHttpByteRange(fbsURL, entry.fbs_fpos, -1);
Constantin Kaplinskyf5821e52008-06-23 11:38:51 +0000240 if (is == null) {
241 return null;
242 }
Constantin Kaplinskya9bb64f2008-06-23 08:59:20 +0000243 return new FbsInputStream(is, entry.timestamp, keyData, entry.fbs_skip);
Constantin Kaplinsky1c2865f2008-06-20 18:50:27 +0000244 }
245
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000246 private static InputStream openHttpByteRange(URL url, long offset, long len)
Constantin Kaplinskyd2af0822008-06-21 11:41:42 +0000247 throws IOException {
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000248 HttpURLConnection conn = (HttpURLConnection)url.openConnection();
249 String rangeSpec = "bytes=" + offset + "-";
250 if (len != -1) {
251 long lastByteOffset = offset + len - 1;
252 rangeSpec += lastByteOffset;
253 }
Constantin Kaplinskyd2af0822008-06-21 11:41:42 +0000254 conn.setRequestProperty("Range", rangeSpec);
255 conn.connect();
Constantin Kaplinsky31b7a242008-06-22 07:24:11 +0000256 InputStream is = conn.getInputStream();
257 if (conn.getResponseCode() != HttpURLConnection.HTTP_PARTIAL) {
258 System.err.println("HTTP server does not support Range request headers");
259 is.close();
260 return null;
261 }
262 return is;
Constantin Kaplinskyd2af0822008-06-21 11:41:42 +0000263 }
264
Constantin Kaplinsky2859fdc2008-06-19 16:07:52 +0000265}