Index: webrtc/libjingle/examples/call/call_main.cc |
diff --git a/webrtc/libjingle/examples/call/call_main.cc b/webrtc/libjingle/examples/call/call_main.cc |
deleted file mode 100644 |
index 2deb78a716c0c4b0727dd6ccd6da035ae46a863d..0000000000000000000000000000000000000000 |
--- a/webrtc/libjingle/examples/call/call_main.cc |
+++ /dev/null |
@@ -1,484 +0,0 @@ |
-/* |
- * libjingle |
- * Copyright 2004--2005, Google Inc. |
- * |
- * Redistribution and use in source and binary forms, with or without |
- * modification, are permitted provided that the following conditions are met: |
- * |
- * 1. Redistributions of source code must retain the above copyright notice, |
- * this list of conditions and the following disclaimer. |
- * 2. Redistributions in binary form must reproduce the above copyright notice, |
- * this list of conditions and the following disclaimer in the documentation |
- * and/or other materials provided with the distribution. |
- * 3. The name of the author may not be used to endorse or promote products |
- * derived from this software without specific prior written permission. |
- * |
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED |
- * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
- * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO |
- * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
- * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; |
- * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
- * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR |
- * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF |
- * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
- */ |
- |
-#include <stdio.h> |
-#include <string.h> |
-#include <time.h> |
- |
-#include <iomanip> |
-#include <iostream> |
-#include <vector> |
- |
-#include "webrtc/base/flags.h" |
-#include "webrtc/base/logging.h" |
-#ifdef OSX |
-#include "webrtc/base/maccocoasocketserver.h" |
-#endif |
-#include "talk/examples/call/callclient.h" |
-#include "talk/examples/call/console.h" |
-#include "talk/examples/call/mediaenginefactory.h" |
-#include "talk/session/media/srtpfilter.h" |
-#include "webrtc/base/pathutils.h" |
-#include "webrtc/base/ssladapter.h" |
-#include "webrtc/base/stream.h" |
-#include "webrtc/base/win32socketserver.h" |
-#include "webrtc/libjingle/session/media/mediasessionclient.h" |
-#include "webrtc/libjingle/xmpp/xmppauth.h" |
-#include "webrtc/libjingle/xmpp/xmppclientsettings.h" |
-#include "webrtc/libjingle/xmpp/xmpppump.h" |
-#include "webrtc/libjingle/xmpp/xmppsocket.h" |
-#include "webrtc/p2p/base/constants.h" |
- |
-class DebugLog : public sigslot::has_slots<> { |
- public: |
- DebugLog() : |
- debug_input_buf_(NULL), debug_input_len_(0), debug_input_alloc_(0), |
- debug_output_buf_(NULL), debug_output_len_(0), debug_output_alloc_(0), |
- censor_password_(false) |
- {} |
- char * debug_input_buf_; |
- int debug_input_len_; |
- int debug_input_alloc_; |
- char * debug_output_buf_; |
- int debug_output_len_; |
- int debug_output_alloc_; |
- bool censor_password_; |
- |
- void Input(const char * data, int len) { |
- if (debug_input_len_ + len > debug_input_alloc_) { |
- char * old_buf = debug_input_buf_; |
- debug_input_alloc_ = 4096; |
- while (debug_input_alloc_ < debug_input_len_ + len) { |
- debug_input_alloc_ *= 2; |
- } |
- debug_input_buf_ = new char[debug_input_alloc_]; |
- memcpy(debug_input_buf_, old_buf, debug_input_len_); |
- delete[] old_buf; |
- } |
- memcpy(debug_input_buf_ + debug_input_len_, data, len); |
- debug_input_len_ += len; |
- DebugPrint(debug_input_buf_, &debug_input_len_, false); |
- } |
- |
- void Output(const char * data, int len) { |
- if (debug_output_len_ + len > debug_output_alloc_) { |
- char * old_buf = debug_output_buf_; |
- debug_output_alloc_ = 4096; |
- while (debug_output_alloc_ < debug_output_len_ + len) { |
- debug_output_alloc_ *= 2; |
- } |
- debug_output_buf_ = new char[debug_output_alloc_]; |
- memcpy(debug_output_buf_, old_buf, debug_output_len_); |
- delete[] old_buf; |
- } |
- memcpy(debug_output_buf_ + debug_output_len_, data, len); |
- debug_output_len_ += len; |
- DebugPrint(debug_output_buf_, &debug_output_len_, true); |
- } |
- |
- static bool IsAuthTag(const char * str, size_t len) { |
- if (str[0] == '<' && str[1] == 'a' && |
- str[2] == 'u' && |
- str[3] == 't' && |
- str[4] == 'h' && |
- str[5] <= ' ') { |
- std::string tag(str, len); |
- |
- if (tag.find("mechanism") != std::string::npos) |
- return true; |
- } |
- return false; |
- } |
- |
- void DebugPrint(char * buf, int * plen, bool output) { |
- int len = *plen; |
- if (len > 0) { |
- time_t tim = time(NULL); |
- struct tm * now = localtime(&tim); |
- char *time_string = asctime(now); |
- if (time_string) { |
- size_t time_len = strlen(time_string); |
- if (time_len > 0) { |
- time_string[time_len-1] = 0; // trim off terminating \n |
- } |
- } |
- LOG(INFO) << (output ? "SEND >>>>>>>>>>>>>>>>" : "RECV <<<<<<<<<<<<<<<<") |
- << " : " << time_string; |
- |
- bool indent; |
- int start = 0, nest = 3; |
- for (int i = 0; i < len; i += 1) { |
- if (buf[i] == '>') { |
- if ((i > 0) && (buf[i-1] == '/')) { |
- indent = false; |
- } else if ((start + 1 < len) && (buf[start + 1] == '/')) { |
- indent = false; |
- nest -= 2; |
- } else { |
- indent = true; |
- } |
- |
- // Output a tag |
- LOG(INFO) << std::setw(nest) << " " |
- << std::string(buf + start, i + 1 - start); |
- |
- if (indent) |
- nest += 2; |
- |
- // Note if it's a PLAIN auth tag |
- if (IsAuthTag(buf + start, i + 1 - start)) { |
- censor_password_ = true; |
- } |
- |
- // incr |
- start = i + 1; |
- } |
- |
- if (buf[i] == '<' && start < i) { |
- if (censor_password_) { |
- LOG(INFO) << std::setw(nest) << " " << "## TEXT REMOVED ##"; |
- censor_password_ = false; |
- } else { |
- LOG(INFO) << std::setw(nest) << " " |
- << std::string(buf + start, i - start); |
- } |
- start = i; |
- } |
- } |
- len = len - start; |
- memcpy(buf, buf + start, len); |
- *plen = len; |
- } |
- } |
-}; |
- |
-static DebugLog debug_log_; |
-static const int DEFAULT_PORT = 5222; |
- |
-#ifdef ANDROID |
-static std::vector<cricket::AudioCodec> codecs; |
-static const cricket::AudioCodec ISAC(103, "ISAC", 40000, 16000, 1, 0); |
- |
-cricket::MediaEngineInterface *CreateAndroidMediaEngine() { |
- cricket::FakeMediaEngine *engine = new cricket::FakeMediaEngine(); |
- |
- codecs.push_back(ISAC); |
- engine->SetAudioCodecs(codecs); |
- return engine; |
-} |
-#endif |
- |
-// TODO: Move this into Console. |
-void Print(const char* chars) { |
- printf("%s", chars); |
- fflush(stdout); |
-} |
- |
-bool GetSecurePolicy(const std::string& in, cricket::SecurePolicy* out) { |
- if (in == "disable") { |
- *out = cricket::SEC_DISABLED; |
- } else if (in == "enable") { |
- *out = cricket::SEC_ENABLED; |
- } else if (in == "require") { |
- *out = cricket::SEC_REQUIRED; |
- } else { |
- return false; |
- } |
- return true; |
-} |
- |
-int main(int argc, char **argv) { |
- // This app has three threads. The main thread will run the XMPP client, |
- // which will print to the screen in its own thread. A second thread |
- // will get input from the console, parse it, and pass the appropriate |
- // message back to the XMPP client's thread. A third thread is used |
- // by MediaSessionClient as its worker thread. |
- |
- // define options |
- DEFINE_string(s, "talk.google.com", "The connection server to use."); |
- DEFINE_string(tls, "require", |
- "Select connection encryption: disable, enable, require."); |
- DEFINE_bool(allowplain, false, "Allow plain authentication."); |
- DEFINE_bool(testserver, false, "Use test server."); |
- DEFINE_string(oauth, "", "OAuth2 access token."); |
- DEFINE_bool(a, false, "Turn on auto accept for incoming calls."); |
- DEFINE_string(signaling, "hybrid", |
- "Initial signaling protocol to use: jingle, gingle, or hybrid."); |
- DEFINE_string(transport, "hybrid", |
- "Initial transport protocol to use: ice, gice, or hybrid."); |
- DEFINE_string(sdes, "enable", |
- "Select SDES media encryption: disable, enable, require."); |
- DEFINE_string(dtls, "disable", |
- "Select DTLS transport encryption: disable, enable, require."); |
- DEFINE_int(portallocator, 0, "Filter out unwanted connection types."); |
- DEFINE_string(pmuc, "groupchat.google.com", "The persistant muc domain."); |
- DEFINE_string(capsnode, "http://code.google.com/p/libjingle/call", |
- "Caps node: A URI identifying the app."); |
- DEFINE_string(capsver, "0.6", |
- "Caps ver: A string identifying the version of the app."); |
- DEFINE_string(voiceinput, NULL, "RTP dump file for voice input."); |
- DEFINE_string(voiceoutput, NULL, "RTP dump file for voice output."); |
- DEFINE_string(videoinput, NULL, "RTP dump file for video input."); |
- DEFINE_string(videooutput, NULL, "RTP dump file for video output."); |
- DEFINE_bool(render, true, "Renders the video."); |
- DEFINE_string(datachannel, "", |
- "Enable a data channel, and choose the type: rtp or sctp."); |
- DEFINE_bool(d, false, "Turn on debugging."); |
- DEFINE_bool(debugsrtp, false, "Enable debugging for srtp."); |
- DEFINE_bool(help, false, "Prints this message"); |
- DEFINE_bool(multisession, false, |
- "Enable support for multiple sessions in calls."); |
- DEFINE_bool(roster, false, |
- "Enable roster messages printed in console."); |
- |
- // parse options |
- rtc::FlagList::SetFlagsFromCommandLine(&argc, argv, true); |
- if (FLAG_help) { |
- rtc::FlagList::Print(NULL, false); |
- return 0; |
- } |
- |
- bool auto_accept = FLAG_a; |
- bool debug = FLAG_d; |
- std::string signaling = FLAG_signaling; |
- std::string transport = FLAG_transport; |
- bool test_server = FLAG_testserver; |
- bool allow_plain = FLAG_allowplain; |
- std::string tls = FLAG_tls; |
- std::string oauth_token = FLAG_oauth; |
- int32 portallocator_flags = FLAG_portallocator; |
- std::string pmuc_domain = FLAG_pmuc; |
- std::string server = FLAG_s; |
- std::string sdes = FLAG_sdes; |
- std::string dtls = FLAG_dtls; |
- std::string caps_node = FLAG_capsnode; |
- std::string caps_ver = FLAG_capsver; |
- bool debugsrtp = FLAG_debugsrtp; |
- bool render = FLAG_render; |
- std::string data_channel = FLAG_datachannel; |
- bool multisession_enabled = FLAG_multisession; |
- rtc::SSLIdentity* ssl_identity = NULL; |
- bool show_roster_messages = FLAG_roster; |
- |
- // Set up debugging. |
- if (debug) { |
- rtc::LogMessage::LogToDebug(rtc::LS_VERBOSE); |
- } |
- |
- if (debugsrtp) { |
- cricket::EnableSrtpDebugging(); |
- } |
- |
- // Set up the crypto subsystem. |
- rtc::InitializeSSL(); |
- |
- // Parse username and password, if present. |
- buzz::Jid jid; |
- std::string username; |
- rtc::InsecureCryptStringImpl pass; |
- if (argc > 1) { |
- username = argv[1]; |
- if (argc > 2) { |
- pass.password() = argv[2]; |
- } |
- } |
- |
- if (username.empty()) { |
- Print("JID: "); |
- std::cin >> username; |
- } |
- if (username.find('@') == std::string::npos) { |
- username.append("@localhost"); |
- } |
- jid = buzz::Jid(username); |
- if (!jid.IsValid() || jid.node() == "") { |
- Print("Invalid JID. JIDs should be in the form user@domain\n"); |
- return 1; |
- } |
- if (pass.password().empty() && !test_server && oauth_token.empty()) { |
- Console::SetEcho(false); |
- Print("Password: "); |
- std::cin >> pass.password(); |
- Console::SetEcho(true); |
- Print("\n"); |
- } |
- |
- // Decide on the connection settings. |
- buzz::XmppClientSettings xcs; |
- xcs.set_user(jid.node()); |
- xcs.set_resource("call"); |
- xcs.set_host(jid.domain()); |
- xcs.set_allow_plain(allow_plain); |
- |
- if (tls == "disable") { |
- xcs.set_use_tls(buzz::TLS_DISABLED); |
- } else if (tls == "enable") { |
- xcs.set_use_tls(buzz::TLS_ENABLED); |
- } else if (tls == "require") { |
- xcs.set_use_tls(buzz::TLS_REQUIRED); |
- } else { |
- Print("Invalid TLS option, must be enable, disable, or require.\n"); |
- return 1; |
- } |
- |
- if (test_server) { |
- pass.password() = jid.node(); |
- xcs.set_allow_plain(true); |
- xcs.set_use_tls(buzz::TLS_DISABLED); |
- xcs.set_test_server_domain("google.com"); |
- } |
- xcs.set_pass(rtc::CryptString(pass)); |
- if (!oauth_token.empty()) { |
- xcs.set_auth_token(buzz::AUTH_MECHANISM_OAUTH2, oauth_token); |
- } |
- |
- std::string host; |
- int port; |
- |
- int colon = server.find(':'); |
- if (colon == -1) { |
- host = server; |
- port = DEFAULT_PORT; |
- } else { |
- host = server.substr(0, colon); |
- port = atoi(server.substr(colon + 1).c_str()); |
- } |
- |
- xcs.set_server(rtc::SocketAddress(host, port)); |
- |
- // Decide on the signaling and crypto settings. |
- cricket::SignalingProtocol signaling_protocol = cricket::PROTOCOL_HYBRID; |
- if (signaling == "jingle") { |
- signaling_protocol = cricket::PROTOCOL_JINGLE; |
- } else if (signaling == "gingle") { |
- signaling_protocol = cricket::PROTOCOL_GINGLE; |
- } else if (signaling == "hybrid") { |
- signaling_protocol = cricket::PROTOCOL_HYBRID; |
- } else { |
- Print("Invalid signaling protocol. Must be jingle, gingle, or hybrid.\n"); |
- return 1; |
- } |
- |
- cricket::TransportProtocol transport_protocol = cricket::ICEPROTO_HYBRID; |
- if (transport == "ice") { |
- transport_protocol = cricket::ICEPROTO_RFC5245; |
- } else if (transport == "gice") { |
- transport_protocol = cricket::ICEPROTO_GOOGLE; |
- } else if (transport == "hybrid") { |
- transport_protocol = cricket::ICEPROTO_HYBRID; |
- } else { |
- Print("Invalid transport protocol. Must be ice, gice, or hybrid.\n"); |
- return 1; |
- } |
- |
- cricket::DataChannelType data_channel_type = cricket::DCT_NONE; |
- if (data_channel == "rtp") { |
- data_channel_type = cricket::DCT_RTP; |
- } else if (data_channel == "sctp") { |
- data_channel_type = cricket::DCT_SCTP; |
- } else if (!data_channel.empty()) { |
- Print("Invalid data channel type. Must be rtp or sctp.\n"); |
- return 1; |
- } |
- |
- cricket::SecurePolicy sdes_policy, dtls_policy; |
- if (!GetSecurePolicy(sdes, &sdes_policy)) { |
- Print("Invalid SDES policy. Must be enable, disable, or require.\n"); |
- return 1; |
- } |
- if (!GetSecurePolicy(dtls, &dtls_policy)) { |
- Print("Invalid DTLS policy. Must be enable, disable, or require.\n"); |
- return 1; |
- } |
- if (dtls_policy != cricket::SEC_DISABLED) { |
- ssl_identity = rtc::SSLIdentity::Generate(jid.Str()); |
- if (!ssl_identity) { |
- Print("Failed to generate identity for DTLS.\n"); |
- return 1; |
- } |
- } |
- |
-#ifdef ANDROID |
- MediaEngineFactory::SetCreateFunction(&CreateAndroidMediaEngine); |
-#endif |
- |
-#if WIN32 |
- // Need to pump messages on our main thread on Windows. |
- rtc::Win32Thread w32_thread; |
- rtc::ThreadManager::Instance()->SetCurrentThread(&w32_thread); |
-#endif |
- rtc::Thread* main_thread = rtc::Thread::Current(); |
-#ifdef OSX |
- rtc::MacCocoaSocketServer ss; |
- rtc::SocketServerScope ss_scope(&ss); |
-#endif |
- |
- buzz::XmppPump pump; |
- CallClient *client = new CallClient(pump.client(), caps_node, caps_ver); |
- |
- if (FLAG_voiceinput || FLAG_voiceoutput || |
- FLAG_videoinput || FLAG_videooutput) { |
- // If any dump file is specified, we use a FileMediaEngine. |
- cricket::MediaEngineInterface* engine = |
- MediaEngineFactory::CreateFileMediaEngine( |
- FLAG_voiceinput, FLAG_voiceoutput, |
- FLAG_videoinput, FLAG_videooutput); |
- client->SetMediaEngine(engine); |
- } |
- |
- Console *console = new Console(main_thread, client); |
- client->SetConsole(console); |
- client->SetAutoAccept(auto_accept); |
- client->SetPmucDomain(pmuc_domain); |
- client->SetPortAllocatorFlags(portallocator_flags); |
- client->SetAllowLocalIps(true); |
- client->SetSignalingProtocol(signaling_protocol); |
- client->SetTransportProtocol(transport_protocol); |
- client->SetSecurePolicy(sdes_policy, dtls_policy); |
- client->SetSslIdentity(ssl_identity); |
- client->SetRender(render); |
- client->SetDataChannelType(data_channel_type); |
- client->SetMultiSessionEnabled(multisession_enabled); |
- client->SetShowRosterMessages(show_roster_messages); |
- console->Start(); |
- |
- if (debug) { |
- pump.client()->SignalLogInput.connect(&debug_log_, &DebugLog::Input); |
- pump.client()->SignalLogOutput.connect(&debug_log_, &DebugLog::Output); |
- } |
- |
- Print(("Logging in to " + server + " as " + jid.Str() + "\n").c_str()); |
- pump.DoLogin(xcs, new buzz::XmppSocket(buzz::TLS_REQUIRED), new XmppAuth()); |
- main_thread->Run(); |
- pump.DoDisconnect(); |
- |
- console->Stop(); |
- delete console; |
- delete client; |
- |
- return 0; |
-} |