// Copyright (c) 2010 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 "remoting/protocol/jingle_session.h" #include "base/crypto/rsa_private_key.h" #include "base/message_loop.h" #include "net/base/cert_verifier.h" #include "net/base/host_port_pair.h" #include "net/base/net_errors.h" #include "net/base/ssl_config_service.h" #include "net/base/x509_certificate.h" #include "net/socket/client_socket_factory.h" #include "net/socket/ssl_client_socket.h" #include "net/socket/ssl_server_socket.h" #include "remoting/base/constants.h" #include "remoting/jingle_glue/channel_socket_adapter.h" #include "remoting/jingle_glue/jingle_thread.h" #include "remoting/jingle_glue/stream_socket_adapter.h" #include "remoting/protocol/jingle_session_manager.h" #include "third_party/libjingle/source/talk/base/thread.h" #include "third_party/libjingle/source/talk/p2p/base/session.h" #include "third_party/libjingle/source/talk/session/tunnel/pseudotcpchannel.h" using cricket::BaseSession; using cricket::PseudoTcpChannel; namespace remoting { namespace protocol { const char JingleSession::kChromotingContentName[] = "chromoting"; namespace { const char kControlChannelName[] = "control"; const char kEventChannelName[] = "event"; const char kVideoChannelName[] = "video"; const char kVideoRtpChannelName[] = "videortp"; const char kVideoRtcpChannelName[] = "videortcp"; // Helper method to create a SSL client socket. net::SSLClientSocket* CreateSSLClientSocket( net::ClientSocket* socket, scoped_refptr cert, net::CertVerifier* cert_verifier) { net::SSLConfig ssl_config; ssl_config.false_start_enabled = false; ssl_config.snap_start_enabled = false; ssl_config.ssl3_enabled = true; ssl_config.tls1_enabled = true; // Certificate provided by the host doesn't need authority. net::SSLConfig::CertAndStatus cert_and_status; cert_and_status.cert_status = net::ERR_CERT_AUTHORITY_INVALID; cert_and_status.cert = cert; ssl_config.allowed_bad_certs.push_back(cert_and_status); // SSLClientSocket takes ownership of the adapter. net::HostPortPair host_and_pair(JingleSession::kChromotingContentName, 0); net::SSLClientSocket* ssl_socket = net::ClientSocketFactory::GetDefaultFactory()->CreateSSLClientSocket( socket, host_and_pair, ssl_config, NULL, cert_verifier); return ssl_socket; } } // namespace // static JingleSession* JingleSession::CreateClientSession( JingleSessionManager* manager) { return new JingleSession(manager, NULL, NULL); } // static JingleSession* JingleSession::CreateServerSession( JingleSessionManager* manager, scoped_refptr certificate, base::RSAPrivateKey* key) { return new JingleSession(manager, certificate, key); } JingleSession::JingleSession( JingleSessionManager* jingle_session_manager, scoped_refptr server_cert, base::RSAPrivateKey* key) : jingle_session_manager_(jingle_session_manager), server_cert_(server_cert), state_(INITIALIZING), closed_(false), cricket_session_(NULL), event_channel_(NULL), video_channel_(NULL), ssl_connections_(0), ALLOW_THIS_IN_INITIALIZER_LIST(connect_callback_( NewCallback(this, &JingleSession::OnSSLConnect))) { // TODO(hclam): Need a better way to clone a key. if (key) { std::vector key_bytes; CHECK(key->ExportPrivateKey(&key_bytes)); key_.reset(base::RSAPrivateKey::CreateFromPrivateKeyInfo(key_bytes)); CHECK(key_.get()); } } JingleSession::~JingleSession() { DCHECK(closed_); } void JingleSession::Init(cricket::Session* cricket_session) { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); cricket_session_ = cricket_session; jid_ = cricket_session_->remote_name(); cert_verifier_.reset(new net::CertVerifier()); cricket_session_->SignalState.connect( this, &JingleSession::OnSessionState); } void JingleSession::CloseInternal(Task* closed_task, int result) { if (MessageLoop::current() != jingle_session_manager_->message_loop()) { jingle_session_manager_->message_loop()->PostTask( FROM_HERE, NewRunnableMethod(this, &JingleSession::CloseInternal, closed_task, result)); return; } if (!closed_) { if (control_ssl_socket_.get()) control_ssl_socket_.reset(); if (control_channel_adapter_.get()) control_channel_adapter_->Close(result); if (control_channel_) { control_channel_->OnSessionTerminate(cricket_session_); control_channel_ = NULL; } if (event_ssl_socket_.get()) event_ssl_socket_.reset(); if (event_channel_adapter_.get()) event_channel_adapter_->Close(result); if (event_channel_) { event_channel_->OnSessionTerminate(cricket_session_); event_channel_ = NULL; } if (video_ssl_socket_.get()) video_ssl_socket_.reset(); if (video_channel_adapter_.get()) video_channel_adapter_->Close(result); if (video_channel_) { video_channel_->OnSessionTerminate(cricket_session_); video_channel_ = NULL; } if (video_rtp_channel_.get()) video_rtp_channel_->Close(result); if (video_rtcp_channel_.get()) video_rtcp_channel_->Close(result); if (cricket_session_) cricket_session_->Terminate(); SetState(CLOSED); closed_ = true; } cert_verifier_.reset(); if (closed_task) { closed_task->Run(); delete closed_task; } } bool JingleSession::HasSession(cricket::Session* cricket_session) { return cricket_session_ == cricket_session; } cricket::Session* JingleSession::ReleaseSession() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); SetState(CLOSED); cricket::Session* session = cricket_session_; if (cricket_session_) cricket_session_->SignalState.disconnect(this); cricket_session_ = NULL; closed_ = true; return session; } void JingleSession::SetStateChangeCallback(StateChangeCallback* callback) { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); DCHECK(callback); state_change_callback_.reset(callback); } net::Socket* JingleSession::control_channel() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); return control_ssl_socket_.get(); } net::Socket* JingleSession::event_channel() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); return event_ssl_socket_.get(); } // TODO(sergeyu): Remove this method after we switch to RTP. net::Socket* JingleSession::video_channel() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); return video_ssl_socket_.get(); } net::Socket* JingleSession::video_rtp_channel() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); return video_rtp_channel_.get(); } net::Socket* JingleSession::video_rtcp_channel() { DCHECK_EQ(jingle_session_manager_->message_loop(), MessageLoop::current()); return video_rtcp_channel_.get(); } const std::string& JingleSession::jid() { // No synchronization is needed because jid_ is not changed // after new connection is passed to JingleChromotocolServer callback. return jid_; } MessageLoop* JingleSession::message_loop() { return jingle_session_manager_->message_loop(); } const CandidateSessionConfig* JingleSession::candidate_config() { DCHECK(candidate_config_.get()); return candidate_config_.get(); } void JingleSession::set_candidate_config( const CandidateSessionConfig* candidate_config) { DCHECK(!candidate_config_.get()); DCHECK(candidate_config); candidate_config_.reset(candidate_config); } scoped_refptr JingleSession::server_certificate() const { return server_cert_; } const SessionConfig* JingleSession::config() { DCHECK(config_.get()); return config_.get(); } void JingleSession::set_config(const SessionConfig* config) { DCHECK(!config_.get()); DCHECK(config); config_.reset(config); } const std::string& JingleSession::initiator_token() { return initiator_token_; } void JingleSession::set_initiator_token(const std::string& initiator_token) { initiator_token_ = initiator_token; } const std::string& JingleSession::receiver_token() { return receiver_token_; } void JingleSession::set_receiver_token(const std::string& receiver_token) { receiver_token_ = receiver_token; } void JingleSession::Close(Task* closed_task) { CloseInternal(closed_task, net::ERR_CONNECTION_CLOSED); } void JingleSession::OnSessionState( BaseSession* session, BaseSession::State state) { DCHECK_EQ(cricket_session_, session); switch (state) { case cricket::Session::STATE_SENTINITIATE: case cricket::Session::STATE_RECEIVEDINITIATE: OnInitiate(); break; case cricket::Session::STATE_SENTACCEPT: case cricket::Session::STATE_RECEIVEDACCEPT: OnAccept(); break; case cricket::Session::STATE_SENTTERMINATE: case cricket::Session::STATE_RECEIVEDTERMINATE: case cricket::Session::STATE_SENTREJECT: case cricket::Session::STATE_RECEIVEDREJECT: OnTerminate(); break; case cricket::Session::STATE_DEINIT: // Close() must have been called before this. NOTREACHED(); break; default: // We don't care about other steates. break; } } void JingleSession::OnInitiate() { jid_ = cricket_session_->remote_name(); std::string content_name; // If we initiate the session, we get to specify the content name. When // accepting one, the remote end specifies it. if (cricket_session_->initiator()) { content_name = kChromotingContentName; } else { const cricket::ContentInfo* content; content = cricket_session_->remote_description()->FirstContentByType( kChromotingXmlNamespace); CHECK(content); content_name = content->name; } // Create video RTP channels. video_rtp_channel_.reset(new TransportChannelSocketAdapter( cricket_session_->CreateChannel(content_name, kVideoRtpChannelName))); video_rtcp_channel_.reset(new TransportChannelSocketAdapter( cricket_session_->CreateChannel(content_name, kVideoRtcpChannelName))); // Create control channel. control_channel_ = new PseudoTcpChannel( jingle_session_manager_->jingle_thread(), cricket_session_); control_channel_->Connect(content_name, kControlChannelName); control_channel_adapter_.reset(new StreamSocketAdapter( control_channel_->GetStream())); // Create event channel. event_channel_ = new PseudoTcpChannel( jingle_session_manager_->jingle_thread(), cricket_session_); event_channel_->Connect(content_name, kEventChannelName); event_channel_adapter_.reset(new StreamSocketAdapter( event_channel_->GetStream())); // Create video channel. // TODO(sergeyu): Remove video channel when we are ready to switch to RTP. video_channel_ = new PseudoTcpChannel( jingle_session_manager_->jingle_thread(), cricket_session_); video_channel_->Connect(content_name, kVideoChannelName); video_channel_adapter_.reset(new StreamSocketAdapter( video_channel_->GetStream())); if (!cricket_session_->initiator()) jingle_session_manager_->AcceptConnection(this, cricket_session_); SetState(CONNECTING); } bool JingleSession::EstablishSSLConnection( net::ClientSocket* adapter, scoped_ptr* ssl_socket) { if (cricket_session_->initiator()) { // Create client SSL socket. net::SSLClientSocket* socket = CreateSSLClientSocket(adapter, server_cert_, cert_verifier_.get()); ssl_socket->reset(socket); int ret = socket->Connect(connect_callback_.get()); if (ret == net::ERR_IO_PENDING) { return true; } else if (ret != net::OK) { LOG(ERROR) << "Failed to establish SSL connection"; cricket_session_->Terminate(); return false; } } else { // Create server SSL socket. net::SSLConfig ssl_config; net::SSLServerSocket* socket = net::CreateSSLServerSocket( adapter, server_cert_, key_.get(), ssl_config); ssl_socket->reset(socket); int ret = socket->Accept(connect_callback_.get()); if (ret == net::ERR_IO_PENDING) { return true; } else if (ret != net::OK) { LOG(ERROR) << "Failed to establish SSL connection"; cricket_session_->Terminate(); return true; } } // Reach here if net::OK is received. connect_callback_->Run(net::OK); return true; } void JingleSession::OnAccept() { // TODO(hclam): Need to close the adapters on failuire otherwise it will // crash in the destructor. // Set the config if we are the one who initiated the session. if (cricket_session_->initiator()) { const cricket::ContentInfo* content = cricket_session_->remote_description()->FirstContentByType( kChromotingXmlNamespace); CHECK(content); const protocol::ContentDescription* content_description = static_cast(content->description); server_cert_ = content_description->certificate(); CHECK(server_cert_); SessionConfig* config = content_description->config()->GetFinalConfig(); // Terminate the session if the config we received is invalid. if (!config || !candidate_config()->IsSupported(config)) { // TODO(sergeyu): Inform the user that the host is misbehaving? LOG(ERROR) << "Terminating outgoing session after an " "invalid session description has been received."; cricket_session_->Terminate(); return; } set_config(config); } bool ret = EstablishSSLConnection(control_channel_adapter_.release(), &control_ssl_socket_); if (ret) { ret = EstablishSSLConnection(event_channel_adapter_.release(), &event_ssl_socket_); } if (ret) { ret = EstablishSSLConnection(video_channel_adapter_.release(), &video_ssl_socket_); } if (!ret) { LOG(ERROR) << "Failed to establish SSL connections"; cricket_session_->Terminate(); } } void JingleSession::OnTerminate() { if (control_channel_adapter_.get()) control_channel_adapter_->Close(net::ERR_CONNECTION_ABORTED); if (control_channel_) { control_channel_->OnSessionTerminate(cricket_session_); control_channel_ = NULL; } if (event_channel_adapter_.get()) event_channel_adapter_->Close(net::ERR_CONNECTION_ABORTED); if (event_channel_) { event_channel_->OnSessionTerminate(cricket_session_); event_channel_ = NULL; } if (video_channel_adapter_.get()) video_channel_adapter_->Close(net::ERR_CONNECTION_ABORTED); if (video_channel_) { video_channel_->OnSessionTerminate(cricket_session_); video_channel_ = NULL; } if (video_rtp_channel_.get()) video_rtp_channel_->Close(net::ERR_CONNECTION_ABORTED); if (video_rtcp_channel_.get()) video_rtcp_channel_->Close(net::ERR_CONNECTION_ABORTED); SetState(CLOSED); closed_ = true; } void JingleSession::OnSSLConnect(int result) { DCHECK(!closed_); if (result != net::OK) { LOG(ERROR) << "Error during SSL connection: " << result; // TODO(hclam): Just setting the state is not enough. Need to invoke a // callback to report failure. CloseInternal(NULL, result); return; } // Number of channels for a jingle session. const int kChannels = 3; // Set the state to connected only of all SSL sockets are connected. if (++ssl_connections_ == kChannels) { SetState(CONNECTED); } CHECK(ssl_connections_ <= kChannels) << "Unexpected SSL connect callback"; } void JingleSession::SetState(State new_state) { if (new_state != state_) { state_ = new_state; if (!closed_ && state_change_callback_.get()) state_change_callback_->Run(new_state); } } } // namespace protocol } // namespace remoting