* @license GNU General Public License, version 2 (GPL-2.0) * * For full copyright and license information, please see * the docs/CREDITS.txt file. * */ use Facebook\WebDriver\WebDriverBy; use Facebook\WebDriver\Exception\WebDriverCurlException; use Facebook\WebDriver\Remote\RemoteWebDriver; use Facebook\WebDriver\Remote\DesiredCapabilities; require_once __DIR__ . '/mock/phpbb_mock_null_installer_task.php'; class phpbb_ui_test_case extends phpbb_test_case { static protected $host = '127.0.0.1'; static protected $port = 8910; /** * @var RemoteWebDriver */ static protected $webDriver; static protected $config; static protected $root_url; static protected $already_installed = false; static protected $install_success = false; static protected $db; /** * Session ID for current test's session (each test makes its own) * @var string */ protected $sid; /** * Language array used by phpBB * @var array */ protected $lang = array(); static public function setUpBeforeClass() { parent::setUpBeforeClass(); if (version_compare(PHP_VERSION, '5.3.19', '<')) { self::markTestSkipped('UI test case requires at least PHP 5.3.19.'); } else if (!class_exists('\Facebook\WebDriver\Remote\RemoteWebDriver')) { self::markTestSkipped( 'Could not find RemoteWebDriver class. ' . 'Run "php ../composer.phar install" from the tests folder.' ); } self::$config = phpbb_test_case_helpers::get_test_config(); self::$root_url = self::$config['phpbb_functional_url']; // Important: this is used both for installation and by // test cases for querying the tables. // Therefore table prefix must be set before a board is // installed, and also before each test case is run. self::$config['table_prefix'] = 'phpbb_'; if (!isset(self::$config['phpbb_functional_url'])) { self::markTestSkipped('phpbb_functional_url was not set in test_config and wasn\'t set as PHPBB_FUNCTIONAL_URL environment variable either.'); } if (!self::$webDriver) { try { $capabilities = DesiredCapabilities::firefox(); self::$webDriver = RemoteWebDriver::create(self::$host . ':' . self::$port, $capabilities); } catch (WebDriverCurlException $e) { self::markTestSkipped('PhantomJS webserver is not running.'); } } if (!self::$already_installed) { self::install_board(); self::$already_installed = true; } } public function setUp() { if (!self::$install_success) { $this->fail('Installing phpBB has failed.'); } // Clear the language array so that things // that were added in other tests are gone $this->lang = array(); $this->add_lang('common'); } protected function tearDown() { parent::tearDown(); if (self::$db instanceof \phpbb\db\driver\driver_interface) { // Close the database connections again this test self::$db->sql_close(); } } static public function visit($path) { self::$webDriver->get(self::$root_url . $path); } static protected function recreate_database($config) { $db_conn_mgr = new phpbb_database_test_connection_manager($config); $db_conn_mgr->recreate_db(); } static public function find_element($type, $value) { return self::$webDriver->findElement(WebDriverBy::$type($value)); } static public function submit($type = 'id', $value = 'submit') { $element = self::find_element($type, $value); $element->click(); } static public function install_board() { global $phpbb_root_path, $phpEx, $db; self::recreate_database(self::$config); $db = self::get_db(); $config_file = $phpbb_root_path . "config.$phpEx"; $config_file_dev = $phpbb_root_path . "config_dev.$phpEx"; $config_file_test = $phpbb_root_path . "config_test.$phpEx"; if (file_exists($config_file)) { if (!file_exists($config_file_dev)) { rename($config_file, $config_file_dev); } else { unlink($config_file); } } $container_builder = new \phpbb\di\container_builder($phpbb_root_path, $phpEx); $container = $container_builder ->with_environment('installer') ->without_extensions() ->without_cache() ->with_custom_parameters([ 'core.disable_super_globals' => false, 'installer.create_config_file.options' => [ 'debug' => true, 'environment' => 'test', ], 'cache.driver.class' => 'phpbb\cache\driver\file' ]) ->without_compiled_container() ->get_container(); $container->register('installer.install_finish.notify_user')->setSynthetic(true); $container->set('installer.install_finish.notify_user', new phpbb_mock_null_installer_task()); $container->compile(); $language = $container->get('language'); $language->add_lang(array('common', 'acp/common', 'acp/board', 'install', 'posting')); $iohandler_factory = $container->get('installer.helper.iohandler_factory'); $iohandler_factory->set_environment('cli'); $iohandler = $iohandler_factory->get(); $parseURL = parse_url(self::$config['phpbb_functional_url']); $output = new \Symfony\Component\Console\Output\NullOutput(); $style = new \Symfony\Component\Console\Style\SymfonyStyle( new \Symfony\Component\Console\Input\ArrayInput(array()), $output ); $iohandler->set_style($style, $output); $installer = $container->get('installer.installer.install'); $installer->set_iohandler($iohandler); // Set data $iohandler->set_input('admin_name', 'admin'); $iohandler->set_input('admin_pass1', 'adminadmin'); $iohandler->set_input('admin_pass2', 'adminadmin'); $iohandler->set_input('board_email', 'nobody@example.com'); $iohandler->set_input('submit_admin', 'submit'); $iohandler->set_input('default_lang', 'en'); $iohandler->set_input('board_name', 'yourdomain.com'); $iohandler->set_input('board_description', 'A short text to describe your forum'); $iohandler->set_input('submit_board', 'submit'); $iohandler->set_input('dbms', str_replace('phpbb\db\driver\\', '', self::$config['dbms'])); $iohandler->set_input('dbhost', self::$config['dbhost']); $iohandler->set_input('dbport', self::$config['dbport']); $iohandler->set_input('dbuser', self::$config['dbuser']); $iohandler->set_input('dbpasswd', self::$config['dbpasswd']); $iohandler->set_input('dbname', self::$config['dbname']); $iohandler->set_input('table_prefix', self::$config['table_prefix']); $iohandler->set_input('submit_database', 'submit'); $iohandler->set_input('email_enable', true); $iohandler->set_input('smtp_delivery', '1'); $iohandler->set_input('smtp_host', 'nxdomain.phpbb.com'); $iohandler->set_input('smtp_auth', 'PLAIN'); $iohandler->set_input('smtp_user', 'nxuser'); $iohandler->set_input('smtp_pass', 'nxpass'); $iohandler->set_input('submit_email', 'submit'); $iohandler->set_input('cookie_secure', '0'); $iohandler->set_input('server_protocol', '0'); $iohandler->set_input('force_server_vars', $parseURL['scheme'] . '://'); $iohandler->set_input('server_name', $parseURL['host']); $iohandler->set_input('server_port', isset($parseURL['port']) ? (int) $parseURL['port'] : 80); $iohandler->set_input('script_path', $parseURL['path']); $iohandler->set_input('submit_server', 'submit'); $installer->run(); copy($config_file, $config_file_test); self::$install_success = true; if (file_exists($phpbb_root_path . 'store/install_config.php')) { self::$install_success = false; @unlink($phpbb_root_path . 'store/install_config.php'); } if (file_exists($phpbb_root_path . 'cache/install_lock')) { @unlink($phpbb_root_path . 'cache/install_lock'); } global $phpbb_container, $cache, $phpbb_dispatcher, $request, $user, $auth, $db, $config, $phpbb_log, $symfony_request, $phpbb_filesystem, $phpbb_path_helper, $phpbb_extension_manager, $template; $phpbb_container->reset(); unset($phpbb_container, $cache, $phpbb_dispatcher, $request, $user, $auth, $db, $config, $phpbb_log, $symfony_request, $phpbb_filesystem, $phpbb_path_helper, $phpbb_extension_manager, $template); } static protected function get_db() { global $phpbb_root_path, $phpEx; // so we don't reopen an open connection if (!(self::$db instanceof \phpbb\db\driver\driver_interface)) { $dbms = self::$config['dbms']; /** @var \phpbb\db\driver\driver_interface $db */ $db = new $dbms(); $db->sql_connect(self::$config['dbhost'], self::$config['dbuser'], self::$config['dbpasswd'], self::$config['dbname'], self::$config['dbport']); self::$db = $db; } return self::$db; } protected function logout() { $this->add_lang('ucp'); if (empty($this->sid)) { return; } $this->visit('ucp.php?sid=' . $this->sid . '&mode=logout'); $this->assertContains($this->lang('REGISTER'), self::$webDriver->getPageSource()); unset($this->sid); } /** * Login to the ACP * You must run login() before calling this. */ protected function admin_login($username = 'admin') { $this->add_lang('acp/common'); // Requires login first! if (empty($this->sid)) { $this->fail('$this->sid is empty. Make sure you call login() before admin_login()'); return; } self::$webDriver->manage()->deleteAllCookies(); $this->visit('adm/index.php?sid=' . $this->sid); $this->assertContains($this->lang('LOGIN_ADMIN_CONFIRM'), self::$webDriver->getPageSource()); self::find_element('cssSelector', 'input[name=username]')->clear()->sendKeys($username); self::find_element('cssSelector', 'input[type=password]')->sendKeys($username . $username); self::find_element('cssSelector', 'input[name=login]')->click(); $this->assertContains($this->lang('ADMIN_PANEL'), $this->find_element('cssSelector', 'h1')->getText()); $cookies = self::$webDriver->manage()->getCookies(); // The session id is stored in a cookie that ends with _sid - we assume there is only one such cookie foreach ($cookies as $cookie) { if (substr($cookie['name'], -4) == '_sid') { $this->sid = $cookie['value']; break; } } $this->assertNotEmpty($this->sid); } protected function add_lang($lang_file) { if (is_array($lang_file)) { foreach ($lang_file as $file) { $this->add_lang($file); } } $lang_path = __DIR__ . "/../../phpBB/language/en/$lang_file.php"; $lang = array(); if (file_exists($lang_path)) { include($lang_path); } $this->lang = array_merge($this->lang, $lang); } protected function add_lang_ext($ext_name, $lang_file) { if (is_array($lang_file)) { foreach ($lang_file as $file) { $this->add_lang_ext($ext_name, $file); } return; } $lang_path = __DIR__ . "/../../phpBB/ext/{$ext_name}/language/en/$lang_file.php"; $lang = array(); if (file_exists($lang_path)) { include($lang_path); } $this->lang = array_merge($this->lang, $lang); } protected function lang() { $args = func_get_args(); $key = $args[0]; if (empty($this->lang[$key])) { throw new RuntimeException('Language key "' . $key . '" could not be found.'); } $args[0] = $this->lang[$key]; return call_user_func_array('sprintf', $args); } /** * assertContains for language strings * * @param string $needle Search string * @param string $haystack Search this * @param string $message Optional failure message */ public function assertContainsLang($needle, $haystack, $message = null) { $this->assertContains(html_entity_decode($this->lang($needle), ENT_QUOTES), $haystack, $message); } /** * assertNotContains for language strings * * @param string $needle Search string * @param string $haystack Search this * @param string $message Optional failure message */ public function assertNotContainsLang($needle, $haystack, $message = null) { $this->assertNotContains(html_entity_decode($this->lang($needle), ENT_QUOTES), $haystack, $message); } protected function login($username = 'admin') { $this->add_lang('ucp'); $this->visit('ucp.php'); $this->assertContains($this->lang('LOGIN_EXPLAIN_UCP'), self::$webDriver->getPageSource()); self::$webDriver->manage()->deleteAllCookies(); self::find_element('cssSelector', 'input[name=username]')->sendKeys($username); self::find_element('cssSelector', 'input[name=password]')->sendKeys($username . $username); self::find_element('cssSelector', 'input[name=login]')->click(); $this->assertNotContains($this->lang('LOGIN'), $this->find_element('className', 'navbar')->getText()); $cookies = self::$webDriver->manage()->getCookies(); // The session id is stored in a cookie that ends with _sid - we assume there is only one such cookie foreach ($cookies as $cookie) { if (substr($cookie['name'], -4) == '_sid') { $this->sid = $cookie['value']; } } $this->assertNotEmpty($this->sid); } /** * Take screenshot. Can be used for debug purposes. * * @throws Exception When screenshot can't be created */ public function take_screenshot() { // Change the Path to your own settings $screenshot = time() . ".png"; self::$webDriver->takeScreenshot($screenshot); } }