| OLD | NEW |
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ppapi/tests/test_websocket.h" | 5 #include "ppapi/tests/test_websocket.h" |
| 6 | 6 |
| 7 #include <string.h> | 7 #include <string.h> |
| 8 #include <vector> |
| 8 | 9 |
| 9 #include "ppapi/c/dev/ppb_websocket_dev.h" | 10 #include "ppapi/c/dev/ppb_websocket_dev.h" |
| 10 #include "ppapi/c/pp_errors.h" | 11 #include "ppapi/c/pp_errors.h" |
| 11 #include "ppapi/c/pp_var.h" | 12 #include "ppapi/c/pp_var.h" |
| 12 #include "ppapi/c/pp_completion_callback.h" | 13 #include "ppapi/c/pp_completion_callback.h" |
| 13 #include "ppapi/c/ppb_core.h" | 14 #include "ppapi/c/ppb_core.h" |
| 14 #include "ppapi/c/ppb_var.h" | 15 #include "ppapi/c/ppb_var.h" |
| 16 #include "ppapi/cpp/dev/websocket_dev.h" |
| 15 #include "ppapi/cpp/instance.h" | 17 #include "ppapi/cpp/instance.h" |
| 16 #include "ppapi/cpp/module.h" | 18 #include "ppapi/cpp/module.h" |
| 17 #include "ppapi/tests/test_utils.h" | 19 #include "ppapi/tests/test_utils.h" |
| 18 #include "ppapi/tests/testing_instance.h" | 20 #include "ppapi/tests/testing_instance.h" |
| 19 | 21 |
| 20 const char kEchoServerURL[] = | 22 const char kEchoServerURL[] = |
| 21 "ws://localhost:8880/websocket/tests/hybi/echo"; | 23 "ws://localhost:8880/websocket/tests/hybi/echo"; |
| 22 | 24 |
| 23 const char kProtocolTestServerURL[] = | 25 const char kProtocolTestServerURL[] = |
| 24 "ws://localhost:8880/websocket/tests/hybi/protocol-test?protocol="; | 26 "ws://localhost:8880/websocket/tests/hybi/protocol-test?protocol="; |
| 25 | 27 |
| 26 const char* const kInvalidURLs[] = { | 28 const char* const kInvalidURLs[] = { |
| 27 "http://www.google.com/invalid_scheme", | 29 "http://www.google.com/invalid_scheme", |
| 28 "ws://www.google.com/invalid#fragment", | 30 "ws://www.google.com/invalid#fragment", |
| 29 "ws://www.google.com:65535/invalid_port", | 31 "ws://www.google.com:65535/invalid_port", |
| 30 NULL | 32 NULL |
| 31 }; | 33 }; |
| 32 | 34 |
| 33 // Connection close code is defined in WebSocket protocol specification. | 35 // Connection close code is defined in WebSocket protocol specification. |
| 34 // The magic number 1000 means gracefull closure without any error. | 36 // The magic number 1000 means gracefull closure without any error. |
| 35 // See section 7.4.1. of RFC 6455. | 37 // See section 7.4.1. of RFC 6455. |
| 36 const uint16_t kCloseCodeNormalClosure = 1000; | 38 const uint16_t kCloseCodeNormalClosure = 1000U; |
| 39 |
| 40 namespace { |
| 41 |
| 42 struct WebSocketEvent { |
| 43 enum EventType { |
| 44 EVENT_OPEN, |
| 45 EVENT_MESSAGE, |
| 46 EVENT_ERROR, |
| 47 EVENT_CLOSE |
| 48 }; |
| 49 |
| 50 WebSocketEvent( |
| 51 EventType type, bool was_clean, uint16_t code, const pp::Var& var) |
| 52 : event_type(type), |
| 53 was_clean(was_clean), |
| 54 code(code), |
| 55 var(var) {} |
| 56 EventType event_type; |
| 57 bool was_clean; |
| 58 uint16_t code; |
| 59 pp::Var var; |
| 60 }; |
| 61 |
| 62 class TestWebSocketClient : public pp::WebSocket_Dev { |
| 63 public: |
| 64 TestWebSocketClient(pp::Instance* instance) |
| 65 : pp::WebSocket_Dev(instance), |
| 66 connected_(false), |
| 67 received_(false), |
| 68 closed_(false), |
| 69 wait_for_connected_(false), |
| 70 wait_for_received_(false), |
| 71 wait_for_closed_(false), |
| 72 instance_(instance->pp_instance()) {} |
| 73 |
| 74 virtual void OnOpen() { |
| 75 events_.push_back(WebSocketEvent(WebSocketEvent::EVENT_OPEN, |
| 76 true, 0U, pp::Var())); |
| 77 connected_ = true; |
| 78 if (wait_for_connected_) { |
| 79 GetTestingInterface()->QuitMessageLoop(instance_); |
| 80 wait_for_connected_ = false; |
| 81 } |
| 82 } |
| 83 |
| 84 virtual void OnMessage(const pp::Var &message) { |
| 85 events_.push_back(WebSocketEvent(WebSocketEvent::EVENT_MESSAGE, |
| 86 true, 0U, message)); |
| 87 received_ = true; |
| 88 if (wait_for_received_) { |
| 89 GetTestingInterface()->QuitMessageLoop(instance_); |
| 90 wait_for_received_ = false; |
| 91 received_ = false; |
| 92 } |
| 93 } |
| 94 |
| 95 virtual void OnError() { |
| 96 events_.push_back(WebSocketEvent(WebSocketEvent::EVENT_ERROR, |
| 97 true, 0U, pp::Var())); |
| 98 } |
| 99 |
| 100 virtual void OnClose( |
| 101 bool was_clean, uint16_t code, const pp::Var& reason) { |
| 102 events_.push_back(WebSocketEvent(WebSocketEvent::EVENT_CLOSE, |
| 103 was_clean, code, reason)); |
| 104 connected_ = true; |
| 105 closed_ = true; |
| 106 if (wait_for_connected_ || wait_for_closed_) { |
| 107 GetTestingInterface()->QuitMessageLoop(instance_); |
| 108 wait_for_connected_ = false; |
| 109 wait_for_closed_ = false; |
| 110 } |
| 111 } |
| 112 |
| 113 void WaitForConnected() { |
| 114 if (!connected_) { |
| 115 wait_for_connected_ = true; |
| 116 GetTestingInterface()->RunMessageLoop(instance_); |
| 117 } |
| 118 } |
| 119 |
| 120 void WaitForReceived() { |
| 121 if (!received_) { |
| 122 wait_for_received_ = true; |
| 123 GetTestingInterface()->RunMessageLoop(instance_); |
| 124 } |
| 125 } |
| 126 |
| 127 void WaitForClosed() { |
| 128 if (!closed_) { |
| 129 wait_for_closed_ = true; |
| 130 GetTestingInterface()->RunMessageLoop(instance_); |
| 131 } |
| 132 } |
| 133 |
| 134 const std::vector<WebSocketEvent>& GetSeenEvents() const { |
| 135 return events_; |
| 136 } |
| 137 |
| 138 private: |
| 139 std::vector<WebSocketEvent> events_; |
| 140 bool connected_; |
| 141 bool received_; |
| 142 bool closed_; |
| 143 bool wait_for_connected_; |
| 144 bool wait_for_received_; |
| 145 bool wait_for_closed_; |
| 146 PP_Instance instance_; |
| 147 }; |
| 148 |
| 149 } // namespace |
| 37 | 150 |
| 38 REGISTER_TEST_CASE(WebSocket); | 151 REGISTER_TEST_CASE(WebSocket); |
| 39 | 152 |
| 40 bool TestWebSocket::Init() { | 153 bool TestWebSocket::Init() { |
| 41 websocket_interface_ = static_cast<const PPB_WebSocket_Dev*>( | 154 websocket_interface_ = static_cast<const PPB_WebSocket_Dev*>( |
| 42 pp::Module::Get()->GetBrowserInterface(PPB_WEBSOCKET_DEV_INTERFACE)); | 155 pp::Module::Get()->GetBrowserInterface(PPB_WEBSOCKET_DEV_INTERFACE)); |
| 43 var_interface_ = static_cast<const PPB_Var*>( | 156 var_interface_ = static_cast<const PPB_Var*>( |
| 44 pp::Module::Get()->GetBrowserInterface(PPB_VAR_INTERFACE)); | 157 pp::Module::Get()->GetBrowserInterface(PPB_VAR_INTERFACE)); |
| 45 core_interface_ = static_cast<const PPB_Core*>( | 158 core_interface_ = static_cast<const PPB_Core*>( |
| 46 pp::Module::Get()->GetBrowserInterface(PPB_CORE_INTERFACE)); | 159 pp::Module::Get()->GetBrowserInterface(PPB_CORE_INTERFACE)); |
| 47 if (!websocket_interface_ || !var_interface_ || !core_interface_) | 160 if (!websocket_interface_ || !var_interface_ || !core_interface_) |
| 48 return false; | 161 return false; |
| 49 | 162 |
| 50 return true; | 163 return true; |
| 51 } | 164 } |
| 52 | 165 |
| 53 void TestWebSocket::RunTests(const std::string& filter) { | 166 void TestWebSocket::RunTests(const std::string& filter) { |
| 54 RUN_TEST(IsWebSocket, filter); | 167 RUN_TEST(IsWebSocket, filter); |
| 55 RUN_TEST(UninitializedPropertiesAccess, filter); | 168 RUN_TEST(UninitializedPropertiesAccess, filter); |
| 56 RUN_TEST(InvalidConnect, filter); | 169 RUN_TEST(InvalidConnect, filter); |
| 57 RUN_TEST(GetURL, filter); | 170 RUN_TEST(GetURL, filter); |
| 58 RUN_TEST(ValidConnect, filter); | 171 RUN_TEST(ValidConnect, filter); |
| 59 RUN_TEST(InvalidClose, filter); | 172 RUN_TEST(InvalidClose, filter); |
| 60 RUN_TEST(ValidClose, filter); | 173 RUN_TEST(ValidClose, filter); |
| 61 RUN_TEST(GetProtocol, filter); | 174 RUN_TEST(GetProtocol, filter); |
| 62 RUN_TEST(TextSendReceive, filter); | 175 RUN_TEST(TextSendReceive, filter); |
| 176 |
| 177 RUN_TEST(CcInvalidConnect, filter); |
| 178 RUN_TEST(CcGetURL, filter); |
| 179 RUN_TEST(CcValidConnect, filter); |
| 180 RUN_TEST(CcGetProtocol, filter); |
| 181 RUN_TEST(CcTextSendReceive, filter); |
| 63 } | 182 } |
| 64 | 183 |
| 65 PP_Var TestWebSocket::CreateVar(const char* string) { | 184 PP_Var TestWebSocket::CreateVar(const char* string) { |
| 66 return var_interface_->VarFromUtf8( | 185 return var_interface_->VarFromUtf8( |
| 67 pp::Module::Get()->pp_module(), string, strlen(string)); | 186 pp::Module::Get()->pp_module(), string, strlen(string)); |
| 68 } | 187 } |
| 69 | 188 |
| 70 void TestWebSocket::ReleaseVar(const PP_Var& var) { | 189 void TestWebSocket::ReleaseVar(const PP_Var& var) { |
| 71 var_interface_->Release(var); | 190 var_interface_->Release(var); |
| 72 } | 191 } |
| (...skipping 15 matching lines...) Expand all Loading... |
| 88 const char* url, int32_t* result, const char* protocol) { | 207 const char* url, int32_t* result, const char* protocol) { |
| 89 PP_Var protocols[] = { PP_MakeUndefined() }; | 208 PP_Var protocols[] = { PP_MakeUndefined() }; |
| 90 PP_Resource ws = websocket_interface_->Create(instance_->pp_instance()); | 209 PP_Resource ws = websocket_interface_->Create(instance_->pp_instance()); |
| 91 if (!ws) | 210 if (!ws) |
| 92 return 0; | 211 return 0; |
| 93 PP_Var url_var = CreateVar(url); | 212 PP_Var url_var = CreateVar(url); |
| 94 TestCompletionCallback callback(instance_->pp_instance(), force_async_); | 213 TestCompletionCallback callback(instance_->pp_instance(), force_async_); |
| 95 int protocol_count = 0; | 214 int protocol_count = 0; |
| 96 if (protocol) { | 215 if (protocol) { |
| 97 protocols[0] = CreateVar(protocol); | 216 protocols[0] = CreateVar(protocol); |
| 98 protocol_count = 1; | 217 protocol_count = 1U; |
| 99 } | 218 } |
| 100 *result = websocket_interface_->Connect( | 219 *result = websocket_interface_->Connect( |
| 101 ws, url_var, protocols, protocol_count, | 220 ws, url_var, protocols, protocol_count, |
| 102 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); | 221 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); |
| 103 ReleaseVar(url_var); | 222 ReleaseVar(url_var); |
| 104 if (protocol) | 223 if (protocol) |
| 105 ReleaseVar(protocols[0]); | 224 ReleaseVar(protocols[0]); |
| 106 if (*result == PP_OK_COMPLETIONPENDING) | 225 if (*result == PP_OK_COMPLETIONPENDING) |
| 107 *result = callback.WaitForResult(); | 226 *result = callback.WaitForResult(); |
| 108 return ws; | 227 return ws; |
| (...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 159 } | 278 } |
| 160 | 279 |
| 161 std::string TestWebSocket::TestInvalidConnect() { | 280 std::string TestWebSocket::TestInvalidConnect() { |
| 162 PP_Var protocols[] = { PP_MakeUndefined() }; | 281 PP_Var protocols[] = { PP_MakeUndefined() }; |
| 163 | 282 |
| 164 PP_Resource ws = websocket_interface_->Create(instance_->pp_instance()); | 283 PP_Resource ws = websocket_interface_->Create(instance_->pp_instance()); |
| 165 ASSERT_TRUE(ws); | 284 ASSERT_TRUE(ws); |
| 166 | 285 |
| 167 TestCompletionCallback callback(instance_->pp_instance(), force_async_); | 286 TestCompletionCallback callback(instance_->pp_instance(), force_async_); |
| 168 int32_t result = websocket_interface_->Connect( | 287 int32_t result = websocket_interface_->Connect( |
| 169 ws, PP_MakeUndefined(), protocols, 1, | 288 ws, PP_MakeUndefined(), protocols, 1U, |
| 170 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); | 289 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); |
| 171 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); | 290 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); |
| 172 | 291 |
| 173 result = websocket_interface_->Connect( | 292 result = websocket_interface_->Connect( |
| 174 ws, PP_MakeUndefined(), protocols, 1, | 293 ws, PP_MakeUndefined(), protocols, 1U, |
| 175 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); | 294 static_cast<pp::CompletionCallback>(callback).pp_completion_callback()); |
| 176 ASSERT_EQ(PP_ERROR_INPROGRESS, result); | 295 ASSERT_EQ(PP_ERROR_INPROGRESS, result); |
| 177 | 296 |
| 178 core_interface_->ReleaseResource(ws); | 297 core_interface_->ReleaseResource(ws); |
| 179 | 298 |
| 180 for (int i = 0; kInvalidURLs[i]; ++i) { | 299 for (int i = 0; kInvalidURLs[i]; ++i) { |
| 181 ws = Connect(kInvalidURLs[i], &result, NULL); | 300 ws = Connect(kInvalidURLs[i], &result, NULL); |
| 182 ASSERT_TRUE(ws); | 301 ASSERT_TRUE(ws); |
| 183 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); | 302 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); |
| 184 | 303 |
| (...skipping 188 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 373 PASS(); | 492 PASS(); |
| 374 } | 493 } |
| 375 | 494 |
| 376 // TODO(toyoshim): Add tests for GetBufferedAmount(). | 495 // TODO(toyoshim): Add tests for GetBufferedAmount(). |
| 377 // For now, the function doesn't work fine because update callback in WebKit is | 496 // For now, the function doesn't work fine because update callback in WebKit is |
| 378 // not landed yet. | 497 // not landed yet. |
| 379 | 498 |
| 380 // TODO(toyoshim): Add tests for didReceiveMessageError(). | 499 // TODO(toyoshim): Add tests for didReceiveMessageError(). |
| 381 | 500 |
| 382 // TODO(toyoshim): Add other function tests. | 501 // TODO(toyoshim): Add other function tests. |
| 502 |
| 503 std::string TestWebSocket::TestCcInvalidConnect() { |
| 504 const pp::Var protocols[] = { pp::Var() }; |
| 505 |
| 506 TestWebSocketClient websocket(instance_); |
| 507 int32_t result = websocket.Connect(pp::Var(), protocols, 1U); |
| 508 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); |
| 509 ASSERT_EQ(0U, websocket.GetSeenEvents().size()); |
| 510 |
| 511 result = websocket.Connect(pp::Var(), protocols, 1U); |
| 512 ASSERT_EQ(PP_ERROR_INPROGRESS, result); |
| 513 ASSERT_EQ(0U, websocket.GetSeenEvents().size()); |
| 514 |
| 515 for (int i = 0; kInvalidURLs[i]; ++i) { |
| 516 TestWebSocketClient ws(instance_); |
| 517 result = ws.Connect(pp::Var(std::string(kInvalidURLs[i])), protocols, 0U); |
| 518 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); |
| 519 ASSERT_EQ(0U, ws.GetSeenEvents().size()); |
| 520 } |
| 521 |
| 522 PASS(); |
| 523 } |
| 524 |
| 525 std::string TestWebSocket::TestCcGetURL() { |
| 526 const pp::Var protocols[] = { pp::Var() }; |
| 527 |
| 528 for (int i = 0; kInvalidURLs[i]; ++i) { |
| 529 TestWebSocketClient websocket(instance_); |
| 530 int32_t result = websocket.Connect( |
| 531 pp::Var(std::string(kInvalidURLs[i])), protocols, 0U); |
| 532 ASSERT_EQ(PP_ERROR_BADARGUMENT, result); |
| 533 pp::Var url = websocket.GetURL(); |
| 534 ASSERT_TRUE(AreEqual(url.pp_var(), kInvalidURLs[i])); |
| 535 ASSERT_EQ(0U, websocket.GetSeenEvents().size()); |
| 536 } |
| 537 |
| 538 PASS(); |
| 539 } |
| 540 |
| 541 std::string TestWebSocket::TestCcValidConnect() { |
| 542 const pp::Var protocols[] = { pp::Var() }; |
| 543 TestWebSocketClient websocket(instance_); |
| 544 int32_t result = websocket.Connect( |
| 545 pp::Var(std::string(kEchoServerURL)), protocols, 0U); |
| 546 ASSERT_EQ(PP_OK_COMPLETIONPENDING, result); |
| 547 websocket.WaitForConnected(); |
| 548 const std::vector<WebSocketEvent>& events = websocket.GetSeenEvents(); |
| 549 ASSERT_EQ(1U, events.size()); |
| 550 ASSERT_EQ(WebSocketEvent::EVENT_OPEN, events[0].event_type); |
| 551 |
| 552 PASS(); |
| 553 } |
| 554 |
| 555 std::string TestWebSocket::TestCcGetProtocol() { |
| 556 const std::string protocol("x-chat"); |
| 557 const pp::Var protocols[] = { pp::Var(protocol) }; |
| 558 std::string url(kProtocolTestServerURL); |
| 559 url += protocol; |
| 560 TestWebSocketClient websocket(instance_); |
| 561 int32_t result = websocket.Connect(pp::Var(url), protocols, 1U); |
| 562 ASSERT_EQ(PP_OK_COMPLETIONPENDING, result); |
| 563 websocket.WaitForConnected(); |
| 564 ASSERT_TRUE(AreEqual(websocket.GetProtocol().pp_var(), protocol.c_str())); |
| 565 const std::vector<WebSocketEvent>& events = websocket.GetSeenEvents(); |
| 566 ASSERT_EQ(1U, events.size()); |
| 567 ASSERT_EQ(WebSocketEvent::EVENT_OPEN, events[0].event_type); |
| 568 |
| 569 PASS(); |
| 570 } |
| 571 |
| 572 std::string TestWebSocket::TestCcTextSendReceive() { |
| 573 const pp::Var protocols[] = { pp::Var() }; |
| 574 TestWebSocketClient websocket(instance_); |
| 575 int32_t result = |
| 576 websocket.Connect(pp::Var(std::string(kEchoServerURL)), protocols, 0U); |
| 577 ASSERT_EQ(PP_OK_COMPLETIONPENDING, result); |
| 578 websocket.WaitForConnected(); |
| 579 |
| 580 // Send 'hello pepper'. |
| 581 std::string message1("hello pepper"); |
| 582 result = websocket.Send(pp::Var(std::string(message1))); |
| 583 ASSERT_EQ(PP_OK, result); |
| 584 |
| 585 // Receive echoed 'hello pepper'. |
| 586 websocket.WaitForReceived(); |
| 587 |
| 588 // Send 'goodbye pepper'. |
| 589 std::string message2("goodbye pepper"); |
| 590 result = websocket.Send(pp::Var(std::string(message2))); |
| 591 |
| 592 // Receive echoed 'goodbye pepper'. |
| 593 websocket.WaitForReceived(); |
| 594 |
| 595 const std::vector<WebSocketEvent>& events = websocket.GetSeenEvents(); |
| 596 ASSERT_EQ(3U, events.size()); |
| 597 ASSERT_EQ(WebSocketEvent::EVENT_OPEN, events[0].event_type); |
| 598 ASSERT_EQ(WebSocketEvent::EVENT_MESSAGE, events[1].event_type); |
| 599 ASSERT_TRUE(AreEqual(events[1].var.pp_var(), message1.c_str())); |
| 600 ASSERT_EQ(WebSocketEvent::EVENT_MESSAGE, events[2].event_type); |
| 601 ASSERT_TRUE(AreEqual(events[2].var.pp_var(), message2.c_str())); |
| 602 |
| 603 PASS(); |
| 604 } |
| OLD | NEW |