Vmeda.Online/local_moodleappbehat/tests/behat/behat_app.php

1339 lines
46 KiB
PHP
Raw Normal View History

<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
// NOTE: no MOODLE_INTERNAL test here, this file may be required by behat before including /config.php.
2021-04-27 15:43:47 +00:00
require_once(__DIR__ . '/../../../../lib/behat/behat_base.php');
require_once(__DIR__ . '/behat_app_helper.php');
use Behat\Behat\Hook\Scope\ScenarioScope;
use Behat\Behat\Hook\Scope\AfterStepScope;
use Behat\Gherkin\Node\TableNode;
use Behat\Mink\Exception\DriverException;
use Behat\Mink\Exception\ExpectationException;
/**
* Moodle App steps definitions.
*
* @package core
* @category test
* @copyright 2018 The Open University
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class behat_app extends behat_app_helper {
2022-03-10 11:07:15 +00:00
/** @var array Config overrides */
protected $appconfig = [
'disableUserTours' => true,
'toastDurations' => [ // Extend toast durations in Behat so they don't disappear too soon.
'short' => 7500,
'long' => 10000,
'sticky' => 0,
],
];
2022-03-10 11:07:15 +00:00
protected $featurepath;
protected $coveragepath;
protected $scenarioslug;
protected $scenariolaststep;
/**
* @BeforeScenario
*/
public function before_scenario(ScenarioScope $scope) {
2023-03-30 10:39:00 +00:00
$feature = $scope->getFeature();
2023-03-30 10:39:00 +00:00
if (!$feature->hasTag('app')) {
return;
}
$steps = $scope->getScenario()->getSteps();
$this->scenarioslug = $this->get_scenario_slug($scope);
$this->scenariolaststep = $steps[count($steps) - 1];
2023-03-30 10:39:00 +00:00
$this->featurepath = dirname($feature->getFile());
$this->coveragepath = get_config('local_moodleappbehat', 'coverage_path') ?: ($this->featurepath . DIRECTORY_SEPARATOR . 'coverage' . DIRECTORY_SEPARATOR);
}
/**
* @AfterStep
*/
public function after_step(AfterStepScope $scope) {
$step = $scope->getStep();
if ($step !== $this->scenariolaststep || empty($this->coveragepath)) {
return;
}
if (!is_dir($this->coveragepath)) {
if (!@mkdir($this->coveragepath, 0777, true)) {
throw new Exception("Cannot create {$this->coveragepath} directory.");
}
}
$coverage = $this->runtime_js('getCoverage()');
if (!is_null($coverage)) {
file_put_contents($this->coveragepath . $this->scenarioslug . '.json', $coverage);
}
}
/**
* Opens the Moodle App in the browser and optionally logs in.
*
* @When I enter the app
* @Given I entered the app as :username
* @param string $username Username
* @throws DriverException Issue with configuration or feature file
* @throws dml_exception Problem with Moodle setup
* @throws ExpectationException Problem with resizing window
*/
public function i_enter_the_app(string $username = null) {
$this->i_launch_the_app();
if (!is_null($username)) {
$this->open_moodleapp_custom_login_url($username);
return;
}
$this->enter_site();
}
/**
* Check whether the current page is the login form.
*/
protected function is_in_login_page(): bool {
$page = $this->getSession()->getPage();
$logininput = $page->find('xpath', '//page-core-login-site//input[@name="url"]');
return !is_null($logininput);
}
/**
* Opens the Moodle App in the browser.
*
* @When I launch the app :runtime
* @When I launch the app
* @param string $runtime Runtime
* @throws DriverException Issue with configuration or feature file
* @throws dml_exception Problem with Moodle setup
* @throws ExpectationException Problem with resizing window
*/
public function i_launch_the_app(string $runtime = '') {
// Go to page and prepare browser for app.
$this->prepare_browser(['skiponboarding' => empty($runtime)]);
}
/**
* Restart the app.
*
* @When I restart the app
*/
public function i_restart_the_app() {
$this->getSession()->visit($this->get_app_url());
$this->i_wait_the_app_to_restart();
}
/**
* @Then I wait the app to restart
*/
public function i_wait_the_app_to_restart() {
// Prepare testing runtime again.
$this->prepare_browser();
}
2022-07-07 10:03:58 +00:00
/**
* @Then I log out in the app
*
* @param bool $force If force logout or not.
*/
public function i_log_out_in_app($force = true) {
$options = json_encode([
'forceLogout' => $force,
]);
$result = $this->zone_js("sites.logout($options)");
if ($result !== 'OK') {
throw new DriverException('Error on log out - ' . $result);
}
$this->i_wait_the_app_to_restart();
}
2021-04-27 15:43:47 +00:00
/**
* Finds elements in the app.
*
* @Then /^I should( not)? find (".+")( inside the .+)? in the app$/
* @param bool $not Whether assert that the element was not found
* @param string $locator Element locator
2022-06-30 11:48:43 +00:00
* @param string $container Container name
2021-04-27 15:43:47 +00:00
*/
2022-06-30 11:48:43 +00:00
public function i_find_in_the_app(bool $not, string $locator, string $container = '') {
$locator = $this->parse_element_locator($locator);
2022-06-30 11:48:43 +00:00
if (!empty($container)) {
preg_match('/^ inside the (.+)$/', $container, $matches);
$container = $matches[1];
}
2022-06-30 11:48:43 +00:00
$options = json_encode(['containerName' => $container]);
2021-04-27 15:43:47 +00:00
2022-06-30 11:48:43 +00:00
$this->spin(function() use ($not, $locator, $options) {
$result = $this->runtime_js("find($locator, $options)");
2021-04-27 15:43:47 +00:00
if ($not && $result === 'OK') {
throw new DriverException('Error, found an element that should not be found');
2021-04-27 15:43:47 +00:00
}
if (!$not && $result !== 'OK') {
throw new DriverException('Error finding element - ' . $result);
2021-04-27 15:43:47 +00:00
}
return true;
});
2021-04-27 15:43:47 +00:00
$this->wait_for_pending_js();
}
/**
* Scroll to an element in the app.
*
* @When /^I scroll to (".+") in the app$/
* @param string $locator Element locator
*/
public function i_scroll_to_in_the_app(string $locator) {
$locator = $this->parse_element_locator($locator);
$this->spin(function() use ($locator) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("scrollTo($locator)");
if ($result !== 'OK') {
throw new DriverException('Error finding element - ' . $result);
}
return true;
});
$this->wait_animations_done();
}
/**
* Load more items in a list with an infinite loader.
*
* @When /^I (should not be able to )?load more items in the app$/
* @param bool $not Whether assert that it is not possible to load more items
*/
public function i_load_more_items_in_the_app(bool $not = false) {
$this->spin(function() use ($not) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js('loadMoreItems()');
if ($not && $result !== 'ERROR: All items are already loaded.') {
throw new DriverException('It should not have been possible to load more items');
}
if (!$not && $result !== 'OK') {
throw new DriverException('Error loading more items - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Trigger swipe gesture.
*
* @When /^I swipe to the (left|right) (in (".+") )?in the app$/
* @param string $direction Swipe direction
* @param bool $hasLocator Whether a reference locator is used.
* @param string $locator Reference locator.
*/
public function i_swipe_in_the_app(string $direction, bool $hasLocator = false, string $locator = '') {
if ($hasLocator) {
$locator = $this->parse_element_locator($locator);
}
$result = $this->zone_js("swipe('$direction'" . ($hasLocator ? ", $locator" : '') . ')');
if ($result !== 'OK') {
throw new DriverException('Error when swiping - ' . $result);
}
$this->wait_animations_done();
}
/**
* Wait for a BBB room to start.
*
* @When I wait for the BigBlueButton room to start
*/
public function i_wait_bbb_room_to_start() {
$windowNames = $this->getSession()->getWindowNames();
$this->getSession()->switchToWindow(array_pop($windowNames));
$this->spin(function($context) {
$joinmodal = $context->getSession()->getPage()->find('css', 'div[role="dialog"][aria-label="How would you like to join the audio?"]');
if ($joinmodal) {
return true;
}
throw new DriverException('BBB room not started');
}, false, 30);
}
/**
* Check if elements are selected in the app.
*
* @Then /^(".+") should( not)? be selected in the app$/
* @param string $locator Element locator
* @param bool $not Whether to assert that the element is not selected
*/
public function be_selected_in_the_app(string $locator, bool $not = false) {
$locator = $this->parse_element_locator($locator);
$this->spin(function() use ($locator, $not) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("isSelected($locator)");
switch ($result) {
case 'YES':
if ($not) {
throw new ExpectationException("Element was selected and shouldn't have", $this->getSession()->getDriver());
}
break;
case 'NO':
if (!$not) {
throw new ExpectationException("Element wasn't selected and should have", $this->getSession()->getDriver());
}
break;
default:
throw new DriverException('Error finding element - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Carries out the login steps for the app, assuming the user is on the app login page. Called
* from behat_auth.php.
*
* @param string $username Username (and password)
* @throws Exception Any error
*/
public function login(string $username) {
$this->i_set_the_field_in_the_app('Username', $username);
$this->i_set_the_field_in_the_app('Password', $username);
// Note there are two 'Log in' texts visible (the title and the button) so we have to use
// a 'near' value here.
$this->i_press_in_the_app('"Log in" "ion-button"');
// Wait until the main page appears.
$this->spin(
function($context) {
$initialpage = $context->getSession()->getPage()->find('xpath', '//page-core-mainmenu') ??
$context->getSession()->getPage()->find('xpath', '//page-core-login-change-password') ??
$context->getSession()->getPage()->find('xpath', '//page-core-user-complete-profile');
if ($initialpage) {
return true;
}
throw new DriverException('Moodle App main page not loaded after login');
}, false, 30);
// Wait for JS to finish as well.
$this->wait_for_pending_js();
}
/**
* Enter site.
*/
protected function enter_site() {
if (!$this->is_in_login_page()) {
// Already in the site.
return;
}
global $CFG;
$this->i_set_the_field_in_the_app('Your site', $CFG->wwwroot);
$this->i_press_in_the_app('"Connect to your site"');
$this->wait_for_pending_js();
}
/**
* Shortcut to let the user enter a course in the app.
*
* @Given I entered the course :coursename as :username in the app
* @Given I entered the course :coursename in the app
* @param string $coursename Course name
* @param string $username Username
* @throws DriverException If the button push doesn't work
*/
public function i_entered_the_course_in_the_app(string $coursename, ?string $username = null) {
$courseid = $this->get_course_id($coursename);
if (!$courseid) {
throw new DriverException("Course '$coursename' not found");
}
if ($username) {
$this->i_launch_the_app();
$this->open_moodleapp_custom_login_url($username, "/course/view.php?id=$courseid", '//page-core-course-index');
} else {
$this->open_moodleapp_custom_url("/course/view.php?id=$courseid", '//page-core-course-index');
}
}
/**
* User enters a course in the app.
*
* @Given I enter the course :coursename as :username in the app
* @Given I enter the course :coursename in the app
* @param string $coursename Course name
* @param string $username Username
* @throws DriverException If the button push doesn't work
*/
public function i_enter_the_course_in_the_app(string $coursename, ?string $username = null) {
if (!is_null($username)) {
$this->i_enter_the_app();
$this->login($username);
}
2022-06-30 11:48:43 +00:00
$mycoursesfound = $this->runtime_js("find({ text: 'My courses', selector: 'ion-tab-button'})");
if ($mycoursesfound !== 'OK') {
// My courses not present enter from Dashboard.
$this->i_press_in_the_app('"Home" "ion-tab-button"');
$this->i_press_in_the_app('"Dashboard"');
$this->i_press_in_the_app('"'.$coursename.'" near "Course overview"');
$this->wait_for_pending_js();
return;
}
$this->i_press_in_the_app('"My courses" "ion-tab-button"');
$this->i_press_in_the_app('"'.$coursename.'"');
$this->wait_for_pending_js();
}
/**
* User enters an activity in a course in the app.
*
* @Given I entered the :activity activity :activityname on course :course as :username in the app
* @Given I entered the :activity activity :activityname on course :course in the app
* @param string $activity Activity
* @param string $activityname Activity name
* @param string $coursename Course name
* @param string $username Username
* @throws DriverException If the button push doesn't work
*/
public function i_enter_the_activity_in_the_app(string $activity, string $activityname, string $coursename, ?string $username = null) {
$cm = $this->get_cm_by_activity_name_and_course($activity, $activityname, $coursename);
if (!$cm) {
throw new DriverException("'$activityname' activity '$activityname' not found");
}
$pageurl = "/mod/$activity/view.php?id={$cm->id}";
if ($username) {
$this->i_launch_the_app();
$this->open_moodleapp_custom_login_url($username, $pageurl);
} else {
$this->open_moodleapp_custom_url($pageurl);
}
}
/**
* Presses standard buttons in the app.
*
* @When /^I press the (back|more menu|page menu|user menu) button in the app$/
* @param string $button Button type
* @throws DriverException If the button push doesn't work
*/
public function i_press_the_standard_button_in_the_app(string $button) {
$this->spin(function() use ($button) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("pressStandard('$button')");
if ($result !== 'OK') {
throw new DriverException('Error pressing standard button - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Presses go back repeatedly in the app.
*
* @When /^I go back( (\d+) times)? in the app$/
* @param int $times
* @throws DriverException If the navigation doesn't work
*/
public function i_go_back_x_times_in_the_app(?string $unused = null, ?int $times = 1) {
if ($times < 1) {
return;
}
$this->spin(function() use ($times) {
$result = $this->runtime_js("goBackTimes($times)");
if ($result !== 'OK') {
throw new DriverException('Error navigating back - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Presses go back repeatedly until the app is in the main menu.
*
* @When /^I go back to the root page in the app$/
* @throws DriverException If the navigation doesn't work
*/
public function i_go_back_to_root_in_the_app() {
$this->spin(function() {
$result = $this->runtime_js("goBackToRoot()");
if ($result !== 'OK') {
throw new DriverException('Error navigating to root - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Receives push notifications.
*
* @When /^I click a push notification in the app for:$/
* @param TableNode $data Table data
*/
public function i_click_a_push_notification(TableNode $data) {
global $DB, $CFG;
$data = (object) $data->getColumnsHash()[0];
if (isset($data->module, $data->discussion)) {
$module = $DB->get_record('course_modules', ['idnumber' => $data->module]);
$discussion = $DB->get_record('forum_discussions', ['name' => $data->discussion]);
$data->name = 'posts';
$data->component = 'mod_forum';
}
$notification = json_encode([
'site' => md5($CFG->behat_wwwroot . $data->username),
'subject' => $data->subject ?? null,
'userfrom' => $data->userfrom ?? null,
'userto' => $data->username ?? null,
'message' => $data->message ?? '',
'title' => $data->title ?? '',
'image' => $data->image ?? null,
'courseid' => $discussion->course ?? null,
'moodlecomponent' => $data->component ?? null,
'name' => $data->name ?? null,
'contexturl' => '',
'notif' => 1,
'customdata' => isset($discussion->id, $module->id, $discussion->forum)
? ['discussionid' => $discussion->id, 'cmid' => $module->id, 'instance' => $discussion->forum]
: null,
'additionalData' => isset($data->subject) || isset($data->userfrom)
? ['foreground' => true, 'notId' => 23, 'notif' => 1] : null,
]);
$this->evaluate_script("pushNotifications.notificationClicked($notification)", true);
$this->wait_for_pending_js();
}
/**
* Replace arguments from the content in the given activity field.
*
* @Given /^I replace the arguments in "([^"]+)" "([^"]+)"$/
* @param string $idnumber Id number
* @param string $field Field
*/
public function i_replace_arguments_in_the_activity(string $idnumber, string $field) {
global $DB;
$coursemodule = $DB->get_record('course_modules', compact('idnumber'));
$module = $DB->get_record('modules', ['id' => $coursemodule->module]);
$activity = $DB->get_record($module->name, ['id' => $coursemodule->instance]);
$DB->update_record($module->name, [
'id' => $coursemodule->instance,
$field => $this->replace_arguments($activity->{$field}),
]);
}
/**
* Opens a custom link.
*
* @Given /^I open a custom link in the app for:$/
* @param TableNode $data Table data
*/
public function i_open_a_custom_link(TableNode $data) {
global $DB;
$data = $data->getColumnsHash()[0];
$title = array_keys($data)[0];
$data = (object) $data;
$username = $data->user ?? '';
switch ($title) {
case 'discussion':
$discussion = $DB->get_record('forum_discussions', ['name' => $data->discussion]);
$pageurl = "/mod/forum/discuss.php?d={$discussion->id}";
break;
case 'assign':
case 'bigbluebuttonbn':
case 'book':
case 'chat':
case 'choice':
case 'data':
case 'feedback':
case 'folder':
case 'forum':
case 'glossary':
case 'h5pactivity':
case 'imscp':
case 'label':
case 'lesson':
case 'lti':
case 'page':
case 'quiz':
case 'resource':
case 'scorm':
case 'survey':
case 'url':
case 'wiki':
case 'workshop':
$name = $data->$title;
$module = $DB->get_record($title, ['name' => $name]);
$cm = get_coursemodule_from_instance($title, $module->id);
$pageurl = "/mod/$title/view.php?id={$cm->id}";
break;
default:
throw new DriverException('Invalid custom link title - ' . $title);
}
$this->open_moodleapp_custom_url($pageurl, '', $username);
}
/**
* Closes a popup by clicking on the 'backdrop' behind it.
*
* @When I close the popup in the app
* @throws DriverException If there isn't a popup to close
*/
public function i_close_the_popup_in_the_app() {
$this->spin(function() {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js('closePopup()');
if ($result !== 'OK') {
throw new DriverException('Error closing popup - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
2022-03-10 11:07:15 +00:00
/**
* Override app config.
*
* @Given /^the app has the following config:$/
* @param TableNode $data Table data
2022-03-10 11:07:15 +00:00
*/
public function the_app_has_the_following_config(TableNode $data) {
foreach ($data->getRows() as $configrow) {
2024-04-15 08:50:59 +00:00
$name = $configrow[0];
$value = $this->replace_wwwroot($configrow[1]);
$this->appconfig[$name] = json_decode($value);
2022-03-10 11:07:15 +00:00
}
}
/**
* Check whether the moodle site is compatible with the current feature file
* and skip it otherwise. This will be checked looking at tags such as @lms_uptoXXX
*
* @Given the Moodle site is compatible with this feature
*/
public function the_moodle_site_is_compatible_with_this_feature() {
$this->check_tags();
}
/**
* Clicks on / touches something that is visible in the app.
*
* Note it is difficult to use the standard 'click on' or 'press' steps because those do not
* distinguish visible elements and the app always has many non-visible elements in the DOM.
*
* @When /^I press (".+") in the app$/
* @param string $locator Element locator
* @throws DriverException If the press doesn't work
*/
public function i_press_in_the_app(string $locator) {
$locator = $this->parse_element_locator($locator);
$this->spin(function() use ($locator) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("press($locator)");
if ($result !== 'OK') {
throw new DriverException('Error pressing item - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Performs a pull to refresh gesture.
*
* @When /^I pull to refresh (?:until I find (".+") )?in the app$/
* @throws DriverException If the gesture is not available
*/
public function i_pull_to_refresh_in_the_app(?string $locator = null) {
$timeout = 0;
$startTime = time();
if (!is_null($locator)) {
$timeout = 60;
$locator = $this->parse_element_locator($locator);
}
do {
$this->spin(function() {
$result = $this->runtime_js('pullToRefresh()');
if ($result !== 'OK') {
throw new DriverException('Error pulling to refresh - ' . $result);
}
return true;
});
$this->wait_animations_done();
if (is_null($locator)) {
return;
}
$result = $this->runtime_js("find($locator)");
if ($result === 'OK') {
return;
}
} while ($timeout > (time() - $startTime));
2022-12-12 09:51:27 +00:00
throw new DriverException('Error finding element after PTR - ' . $result);
}
/**
* Checks if elements can be pressed in the app.
*
* @Then /^I should( not)? be able to press (".+") in the app$/
* @param bool $not Whether to assert that the element cannot be pressed
* @param string $locator Element locator
*/
public function i_should_be_able_to_press_in_the_app(bool $not, string $locator) {
$locator = $this->parse_element_locator($locator);
$this->spin(function() use ($not, $locator) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("find($locator, { onlyClickable: true })");
if ($not && $result === 'OK') {
throw new DriverException('Error, found a clickable element that should not be found');
}
if (!$not && $result !== 'OK') {
throw new DriverException('Error finding clickable element - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Select an item from a list of options, such as a radio button.
*
* It may be necessary to use this step instead of "I press..." because radio buttons in Ionic are initialized
* with JavaScript, and clicks may not work until they are initialized properly which may cause flaky tests due
* to race conditions.
*
* @Then /^I (unselect|select) (".+") in the app$/
* @param string $selectedtext Text inidicating if the element should be selected or unselected
* @param string $locator Element locator
* @throws DriverException If the press doesn't work
*/
public function i_select_in_the_app(string $selectedtext, string $locator) {
2021-05-31 14:39:45 +00:00
$selected = $selectedtext === 'select' ? 'YES' : 'NO';
$locator = $this->parse_element_locator($locator);
2021-05-31 14:39:45 +00:00
$this->spin(function() use ($selectedtext, $selected, $locator) {
2021-05-31 14:39:45 +00:00
// Don't do anything if the item is already in the expected state.
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("isSelected($locator)");
2021-05-31 14:39:45 +00:00
if ($result === $selected) {
return true;
}
// Press element.
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("press($locator)");
2021-05-31 14:39:45 +00:00
if ($result !== 'OK') {
throw new DriverException('Error pressing element - ' . $result);
2021-05-31 14:39:45 +00:00
}
// Check that it worked as expected.
$this->wait_for_pending_js();
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("isSelected($locator)");
2021-05-31 14:39:45 +00:00
switch ($result) {
case 'YES':
case 'NO':
if ($result !== $selected) {
throw new ExpectationException("Item wasn't $selectedtext after pressing it", $this->getSession()->getDriver());
}
return true;
default:
throw new DriverException('Error finding item - ' . $result);
}
});
$this->wait_for_pending_js();
}
/**
* Sets a field to the given text value in the app.
*
* @Given /^I set the field "((?:[^"]|\\")+)" to "((?:[^"]|\\")*)" in the app$/
2024-01-11 14:25:49 +00:00
* @param string $field Text identifying the field.
* @param string $value Value to set. In select fields, this can be either the value or text included in the select option.
* @throws DriverException If the field set doesn't work.
*/
public function i_set_the_field_in_the_app(string $field, string $value) {
$field = addslashes_js($field);
$value = addslashes_js($value);
$this->spin(function() use ($field, $value) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js("setField('$field', '$value')");
if ($result !== 'OK') {
throw new DriverException('Error setting field "' . $field . '" - ' . $result);
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Fills a form with field/value data.
*
* @Given /^I set the following fields to these values in the app:$/
* @param TableNode $data
*/
public function i_set_the_following_fields_to_these_values_in_the_app(TableNode $data) {
$datahash = $data->getRowsHash();
// The action depends on the field type.
foreach ($datahash as $locator => $value) {
$this->i_set_the_field_in_the_app($locator, $value);
}
}
2023-03-30 10:39:00 +00:00
/**
* Uploads a file to a file input, the file path should be relative to a fixtures folder next to the feature file.
* The ìnput locator can match a container with a file input inside, it doesn't have to be the input itself.
*
* @Given /^I upload "((?:[^"]|\\")+)" to (".+") in the app$/
* @param string $filename
* @param string $inputlocator
*/
public function i_upload_a_file_in_the_app(string $filename, string $inputlocator) {
$filepath = str_replace('/', DIRECTORY_SEPARATOR, "{$this->featurepath}/fixtures/$filename");
$inputlocator = $this->parse_element_locator($inputlocator);
$id = $this->spin(function() use ($inputlocator) {
$result = $this->runtime_js("getFileInputId($inputlocator)");
if (str_starts_with($result, 'ERROR')) {
throw new DriverException('Error finding input - ' . $result);
}
return $result;
});
$this->wait_for_pending_js();
$fileinput = $this ->getSession()->getPage()->findById($id);
$fileinput->attachFile($filepath);
}
/**
* Checks a field matches a certain value in the app.
*
* Currently this only works for input fields which must be identified using a partial or
* exact match on the placeholder text.
*
* @Given /^the field "((?:[^"]|\\")+)" matches value "((?:[^"]|\\")*)" in the app$/
* @param string $field Text identifying field
* @param string $value Value for field
* @throws DriverException If the field isn't found
* @throws ExpectationException If the field value is different to the expected value
*/
public function the_field_matches_value_in_the_app(string $field, string $value) {
$field = addslashes_js($field);
$value = addslashes_js($value);
$this->spin(function() use ($field, $value) {
$result = $this->runtime_js("fieldMatches('$field', '$value')");
if ($result !== 'OK') {
if (str_contains($result, 'No element matches')) {
throw new DriverException('Error field matches value - ' . $result);
} else {
throw new ExpectationException(
'Error field matches value - ' . $result,
$this->getSession()->getDriver()
);
}
}
return true;
});
$this->wait_for_pending_js();
}
/**
* Checks that the current header stripe in the app contains the expected text.
*
* This can be used to see if the app went to the expected page.
*
* @Then /^the header should be "((?:[^"]|\\")+)" in the app$/
* @param string $text Expected header text
* @throws DriverException If the header can't be retrieved
* @throws ExpectationException If the header text is different to the expected value
*/
public function the_header_should_be_in_the_app(string $text) {
2021-04-27 15:43:47 +00:00
$this->spin(function() use ($text) {
2022-06-30 11:48:43 +00:00
$result = $this->runtime_js('getHeader()');
2021-04-27 15:43:47 +00:00
if (substr($result, 0, 3) !== 'OK:') {
throw new DriverException('Error getting header - ' . $result);
}
2021-04-27 15:43:47 +00:00
$header = substr($result, 3);
if (trim($header) !== trim($text)) {
throw new ExpectationException(
"The header text was not as expected: '$header'",
$this->getSession()->getDriver()
);
}
return true;
});
}
/**
* Check that the app opened a new browser tab.
*
* @Then /^the app should( not)? have opened a browser tab(?: with url "(?P<pattern>[^"]+)")?$/
* @param bool $not Whether to check if the app did not open a new browser tab
* @param string $urlpattern Url pattern
*/
public function the_app_should_have_opened_a_browser_tab(bool $not = false, ?string $urlpattern = null) {
$this->spin(function() use ($not, $urlpattern) {
$windowNames = $this->getSession()->getWindowNames();
$openedbrowsertab = count($windowNames) === 2;
if ((!$not && !$openedbrowsertab) || ($not && $openedbrowsertab && is_null($urlpattern))) {
throw new ExpectationException(
$not
? 'Did not expect the app to have opened a browser tab'
: 'Expected the app to have opened a browser tab',
$this->getSession()->getDriver()
);
}
if (!is_null($urlpattern)) {
$this->getSession()->switchToWindow($windowNames[1]);
$windowurl = $this->getSession()->getCurrentUrl();
$windowhaspattern = (bool)preg_match("/$urlpattern/", $windowurl);
$this->getSession()->switchToWindow($windowNames[0]);
if ($not === $windowhaspattern) {
throw new ExpectationException(
$not
? "Did not expect the app to have opened a browser tab with pattern '$urlpattern'"
: "Browser tab url does not match pattern '$urlpattern', it is '$windowurl'",
$this->getSession()->getDriver()
);
}
}
return true;
});
}
2024-02-07 11:55:12 +00:00
/**
* Check that the app opened a url.
*
* @Then /^the app should( not)? have opened url "([^"]+)"(?: with contents "([^"]+)")?(?: (once|\d+ times))?$/
* @param bool $not Whether to check if the app did not open the url
* @param string $urlpattern Url pattern
* @param string $contents Url contents
* @param string $times How many times the url should have been opened
*/
public function the_app_should_have_opened_url(bool $not, string $urlpattern, ?string $contents = null, ?string $times = null) {
if (is_null($times) || $times === 'once') {
$times = 1;
} else {
$times = intval(substr($times, 0, strlen($times) - 6));
}
$this->spin(function() use ($not, $urlpattern, $contents, $times) {
$result = $this->runtime_js("hasOpenedUrl('$urlpattern', '$contents', $times)");
// TODO process times
if ($not && $result === 'OK') {
throw new DriverException('Error, an url was opened that should not have');
}
if (!$not && $result !== 'OK') {
throw new DriverException('Error asserting that url was opened - ' . $result);
}
return true;
});
}
2024-04-10 09:12:53 +00:00
/**
* Check that an event has been logged.
*
* @Then /^the following events should( not)? have been logged for (".+"|the system) in the app:$/
*/
public function the_event_should_have_been_logged(bool $not, string $username, TableNode $data) {
$userid = $this->get_event_userid($username);
foreach ($data->getColumnsHash() as $event) {
$eventname = $event['name'];
$logs = $this->get_event_logs($userid, $event);
if (!$not && empty($logs)) {
throw new ExpectationException("Logs for event '$eventname' not found", $this->getSession()->getDriver());
}
if ($not && !empty($logs) && empty($event['other'])) {
throw new ExpectationException("Logs for event '$eventname' found, but shouldn't have", $this->getSession()->getDriver());
}
if (!empty($event['other'])) {
$log = $this->find_event_log_with_other($logs, json_decode($event['other'], true));
if (!$not && is_null($log)) {
throw new ExpectationException("Other data for event '$eventname' does not match", $this->getSession()->getDriver());
}
if ($not && !is_null($log)) {
throw new ExpectationException("Logs for event '$eventname' found, but shouldn't have", $this->getSession()->getDriver());
}
}
}
}
/**
* Switches to a newly-opened browser tab.
*
* This assumes the app opened a new tab.
*
* @Given I switch to the browser tab opened by the app
* @throws DriverException If there aren't exactly 2 tabs open
*/
public function i_switch_to_the_browser_tab_opened_by_the_app() {
$windowNames = $this->getSession()->getWindowNames();
if (count($windowNames) !== 2) {
throw new DriverException('Expected to see 2 tabs open, not ' . count($windowNames));
}
$this->getSession()->switchToWindow($windowNames[1]);
}
/**
* Switches to the app if the user is in a browser tab.
*
* @Given I switch back to the app
*/
public function i_switch_back_to_the_app() {
$windowNames = $this->getSession()->getWindowNames();
if (count($windowNames) > 1) {
$this->getSession()->switchToWindow($windowNames[0]);
}
}
/**
* Force cron tasks instead of waiting for the next scheduled execution.
*
* @When I run cron tasks in the app
*/
public function i_run_cron_tasks_in_the_app() {
2022-06-30 11:48:43 +00:00
$this->zone_js('cronDelegate.forceSyncExecution()');
$this->wait_for_pending_js();
}
/**
* Wait until loading has finished.
*
* @When I wait loading to finish in the app
*/
public function i_wait_loading_to_finish_in_the_app() {
2022-06-30 11:48:43 +00:00
$this->runtime_js('waitLoadingToFinish()');
$this->wait_for_pending_js();
}
/**
* Closes the current browser tab.
*
* This assumes it was opened by the app and you will now get back to the app.
*
* @Given I close the browser tab opened by the app
* @throws DriverException If there aren't exactly 2 tabs open
*/
public function i_close_the_browser_tab_opened_by_the_app() {
$names = $this->getSession()->getWindowNames();
if (count($names) !== 2) {
throw new DriverException('Expected to see 2 tabs open, not ' . count($names));
}
// Make sure the browser tab is selected.
if ($this->getSession()->getWindowName() !== $names[1]) {
$this->getSession()->switchToWindow($names[1]);
}
2022-07-07 08:39:36 +00:00
$this->evaluate_script('window.close()');
$this->getSession()->switchToWindow($names[0]);
}
/**
* Switch navigator online mode.
*
* @Given /^I switch offline mode to "(true|false)"$/
* @param string $offline New value for navigator online mode
* @throws DriverException If the navigator.online mode is not available
* @deprecated since 4.1 use i_switch_network_connection instead.
*/
public function i_switch_offline_mode(string $offline) {
$this->i_switch_network_connection($offline == 'true' ? 'offline' : 'wifi');
}
/**
* Switch network connection.
*
* @When /^I switch network connection to (wifi|cellular|offline)$/
* @param string $more New network mode.
* @throws DriverException If the navigator.online mode is not available
*/
public function i_switch_network_connection(string $mode) {
switch ($mode) {
case 'wifi':
$this->runtime_js("network.setForceConnectionMode('$mode');");
break;
case 'cellular':
$this->runtime_js("network.setForceConnectionMode('$mode');");
break;
case 'offline':
$this->runtime_js("network.setForceConnectionMode('none');");
break;
default:
break;
}
}
/**
* Open a browser tab with a certain URL.
*
* @Then /^I open a browser tab with url "(?P<pattern>[^"]+)"$/
* @param string $url URL
*/
public function i_open_a_browser_tab_with_url(string $url) {
$this->execute_script("window.open('$url', '_system');");
$windowNames = $this->getSession()->getWindowNames();
$this->getSession()->switchToWindow($windowNames[1]);
}
/**
* Send pending notifications.
*
* @Then /^I flush pending notifications in the app$/
*/
public function i_flush_notifications() {
$this->runtime_js("flushNotifications()");
}
/**
* Check if a notification has been triggered and is present.
*
* @Then /^a notification with title (".+") should( not)? be present in the app$/
* @param string $title Notification title
* @param bool $not Whether assert that the notification was not found
*/
public function notification_present_in_the_app(string $title, bool $not = false) {
$this->spin(function() use ($not, $title) {
$result = $this->runtime_js("notificationIsPresentWithText($title)");
if ($not && $result === 'YES') {
throw new ExpectationException("Notification is present", $this->getSession()->getDriver());
}
if (!$not && $result === 'NO') {
throw new ExpectationException("Notification is not present", $this->getSession()->getDriver());
}
if ($result !== 'YES' && $result !== 'NO') {
throw new DriverException('Error checking notification - ' . $result);
}
return true;
});
}
/**
* Check if a notification has been scheduled.
*
* @Then /^a notification with title (".+") should( not)? be scheduled(?: (\d+) minutes before the "(.+)" assignment due date)? in the app$/
* @param string $title Notification title
* @param bool $not Whether assert that the notification was not scheduled
* @param int $minutes Minutes before the assignment at which the notification was scheduled
* @param string $assignment Assignment for which the notification was scheduled
*/
public function notification_scheduled_in_the_app(string $title, bool $not = false, ?int $minutes = null, ?string $assignment = null) {
if (!is_null($minutes)) {
global $DB;
$assign = $DB->get_record('assign', ['name' => $assignment]);
if (!$assign) {
throw new ExpectationException("Couldn't find '$assignment' assignment", $this->getSession()->getDriver());
}
$date = ($assign->duedate - $minutes * 60) * 1000;
} else {
$date = 'undefined';
}
$this->spin(function() use ($not, $title, $date) {
$result = $this->runtime_js("notificationIsScheduledWithText($title, $date)");
if ($not && $result === 'YES') {
throw new ExpectationException("Notification is scheduled", $this->getSession()->getDriver());
}
if (!$not && $result === 'NO') {
throw new ExpectationException("Notification is not scheduled", $this->getSession()->getDriver());
}
if ($result !== 'YES' && $result !== 'NO') {
throw new DriverException('Error checking scheduled notification - ' . $result);
}
return true;
});
}
/**
* Close a notification present in the app
*
* @Then /^I close a notification with title (".+") in the app$/
* @param string $title Notification title
*/
public function close_notification_app(string $title) {
$result = $this->runtime_js("closeNotification($title)");
if ($result !== 'OK') {
throw new DriverException('Error closing notification - ' . $result);
}
return true;
}
/**
* View a specific month in the calendar in the app.
*
* @When /^I open the calendar for "(?P<month>\d+)" "(?P<year>\d+)" in the app$/
* @param int $month the month selected as a number
* @param int $year the four digit year
*/
public function i_open_the_calendar_for($month, $year) {
$options = json_encode([
'params' => [
'month' => $month,
'year' => $year,
],
]);
$this->zone_js("navigator.navigateToSitePath('/calendar/index', $options)");
}
/**
* Change the viewport size in the browser running the Moodle App.
*
* @Given /^I change viewport size to "(?P<width>\d+)x(?P<height>\d+)" in the app$/
* @param int $width Width
* @param int $height Height
*/
public function i_change_viewport_size_in_the_app(int $width, int $height) {
$this->resize_app_window($width, $height);
}
/**
* Wait until Toast disappears.
*
* @When I wait toast to dismiss in the app
*/
public function i_wait_toast_to_dismiss_in_the_app() {
$this->runtime_js('waitToastDismiss()');
}
}