• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "jingle/glue/chrome_async_socket.h"
6 
7 #include <deque>
8 #include <string>
9 
10 #include "base/basictypes.h"
11 #include "base/logging.h"
12 #include "base/memory/scoped_ptr.h"
13 #include "base/message_loop/message_loop.h"
14 #include "jingle/glue/resolving_client_socket_factory.h"
15 #include "net/base/address_list.h"
16 #include "net/base/net_errors.h"
17 #include "net/base/net_util.h"
18 #include "net/cert/mock_cert_verifier.h"
19 #include "net/http/transport_security_state.h"
20 #include "net/socket/socket_test_util.h"
21 #include "net/socket/ssl_client_socket.h"
22 #include "net/ssl/ssl_config_service.h"
23 #include "net/url_request/url_request_context_getter.h"
24 #include "testing/gtest/include/gtest/gtest.h"
25 #include "third_party/libjingle/source/talk/base/ipaddress.h"
26 #include "third_party/libjingle/source/talk/base/sigslot.h"
27 #include "third_party/libjingle/source/talk/base/socketaddress.h"
28 
29 namespace jingle_glue {
30 
31 namespace {
32 
33 // Data provider that handles reads/writes for ChromeAsyncSocket.
34 class AsyncSocketDataProvider : public net::SocketDataProvider {
35  public:
AsyncSocketDataProvider()36   AsyncSocketDataProvider() : has_pending_read_(false) {}
37 
~AsyncSocketDataProvider()38   virtual ~AsyncSocketDataProvider() {
39     EXPECT_TRUE(writes_.empty());
40     EXPECT_TRUE(reads_.empty());
41   }
42 
43   // If there's no read, sets the "has pending read" flag.  Otherwise,
44   // pops the next read.
GetNextRead()45   virtual net::MockRead GetNextRead() OVERRIDE {
46     if (reads_.empty()) {
47       DCHECK(!has_pending_read_);
48       has_pending_read_ = true;
49       const net::MockRead pending_read(net::SYNCHRONOUS, net::ERR_IO_PENDING);
50       return pending_read;
51     }
52     net::MockRead mock_read = reads_.front();
53     reads_.pop_front();
54     return mock_read;
55   }
56 
57   // Simply pops the next write and, if applicable, compares it to
58   // |data|.
OnWrite(const std::string & data)59   virtual net::MockWriteResult OnWrite(const std::string& data) OVERRIDE {
60     DCHECK(!writes_.empty());
61     net::MockWrite mock_write = writes_.front();
62     writes_.pop_front();
63     if (mock_write.result != net::OK) {
64       return net::MockWriteResult(mock_write.mode, mock_write.result);
65     }
66     std::string expected_data(mock_write.data, mock_write.data_len);
67     EXPECT_EQ(expected_data, data);
68     if (expected_data != data) {
69       return net::MockWriteResult(net::SYNCHRONOUS, net::ERR_UNEXPECTED);
70     }
71     return net::MockWriteResult(mock_write.mode, data.size());
72   }
73 
74   // We ignore resets so we can pre-load the socket data provider with
75   // read/write events.
Reset()76   virtual void Reset() OVERRIDE {}
77 
78   // If there is a pending read, completes it with the given read.
79   // Otherwise, queues up the given read.
AddRead(const net::MockRead & mock_read)80   void AddRead(const net::MockRead& mock_read) {
81     DCHECK_NE(mock_read.result, net::ERR_IO_PENDING);
82     if (has_pending_read_) {
83       socket()->OnReadComplete(mock_read);
84       has_pending_read_ = false;
85       return;
86     }
87     reads_.push_back(mock_read);
88   }
89 
90   // Simply queues up the given write.
AddWrite(const net::MockWrite & mock_write)91   void AddWrite(const net::MockWrite& mock_write) {
92     writes_.push_back(mock_write);
93   }
94 
95  private:
96   std::deque<net::MockRead> reads_;
97   bool has_pending_read_;
98 
99   std::deque<net::MockWrite> writes_;
100 
101   DISALLOW_COPY_AND_ASSIGN(AsyncSocketDataProvider);
102 };
103 
104 class MockXmppClientSocketFactory : public ResolvingClientSocketFactory {
105  public:
MockXmppClientSocketFactory(net::ClientSocketFactory * mock_client_socket_factory,const net::AddressList & address_list)106   MockXmppClientSocketFactory(
107       net::ClientSocketFactory* mock_client_socket_factory,
108       const net::AddressList& address_list)
109           : mock_client_socket_factory_(mock_client_socket_factory),
110             address_list_(address_list),
111             cert_verifier_(new net::MockCertVerifier),
112             transport_security_state_(new net::TransportSecurityState) {
113   }
114 
115   // ResolvingClientSocketFactory implementation.
CreateTransportClientSocket(const net::HostPortPair & host_and_port)116   virtual scoped_ptr<net::StreamSocket> CreateTransportClientSocket(
117       const net::HostPortPair& host_and_port) OVERRIDE {
118     return mock_client_socket_factory_->CreateTransportClientSocket(
119         address_list_, NULL, net::NetLog::Source());
120   }
121 
CreateSSLClientSocket(scoped_ptr<net::ClientSocketHandle> transport_socket,const net::HostPortPair & host_and_port)122   virtual scoped_ptr<net::SSLClientSocket> CreateSSLClientSocket(
123       scoped_ptr<net::ClientSocketHandle> transport_socket,
124       const net::HostPortPair& host_and_port) OVERRIDE {
125     net::SSLClientSocketContext context;
126     context.cert_verifier = cert_verifier_.get();
127     context.transport_security_state = transport_security_state_.get();
128     return mock_client_socket_factory_->CreateSSLClientSocket(
129         transport_socket.Pass(), host_and_port, ssl_config_, context);
130   }
131 
132  private:
133   scoped_ptr<net::ClientSocketFactory> mock_client_socket_factory_;
134   net::AddressList address_list_;
135   net::SSLConfig ssl_config_;
136   scoped_ptr<net::CertVerifier> cert_verifier_;
137   scoped_ptr<net::TransportSecurityState> transport_security_state_;
138 };
139 
140 class ChromeAsyncSocketTest
141     : public testing::Test,
142       public sigslot::has_slots<> {
143  protected:
ChromeAsyncSocketTest()144   ChromeAsyncSocketTest()
145       : ssl_socket_data_provider_(net::ASYNC, net::OK),
146         addr_("localhost", 35) {}
147 
~ChromeAsyncSocketTest()148   virtual ~ChromeAsyncSocketTest() {}
149 
SetUp()150   virtual void SetUp() {
151     scoped_ptr<net::MockClientSocketFactory> mock_client_socket_factory(
152         new net::MockClientSocketFactory());
153     mock_client_socket_factory->AddSocketDataProvider(
154         &async_socket_data_provider_);
155     mock_client_socket_factory->AddSSLSocketDataProvider(
156         &ssl_socket_data_provider_);
157 
158     // Fake DNS resolution for |addr_| and pass it to the factory.
159     net::IPAddressNumber resolved_addr;
160     EXPECT_TRUE(net::ParseIPLiteralToNumber("127.0.0.1", &resolved_addr));
161     const net::AddressList address_list =
162         net::AddressList::CreateFromIPAddress(resolved_addr, addr_.port());
163     scoped_ptr<MockXmppClientSocketFactory> mock_xmpp_client_socket_factory(
164         new MockXmppClientSocketFactory(
165             mock_client_socket_factory.release(),
166             address_list));
167     chrome_async_socket_.reset(
168         new ChromeAsyncSocket(mock_xmpp_client_socket_factory.release(),
169                               14, 20)),
170 
171     chrome_async_socket_->SignalConnected.connect(
172         this, &ChromeAsyncSocketTest::OnConnect);
173     chrome_async_socket_->SignalSSLConnected.connect(
174         this, &ChromeAsyncSocketTest::OnSSLConnect);
175     chrome_async_socket_->SignalClosed.connect(
176         this, &ChromeAsyncSocketTest::OnClose);
177     chrome_async_socket_->SignalRead.connect(
178         this, &ChromeAsyncSocketTest::OnRead);
179     chrome_async_socket_->SignalError.connect(
180         this, &ChromeAsyncSocketTest::OnError);
181   }
182 
TearDown()183   virtual void TearDown() {
184     // Run any tasks that we forgot to pump.
185     message_loop_.RunUntilIdle();
186     ExpectClosed();
187     ExpectNoSignal();
188     chrome_async_socket_.reset();
189   }
190 
191   enum Signal {
192     SIGNAL_CONNECT,
193     SIGNAL_SSL_CONNECT,
194     SIGNAL_CLOSE,
195     SIGNAL_READ,
196     SIGNAL_ERROR,
197   };
198 
199   // Helper struct that records the state at the time of a signal.
200 
201   struct SignalSocketState {
SignalSocketStatejingle_glue::__anon4fb7544f0111::ChromeAsyncSocketTest::SignalSocketState202     SignalSocketState()
203         : signal(SIGNAL_ERROR),
204           state(ChromeAsyncSocket::STATE_CLOSED),
205           error(ChromeAsyncSocket::ERROR_NONE),
206           net_error(net::OK) {}
207 
SignalSocketStatejingle_glue::__anon4fb7544f0111::ChromeAsyncSocketTest::SignalSocketState208     SignalSocketState(
209         Signal signal,
210         ChromeAsyncSocket::State state,
211         ChromeAsyncSocket::Error error,
212         net::Error net_error)
213         : signal(signal),
214           state(state),
215           error(error),
216           net_error(net_error) {}
217 
IsEqualjingle_glue::__anon4fb7544f0111::ChromeAsyncSocketTest::SignalSocketState218     bool IsEqual(const SignalSocketState& other) const {
219       return
220           (signal == other.signal) &&
221           (state == other.state) &&
222           (error == other.error) &&
223           (net_error == other.net_error);
224     }
225 
FromAsyncSocketjingle_glue::__anon4fb7544f0111::ChromeAsyncSocketTest::SignalSocketState226     static SignalSocketState FromAsyncSocket(
227         Signal signal,
228         buzz::AsyncSocket* async_socket) {
229       return SignalSocketState(signal,
230                                async_socket->state(),
231                                async_socket->error(),
232                                static_cast<net::Error>(
233                                    async_socket->GetError()));
234     }
235 
NoErrorjingle_glue::__anon4fb7544f0111::ChromeAsyncSocketTest::SignalSocketState236     static SignalSocketState NoError(
237         Signal signal, buzz::AsyncSocket::State state) {
238         return SignalSocketState(signal, state,
239                                  buzz::AsyncSocket::ERROR_NONE,
240                                  net::OK);
241     }
242 
243     Signal signal;
244     ChromeAsyncSocket::State state;
245     ChromeAsyncSocket::Error error;
246     net::Error net_error;
247   };
248 
CaptureSocketState(Signal signal)249   void CaptureSocketState(Signal signal) {
250     signal_socket_states_.push_back(
251         SignalSocketState::FromAsyncSocket(
252             signal, chrome_async_socket_.get()));
253   }
254 
OnConnect()255   void OnConnect() {
256     CaptureSocketState(SIGNAL_CONNECT);
257   }
258 
OnSSLConnect()259   void OnSSLConnect() {
260     CaptureSocketState(SIGNAL_SSL_CONNECT);
261   }
262 
OnClose()263   void OnClose() {
264     CaptureSocketState(SIGNAL_CLOSE);
265   }
266 
OnRead()267   void OnRead() {
268     CaptureSocketState(SIGNAL_READ);
269   }
270 
OnError()271   void OnError() {
272     ADD_FAILURE();
273   }
274 
275   // State expect functions.
276 
ExpectState(ChromeAsyncSocket::State state,ChromeAsyncSocket::Error error,net::Error net_error)277   void ExpectState(ChromeAsyncSocket::State state,
278                    ChromeAsyncSocket::Error error,
279                    net::Error net_error) {
280     EXPECT_EQ(state, chrome_async_socket_->state());
281     EXPECT_EQ(error, chrome_async_socket_->error());
282     EXPECT_EQ(net_error, chrome_async_socket_->GetError());
283   }
284 
ExpectNonErrorState(ChromeAsyncSocket::State state)285   void ExpectNonErrorState(ChromeAsyncSocket::State state) {
286     ExpectState(state, ChromeAsyncSocket::ERROR_NONE, net::OK);
287   }
288 
ExpectErrorState(ChromeAsyncSocket::State state,ChromeAsyncSocket::Error error)289   void ExpectErrorState(ChromeAsyncSocket::State state,
290                         ChromeAsyncSocket::Error error) {
291     ExpectState(state, error, net::OK);
292   }
293 
ExpectClosed()294   void ExpectClosed() {
295     ExpectNonErrorState(ChromeAsyncSocket::STATE_CLOSED);
296   }
297 
298   // Signal expect functions.
299 
ExpectNoSignal()300   void ExpectNoSignal() {
301     if (!signal_socket_states_.empty()) {
302       ADD_FAILURE() << signal_socket_states_.front().signal;
303     }
304   }
305 
ExpectSignalSocketState(SignalSocketState expected_signal_socket_state)306   void ExpectSignalSocketState(
307       SignalSocketState expected_signal_socket_state) {
308     if (signal_socket_states_.empty()) {
309       ADD_FAILURE() << expected_signal_socket_state.signal;
310       return;
311     }
312     EXPECT_TRUE(expected_signal_socket_state.IsEqual(
313         signal_socket_states_.front()))
314         << signal_socket_states_.front().signal;
315     signal_socket_states_.pop_front();
316   }
317 
ExpectReadSignal()318   void ExpectReadSignal() {
319     ExpectSignalSocketState(
320         SignalSocketState::NoError(
321             SIGNAL_READ, ChromeAsyncSocket::STATE_OPEN));
322   }
323 
ExpectSSLConnectSignal()324   void ExpectSSLConnectSignal() {
325     ExpectSignalSocketState(
326         SignalSocketState::NoError(SIGNAL_SSL_CONNECT,
327                                    ChromeAsyncSocket::STATE_TLS_OPEN));
328   }
329 
ExpectSSLReadSignal()330   void ExpectSSLReadSignal() {
331     ExpectSignalSocketState(
332         SignalSocketState::NoError(
333             SIGNAL_READ, ChromeAsyncSocket::STATE_TLS_OPEN));
334   }
335 
336   // Open/close utility functions.
337 
DoOpenClosed()338   void DoOpenClosed() {
339     ExpectClosed();
340     async_socket_data_provider_.set_connect_data(
341         net::MockConnect(net::SYNCHRONOUS, net::OK));
342     EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
343     ExpectNonErrorState(ChromeAsyncSocket::STATE_CONNECTING);
344 
345     message_loop_.RunUntilIdle();
346     // We may not necessarily be open; may have been other events
347     // queued up.
348     ExpectSignalSocketState(
349         SignalSocketState::NoError(
350             SIGNAL_CONNECT, ChromeAsyncSocket::STATE_OPEN));
351   }
352 
DoCloseOpened(SignalSocketState expected_signal_socket_state)353   void DoCloseOpened(SignalSocketState expected_signal_socket_state) {
354     // We may be in an error state, so just compare state().
355     EXPECT_EQ(ChromeAsyncSocket::STATE_OPEN, chrome_async_socket_->state());
356     EXPECT_TRUE(chrome_async_socket_->Close());
357     ExpectSignalSocketState(expected_signal_socket_state);
358     ExpectNonErrorState(ChromeAsyncSocket::STATE_CLOSED);
359   }
360 
DoCloseOpenedNoError()361   void DoCloseOpenedNoError() {
362     DoCloseOpened(
363         SignalSocketState::NoError(
364             SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
365   }
366 
DoSSLOpenClosed()367   void DoSSLOpenClosed() {
368     const char kDummyData[] = "dummy_data";
369     async_socket_data_provider_.AddRead(net::MockRead(kDummyData));
370     DoOpenClosed();
371     ExpectReadSignal();
372     EXPECT_EQ(kDummyData, DrainRead(1));
373 
374     EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
375     message_loop_.RunUntilIdle();
376     ExpectSSLConnectSignal();
377     ExpectNoSignal();
378     ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_OPEN);
379   }
380 
DoSSLCloseOpened(SignalSocketState expected_signal_socket_state)381   void DoSSLCloseOpened(SignalSocketState expected_signal_socket_state) {
382     // We may be in an error state, so just compare state().
383     EXPECT_EQ(ChromeAsyncSocket::STATE_TLS_OPEN,
384               chrome_async_socket_->state());
385     EXPECT_TRUE(chrome_async_socket_->Close());
386     ExpectSignalSocketState(expected_signal_socket_state);
387     ExpectNonErrorState(ChromeAsyncSocket::STATE_CLOSED);
388   }
389 
DoSSLCloseOpenedNoError()390   void DoSSLCloseOpenedNoError() {
391     DoSSLCloseOpened(
392         SignalSocketState::NoError(
393             SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
394   }
395 
396   // Read utility fucntions.
397 
DrainRead(size_t buf_size)398   std::string DrainRead(size_t buf_size) {
399     std::string read;
400     scoped_ptr<char[]> buf(new char[buf_size]);
401     size_t len_read;
402     while (true) {
403       bool success =
404           chrome_async_socket_->Read(buf.get(), buf_size, &len_read);
405       if (!success) {
406         ADD_FAILURE();
407         break;
408       }
409       if (len_read == 0U) {
410         break;
411       }
412       read.append(buf.get(), len_read);
413     }
414     return read;
415   }
416 
417   // ChromeAsyncSocket expects a message loop.
418   base::MessageLoop message_loop_;
419 
420   AsyncSocketDataProvider async_socket_data_provider_;
421   net::SSLSocketDataProvider ssl_socket_data_provider_;
422 
423   scoped_ptr<ChromeAsyncSocket> chrome_async_socket_;
424   std::deque<SignalSocketState> signal_socket_states_;
425   const talk_base::SocketAddress addr_;
426 
427  private:
428   DISALLOW_COPY_AND_ASSIGN(ChromeAsyncSocketTest);
429 };
430 
TEST_F(ChromeAsyncSocketTest,InitialState)431 TEST_F(ChromeAsyncSocketTest, InitialState) {
432   ExpectClosed();
433   ExpectNoSignal();
434 }
435 
TEST_F(ChromeAsyncSocketTest,EmptyClose)436 TEST_F(ChromeAsyncSocketTest, EmptyClose) {
437   ExpectClosed();
438   EXPECT_TRUE(chrome_async_socket_->Close());
439   ExpectClosed();
440 }
441 
TEST_F(ChromeAsyncSocketTest,ImmediateConnectAndClose)442 TEST_F(ChromeAsyncSocketTest, ImmediateConnectAndClose) {
443   DoOpenClosed();
444 
445   ExpectNonErrorState(ChromeAsyncSocket::STATE_OPEN);
446 
447   DoCloseOpenedNoError();
448 }
449 
450 // After this, no need to test immediate successful connect and
451 // Close() so thoroughly.
452 
TEST_F(ChromeAsyncSocketTest,DoubleClose)453 TEST_F(ChromeAsyncSocketTest, DoubleClose) {
454   DoOpenClosed();
455 
456   EXPECT_TRUE(chrome_async_socket_->Close());
457   ExpectClosed();
458   ExpectSignalSocketState(
459       SignalSocketState::NoError(
460           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
461 
462   EXPECT_TRUE(chrome_async_socket_->Close());
463   ExpectClosed();
464 }
465 
TEST_F(ChromeAsyncSocketTest,NoHostnameConnect)466 TEST_F(ChromeAsyncSocketTest, NoHostnameConnect) {
467   talk_base::IPAddress ip_address;
468   EXPECT_TRUE(talk_base::IPFromString("127.0.0.1", &ip_address));
469   const talk_base::SocketAddress no_hostname_addr(ip_address, addr_.port());
470   EXPECT_FALSE(chrome_async_socket_->Connect(no_hostname_addr));
471   ExpectErrorState(ChromeAsyncSocket::STATE_CLOSED,
472                    ChromeAsyncSocket::ERROR_DNS);
473 
474   EXPECT_TRUE(chrome_async_socket_->Close());
475   ExpectClosed();
476 }
477 
TEST_F(ChromeAsyncSocketTest,ZeroPortConnect)478 TEST_F(ChromeAsyncSocketTest, ZeroPortConnect) {
479   const talk_base::SocketAddress zero_port_addr(addr_.hostname(), 0);
480   EXPECT_FALSE(chrome_async_socket_->Connect(zero_port_addr));
481   ExpectErrorState(ChromeAsyncSocket::STATE_CLOSED,
482                    ChromeAsyncSocket::ERROR_DNS);
483 
484   EXPECT_TRUE(chrome_async_socket_->Close());
485   ExpectClosed();
486 }
487 
TEST_F(ChromeAsyncSocketTest,DoubleConnect)488 TEST_F(ChromeAsyncSocketTest, DoubleConnect) {
489   EXPECT_DEBUG_DEATH({
490     DoOpenClosed();
491 
492     EXPECT_FALSE(chrome_async_socket_->Connect(addr_));
493     ExpectErrorState(ChromeAsyncSocket::STATE_OPEN,
494                      ChromeAsyncSocket::ERROR_WRONGSTATE);
495 
496     DoCloseOpened(
497         SignalSocketState(SIGNAL_CLOSE,
498                           ChromeAsyncSocket::STATE_CLOSED,
499                           ChromeAsyncSocket::ERROR_WRONGSTATE,
500                           net::OK));
501   }, "non-closed socket");
502 }
503 
TEST_F(ChromeAsyncSocketTest,ImmediateConnectCloseBeforeRead)504 TEST_F(ChromeAsyncSocketTest, ImmediateConnectCloseBeforeRead) {
505   DoOpenClosed();
506 
507   EXPECT_TRUE(chrome_async_socket_->Close());
508   ExpectClosed();
509   ExpectSignalSocketState(
510       SignalSocketState::NoError(
511           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
512 
513   message_loop_.RunUntilIdle();
514 }
515 
TEST_F(ChromeAsyncSocketTest,HangingConnect)516 TEST_F(ChromeAsyncSocketTest, HangingConnect) {
517   EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
518   ExpectNonErrorState(ChromeAsyncSocket::STATE_CONNECTING);
519   ExpectNoSignal();
520 
521   EXPECT_TRUE(chrome_async_socket_->Close());
522   ExpectClosed();
523   ExpectSignalSocketState(
524       SignalSocketState::NoError(
525           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
526 }
527 
TEST_F(ChromeAsyncSocketTest,PendingConnect)528 TEST_F(ChromeAsyncSocketTest, PendingConnect) {
529   async_socket_data_provider_.set_connect_data(
530       net::MockConnect(net::ASYNC, net::OK));
531   EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
532   ExpectNonErrorState(ChromeAsyncSocket::STATE_CONNECTING);
533   ExpectNoSignal();
534 
535   message_loop_.RunUntilIdle();
536   ExpectNonErrorState(ChromeAsyncSocket::STATE_OPEN);
537   ExpectSignalSocketState(
538       SignalSocketState::NoError(
539           SIGNAL_CONNECT, ChromeAsyncSocket::STATE_OPEN));
540   ExpectNoSignal();
541 
542   message_loop_.RunUntilIdle();
543 
544   DoCloseOpenedNoError();
545 }
546 
547 // After this no need to test successful pending connect so
548 // thoroughly.
549 
TEST_F(ChromeAsyncSocketTest,PendingConnectCloseBeforeRead)550 TEST_F(ChromeAsyncSocketTest, PendingConnectCloseBeforeRead) {
551   async_socket_data_provider_.set_connect_data(
552       net::MockConnect(net::ASYNC, net::OK));
553   EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
554 
555   message_loop_.RunUntilIdle();
556   ExpectSignalSocketState(
557       SignalSocketState::NoError(
558           SIGNAL_CONNECT, ChromeAsyncSocket::STATE_OPEN));
559 
560   DoCloseOpenedNoError();
561 
562   message_loop_.RunUntilIdle();
563 }
564 
TEST_F(ChromeAsyncSocketTest,PendingConnectError)565 TEST_F(ChromeAsyncSocketTest, PendingConnectError) {
566   async_socket_data_provider_.set_connect_data(
567       net::MockConnect(net::ASYNC, net::ERR_TIMED_OUT));
568   EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
569 
570   message_loop_.RunUntilIdle();
571 
572   ExpectSignalSocketState(
573       SignalSocketState(
574           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
575           ChromeAsyncSocket::ERROR_WINSOCK, net::ERR_TIMED_OUT));
576 }
577 
578 // After this we can assume Connect() and Close() work as expected.
579 
TEST_F(ChromeAsyncSocketTest,EmptyRead)580 TEST_F(ChromeAsyncSocketTest, EmptyRead) {
581   DoOpenClosed();
582 
583   char buf[4096];
584   size_t len_read = 10000U;
585   EXPECT_TRUE(chrome_async_socket_->Read(buf, sizeof(buf), &len_read));
586   EXPECT_EQ(0U, len_read);
587 
588   DoCloseOpenedNoError();
589 }
590 
TEST_F(ChromeAsyncSocketTest,WrongRead)591 TEST_F(ChromeAsyncSocketTest, WrongRead) {
592   EXPECT_DEBUG_DEATH({
593     async_socket_data_provider_.set_connect_data(
594         net::MockConnect(net::ASYNC, net::OK));
595     EXPECT_TRUE(chrome_async_socket_->Connect(addr_));
596     ExpectNonErrorState(ChromeAsyncSocket::STATE_CONNECTING);
597     ExpectNoSignal();
598 
599     char buf[4096];
600     size_t len_read;
601     EXPECT_FALSE(chrome_async_socket_->Read(buf, sizeof(buf), &len_read));
602     ExpectErrorState(ChromeAsyncSocket::STATE_CONNECTING,
603                      ChromeAsyncSocket::ERROR_WRONGSTATE);
604     EXPECT_TRUE(chrome_async_socket_->Close());
605     ExpectSignalSocketState(
606         SignalSocketState(
607             SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
608             ChromeAsyncSocket::ERROR_WRONGSTATE, net::OK));
609   }, "non-open");
610 }
611 
TEST_F(ChromeAsyncSocketTest,WrongReadClosed)612 TEST_F(ChromeAsyncSocketTest, WrongReadClosed) {
613   char buf[4096];
614   size_t len_read;
615   EXPECT_FALSE(chrome_async_socket_->Read(buf, sizeof(buf), &len_read));
616   ExpectErrorState(ChromeAsyncSocket::STATE_CLOSED,
617                    ChromeAsyncSocket::ERROR_WRONGSTATE);
618   EXPECT_TRUE(chrome_async_socket_->Close());
619 }
620 
621 const char kReadData[] = "mydatatoread";
622 
TEST_F(ChromeAsyncSocketTest,Read)623 TEST_F(ChromeAsyncSocketTest, Read) {
624   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
625   DoOpenClosed();
626 
627   ExpectReadSignal();
628   ExpectNoSignal();
629 
630   EXPECT_EQ(kReadData, DrainRead(1));
631 
632   message_loop_.RunUntilIdle();
633 
634   DoCloseOpenedNoError();
635 }
636 
TEST_F(ChromeAsyncSocketTest,ReadTwice)637 TEST_F(ChromeAsyncSocketTest, ReadTwice) {
638   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
639   DoOpenClosed();
640 
641   ExpectReadSignal();
642   ExpectNoSignal();
643 
644   EXPECT_EQ(kReadData, DrainRead(1));
645 
646   message_loop_.RunUntilIdle();
647 
648   const char kReadData2[] = "mydatatoread2";
649   async_socket_data_provider_.AddRead(net::MockRead(kReadData2));
650 
651   ExpectReadSignal();
652   ExpectNoSignal();
653 
654   EXPECT_EQ(kReadData2, DrainRead(1));
655 
656   DoCloseOpenedNoError();
657 }
658 
TEST_F(ChromeAsyncSocketTest,ReadError)659 TEST_F(ChromeAsyncSocketTest, ReadError) {
660   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
661   DoOpenClosed();
662 
663   ExpectReadSignal();
664   ExpectNoSignal();
665 
666   EXPECT_EQ(kReadData, DrainRead(1));
667 
668   message_loop_.RunUntilIdle();
669 
670   async_socket_data_provider_.AddRead(
671       net::MockRead(net::SYNCHRONOUS, net::ERR_TIMED_OUT));
672 
673   ExpectSignalSocketState(
674       SignalSocketState(
675           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
676           ChromeAsyncSocket::ERROR_WINSOCK, net::ERR_TIMED_OUT));
677 }
678 
TEST_F(ChromeAsyncSocketTest,ReadEmpty)679 TEST_F(ChromeAsyncSocketTest, ReadEmpty) {
680   async_socket_data_provider_.AddRead(net::MockRead(""));
681   DoOpenClosed();
682 
683   ExpectSignalSocketState(
684       SignalSocketState::NoError(
685           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
686 }
687 
TEST_F(ChromeAsyncSocketTest,PendingRead)688 TEST_F(ChromeAsyncSocketTest, PendingRead) {
689   DoOpenClosed();
690 
691   ExpectNoSignal();
692 
693   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
694 
695   ExpectSignalSocketState(
696       SignalSocketState::NoError(
697           SIGNAL_READ, ChromeAsyncSocket::STATE_OPEN));
698   ExpectNoSignal();
699 
700   EXPECT_EQ(kReadData, DrainRead(1));
701 
702   message_loop_.RunUntilIdle();
703 
704   DoCloseOpenedNoError();
705 }
706 
TEST_F(ChromeAsyncSocketTest,PendingEmptyRead)707 TEST_F(ChromeAsyncSocketTest, PendingEmptyRead) {
708   DoOpenClosed();
709 
710   ExpectNoSignal();
711 
712   async_socket_data_provider_.AddRead(net::MockRead(""));
713 
714   ExpectSignalSocketState(
715       SignalSocketState::NoError(
716           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED));
717 }
718 
TEST_F(ChromeAsyncSocketTest,PendingReadError)719 TEST_F(ChromeAsyncSocketTest, PendingReadError) {
720   DoOpenClosed();
721 
722   ExpectNoSignal();
723 
724   async_socket_data_provider_.AddRead(
725       net::MockRead(net::ASYNC, net::ERR_TIMED_OUT));
726 
727   ExpectSignalSocketState(
728       SignalSocketState(
729           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
730           ChromeAsyncSocket::ERROR_WINSOCK, net::ERR_TIMED_OUT));
731 }
732 
733 // After this we can assume non-SSL Read() works as expected.
734 
TEST_F(ChromeAsyncSocketTest,WrongWrite)735 TEST_F(ChromeAsyncSocketTest, WrongWrite) {
736   EXPECT_DEBUG_DEATH({
737     std::string data("foo");
738     EXPECT_FALSE(chrome_async_socket_->Write(data.data(), data.size()));
739     ExpectErrorState(ChromeAsyncSocket::STATE_CLOSED,
740                      ChromeAsyncSocket::ERROR_WRONGSTATE);
741     EXPECT_TRUE(chrome_async_socket_->Close());
742   }, "non-open");
743 }
744 
745 const char kWriteData[] = "mydatatowrite";
746 
TEST_F(ChromeAsyncSocketTest,SyncWrite)747 TEST_F(ChromeAsyncSocketTest, SyncWrite) {
748   async_socket_data_provider_.AddWrite(
749       net::MockWrite(net::SYNCHRONOUS, kWriteData, 3));
750   async_socket_data_provider_.AddWrite(
751       net::MockWrite(net::SYNCHRONOUS, kWriteData + 3, 5));
752   async_socket_data_provider_.AddWrite(
753       net::MockWrite(net::SYNCHRONOUS,
754                      kWriteData + 8, arraysize(kWriteData) - 8));
755   DoOpenClosed();
756 
757   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
758   message_loop_.RunUntilIdle();
759   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 3, 5));
760   message_loop_.RunUntilIdle();
761   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 8,
762                                           arraysize(kWriteData) - 8));
763   message_loop_.RunUntilIdle();
764 
765   ExpectNoSignal();
766 
767   DoCloseOpenedNoError();
768 }
769 
TEST_F(ChromeAsyncSocketTest,AsyncWrite)770 TEST_F(ChromeAsyncSocketTest, AsyncWrite) {
771   DoOpenClosed();
772 
773   async_socket_data_provider_.AddWrite(
774       net::MockWrite(net::ASYNC, kWriteData, 3));
775   async_socket_data_provider_.AddWrite(
776       net::MockWrite(net::ASYNC, kWriteData + 3, 5));
777   async_socket_data_provider_.AddWrite(
778       net::MockWrite(net::ASYNC, kWriteData + 8, arraysize(kWriteData) - 8));
779 
780   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
781   message_loop_.RunUntilIdle();
782   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 3, 5));
783   message_loop_.RunUntilIdle();
784   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 8,
785                                           arraysize(kWriteData) - 8));
786   message_loop_.RunUntilIdle();
787 
788   ExpectNoSignal();
789 
790   DoCloseOpenedNoError();
791 }
792 
TEST_F(ChromeAsyncSocketTest,AsyncWriteError)793 TEST_F(ChromeAsyncSocketTest, AsyncWriteError) {
794   DoOpenClosed();
795 
796   async_socket_data_provider_.AddWrite(
797       net::MockWrite(net::ASYNC, kWriteData, 3));
798   async_socket_data_provider_.AddWrite(
799       net::MockWrite(net::ASYNC, kWriteData + 3, 5));
800   async_socket_data_provider_.AddWrite(
801       net::MockWrite(net::ASYNC, net::ERR_TIMED_OUT));
802 
803   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
804   message_loop_.RunUntilIdle();
805   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 3, 5));
806   message_loop_.RunUntilIdle();
807   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 8,
808                                           arraysize(kWriteData) - 8));
809   message_loop_.RunUntilIdle();
810 
811   ExpectSignalSocketState(
812       SignalSocketState(
813           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
814           ChromeAsyncSocket::ERROR_WINSOCK, net::ERR_TIMED_OUT));
815 }
816 
TEST_F(ChromeAsyncSocketTest,LargeWrite)817 TEST_F(ChromeAsyncSocketTest, LargeWrite) {
818   EXPECT_DEBUG_DEATH({
819     DoOpenClosed();
820 
821     std::string large_data(100, 'x');
822     EXPECT_FALSE(chrome_async_socket_->Write(large_data.data(),
823                                              large_data.size()));
824     ExpectState(ChromeAsyncSocket::STATE_OPEN,
825                 ChromeAsyncSocket::ERROR_WINSOCK,
826                 net::ERR_INSUFFICIENT_RESOURCES);
827     DoCloseOpened(
828         SignalSocketState(
829             SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
830             ChromeAsyncSocket::ERROR_WINSOCK,
831             net::ERR_INSUFFICIENT_RESOURCES));
832     }, "exceed the max write buffer");
833 }
834 
TEST_F(ChromeAsyncSocketTest,LargeAccumulatedWrite)835 TEST_F(ChromeAsyncSocketTest, LargeAccumulatedWrite) {
836   EXPECT_DEBUG_DEATH({
837     DoOpenClosed();
838 
839     std::string data(15, 'x');
840     EXPECT_TRUE(chrome_async_socket_->Write(data.data(), data.size()));
841     EXPECT_FALSE(chrome_async_socket_->Write(data.data(), data.size()));
842     ExpectState(ChromeAsyncSocket::STATE_OPEN,
843                 ChromeAsyncSocket::ERROR_WINSOCK,
844                 net::ERR_INSUFFICIENT_RESOURCES);
845     DoCloseOpened(
846         SignalSocketState(
847             SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
848             ChromeAsyncSocket::ERROR_WINSOCK,
849             net::ERR_INSUFFICIENT_RESOURCES));
850     }, "exceed the max write buffer");
851 }
852 
853 // After this we can assume non-SSL I/O works as expected.
854 
TEST_F(ChromeAsyncSocketTest,HangingSSLConnect)855 TEST_F(ChromeAsyncSocketTest, HangingSSLConnect) {
856   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
857   DoOpenClosed();
858   ExpectReadSignal();
859 
860   EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
861   ExpectNoSignal();
862 
863   ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_CONNECTING);
864   EXPECT_TRUE(chrome_async_socket_->Close());
865   ExpectSignalSocketState(
866       SignalSocketState::NoError(SIGNAL_CLOSE,
867                                  ChromeAsyncSocket::STATE_CLOSED));
868   ExpectNonErrorState(ChromeAsyncSocket::STATE_CLOSED);
869 }
870 
TEST_F(ChromeAsyncSocketTest,ImmediateSSLConnect)871 TEST_F(ChromeAsyncSocketTest, ImmediateSSLConnect) {
872   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
873   DoOpenClosed();
874   ExpectReadSignal();
875 
876   EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
877   message_loop_.RunUntilIdle();
878   ExpectSSLConnectSignal();
879   ExpectNoSignal();
880   ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_OPEN);
881 
882   DoSSLCloseOpenedNoError();
883 }
884 
TEST_F(ChromeAsyncSocketTest,DoubleSSLConnect)885 TEST_F(ChromeAsyncSocketTest, DoubleSSLConnect) {
886   EXPECT_DEBUG_DEATH({
887     async_socket_data_provider_.AddRead(net::MockRead(kReadData));
888     DoOpenClosed();
889     ExpectReadSignal();
890 
891     EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
892     message_loop_.RunUntilIdle();
893     ExpectSSLConnectSignal();
894     ExpectNoSignal();
895     ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_OPEN);
896 
897     EXPECT_FALSE(chrome_async_socket_->StartTls("fakedomain.com"));
898 
899     DoSSLCloseOpened(
900         SignalSocketState(SIGNAL_CLOSE,
901                           ChromeAsyncSocket::STATE_CLOSED,
902                           ChromeAsyncSocket::ERROR_WRONGSTATE,
903                           net::OK));
904   }, "wrong state");
905 }
906 
TEST_F(ChromeAsyncSocketTest,FailedSSLConnect)907 TEST_F(ChromeAsyncSocketTest, FailedSSLConnect) {
908   ssl_socket_data_provider_.connect =
909       net::MockConnect(net::ASYNC, net::ERR_CERT_COMMON_NAME_INVALID),
910 
911   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
912   DoOpenClosed();
913   ExpectReadSignal();
914 
915   EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
916   message_loop_.RunUntilIdle();
917   ExpectSignalSocketState(
918       SignalSocketState(
919           SIGNAL_CLOSE, ChromeAsyncSocket::STATE_CLOSED,
920           ChromeAsyncSocket::ERROR_WINSOCK,
921           net::ERR_CERT_COMMON_NAME_INVALID));
922 
923   EXPECT_TRUE(chrome_async_socket_->Close());
924   ExpectClosed();
925 }
926 
TEST_F(ChromeAsyncSocketTest,ReadDuringSSLConnecting)927 TEST_F(ChromeAsyncSocketTest, ReadDuringSSLConnecting) {
928   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
929   DoOpenClosed();
930   ExpectReadSignal();
931   EXPECT_EQ(kReadData, DrainRead(1));
932 
933   EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
934   ExpectNoSignal();
935 
936   // Shouldn't do anything.
937   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
938 
939   char buf[4096];
940   size_t len_read = 10000U;
941   EXPECT_TRUE(chrome_async_socket_->Read(buf, sizeof(buf), &len_read));
942   EXPECT_EQ(0U, len_read);
943 
944   message_loop_.RunUntilIdle();
945   ExpectSSLConnectSignal();
946   ExpectSSLReadSignal();
947   ExpectNoSignal();
948   ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_OPEN);
949 
950   len_read = 10000U;
951   EXPECT_TRUE(chrome_async_socket_->Read(buf, sizeof(buf), &len_read));
952   EXPECT_EQ(kReadData, std::string(buf, len_read));
953 
954   DoSSLCloseOpenedNoError();
955 }
956 
TEST_F(ChromeAsyncSocketTest,WriteDuringSSLConnecting)957 TEST_F(ChromeAsyncSocketTest, WriteDuringSSLConnecting) {
958   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
959   DoOpenClosed();
960   ExpectReadSignal();
961 
962   EXPECT_TRUE(chrome_async_socket_->StartTls("fakedomain.com"));
963   ExpectNoSignal();
964   ExpectNonErrorState(ChromeAsyncSocket::STATE_TLS_CONNECTING);
965 
966   async_socket_data_provider_.AddWrite(
967       net::MockWrite(net::ASYNC, kWriteData, 3));
968 
969   // Shouldn't do anything.
970   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
971 
972   // TODO(akalin): Figure out how to test that the write happens
973   // *after* the SSL connect.
974 
975   message_loop_.RunUntilIdle();
976   ExpectSSLConnectSignal();
977   ExpectNoSignal();
978 
979   message_loop_.RunUntilIdle();
980 
981   DoSSLCloseOpenedNoError();
982 }
983 
TEST_F(ChromeAsyncSocketTest,SSLConnectDuringPendingRead)984 TEST_F(ChromeAsyncSocketTest, SSLConnectDuringPendingRead) {
985   EXPECT_DEBUG_DEATH({
986     DoOpenClosed();
987 
988     EXPECT_FALSE(chrome_async_socket_->StartTls("fakedomain.com"));
989 
990     DoCloseOpened(
991         SignalSocketState(SIGNAL_CLOSE,
992                           ChromeAsyncSocket::STATE_CLOSED,
993                           ChromeAsyncSocket::ERROR_WRONGSTATE,
994                           net::OK));
995   }, "wrong state");
996 }
997 
TEST_F(ChromeAsyncSocketTest,SSLConnectDuringPostedWrite)998 TEST_F(ChromeAsyncSocketTest, SSLConnectDuringPostedWrite) {
999   EXPECT_DEBUG_DEATH({
1000     DoOpenClosed();
1001 
1002     async_socket_data_provider_.AddWrite(
1003         net::MockWrite(net::ASYNC, kWriteData, 3));
1004     EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
1005 
1006     EXPECT_FALSE(chrome_async_socket_->StartTls("fakedomain.com"));
1007 
1008     message_loop_.RunUntilIdle();
1009 
1010     DoCloseOpened(
1011         SignalSocketState(SIGNAL_CLOSE,
1012                           ChromeAsyncSocket::STATE_CLOSED,
1013                           ChromeAsyncSocket::ERROR_WRONGSTATE,
1014                           net::OK));
1015   }, "wrong state");
1016 }
1017 
1018 // After this we can assume SSL connect works as expected.
1019 
TEST_F(ChromeAsyncSocketTest,SSLRead)1020 TEST_F(ChromeAsyncSocketTest, SSLRead) {
1021   DoSSLOpenClosed();
1022   async_socket_data_provider_.AddRead(net::MockRead(kReadData));
1023   message_loop_.RunUntilIdle();
1024 
1025   ExpectSSLReadSignal();
1026   ExpectNoSignal();
1027 
1028   EXPECT_EQ(kReadData, DrainRead(1));
1029 
1030   message_loop_.RunUntilIdle();
1031 
1032   DoSSLCloseOpenedNoError();
1033 }
1034 
TEST_F(ChromeAsyncSocketTest,SSLSyncWrite)1035 TEST_F(ChromeAsyncSocketTest, SSLSyncWrite) {
1036   async_socket_data_provider_.AddWrite(
1037       net::MockWrite(net::SYNCHRONOUS, kWriteData, 3));
1038   async_socket_data_provider_.AddWrite(
1039       net::MockWrite(net::SYNCHRONOUS, kWriteData + 3, 5));
1040   async_socket_data_provider_.AddWrite(
1041       net::MockWrite(net::SYNCHRONOUS,
1042                      kWriteData + 8, arraysize(kWriteData) - 8));
1043   DoSSLOpenClosed();
1044 
1045   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
1046   message_loop_.RunUntilIdle();
1047   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 3, 5));
1048   message_loop_.RunUntilIdle();
1049   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 8,
1050                                           arraysize(kWriteData) - 8));
1051   message_loop_.RunUntilIdle();
1052 
1053   ExpectNoSignal();
1054 
1055   DoSSLCloseOpenedNoError();
1056 }
1057 
TEST_F(ChromeAsyncSocketTest,SSLAsyncWrite)1058 TEST_F(ChromeAsyncSocketTest, SSLAsyncWrite) {
1059   DoSSLOpenClosed();
1060 
1061   async_socket_data_provider_.AddWrite(
1062       net::MockWrite(net::ASYNC, kWriteData, 3));
1063   async_socket_data_provider_.AddWrite(
1064       net::MockWrite(net::ASYNC, kWriteData + 3, 5));
1065   async_socket_data_provider_.AddWrite(
1066       net::MockWrite(net::ASYNC, kWriteData + 8, arraysize(kWriteData) - 8));
1067 
1068   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData, 3));
1069   message_loop_.RunUntilIdle();
1070   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 3, 5));
1071   message_loop_.RunUntilIdle();
1072   EXPECT_TRUE(chrome_async_socket_->Write(kWriteData + 8,
1073                                           arraysize(kWriteData) - 8));
1074   message_loop_.RunUntilIdle();
1075 
1076   ExpectNoSignal();
1077 
1078   DoSSLCloseOpenedNoError();
1079 }
1080 
1081 }  // namespace
1082 
1083 }  // namespace jingle_glue
1084