From ad29c3c439947f136d3529da88bd546d24d21a31 Mon Sep 17 00:00:00 2001 From: marc tobias Date: Mon, 17 Jun 2024 00:46:34 +0200 Subject: [PATCH] add headless-browser testing of UI --- .github/workflows/ci.yml | 1 + composer.json | 5 +- composer.lock | 236 ++++++++++++++++++++++++++++++++++++--- test/PageTest.php | 105 +++++++++++++++++ test/WebServerHelper.php | 40 +++++++ 5 files changed, 371 insertions(+), 16 deletions(-) create mode 100644 test/PageTest.php create mode 100644 test/WebServerHelper.php diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 045e4f3..3b2d66f 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -31,4 +31,5 @@ jobs: - name: Run tests run: | composer validate + cp www/config.php.sample www/config.php ./vendor/bin/phpunit --display-warnings test/ diff --git a/composer.json b/composer.json index 1c48952..97e09d3 100644 --- a/composer.json +++ b/composer.json @@ -3,10 +3,11 @@ "description": "Web-based OpenStreetMap Editor", "type": "project", "require": { - "jbelien/oauth2-openstreetmap": "^0.1.2" + "jbelien/oauth2-openstreetmap": "^0.1.2" }, "require-dev": { - "phpunit/phpunit": "^11.2" + "phpunit/phpunit": "^11.2", + "php-webdriver/webdriver": "^1.15" }, "license": "WTFPL" } diff --git a/composer.lock b/composer.lock index c78c126..55953e8 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "c31154080b9231643944c534b5e3614f", + "content-hash": "4b0b3b74d82e3a677503a1aeba7523ee", "packages": [ { "name": "guzzlehttp/guzzle", @@ -774,16 +774,16 @@ "packages-dev": [ { "name": "myclabs/deep-copy", - "version": "1.11.1", + "version": "1.12.0", "source": { "type": "git", "url": "https://github.com/myclabs/DeepCopy.git", - "reference": "7284c22080590fb39f2ffa3e9057f10a4ddd0e0c" + "reference": "3a6b9a42cd8f8771bd4295d13e1423fa7f3d942c" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/7284c22080590fb39f2ffa3e9057f10a4ddd0e0c", - "reference": "7284c22080590fb39f2ffa3e9057f10a4ddd0e0c", + "url": "https://api.github.com/repos/myclabs/DeepCopy/zipball/3a6b9a42cd8f8771bd4295d13e1423fa7f3d942c", + "reference": "3a6b9a42cd8f8771bd4295d13e1423fa7f3d942c", "shasum": "" }, "require": { @@ -791,11 +791,12 @@ }, "conflict": { "doctrine/collections": "<1.6.8", - "doctrine/common": "<2.13.3 || >=3,<3.2.2" + "doctrine/common": "<2.13.3 || >=3 <3.2.2" }, "require-dev": { "doctrine/collections": "^1.6.8", "doctrine/common": "^2.13.3 || ^3.2.2", + "phpspec/prophecy": "^1.10", "phpunit/phpunit": "^7.5.20 || ^8.5.23 || ^9.5.13" }, "type": "library", @@ -821,7 +822,7 @@ ], "support": { "issues": "https://github.com/myclabs/DeepCopy/issues", - "source": "https://github.com/myclabs/DeepCopy/tree/1.11.1" + "source": "https://github.com/myclabs/DeepCopy/tree/1.12.0" }, "funding": [ { @@ -829,7 +830,7 @@ "type": "tidelift" } ], - "time": "2023-03-08T13:26:56+00:00" + "time": "2024-06-12T14:39:25+00:00" }, { "name": "nikic/php-parser", @@ -1007,6 +1008,72 @@ }, "time": "2022-02-21T01:04:05+00:00" }, + { + "name": "php-webdriver/webdriver", + "version": "1.15.1", + "source": { + "type": "git", + "url": "https://github.com/php-webdriver/php-webdriver.git", + "reference": "cd52d9342c5aa738c2e75a67e47a1b6df97154e8" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-webdriver/php-webdriver/zipball/cd52d9342c5aa738c2e75a67e47a1b6df97154e8", + "reference": "cd52d9342c5aa738c2e75a67e47a1b6df97154e8", + "shasum": "" + }, + "require": { + "ext-curl": "*", + "ext-json": "*", + "ext-zip": "*", + "php": "^7.3 || ^8.0", + "symfony/polyfill-mbstring": "^1.12", + "symfony/process": "^5.0 || ^6.0 || ^7.0" + }, + "replace": { + "facebook/webdriver": "*" + }, + "require-dev": { + "ergebnis/composer-normalize": "^2.20.0", + "ondram/ci-detector": "^4.0", + "php-coveralls/php-coveralls": "^2.4", + "php-mock/php-mock-phpunit": "^2.0", + "php-parallel-lint/php-parallel-lint": "^1.2", + "phpunit/phpunit": "^9.3", + "squizlabs/php_codesniffer": "^3.5", + "symfony/var-dumper": "^5.0 || ^6.0" + }, + "suggest": { + "ext-SimpleXML": "For Firefox profile creation" + }, + "type": "library", + "autoload": { + "files": [ + "lib/Exception/TimeoutException.php" + ], + "psr-4": { + "Facebook\\WebDriver\\": "lib/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "description": "A PHP client for Selenium WebDriver. Previously facebook/webdriver.", + "homepage": "https://github.com/php-webdriver/php-webdriver", + "keywords": [ + "Chromedriver", + "geckodriver", + "php", + "selenium", + "webdriver" + ], + "support": { + "issues": "https://github.com/php-webdriver/php-webdriver/issues", + "source": "https://github.com/php-webdriver/php-webdriver/tree/1.15.1" + }, + "time": "2023-10-20T12:21:20+00:00" + }, { "name": "phpunit/php-code-coverage", "version": "11.0.3", @@ -1332,16 +1399,16 @@ }, { "name": "phpunit/phpunit", - "version": "11.2.0", + "version": "11.2.2", "source": { "type": "git", "url": "https://github.com/sebastianbergmann/phpunit.git", - "reference": "705eba0190afe04bc057f565ad843267717cf109" + "reference": "3e1843a58adc9c433ee6170bdee7d615f7ccc20b" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/705eba0190afe04bc057f565ad843267717cf109", - "reference": "705eba0190afe04bc057f565ad843267717cf109", + "url": "https://api.github.com/repos/sebastianbergmann/phpunit/zipball/3e1843a58adc9c433ee6170bdee7d615f7ccc20b", + "reference": "3e1843a58adc9c433ee6170bdee7d615f7ccc20b", "shasum": "" }, "require": { @@ -1412,7 +1479,7 @@ "support": { "issues": "https://github.com/sebastianbergmann/phpunit/issues", "security": "https://github.com/sebastianbergmann/phpunit/security/policy", - "source": "https://github.com/sebastianbergmann/phpunit/tree/11.2.0" + "source": "https://github.com/sebastianbergmann/phpunit/tree/11.2.2" }, "funding": [ { @@ -1428,7 +1495,7 @@ "type": "tidelift" } ], - "time": "2024-06-07T04:48:50+00:00" + "time": "2024-06-15T09:14:53+00:00" }, { "name": "sebastian/cli-parser", @@ -2353,6 +2420,147 @@ ], "time": "2024-02-02T06:10:47+00:00" }, + { + "name": "symfony/polyfill-mbstring", + "version": "v1.29.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-mbstring.git", + "reference": "9773676c8a1bb1f8d4340a62efe641cf76eda7ec" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/9773676c8a1bb1f8d4340a62efe641cf76eda7ec", + "reference": "9773676c8a1bb1f8d4340a62efe641cf76eda7ec", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "provide": { + "ext-mbstring": "*" + }, + "suggest": { + "ext-mbstring": "For best performance" + }, + "type": "library", + "extra": { + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Mbstring\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for the Mbstring extension", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "mbstring", + "polyfill", + "portable", + "shim" + ], + "support": { + "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.29.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-01-29T20:11:03+00:00" + }, + { + "name": "symfony/process", + "version": "v7.1.1", + "source": { + "type": "git", + "url": "https://github.com/symfony/process.git", + "reference": "febf90124323a093c7ee06fdb30e765ca3c20028" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/process/zipball/febf90124323a093c7ee06fdb30e765ca3c20028", + "reference": "febf90124323a093c7ee06fdb30e765ca3c20028", + "shasum": "" + }, + "require": { + "php": ">=8.2" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Process\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Executes commands in sub-processes", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/process/tree/v7.1.1" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2024-05-31T14:57:53+00:00" + }, { "name": "theseer/tokenizer", "version": "1.2.3", diff --git a/test/PageTest.php b/test/PageTest.php new file mode 100644 index 0000000..98f52d0 --- /dev/null +++ b/test/PageTest.php @@ -0,0 +1,105 @@ +setBinary('/home/user/Downloads/my_chrome_binary'); + $chromeOptions->addArguments(['--headless']); + + $capabilities = DesiredCapabilities::chrome(); + $capabilities->setCapability(ChromeOptions::CAPABILITY_W3C, $chromeOptions); + + $this->webDriver = ChromeDriver::start($capabilities); + } + + public function tearDown(): void + { + $this->webDriver->quit(); + } + + private function elementByCSS($query) + { + return $this->webDriver->findElement(WebDriverBy::cssSelector($query)); + } + + public function testPage() + { + $this->webDriver->get(WebServerHelper::url()); + + sleep(2); + // Wait for map (Javacript) to load + $this->webDriver->wait()->until( + WebDriverExpectedCondition::presenceOfElementLocated(WebDriverBy::cssSelector('.leaflet-map-pane')) + ); + + + $h2 = $this->elementByCSS('h2'); + $this->assertEquals($h2->getText(), 'Level0 OpenStreetMap Editor'); + + + $input = $this->elementByCSS('input[name="url"]'); + $input->sendKeys(array('w105957600', WebDriverKeys::ENTER)); + + sleep(2); + + $textarea = $this->elementByCSS('textarea'); + $this->assertStringContainsString("way 105957600\n", $textarea->getText()); + $this->assertStringContainsString(" addr:street = Downing Street\n", $textarea->getText()); + + + // Button "Add to Editor" + $input = $this->elementByCSS('input[name="url"]'); + $input->sendKeys('n3815077900'); + $this->elementByCSS('input[type="submit"][name="add"]')->click(); + sleep(2); + $textarea = $this->elementByCSS('textarea'); + $this->assertStringContainsString("way 105957600\n", $textarea->getText()); + $this->assertStringContainsString("node 3815077900: ", $textarea->getText()); + + + // Button "Check for conflicts" + $this->elementByCSS('input[name="check"]')->click(); + $this->assertStringContainsString('Nothing is modified', $this->webDriver->getPageSource()); + + + // Button "Show osmChange" + $this->elementByCSS('input[name="showosc"]')->click(); + $this->assertStringContainsString('this will be uploaded to the server', $this->webDriver->getPageSource()); + $this->assertMatchesRegularExpression('/<osmChange [^&]+>\s*<\/osmChange>/', $this->webDriver->getPageSource()); + } +} diff --git a/test/WebServerHelper.php b/test/WebServerHelper.php new file mode 100644 index 0000000..c910e80 --- /dev/null +++ b/test/WebServerHelper.php @@ -0,0 +1,40 @@ + /dev/null 2>&1 & echo $!', + self::$host, + self::$port, + escapeshellarg(self::$docRoot) + ); + + $output = []; + exec($command, $output); + self::$pid = (int) $output[0]; + + file_put_contents(__DIR__ . '/server.pid', self::$pid); + sleep(1); + } + + public static function stop() + { + $pid = (int) @file_get_contents(__DIR__ . '/server.pid'); + if ($pid) { + exec('kill ' . $pid); + unlink(__DIR__ . '/server.pid'); + } + } +}