3 declare(strict_types
=1);
5 namespace PhpMyAdmin\Tests\Selenium
;
7 use PHPUnit\Framework\Attributes\CoversNothing
;
8 use PHPUnit\Framework\Attributes\Large
;
12 class TrackingTest
extends TestBase
15 * Setup the browser environment to run the selenium test case
17 protected function setUp(): void
22 'USE `' . $this->databaseName
. '`;'
23 . 'CREATE TABLE `test_table` ('
24 . ' `id` int(11) NOT NULL AUTO_INCREMENT,'
25 . ' `val` int(11) NOT NULL,'
26 . ' PRIMARY KEY (`id`)'
28 . 'CREATE TABLE `test_table_2` ('
29 . ' `id` int(11) NOT NULL AUTO_INCREMENT,'
30 . ' `val` int(11) NOT NULL,'
31 . ' PRIMARY KEY (`id`)'
33 . 'INSERT INTO `test_table` (val) VALUES (2), (3);',
37 $this->skipIfNotPMADB();
39 $this->navigateDatabase($this->databaseName
);
42 $this->waitForElement('partialLinkText', 'Tracking')->click();
45 $this->waitForElement('partialLinkText', 'Track table');
46 $this->byXPath("(//a[contains(., 'Track table')])[1]")->click();
49 $this->waitForElement('name', 'delete')->click();
50 $this->byCssSelector("input[value='Create version']")->click();
51 $this->waitForElement('id', 'versions');
55 * Tests basic tracking functionality
57 public function testTrackingData(): void
59 $this->executeSqlAndReturnToTableTracking();
61 $this->byPartialLinkText('Tracking report')->click();
62 $this->waitForElement('xpath', "//h3[contains(., 'Tracking report')]");
64 self
::assertStringContainsString(
65 'DROP TABLE IF EXISTS `test_table`',
66 $this->getCellByTableId('ddl_versions', 1, 4),
69 self
::assertStringContainsString(
70 'CREATE TABLE `test_table` (',
71 $this->getCellByTableId('ddl_versions', 2, 4),
74 self
::assertStringContainsString(
75 'UPDATE test_table SET val = val + 1',
76 $this->getCellByTableId('dml_versions', 1, 4),
79 self
::assertStringNotContainsString(
80 'DELETE FROM test_table WHERE val = 3',
81 $this->byId('dml_versions')->getText(),
85 $this->selectByLabel($this->byName('log_type'), 'Structure only');
87 $this->byCssSelector("input[value='Go']")->click();
92 $this->isElementPresent('id', 'dml_versions'),
95 self
::assertStringContainsString(
96 'DROP TABLE IF EXISTS `test_table`',
97 $this->getCellByTableId('ddl_versions', 1, 4),
100 self
::assertStringContainsString(
101 'CREATE TABLE `test_table` (',
102 $this->getCellByTableId('ddl_versions', 2, 4),
106 $this->selectByLabel($this->waitForElement('name', 'log_type'), 'Data only');
108 $this->byCssSelector("input[value='Go']")->click();
113 $this->isElementPresent('id', 'ddl_versions'),
116 self
::assertStringContainsString(
117 'UPDATE test_table SET val = val + 1',
118 $this->getCellByTableId('dml_versions', 1, 4),
121 self
::assertStringNotContainsString(
122 'DELETE FROM test_table WHERE val = 3',
123 $this->byId('dml_versions')->getText(),
128 * Tests deactivation of tracking
130 public function testDeactivateTracking(): void
132 $this->byCssSelector("input[value='Deactivate now']")->click();
133 $this->waitForElement('cssSelector', "input[value='Activate now']");
134 $this->executeSqlAndReturnToTableTracking();
136 $this->isElementPresent('id', 'dml_versions'),
141 * Tests dropping a tracking
143 public function testDropTracking(): void
145 $this->navigateDatabase($this->databaseName
, true);
148 $this->byPartialLinkText('Tracking')->click();
151 $this->waitForElement('id', 'versions');
153 $ele = $this->waitForElement('cssSelector', 'table#versions tbody tr:nth-child(1) td:nth-child(7) a');
157 $this->waitForElement('id', 'functionConfirmOkButton')->click();
160 $success = $this->waitForElement('cssSelector', '.alert-success');
161 self
::assertStringContainsString('Tracking data deleted successfully.', $success->getText());
163 // Can not use getCellByTableId,
164 // since this is under 'th' and not 'td'
165 self
::assertStringContainsString(
167 $this->waitForElement(
169 'table#noversions tbody tr:nth-child(1) th:nth-child(2)',
172 self
::assertStringContainsString(
174 $this->waitForElement(
176 'table#noversions tbody tr:nth-child(2) th:nth-child(2)',
182 * Tests structure snapshot of a tracking
184 public function testStructureSnapshot(): void
186 $this->byPartialLinkText('Structure snapshot')->click();
187 $this->waitForElement('id', 'tablestructure');
189 self
::assertStringContainsString(
191 $this->getCellByTableId('tablestructure', 1, 2),
194 self
::assertStringContainsString(
196 $this->getCellByTableId('tablestructure', 2, 2),
199 self
::assertStringContainsString(
201 $this->getCellByTableId('tablestructure_indexes', 1, 1),
204 self
::assertStringContainsString(
206 $this->getCellByTableId('tablestructure_indexes', 1, 5),
211 * Goes to SQL tab, executes queries, returns to tracking page
213 private function executeSqlAndReturnToTableTracking(): void
215 $this->byPartialLinkText('SQL')->click();
218 $this->waitForElement('id', 'queryfieldscontainer');
219 $this->typeInTextArea(';UPDATE test_table SET val = val + 1; DELETE FROM test_table WHERE val = 3');
220 $this->scrollToBottom();
221 $this->byCssSelector("input[value='Go']")->click();
223 $this->waitForElement('className', 'alert-success');
226 $this->byPartialLinkText('Tracking')->click();
228 $this->waitForElement('id', 'versions');