| // Copyright 2015 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "blimp/net/engine_authentication_handler.h" |
| |
| #include "base/callback_helpers.h" |
| #include "base/logging.h" |
| #include "base/timer/timer.h" |
| #include "blimp/common/proto/blimp_message.pb.h" |
| #include "blimp/net/blimp_connection.h" |
| #include "blimp/net/blimp_message_processor.h" |
| #include "blimp/net/blimp_transport.h" |
| #include "blimp/net/connection_error_observer.h" |
| #include "net/base/completion_callback.h" |
| #include "net/base/net_errors.h" |
| |
| namespace blimp { |
| |
| namespace { |
| // Expect Client to send the StartConnection within ten seconds of becoming |
| // connected. |
| const int kAuthTimeoutDurationInSeconds = 10; |
| |
| // Authenticates one connection. It deletes itself when |
| // * the connection is authenticated and passed to |connection_handler|. |
| // * the connection gets into an error state. |
| // * the auth message does not arrive within a reasonable time. |
| class Authenticator : public ConnectionErrorObserver, |
| public BlimpMessageProcessor { |
| public: |
| explicit Authenticator(scoped_ptr<BlimpConnection> connection, |
| base::WeakPtr<ConnectionHandler> connection_handler); |
| ~Authenticator() override; |
| |
| private: |
| // Processes authentication result and deletes |this|. |
| void OnConnectionAuthenticated(bool authenticated); |
| |
| // Handles timeout waiting for auth message, and deletes |this|. |
| void OnAuthenticationTimeout(); |
| |
| // ConnectionErrorObserver implementation. |
| void OnConnectionError(int error) override; |
| |
| // BlimpMessageProcessor implementation. |
| void ProcessMessage(scoped_ptr<BlimpMessage> message, |
| const net::CompletionCallback& callback) override; |
| |
| // The connection to be authenticated. |
| scoped_ptr<BlimpConnection> connection_; |
| |
| // Handler to pass successfully authenticated connections to. |
| base::WeakPtr<ConnectionHandler> connection_handler_; |
| |
| // A timer to fail authentication on timeout. |
| base::OneShotTimer timeout_timer_; |
| |
| DISALLOW_COPY_AND_ASSIGN(Authenticator); |
| }; |
| |
| Authenticator::Authenticator( |
| scoped_ptr<BlimpConnection> connection, |
| base::WeakPtr<ConnectionHandler> connection_handler) |
| : connection_(std::move(connection)), |
| connection_handler_(connection_handler) { |
| DVLOG(1) << "Authenticator object created."; |
| |
| // Observe for errors that might occur during the authentication phase. |
| connection_->AddConnectionErrorObserver(this); |
| connection_->SetIncomingMessageProcessor(this); |
| timeout_timer_.Start( |
| FROM_HERE, base::TimeDelta::FromSeconds(kAuthTimeoutDurationInSeconds), |
| this, &Authenticator::OnAuthenticationTimeout); |
| } |
| |
| Authenticator::~Authenticator() {} |
| |
| void Authenticator::OnConnectionAuthenticated(bool authenticated) { |
| DVLOG(1) << "OnConnectionAuthenticated result=" << authenticated; |
| |
| if (authenticated && connection_handler_) { |
| // Authentication is successful. Stop observing connection errors. |
| connection_->RemoveConnectionErrorObserver(this); |
| connection_handler_->HandleConnection(std::move(connection_)); |
| } |
| |
| delete this; |
| } |
| |
| void Authenticator::OnAuthenticationTimeout() { |
| DVLOG(1) << "Connection authentication timeout"; |
| OnConnectionAuthenticated(false); |
| } |
| |
| void Authenticator::OnConnectionError(int error) { |
| DVLOG(1) << "Connection error before authenticated " |
| << net::ErrorToString(error); |
| OnConnectionAuthenticated(false); |
| } |
| |
| void Authenticator::ProcessMessage(scoped_ptr<BlimpMessage> message, |
| const net::CompletionCallback& callback) { |
| if (message->type() == BlimpMessage::PROTOCOL_CONTROL) { |
| DVLOG(1) << "Authentication challenge received: " |
| << message->protocol_control().start_connection().client_token(); |
| OnConnectionAuthenticated(true); |
| } else { |
| DVLOG(1) << "Expected START_CONNECTION message, got " << message |
| << " instead."; |
| OnConnectionAuthenticated(false); |
| } |
| |
| callback.Run(net::OK); |
| } |
| |
| } // namespace |
| |
| EngineAuthenticationHandler::EngineAuthenticationHandler( |
| ConnectionHandler* connection_handler) |
| : connection_handler_weak_factory_(connection_handler) {} |
| |
| EngineAuthenticationHandler::~EngineAuthenticationHandler() {} |
| |
| void EngineAuthenticationHandler::HandleConnection( |
| scoped_ptr<BlimpConnection> connection) { |
| // Authenticator manages its own lifetime. |
| new Authenticator(std::move(connection), |
| connection_handler_weak_factory_.GetWeakPtr()); |
| } |
| |
| } // namespace blimp |