// Copyright 2013 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 <stdio.h> #include <string> #include "base/at_exit.h" #include "base/bind.h" #include "base/cancelable_callback.h" #include "base/command_line.h" #include "base/files/file_util.h" #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_split.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/time/time.h" #include "components/wifi/wifi_service.h" #if defined(OS_MACOSX) #include "base/mac/scoped_nsautorelease_pool.h" #endif namespace wifi { class WiFiTest { public: WiFiTest() {} ~WiFiTest() {} enum Result { RESULT_ERROR = -2, RESULT_WRONG_USAGE = -1, RESULT_OK = 0, RESULT_PENDING = 1, }; Result Main(int argc, const char* argv[]); private: bool ParseCommandLine(int argc, const char* argv[]); void Start() {} void Finish(Result result) { DCHECK_NE(RESULT_PENDING, result); result_ = result; if (base::MessageLoop::current()) base::MessageLoop::current()->Quit(); } void OnNetworksChanged( const WiFiService::NetworkGuidList& network_guid_list) { VLOG(0) << "Networks Changed: " << network_guid_list[0]; base::DictionaryValue properties; std::string error; wifi_service_->GetProperties(network_guid_list[0], &properties, &error); VLOG(0) << error << ":\n" << properties; } void OnNetworkListChanged( const WiFiService::NetworkGuidList& network_guid_list) { VLOG(0) << "Network List Changed: " << network_guid_list.size(); } #if defined(OS_MACOSX) // Without this there will be a mem leak on osx. base::mac::ScopedNSAutoreleasePool scoped_pool_; #endif scoped_ptr<WiFiService> wifi_service_; // Need AtExitManager to support AsWeakPtr (in NetLog). base::AtExitManager exit_manager_; Result result_; }; WiFiTest::Result WiFiTest::Main(int argc, const char* argv[]) { if (!ParseCommandLine(argc, argv)) { VLOG(0) << "Usage: " << argv[0] << " [--list]" " [--get_key]" " [--get_properties]" " [--create]" " [--connect]" " [--disconnect]" " [--scan]" " [--network_guid=<network_guid>]" " [--frequency=0|2400|5000]" " [--security=none|WEP-PSK|WPA-PSK|WPA2-PSK]" " [--password=<wifi_password>]" " [<network_guid>]\n"; return RESULT_WRONG_USAGE; } base::MessageLoopForIO loop; result_ = RESULT_PENDING; return result_; } bool WiFiTest::ParseCommandLine(int argc, const char* argv[]) { CommandLine::Init(argc, argv); const CommandLine& parsed_command_line = *CommandLine::ForCurrentProcess(); std::string network_guid = parsed_command_line.GetSwitchValueASCII("network_guid"); std::string frequency = parsed_command_line.GetSwitchValueASCII("frequency"); std::string password = parsed_command_line.GetSwitchValueASCII("password"); std::string security = parsed_command_line.GetSwitchValueASCII("security"); if (parsed_command_line.GetArgs().size() == 1) { #if defined(OS_WIN) network_guid = base::UTF16ToASCII(parsed_command_line.GetArgs()[0]); #else network_guid = parsed_command_line.GetArgs()[0]; #endif } #if defined(OS_WIN) if (parsed_command_line.HasSwitch("debug")) MessageBoxA(NULL, __FUNCTION__, "Debug Me!", MB_OK); #endif base::MessageLoopForIO loop; wifi_service_.reset(WiFiService::Create()); wifi_service_->Initialize(loop.message_loop_proxy()); if (parsed_command_line.HasSwitch("list")) { base::ListValue network_list; wifi_service_->GetVisibleNetworks(std::string(), &network_list, true); VLOG(0) << network_list; return true; } if (parsed_command_line.HasSwitch("get_properties")) { if (network_guid.length() > 0) { base::DictionaryValue properties; std::string error; wifi_service_->GetProperties(network_guid, &properties, &error); VLOG(0) << error << ":\n" << properties; return true; } } // Optional properties (frequency, password) to use for connect or create. scoped_ptr<base::DictionaryValue> properties(new base::DictionaryValue()); if (!frequency.empty()) { int value = 0; if (base::StringToInt(frequency, &value)) { properties->SetInteger("WiFi.Frequency", value); // fall through to connect. } } if (!password.empty()) properties->SetString("WiFi.Passphrase", password); if (!security.empty()) properties->SetString("WiFi.Security", security); if (parsed_command_line.HasSwitch("create")) { if (!network_guid.empty()) { std::string error; std::string new_network_guid; properties->SetString("WiFi.SSID", network_guid); VLOG(0) << "Creating Network: " << *properties; wifi_service_->CreateNetwork( false, properties.Pass(), &new_network_guid, &error); VLOG(0) << error << ":\n" << new_network_guid; return true; } } if (parsed_command_line.HasSwitch("connect")) { if (!network_guid.empty()) { std::string error; if (!properties->empty()) { VLOG(0) << "Using connect properties: " << *properties; wifi_service_->SetProperties(network_guid, properties.Pass(), &error); } wifi_service_->SetEventObservers( loop.message_loop_proxy(), base::Bind(&WiFiTest::OnNetworksChanged, base::Unretained(this)), base::Bind(&WiFiTest::OnNetworkListChanged, base::Unretained(this))); wifi_service_->StartConnect(network_guid, &error); VLOG(0) << error; if (error.empty()) base::MessageLoop::current()->Run(); return true; } } if (parsed_command_line.HasSwitch("disconnect")) { if (network_guid.length() > 0) { std::string error; wifi_service_->StartDisconnect(network_guid, &error); VLOG(0) << error; return true; } } if (parsed_command_line.HasSwitch("get_key")) { if (network_guid.length() > 0) { std::string error; std::string key_data; wifi_service_->GetKeyFromSystem(network_guid, &key_data, &error); VLOG(0) << key_data << error; return true; } } if (parsed_command_line.HasSwitch("scan")) { wifi_service_->SetEventObservers( loop.message_loop_proxy(), base::Bind(&WiFiTest::OnNetworksChanged, base::Unretained(this)), base::Bind(&WiFiTest::OnNetworkListChanged, base::Unretained(this))); wifi_service_->RequestNetworkScan(); base::MessageLoop::current()->Run(); return true; } return false; } } // namespace wifi int main(int argc, const char* argv[]) { CommandLine::Init(argc, argv); logging::LoggingSettings settings; settings.logging_dest = logging::LOG_TO_SYSTEM_DEBUG_LOG; logging::InitLogging(settings); wifi::WiFiTest wifi_test; return wifi_test.Main(argc, argv); }