Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 abstract class TlsSocket implements Socket { | |
| 6 /** | |
| 7 * Constructs a new secure socket and connect it to the given | |
| 8 * host on the given port. The returned socket is not yet connected | |
| 9 * but ready for registration of callbacks. | |
| 10 */ | |
| 11 factory TlsSocket(String host, int port) => new _TlsSocket(host, port); | |
| 12 | |
| 13 /** | |
| 14 * Initializes the TLS library with the path to a certificate database | |
| 15 * containing root certificates for verifying certificate paths on | |
| 16 * client connections, and server certificates to provide on server | |
| 17 * connections. | |
| 18 */ | |
| 19 external static void setCertificateDatabase(String pkcertDirectory); | |
| 20 } | |
| 21 | |
| 22 class _TlsSocket implements TlsSocket { | |
|
Anders Johnsen
2012/11/01 07:27:21
A higher level question: Would it make sense to ex
Bill Hesse
2012/11/11 22:34:34
I can't see a way anyone could really use it, beca
| |
| 23 static final int _BUFFER_SIZE = 2048; | |
|
Mads Ager (google)
2012/11/01 10:09:01
The name here is using an '_' in from of it and no
Bill Hesse
2012/11/11 22:34:34
Constant removed.
On 2012/11/01 10:09:01, Mads Age
| |
| 24 | |
| 25 // Status states | |
| 26 static final int NOT_CONNECTED = 200; | |
| 27 static final int HANDSHAKE = 201; | |
| 28 static final int CONNECTED = 202; | |
| 29 static final int CLOSED = 203; | |
| 30 | |
| 31 // Buffer identifiers. | |
| 32 static final int kReadPlaintext = 0; | |
|
Mads Ager (google)
2012/11/01 10:09:01
This is using a different naming style than the ot
Bill Hesse
2012/11/11 22:34:34
Should we be consistent? Which do you prefer?
It
Mads Ager (google)
2012/11/12 11:39:08
Of course we should be consistent!
The are all ju
Bob Nystrom
2012/11/14 18:06:36
Style nit. These are technically not constants, th
| |
| 33 static final int kWritePlaintext = 1; | |
| 34 static final int kReadEncrypted = 2; | |
| 35 static final int kWriteEncrypted = 3; | |
| 36 static final int kNumBuffers = 4; | |
| 37 | |
| 38 // Constructs a new secure client socket. | |
| 39 _TlsSocket(String host, int port) | |
| 40 : _socket = new Socket(host, port), | |
| 41 _tlsFilter = new _TlsFilter() { | |
| 42 _socket.onConnect = _tlsConnectHandler; | |
| 43 _socket.onWrite = _tlsWriteHandler; | |
| 44 _socket.onData = _tlsDataHandler; | |
| 45 _socket.onClosed = _tlsCloseHandler; | |
| 46 _tlsFilter.init(); | |
| 47 _tlsFilter.registerHandshakeCallbacks(_tlsHandshakeStartHandler, | |
| 48 _tlsHandshakeFinishHandler); | |
| 49 } | |
| 50 | |
| 51 void set onConnect(void callback()) { | |
| 52 _socketConnectHandler = callback; | |
| 53 } | |
| 54 | |
| 55 void set onWrite(void callback()) { | |
| 56 _socketWriteHandler = callback; | |
| 57 // Reset the one-shot onWrite handler. | |
| 58 _socket.onWrite = _tlsWriteHandler; | |
| 59 } | |
| 60 | |
| 61 void set onData(void callback()) { | |
| 62 _socketDataHandler = callback; | |
| 63 } | |
| 64 | |
| 65 void set onClosed(void callback()) { | |
| 66 _socketCloseHandler = callback; | |
| 67 } | |
| 68 | |
| 69 void _tlsConnectHandler() { | |
| 70 _connectPending = true; | |
| 71 _tlsFilter.connect(); | |
| 72 } | |
| 73 | |
| 74 void _tlsWriteHandler() { | |
| 75 if (_status == HANDSHAKE) { | |
| 76 _writeEncryptedData(); | |
| 77 _readEncryptedData(); | |
| 78 _tlsFilter.connect(); | |
| 79 // Only do this if we have more data to write. | |
| 80 if (_tlsFilter.buffers[kWriteEncrypted].length > 0) { | |
| 81 _socket.onWrite = _tlsWriteHandler; | |
| 82 } | |
| 83 } else if (_status == CONNECTED) { | |
| 84 if (_socketWriteHandler != null) { | |
| 85 _socketWriteHandler(); | |
| 86 } | |
| 87 } | |
| 88 } | |
| 89 | |
| 90 void _tlsDataHandler() { | |
| 91 if (_status == HANDSHAKE) { | |
| 92 _readEncryptedData(); | |
| 93 _writeEncryptedData(); | |
| 94 _tlsFilter.connect(); | |
| 95 _socket.onWrite = _tlsWriteHandler; | |
| 96 } else { | |
| 97 if (scheduledDataEvent != null) { | |
| 98 scheduledDataEvent.cancel(); | |
| 99 scheduledDataEvent = null; | |
| 100 } | |
| 101 if (_socketDataHandler != null) { | |
| 102 _readEncryptedData(); | |
| 103 _socketDataHandler(); | |
| 104 } | |
| 105 } | |
| 106 } | |
| 107 | |
| 108 void _tlsCloseHandler() { | |
| 109 _socketClosed = true; | |
| 110 _status = CLOSED; | |
| 111 _socket.close(); | |
| 112 if (_filterEmpty) { | |
| 113 _fireCloseEvent(); | |
| 114 } else { | |
| 115 _fireCloseEventPending = true; | |
| 116 } | |
| 117 } | |
| 118 | |
| 119 void _tlsHandshakeStartHandler() { | |
| 120 _status = HANDSHAKE; | |
| 121 _socket.onWrite = _tlsWriteHandler; | |
| 122 } | |
| 123 | |
| 124 void _tlsHandshakeFinishHandler() { | |
| 125 _status = CONNECTED; | |
| 126 if (_connectPending && _socketConnectHandler != null) { | |
| 127 _connectPending = false; | |
| 128 _socketConnectHandler(); | |
| 129 } | |
| 130 } | |
| 131 | |
| 132 void _fireCloseEvent() { | |
| 133 _fireCloseEventPending = false; | |
| 134 _tlsFilter.destroy(); | |
| 135 _tlsFilter = null; | |
| 136 if (scheduledDataEvent != null) { | |
| 137 scheduledDataEvent.cancel(); | |
| 138 } | |
| 139 if (_socketCloseHandler != null) { | |
| 140 _socketCloseHandler(); | |
| 141 } | |
| 142 } | |
| 143 | |
| 144 void close([bool halfClose]) { | |
| 145 _socket.close(halfClose); | |
| 146 } | |
| 147 | |
| 148 int readList(List<int> data, int offset, int bytes) { | |
| 149 print("Entering readList"); | |
|
Mads Ager (google)
2012/11/01 10:09:01
Debug printing.
Bill Hesse
2012/11/11 22:34:34
Done.
| |
| 150 _readEncryptedData(); | |
| 151 if (offset < 0 || bytes < 0 || offset + bytes > data.length) { | |
| 152 throw new IllegalArgumentException( | |
| 153 "Invalid offset or bytes in TlsSocket.readList"); | |
| 154 } | |
| 155 int bytesRead = 0; | |
| 156 var buffer = _tlsFilter.buffers[kReadPlaintext]; | |
| 157 if (buffer.length == 0 && buffer.start != 0) { | |
| 158 throw "Unexpected buffer state in TlsSocket.readList"; | |
| 159 } | |
| 160 if (buffer.length > 0) { | |
| 161 int toRead = min(bytes, buffer.length); | |
| 162 data.setRange(offset, toRead, buffer.data, buffer.start); | |
| 163 buffer.advanceStart(toRead); | |
| 164 bytesRead += toRead; | |
| 165 } | |
| 166 print("Before processBuffer(readPlaintext)"); | |
| 167 int newBytes = _tlsFilter.processBuffer(kReadPlaintext); | |
| 168 if (newBytes > 0) { | |
| 169 buffer.length += newBytes; | |
| 170 } | |
| 171 if (bytes - bytesRead > 0 && buffer.length > 0) { | |
| 172 int toRead = min(bytes - bytesRead, buffer.length); | |
| 173 data.setRange(offset + bytesRead, toRead, buffer.data, buffer.start); | |
| 174 buffer.advanceStart(toRead); | |
| 175 bytesRead += toRead; | |
| 176 } | |
| 177 | |
| 178 // If bytesRead is 0, then something is blocked or empty, and | |
| 179 // we are guaranteed an event when it becomes unblocked. | |
| 180 // Otherwise, give an event if there is data available, and | |
| 181 // there has been a read call since the last data event. | |
| 182 // This gives the invariant that: | |
| 183 // If there is data available, and there has been a read after the | |
| 184 // last data event (or no previous one fired), then we are guaranteed | |
| 185 // to get a data event. | |
| 186 _filterEmpty = (bytesRead == 0); | |
| 187 if (bytesRead > 0 && scheduledDataEvent == null) { | |
| 188 scheduledDataEvent = new Timer(0, (_) => _tlsDataHandler()); | |
| 189 } else if (bytesRead == 0) { | |
| 190 if (_fireCloseEventPending) { | |
| 191 _fireCloseEvent(); | |
| 192 } else if (scheduledDataEvent != null) { | |
| 193 scheduledDataEvent.cancel(); | |
| 194 scheduledDataEvent = null; | |
| 195 } | |
| 196 } | |
| 197 return bytesRead; | |
| 198 } | |
| 199 | |
| 200 | |
| 201 // Write the data to the socket, and flush it as much as possible | |
| 202 // without blocking. If not all the data is written, enable the | |
| 203 // onWrite event. If data is not all flushed, add handlers to all | |
| 204 // relevant events. | |
| 205 int writeList(List<int> data, int offset, int bytes) { | |
| 206 _writeEncryptedData(); // Tries to flush all post-filter stages. | |
| 207 var buffer = _tlsFilter.buffers[kWritePlaintext]; | |
| 208 if (bytes > buffer.free) { | |
| 209 bytes = buffer.free; | |
| 210 } | |
| 211 if (bytes > 0) { | |
| 212 buffer.data.setRange(buffer.start + buffer.length, bytes, data, offset); | |
| 213 buffer.length += bytes; | |
| 214 } | |
| 215 int bytesWritten = _tlsFilter.processBuffer(kWritePlaintext); | |
| 216 buffer.advanceStart(bytesWritten); | |
| 217 print("bytesWritten $bytesWritten"); | |
| 218 _readEncryptedData(); | |
| 219 _writeEncryptedData(); | |
| 220 return bytes; | |
| 221 } | |
| 222 | |
| 223 void _readEncryptedData() { | |
| 224 // Read from the socket and write to the filter. | |
| 225 print("Entering readEncryptedData"); | |
|
Anders Johnsen
2012/11/01 07:27:21
Debug info.
| |
| 226 var buffer = _tlsFilter.buffers[kReadEncrypted]; | |
| 227 while (true) { | |
| 228 print(" buffer.length: ${buffer.length}"); | |
|
Anders Johnsen
2012/11/01 07:27:21
Debug info.
| |
| 229 if (buffer.length > 0) { | |
| 230 int bytes = _tlsFilter.processBuffer(kReadEncrypted); | |
| 231 if (bytes > 0) { | |
| 232 buffer.advanceStart(bytes); | |
| 233 } else { | |
| 234 break; | |
| 235 } | |
| 236 } else if (!_socketClosed) { | |
| 237 int bytes = _socket.readList(buffer.data, | |
| 238 buffer.start + buffer.length, | |
| 239 buffer.free); | |
| 240 print(" bytes read from socket: $bytes"); | |
|
Anders Johnsen
2012/11/01 07:27:21
Debug info.
| |
| 241 if (bytes <= 0) break; | |
| 242 buffer.length += bytes; | |
| 243 } else { | |
| 244 break; // Socket is closed and read buffer is empty. | |
| 245 } | |
| 246 } | |
| 247 } | |
| 248 | |
| 249 void _writeEncryptedData() { | |
| 250 // Write from the filter to the socket. | |
| 251 var buffer = _tlsFilter.buffers[kWriteEncrypted]; | |
| 252 while (true) { | |
| 253 print("top of while loop: buffer.length = ${buffer.length}"); | |
|
Anders Johnsen
2012/11/01 07:27:21
Debug info
| |
| 254 if (buffer.length > 0) { | |
| 255 int bytes = _socket.writeList(buffer.data, buffer.start, buffer.length); | |
| 256 if (bytes <= 0) break; | |
| 257 buffer.advanceStart(bytes); | |
| 258 } else { | |
| 259 if (buffer.start != 0 || buffer.length != 0) { | |
| 260 throw "Unexpected state in _writeEncryptedData"; | |
| 261 } | |
| 262 int bytes = _tlsFilter.processBuffer(kWriteEncrypted); | |
| 263 print("foo writeEncrypted processBuffer returned $bytes"); | |
|
Anders Johnsen
2012/11/01 07:27:21
Debug info.
| |
| 264 if (bytes <= 0) break; | |
| 265 buffer.length += bytes; | |
| 266 } | |
| 267 } | |
| 268 } | |
| 269 | |
| 270 // _TlsSocket cannot extend _Socket and use _Socket's factory constructor. | |
| 271 Socket _socket; | |
| 272 | |
| 273 var _status = NOT_CONNECTED; | |
| 274 bool _socketClosed = false; | |
| 275 bool _filterEmpty = false; | |
| 276 bool _connectPending = false; | |
| 277 bool _fireCloseEventPending = false; | |
| 278 Function _socketConnectHandler; | |
| 279 Function _socketWriteHandler; | |
| 280 Function _socketDataHandler; | |
| 281 Function _socketCloseHandler; | |
| 282 Timer scheduledDataEvent; | |
| 283 | |
| 284 var _tlsFilter; | |
| 285 } | |
| 286 | |
| 287 class _TlsExternalBuffer { | |
| 288 static final int kSize = 8 * 1024; | |
| 289 _TlsExternalBuffer() : start = 0, length = 0; | |
| 290 | |
| 291 void advanceStart(int numBytes) { | |
| 292 start += numBytes; | |
| 293 length -= numBytes; | |
| 294 if (length == 0) { | |
| 295 start = 0; | |
| 296 } | |
| 297 } | |
| 298 | |
| 299 int get free() => kSize - start - length; | |
|
Anders Johnsen
2012/11/01 07:27:21
remove ().
Anders Johnsen
2012/11/01 07:27:21
Please add (...) to explicit express precedence fo
Bill Hesse
2012/11/11 22:34:34
Done.
| |
| 300 | |
| 301 List data; // This will be a ExternalByteArray, backed by C allocated data. | |
| 302 int start; | |
| 303 int length; | |
| 304 } | |
| 305 | |
| 306 /** | |
| 307 * _TlsFilter wraps a filter that encrypts and decrypts data travelling | |
| 308 * over a TLS encrypted socket. The filter also handles the handshaking | |
| 309 * and certificate verification. | |
| 310 * | |
| 311 * The filter exposes its input and output buffers as Dart objects that | |
| 312 * are backed by an external C array of bytes, so that both Dart code and | |
| 313 * native code can access the same data. | |
| 314 */ | |
| 315 class _TlsFilter extends NativeFieldWrapperClass1 { | |
|
Anders Johnsen
2012/11/01 07:27:21
AFAIK, this is VM specific, I think this should be
Bill Hesse
2012/11/11 22:34:34
Done.
We have created an abstract interface TlsFi
| |
| 316 _TlsFilter() { | |
| 317 buffers = new List<_TlsExternalBuffer>(_TlsSocket.kNumBuffers); | |
| 318 for (int i = 0; i < _TlsSocket.kNumBuffers; ++i) { | |
| 319 buffers[i] = new _TlsExternalBuffer(); | |
| 320 } | |
| 321 } | |
| 322 | |
| 323 external void init(); | |
| 324 | |
| 325 external void connect(); | |
| 326 | |
| 327 external void registerHandshakeCallbacks(Function startHandshakeHandler, | |
| 328 Function finishHandshakeHandler); | |
| 329 external int processBuffer(int bufferIndex); | |
| 330 external void destroy(); | |
| 331 | |
| 332 List<_TlsExternalBuffer> buffers; | |
| 333 } | |
| OLD | NEW |