dummyDbi = $this->createDbiDummy(); $this->dbi = $this->createDatabaseInterface($this->dummyDbi); $GLOBALS['dbi'] = $this->dbi; $GLOBALS['server'] = 1; $GLOBALS['PMA_PHP_SELF'] = 'index.php'; $GLOBALS['cfg']['ServerDefault'] = 1; $GLOBALS['text_dir'] = 'ltr'; $GLOBALS['db'] = 'db'; $GLOBALS['table'] = 'table'; $GLOBALS['cfg']['LimitChars'] = 50; $GLOBALS['cfg']['LongtextDoubleTextarea'] = false; $GLOBALS['cfg']['ShowFieldTypesInDataEditView'] = true; $GLOBALS['cfg']['ShowFunctionFields'] = true; $GLOBALS['cfg']['ProtectBinary'] = 'blob'; $GLOBALS['cfg']['MaxSizeForInputField'] = 10; $GLOBALS['cfg']['MinSizeForInputField'] = 2; $GLOBALS['cfg']['TextareaRows'] = 5; $GLOBALS['cfg']['TextareaCols'] = 4; $GLOBALS['cfg']['CharTextareaRows'] = 5; $GLOBALS['cfg']['CharTextareaCols'] = 6; $GLOBALS['cfg']['AllowThirdPartyFraming'] = false; $GLOBALS['cfg']['SendErrorReports'] = 'ask'; $GLOBALS['cfg']['DefaultTabDatabase'] = 'structure'; $GLOBALS['cfg']['ShowDatabasesNavigationAsTree'] = true; $GLOBALS['cfg']['DefaultTabTable'] = 'browse'; $GLOBALS['cfg']['NavigationTreeDefaultTabTable'] = 'structure'; $GLOBALS['cfg']['NavigationTreeDefaultTabTable2'] = ''; $GLOBALS['cfg']['Confirm'] = true; $GLOBALS['cfg']['LoginCookieValidity'] = 1440; $GLOBALS['cfg']['enable_drag_drop_import'] = true; $this->insertEdit = new InsertEdit( $this->dbi, new Relation($this->dbi), new Transformations(), new FileListing(), new Template() ); } /** * Teardown all objects */ protected function tearDown(): void { parent::tearDown(); $response = new ReflectionProperty(ResponseRenderer::class, 'instance'); $response->setAccessible(true); $response->setValue(null); $response->setAccessible(false); } /** * Test for getFormParametersForInsertForm */ public function testGetFormParametersForInsertForm(): void { $where_clause = [ 'foo' => 'bar ', '1' => ' test', ]; $_POST['clause_is_unique'] = false; $_POST['sql_query'] = 'SELECT a'; $GLOBALS['goto'] = 'index.php'; $result = $this->insertEdit->getFormParametersForInsertForm( 'dbname', 'tablename', [], $where_clause, 'localhost' ); $this->assertEquals( [ 'db' => 'dbname', 'table' => 'tablename', 'goto' => 'index.php', 'err_url' => 'localhost', 'sql_query' => 'SELECT a', 'where_clause[foo]' => 'bar', 'where_clause[1]' => 'test', 'clause_is_unique' => false, ], $result ); } /** * Test for getWhereClauseArray */ public function testGetWhereClauseArray(): void { $this->assertEquals( [], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getWhereClauseArray', [null] ) ); $this->assertEquals( [ 1, 2, 3, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getWhereClauseArray', [[1, 2, 3]] ) ); $this->assertEquals( ['clause'], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getWhereClauseArray', ['clause'] ) ); } /** * Test for analyzeWhereClauses */ public function testAnalyzeWhereClause(): void { $clauses = [ 'a=1', 'b="fo\o"', ]; $resultStub1 = $this->createMock(DummyResult::class); $resultStub2 = $this->createMock(DummyResult::class); $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->exactly(2)) ->method('query') ->willReturnOnConsecutiveCalls($resultStub1, $resultStub2); $resultStub1->expects($this->once()) ->method('fetchAssoc') ->will($this->returnValue(['assoc1'])); $resultStub2->expects($this->once()) ->method('fetchAssoc') ->will($this->returnValue(['assoc2'])); $dbi->expects($this->exactly(2)) ->method('getFieldsMeta') ->willReturnOnConsecutiveCalls( [], [] ); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'analyzeWhereClauses', [ $clauses, 'table', 'db', ] ); $this->assertSame( [ [ 'a=1', 'b="fo\\\\o"', ], [ $resultStub1, $resultStub2, ], [ ['assoc1'], ['assoc2'], ], false, ], $result ); } /** * Test for showEmptyResultMessageOrSetUniqueCondition */ public function testShowEmptyResultMessageOrSetUniqueCondition(): void { $temp = new stdClass(); $temp->orgname = 'orgname'; $temp->table = 'table'; $meta_arr = [new FieldMetadata(MYSQLI_TYPE_DECIMAL, MYSQLI_PRI_KEY_FLAG, $temp)]; $resultStub = $this->createMock(DummyResult::class); $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('getFieldsMeta') ->with($resultStub) ->will($this->returnValue($meta_arr)); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'showEmptyResultMessageOrSetUniqueCondition', [ ['1' => ['1' => 1]], 1, [], 'SELECT', ['1' => $resultStub], ] ); $this->assertTrue($result); // case 2 $GLOBALS['cfg']['ShowSQL'] = false; $responseMock = $this->getMockBuilder(ResponseRenderer::class) ->disableOriginalConstructor() ->onlyMethods(['addHtml']) ->getMock(); $restoreInstance = ResponseRenderer::getInstance(); $response = new ReflectionProperty(ResponseRenderer::class, 'instance'); $response->setAccessible(true); $response->setValue($responseMock); $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'showEmptyResultMessageOrSetUniqueCondition', [ [false], 0, ['1'], 'SELECT', ['1' => 'result1'], ] ); $response->setValue($restoreInstance); $this->assertFalse($result); } public function dataProviderConfigValueInsertRows(): array { return [ [ 2, [ false, false, ], ], [ '2', [ false, false, ], ], [ 3, [ false, false, false, ], ], [ '3', [ false, false, false, ], ], ]; } /** * Test for loadFirstRow * * @param string|int $configValue * * @dataProvider dataProviderConfigValueInsertRows */ public function testLoadFirstRow($configValue, array $rowsValue): void { $GLOBALS['cfg']['InsertRows'] = $configValue; $resultStub = $this->createMock(DummyResult::class); $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('query') ->with( 'SELECT * FROM `db`.`table` LIMIT 1;' ) ->will($this->returnValue($resultStub)); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'loadFirstRow', ['table', 'db'] ); $this->assertEquals( [ $resultStub, $rowsValue, ], $result ); } /** * Test for urlParamsInEditMode */ public function testUrlParamsInEditMode(): void { $where_clause_array = [ 'foo=1', 'bar=2', ]; $_POST['sql_query'] = 'SELECT 1'; $result = $this->insertEdit->urlParamsInEditMode([1], $where_clause_array); $this->assertEquals( [ '0' => 1, 'where_clause' => 'bar=2', 'sql_query' => 'SELECT 1', ], $result ); } /** * Test for showTypeOrFunction */ public function testShowTypeOrFunction(): void { $GLOBALS['cfg']['ShowFieldTypesInDataEditView'] = true; $GLOBALS['cfg']['ServerDefault'] = 1; $url_params = ['ShowFunctionFields' => 2]; $result = $this->insertEdit->showTypeOrFunction('function', $url_params, false); $this->assertStringContainsString('index.php?route=/table/change', $result); $this->assertStringContainsString( 'ShowFunctionFields=1&ShowFieldTypesInDataEditView=1&goto=index.php%3Froute%3D%2Fsql', $result ); $this->assertStringContainsString('Function', $result); // case 2 $result = $this->insertEdit->showTypeOrFunction('function', $url_params, true); $this->assertStringContainsString('index.php?route=/table/change', $result); $this->assertStringContainsString( 'ShowFunctionFields=0&ShowFieldTypesInDataEditView=1&goto=index.php%3Froute%3D%2Fsql', $result ); $this->assertStringContainsString('Function', $result); // case 3 $result = $this->insertEdit->showTypeOrFunction('type', $url_params, false); $this->assertStringContainsString('index.php?route=/table/change', $result); $this->assertStringContainsString( 'ShowFunctionFields=1&ShowFieldTypesInDataEditView=1&goto=index.php%3Froute%3D%2Fsql', $result ); $this->assertStringContainsString('Type', $result); // case 4 $result = $this->insertEdit->showTypeOrFunction('type', $url_params, true); $this->assertStringContainsString('index.php?route=/table/change', $result); $this->assertStringContainsString( 'ShowFunctionFields=1&ShowFieldTypesInDataEditView=0&goto=index.php%3Froute%3D%2Fsql', $result ); $this->assertStringContainsString('Type', $result); } /** * Test for analyzeTableColumnsArray */ public function testAnalyzeTableColumnsArray(): void { $column = [ 'Field' => '1<2', 'Field_md5' => 'pswd', 'Type' => 'float(10, 1)', ]; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'analyzeTableColumnsArray', [ $column, [], false, ] ); $this->assertEquals($result['Field_md5'], '4342210df36bf2ff2c4e2a997a6d4089'); $this->assertEquals($result['True_Type'], 'float'); $this->assertEquals($result['len'], 100); $this->assertEquals($result['Field_title'], '1<2'); $this->assertEquals($result['is_binary'], false); $this->assertEquals($result['is_blob'], false); $this->assertEquals($result['is_char'], false); $this->assertEquals($result['pma_type'], 'float(10, 1)'); $this->assertEquals($result['wrap'], ' text-nowrap'); $this->assertEquals($result['Field'], '1<2'); } /** * Test for getColumnTitle */ public function testGetColumnTitle(): void { $column = []; $column['Field'] = 'f1<'; $this->assertEquals( $this->callFunction( $this->insertEdit, InsertEdit::class, 'getColumnTitle', [ $column, [], ] ), 'f1<' ); $comments = []; $comments['f1<'] = 'comment>'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getColumnTitle', [ $column, $comments, ] ); $this->assertStringContainsString('title="comment>"', $result); $this->assertStringContainsString('f1<', $result); } /** * Test for isColumn */ public function testIsColumn(): void { $column = []; $types = [ 'binary', 'varbinary', ]; $column['Type'] = 'binaryfoo'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'Binaryfoo'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'varbinaryfoo'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'barbinaryfoo'; $this->assertFalse($this->insertEdit->isColumn($column, $types)); $types = [ 'char', 'varchar', ]; $column['Type'] = 'char(10)'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'VarChar(20)'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'foochar'; $this->assertFalse($this->insertEdit->isColumn($column, $types)); $types = [ 'blob', 'tinyblob', 'mediumblob', 'longblob', ]; $column['Type'] = 'blob'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'bloB'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'mediumBloB'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'tinyblobabc'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'longblob'; $this->assertTrue($this->insertEdit->isColumn($column, $types)); $column['Type'] = 'foolongblobbar'; $this->assertFalse($this->insertEdit->isColumn($column, $types)); } /** * Test for getEnumSetAndTimestampColumns */ public function testGetEnumAndTimestampColumns(): void { $column = []; $column['True_Type'] = 'set'; $this->assertEquals( [ 'set', '', false, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, false, ] ) ); $column['True_Type'] = 'enum'; $this->assertEquals( [ 'enum', '', false, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, false, ] ) ); $column['True_Type'] = 'timestamp'; $column['Type'] = 'date'; $this->assertEquals( [ 'date', ' text-nowrap', true, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, false, ] ) ); $column['True_Type'] = 'timestamp'; $column['Type'] = 'date'; $this->assertEquals( [ 'date', ' text-nowrap', false, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, true, ] ) ); $column['True_Type'] = 'SET'; $column['Type'] = 'num'; $this->assertEquals( [ 'num', ' text-nowrap', false, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, false, ] ) ); $column['True_Type'] = ''; $column['Type'] = 'num'; $this->assertEquals( [ 'num', ' text-nowrap', false, ], $this->callFunction( $this->insertEdit, InsertEdit::class, 'getEnumSetAndTimestampColumns', [ $column, false, ] ) ); } /** * Test for getNullifyCodeForNullColumn */ public function testGetNullifyCodeForNullColumn(): void { $column = $foreignData = []; $foreigners = [ 'foreign_keys_data' => [], ]; $column['Field'] = 'f'; $column['True_Type'] = 'enum'; $column['Type'] = 'ababababababababababa'; $this->assertEquals( '1', $this->callFunction( $this->insertEdit, InsertEdit::class, 'getNullifyCodeForNullColumn', [ $column, $foreigners, [], ] ) ); $column['True_Type'] = 'enum'; $column['Type'] = 'abababababababababab'; $this->assertEquals( '2', $this->callFunction( $this->insertEdit, InsertEdit::class, 'getNullifyCodeForNullColumn', [ $column, $foreigners, [], ] ) ); $column['True_Type'] = 'set'; $this->assertEquals( '3', $this->callFunction( $this->insertEdit, InsertEdit::class, 'getNullifyCodeForNullColumn', [ $column, $foreigners, [], ] ) ); $column['True_Type'] = ''; $foreigners['f'] = true; $foreignData['foreign_link'] = ''; $this->assertEquals( '4', $this->callFunction( $this->insertEdit, InsertEdit::class, 'getNullifyCodeForNullColumn', [ $column, $foreigners, $foreignData, ] ) ); } /** * Test for getTextarea */ public function testGetTextarea(): void { $GLOBALS['cfg']['TextareaRows'] = 20; $GLOBALS['cfg']['TextareaCols'] = 10; $GLOBALS['cfg']['CharTextareaRows'] = 7; $GLOBALS['cfg']['CharTextareaCols'] = 1; $GLOBALS['cfg']['LimitChars'] = 20; $column = []; $column['is_char'] = true; $column['Type'] = 'char(10)'; $column['True_Type'] = 'char'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getTextarea', [ $column, 'a', 'b', '', 2, 0, 1, 'abc/', 'foobar', 'CHAR', false, ] ); $this->assertStringContainsString( '', $result ); // case 2: (else) $column['is_char'] = false; $column['Extra'] = 'auto_increment'; $column['pma_type'] = 'timestamp'; $column['True_Type'] = 'timestamp'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getValueColumnForOtherDatatypes', [ $column, 'defchar', 'a', 'b', 'c', 22, '<', 12, 1, '/', '<', "foo\nbar", $extracted_columnspec, false, ] ); $this->assertEquals( "a\n" . '' . '', $result ); // case 3: (else -> datetime) $column['pma_type'] = 'datetime'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getValueColumnForOtherDatatypes', [ $column, 'defchar', 'a', 'b', 'c', 22, '<', 12, 1, '/', '<', "foo\nbar", $extracted_columnspec, false, ] ); $this->assertStringContainsString('', $result); // case 4: (else -> date) $column['pma_type'] = 'date'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getValueColumnForOtherDatatypes', [ $column, 'defchar', 'a', 'b', 'c', 22, '<', 12, 1, '/', '<', "foo\nbar", $extracted_columnspec, false, ] ); $this->assertStringContainsString('', $result); } /** * Test for getColumnSize */ public function testGetColumnSize(): void { $column = []; $column['is_char'] = true; $spec_in_brackets = '45'; $GLOBALS['cfg']['MinSizeForInputField'] = 30; $GLOBALS['cfg']['MaxSizeForInputField'] = 40; $this->assertEquals( 40, $this->callFunction( $this->insertEdit, InsertEdit::class, 'getColumnSize', [ $column, $spec_in_brackets, ] ) ); $this->assertEquals('textarea', $GLOBALS['cfg']['CharEditing']); // case 2 $column['is_char'] = false; $column['len'] = 20; $this->assertEquals( 30, $this->callFunction( $this->insertEdit, InsertEdit::class, 'getColumnSize', [ $column, $spec_in_brackets, ] ) ); } /** * Test for getContinueInsertionForm */ public function testGetContinueInsertionForm(): void { $where_clause_array = ['ainsertEdit->getContinueInsertionForm('tbl', 'db', $where_clause_array, 'localhost'); $this->assertStringContainsString( '
', $result ); $this->assertStringContainsString('', $result); $this->assertStringContainsString('', $result); $this->assertStringContainsString('', $result); $this->assertStringContainsString('', $result); $this->assertStringContainsString('', $result); $this->assertStringContainsString('', $result); } public function testIsWhereClauseNumeric(): void { $this->assertFalse(InsertEdit::isWhereClauseNumeric(null)); $this->assertFalse(InsertEdit::isWhereClauseNumeric('')); $this->assertFalse(InsertEdit::isWhereClauseNumeric([])); $this->assertTrue(InsertEdit::isWhereClauseNumeric('`actor`.`actor_id` = 1')); $this->assertTrue(InsertEdit::isWhereClauseNumeric(['`actor`.`actor_id` = 1'])); $this->assertFalse(InsertEdit::isWhereClauseNumeric('`actor`.`first_name` = \'value\'')); $this->assertFalse(InsertEdit::isWhereClauseNumeric(['`actor`.`first_name` = \'value\''])); } /** * Test for getHeadAndFootOfInsertRowTable */ public function testGetHeadAndFootOfInsertRowTable(): void { $GLOBALS['cfg']['ShowFieldTypesInDataEditView'] = true; $GLOBALS['cfg']['ShowFunctionFields'] = true; $GLOBALS['cfg']['ServerDefault'] = 1; $url_params = ['ShowFunctionFields' => 2]; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getHeadAndFootOfInsertRowTable', [$url_params] ); $this->assertStringContainsString('index.php?route=/table/change', $result); $this->assertStringContainsString('ShowFunctionFields=1&ShowFieldTypesInDataEditView=0', $result); $this->assertStringContainsString('ShowFunctionFields=0&ShowFieldTypesInDataEditView=1', $result); } /** * Test for getSpecialCharsAndBackupFieldForExistingRow */ public function testGetSpecialCharsAndBackupFieldForExistingRow(): void { $column = $current_row = $extracted_columnspec = []; $column['Field'] = 'f'; $current_row['f'] = null; $_POST['default_action'] = 'insert'; $column['Key'] = 'PRI'; $column['Extra'] = 'fooauto_increment'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, [], [], 'a', false, ] ); $this->assertEquals( [ true, null, null, null, '', ], $result ); // Case 2 (bit) unset($_POST['default_action']); $current_row['f'] = '123'; $extracted_columnspec['spec_in_brackets'] = '20'; $column['True_Type'] = 'bit'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, $extracted_columnspec, [], 'a', false, ] ); $this->assertEquals( [ false, '', '00000000000001111011', null, '', ], $result ); $current_row['f'] = 'abcd'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, $extracted_columnspec, [], 'a', true, ] ); $this->assertEquals( [ false, '', 'abcd', null, '', ], $result ); // Case 3 (bit) $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $current_row['f'] = '123'; $extracted_columnspec['spec_in_brackets'] = '20'; $column['True_Type'] = 'int'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, $extracted_columnspec, ['int'], 'a', false, ] ); $this->assertEquals( [ false, '', "'',", null, '', ], $result ); // Case 4 (else) $column['is_binary'] = false; $column['is_blob'] = true; $GLOBALS['cfg']['ProtectBinary'] = false; $current_row['f'] = '11001'; $extracted_columnspec['spec_in_brackets'] = '20'; $column['True_Type'] = 'char'; $GLOBALS['cfg']['ShowFunctionFields'] = true; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, $extracted_columnspec, ['int'], 'a', false, ] ); $this->assertEquals( [ false, '3131303031', '3131303031', '3131303031', '', ], $result ); // Case 5 $current_row['f'] = "11001\x00"; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForExistingRow', [ $current_row, $column, $extracted_columnspec, ['int'], 'a', false, ] ); $this->assertEquals( [ false, '313130303100', '313130303100', '313130303100', '', ], $result ); } /** * Test for getSpecialCharsAndBackupFieldForInsertingMode */ public function testGetSpecialCharsAndBackupFieldForInsertingMode(): void { $column = []; $column['True_Type'] = 'bit'; $column['Default'] = 'b\'101\''; $column['is_binary'] = true; $GLOBALS['cfg']['ProtectBinary'] = false; $GLOBALS['cfg']['ShowFunctionFields'] = true; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForInsertingMode', [$column] ); $this->assertEquals( [ false, 'b\'101\'', '101', '', '101', ], $result ); // case 2 unset($column['Default']); $column['True_Type'] = 'char'; $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getSpecialCharsAndBackupFieldForInsertingMode', [$column] ); $this->assertEquals( [ true, '', '', '', '', ], $result ); } /** * Test for getParamsForUpdateOrInsert */ public function testGetParamsForUpdateOrInsert(): void { $_POST['where_clause'] = 'LIMIT 1'; $_POST['submit_type'] = 'showinsert'; $result = $this->insertEdit->getParamsForUpdateOrInsert(); $this->assertEquals( [ ['LIMIT 1'], true, true, false, ], $result ); // case 2 (else) unset($_POST['where_clause']); $_POST['fields']['multi_edit'] = [ 'a' => 'b', 'c' => 'd', ]; $result = $this->insertEdit->getParamsForUpdateOrInsert(); $this->assertEquals( [ [ 'a', 'c', ], false, true, false, ], $result ); } /** * Test for setSessionForEditNext */ public function testSetSessionForEditNext(): void { $temp = new stdClass(); $temp->orgname = 'orgname'; $temp->table = 'table'; $temp->orgtable = 'table'; $meta_arr = [new FieldMetadata(MYSQLI_TYPE_DECIMAL, MYSQLI_PRI_KEY_FLAG, $temp)]; $row = ['1' => 1]; $resultStub = $this->createMock(DummyResult::class); $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('query') ->with('SELECT * FROM `db`.`table` WHERE `a` > 2 LIMIT 1;') ->will($this->returnValue($resultStub)); $resultStub->expects($this->once()) ->method('fetchRow') ->will($this->returnValue($row)); $dbi->expects($this->once()) ->method('getFieldsMeta') ->with($resultStub) ->will($this->returnValue($meta_arr)); $GLOBALS['dbi'] = $dbi; $GLOBALS['db'] = 'db'; $GLOBALS['table'] = 'table'; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $this->insertEdit->setSessionForEditNext('`a` = 2'); $this->assertEquals('CONCAT(`table`.`orgname`) IS NULL', $_SESSION['edit_next']); } /** * Test for getGotoInclude */ public function testGetGotoInclude(): void { $GLOBALS['goto'] = '123.php'; $GLOBALS['table'] = ''; $this->assertEquals( '/database/sql', $this->insertEdit->getGotoInclude('index') ); $GLOBALS['table'] = 'tbl'; $this->assertEquals( '/table/sql', $this->insertEdit->getGotoInclude('index') ); $GLOBALS['goto'] = 'index.php?route=/database/sql'; $this->assertEquals( '/database/sql', $this->insertEdit->getGotoInclude('index') ); $this->assertEquals('', $GLOBALS['table']); $_POST['after_insert'] = 'new_insert'; $this->assertEquals( '/table/change', $this->insertEdit->getGotoInclude('index') ); } /** * Test for getErrorUrl */ public function testGetErrorUrl(): void { $GLOBALS['cfg']['ServerDefault'] = 1; $this->assertEquals( 'index.php?route=/table/change&lang=en', $this->insertEdit->getErrorUrl([]) ); $_POST['err_url'] = 'localhost'; $this->assertEquals( 'localhost', $this->insertEdit->getErrorUrl([]) ); } /** * Test for buildSqlQuery */ public function testBuildSqlQuery(): void { $GLOBALS['db'] = 'db'; $GLOBALS['table'] = 'table'; $query_fields = [ 'a', 'b', ]; $value_sets = [ 1, 2, ]; $this->assertEquals( ['INSERT IGNORE INTO `table` (a, b) VALUES (1), (2)'], $this->insertEdit->buildSqlQuery(true, $query_fields, $value_sets) ); $this->assertEquals( ['INSERT INTO `table` (a, b) VALUES (1), (2)'], $this->insertEdit->buildSqlQuery(false, $query_fields, $value_sets) ); } /** * Test for executeSqlQuery */ public function testExecuteSqlQuery(): void { $query = [ 'SELECT * FROM `test_db`.`test_table`;', 'SELECT * FROM `test_db`.`test_table_yaml`;', ]; $GLOBALS['sql_query'] = 'SELECT * FROM `test_db`.`test_table`;'; $GLOBALS['cfg']['IgnoreMultiSubmitErrors'] = false; $_POST['submit_type'] = ''; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->executeSqlQuery([], $query); $this->assertEquals(['sql_query' => 'SELECT * FROM `test_db`.`test_table`;'], $result[0]); $this->assertEquals([], $result[3]); $this->assertEquals('SELECT * FROM `test_db`.`test_table`;', $result[5]); } /** * Test for executeSqlQuery */ public function testExecuteSqlQueryWithTryQuery(): void { $query = [ 'SELECT * FROM `test_db`.`test_table`;', 'SELECT * FROM `test_db`.`test_table_yaml`;', ]; $GLOBALS['sql_query'] = 'SELECT * FROM `test_db`.`test_table`;'; $GLOBALS['cfg']['IgnoreMultiSubmitErrors'] = true; $_POST['submit_type'] = ''; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->executeSqlQuery([], $query); $this->assertEquals(['sql_query' => 'SELECT * FROM `test_db`.`test_table`;'], $result[0]); $this->assertEquals([], $result[3]); $this->assertEquals('SELECT * FROM `test_db`.`test_table`;', $result[5]); } /** * Test for getWarningMessages */ public function testGetWarningMessages(): void { $warnings = [ Warning::fromArray(['Level' => 'Error', 'Code' => '1001', 'Message' => 'Message 1']), Warning::fromArray(['Level' => 'Warning', 'Code' => '1002', 'Message' => 'Message 2']), ]; $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('getWarnings') ->will($this->returnValue($warnings)); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getWarningMessages', [] ); $this->assertEquals(['Error: #1001 Message 1', 'Warning: #1002 Message 2'], $result); } /** * Test for getDisplayValueForForeignTableColumn */ public function testGetDisplayValueForForeignTableColumn(): void { $map = []; $map['f']['foreign_db'] = 'information_schema'; $map['f']['foreign_table'] = 'TABLES'; $map['f']['foreign_field'] = 'f'; $resultStub = $this->createMock(DummyResult::class); $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('tryQuery') ->with( 'SELECT `TABLE_COMMENT` FROM `information_schema`.`TABLES` WHERE `f`=1' ) ->will($this->returnValue($resultStub)); $resultStub->expects($this->once()) ->method('numRows') ->will($this->returnValue('2')); $resultStub->expects($this->once()) ->method('fetchValue') ->with(0) ->will($this->returnValue('2')); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->getDisplayValueForForeignTableColumn('=1', $map, 'f'); $this->assertEquals(2, $result); } /** * Test for getLinkForRelationalDisplayField */ public function testGetLinkForRelationalDisplayField(): void { $GLOBALS['cfg']['ServerDefault'] = 1; $_SESSION['tmpval']['relational_display'] = 'K'; $map = []; $map['f']['foreign_db'] = 'information_schema'; $map['f']['foreign_table'] = 'TABLES'; $map['f']['foreign_field'] = 'f'; $result = $this->insertEdit->getLinkForRelationalDisplayField($map, 'f', '=1', 'a>', 'b<'); $sqlSignature = Core::signSqlQuery('SELECT * FROM `information_schema`.`TABLES` WHERE `f`=1'); $this->assertEquals( 'b<', $result ); $_SESSION['tmpval']['relational_display'] = 'D'; $result = $this->insertEdit->getLinkForRelationalDisplayField($map, 'f', '=1', 'a>', 'b<'); $this->assertEquals( 'a>', $result ); } /** * Test for transformEditedValues */ public function testTransformEditedValues(): void { $_SESSION[' HMAC_secret '] = hash('sha1', 'test'); $edited_values = [ ['c' => 'cname'], ]; $GLOBALS['cfg']['DefaultTransformations']['PreApPend'] = [ '', '', ]; $GLOBALS['cfg']['ServerDefault'] = 1; $_POST['where_clause'] = '1'; $_POST['where_clause_sign'] = Core::signSqlQuery($_POST['where_clause']); $transformation = ['transformation_options' => "'','option ,, quoted',abd"]; $result = $this->insertEdit->transformEditedValues( 'db', 'table', $transformation, $edited_values, 'Text_Plain_PreApPend.php', 'c', ['a' => 'b'], 'transformation' ); $this->assertEquals( [ 'a' => 'b', 'transformations' => ['cnameoption ,, quoted'], ], $result ); } /** * Test for getQueryValuesForInsertAndUpdateInMultipleEdit */ public function testGetQueryValuesForInsertAndUpdateInMultipleEdit(): void { $multi_edit_columns_name = ['0' => 'fld']; $result = $this->insertEdit->getQueryValuesForInsertAndUpdateInMultipleEdit( $multi_edit_columns_name, [], '', [], [], true, [1], [2], 'foo', [], '0', [] ); $this->assertEquals( [ [ 1, 'foo', ], [ 2, '`fld`', ], ], $result ); $result = $this->insertEdit->getQueryValuesForInsertAndUpdateInMultipleEdit( $multi_edit_columns_name, [], '', [], [], false, [1], [2], 'foo', [], '0', ['a'] ); $this->assertEquals( [ [ 1, '`fld` = foo', ], [2], ], $result ); $result = $this->insertEdit->getQueryValuesForInsertAndUpdateInMultipleEdit( $multi_edit_columns_name, ['b'], "'`c`'", ['c'], [], false, [1], [2], 'foo', [], '0', ['a'] ); $this->assertEquals( [ [1], [2], ], $result ); $result = $this->insertEdit->getQueryValuesForInsertAndUpdateInMultipleEdit( $multi_edit_columns_name, ['b'], "'`c`'", ['c'], [3], false, [1], [2], 'foo', [], '0', [] ); $this->assertEquals( [ [ 1, '`fld` = foo', ], [2], ], $result ); } /** * Test for getCurrentValueAsAnArrayForMultipleEdit */ public function testGetCurrentValueAsAnArrayForMultipleEdit(): void { // case 2 $multi_edit_function = 'UUID'; $this->dummyDbi->addResult( 'SELECT UUID()', [ ['uuid1234'],// Minimal working setup for 2FA ] ); $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->getCurrentValueAsAnArrayForMultipleEdit( $multi_edit_function, null, 'currVal' ); $this->assertEquals("'uuid1234'", $result); // case 3 $multi_edit_function = 'AES_ENCRYPT'; $multi_edit_salt = ''; $result = $this->insertEdit->getCurrentValueAsAnArrayForMultipleEdit( $multi_edit_function, $multi_edit_salt, "'" ); $this->assertEquals("AES_ENCRYPT('\\'','')", $result); // case 4 $multi_edit_function = 'ABS'; $result = $this->insertEdit->getCurrentValueAsAnArrayForMultipleEdit( $multi_edit_function, null, "'" ); $this->assertEquals("ABS('\\'')", $result); // case 5 $multi_edit_function = 'RAND'; $result = $this->insertEdit->getCurrentValueAsAnArrayForMultipleEdit( $multi_edit_function, null, '' ); $this->assertEquals('RAND()', $result); // case 6 $multi_edit_function = 'PHP_PASSWORD_HASH'; $result = $this->insertEdit->getCurrentValueAsAnArrayForMultipleEdit( $multi_edit_function, null, "a'c" ); $this->assertTrue(password_verify("a'c", mb_substr($result, 1, -1))); } /** * Test for getCurrentValueForDifferentTypes */ public function testGetCurrentValueForDifferentTypes(): void { $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->getCurrentValueForDifferentTypes( '123', '0', [], '', [], [], [], [], true, true, '', 'test_table' ); $this->assertEquals('123', $result); // case 2 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['test'], '', [1], [], [], [], true, true, '', 'test_table' ); $this->assertEquals('NULL', $result); // case 3 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['test'], '', [], [], [], [], true, true, '', 'test_table' ); $this->assertEquals("''", $result); // case 4 $_POST['fields']['multi_edit'][0][0] = []; $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['set'], '', [], [], [], [], true, true, '', 'test_table' ); $this->assertEquals("''", $result); // case 5 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['protected'], '', [], ['name'], [], [], true, true, '`id` = 4', 'test_table' ); $this->assertEquals('0x313031', $result); // case 6 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['protected'], '', [], ['a'], [], [], true, true, '', 'test_table' ); $this->assertEquals('', $result); // case 7 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['bit'], '20\'12', [], ['a'], [], [], true, true, '', 'test_table' ); $this->assertEquals("b'00010'", $result); // case 7 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['date'], '20\'12', [], ['a'], [], [], true, true, '', 'test_table' ); $this->assertEquals("'20\\'12'", $result); // case 8 $_POST['fields']['multi_edit'][0][0] = []; $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['set'], '', [], [], [1], [], true, true, '', 'test_table' ); $this->assertEquals('NULL', $result); // case 9 $result = $this->insertEdit->getCurrentValueForDifferentTypes( false, '0', ['protected'], '', [], ['a'], [], [1], true, true, '', 'test_table' ); $this->assertEquals("''", $result); } /** * Test for verifyWhetherValueCanBeTruncatedAndAppendExtraData */ public function testVerifyWhetherValueCanBeTruncatedAndAppendExtraData(): void { $extra_data = ['isNeedToRecheck' => true]; $_POST['where_clause'] = []; $_POST['where_clause'][0] = 1; $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $resultStub = $this->createMock(DummyResult::class); $dbi->expects($this->exactly(3)) ->method('tryQuery') ->with('SELECT `table`.`a` FROM `db`.`table` WHERE 1') ->willReturn($resultStub); $meta1 = new FieldMetadata(MYSQLI_TYPE_TINY, 0, (object) []); $meta2 = new FieldMetadata(MYSQLI_TYPE_TINY, 0, (object) []); $meta3 = new FieldMetadata(MYSQLI_TYPE_TIMESTAMP, 0, (object) []); $dbi->expects($this->exactly(3)) ->method('getFieldsMeta') ->will($this->onConsecutiveCalls([$meta1], [$meta2], [$meta3])); $resultStub->expects($this->exactly(3)) ->method('fetchValue') ->will($this->onConsecutiveCalls(false, '123', '2013-08-28 06:34:14')); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $this->insertEdit->verifyWhetherValueCanBeTruncatedAndAppendExtraData('db', 'table', 'a', $extra_data); $this->assertFalse($extra_data['isNeedToRecheck']); $this->insertEdit->verifyWhetherValueCanBeTruncatedAndAppendExtraData('db', 'table', 'a', $extra_data); $this->assertEquals('123', $extra_data['truncatableFieldValue']); $this->assertTrue($extra_data['isNeedToRecheck']); $this->insertEdit->verifyWhetherValueCanBeTruncatedAndAppendExtraData('db', 'table', 'a', $extra_data); $this->assertEquals('2013-08-28 06:34:14.000000', $extra_data['truncatableFieldValue']); $this->assertTrue($extra_data['isNeedToRecheck']); } /** * Test for getTableColumns */ public function testGetTableColumns(): void { $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('selectDb') ->with('db'); $dbi->expects($this->once()) ->method('getColumns') ->with('db', 'table') ->will($this->returnValue([ ['a' => 'b', 'c' => 'd'], ['e' => 'f', 'g' => 'h'], ])); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->getTableColumns('db', 'table'); $this->assertEquals( [ ['a' => 'b', 'c' => 'd'], ['e' => 'f', 'g' => 'h'], ], $result ); } /** * Test for determineInsertOrEdit */ public function testDetermineInsertOrEdit(): void { $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $resultStub = $this->createMock(DummyResult::class); $dbi->expects($this->exactly(2)) ->method('query') ->will($this->returnValue($resultStub)); $GLOBALS['dbi'] = $dbi; $_POST['where_clause'] = '1'; $_SESSION['edit_next'] = '1'; $_POST['ShowFunctionFields'] = true; $_POST['ShowFieldTypesInDataEditView'] = true; $_POST['after_insert'] = 'edit_next'; $GLOBALS['cfg']['InsertRows'] = 2; $GLOBALS['cfg']['ShowSQL'] = false; $_POST['default_action'] = 'insert'; $responseMock = $this->getMockBuilder(ResponseRenderer::class) ->disableOriginalConstructor() ->onlyMethods(['addHtml']) ->getMock(); $restoreInstance = ResponseRenderer::getInstance(); $response = new ReflectionProperty(ResponseRenderer::class, 'instance'); $response->setAccessible(true); $response->setValue($responseMock); $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $result = $this->insertEdit->determineInsertOrEdit('1', 'db', 'table'); $this->assertEquals( [ false, null, [1], null, [$resultStub], [[]], false, 'edit_next', ], $result ); // case 2 unset($_POST['where_clause']); unset($_SESSION['edit_next']); $_POST['default_action'] = ''; $result = $this->insertEdit->determineInsertOrEdit(null, 'db', 'table'); $response->setValue($restoreInstance); $this->assertEquals( [ true, null, [], null, $resultStub, [ false, false, ], false, 'edit_next', ], $result ); } /** * Test for getCommentsMap */ public function testGetCommentsMap(): void { $GLOBALS['cfg']['ShowPropertyComments'] = false; $dbi = $this->getMockBuilder(DatabaseInterface::class) ->disableOriginalConstructor() ->getMock(); $dbi->expects($this->once()) ->method('getColumns') ->with('db', 'table', true) ->will( $this->returnValue( [ [ 'Comment' => 'b', 'Field' => 'd', ], ] ) ); $dbi->expects($this->any()) ->method('getTable') ->will( $this->returnValue( new Table('table', 'db') ) ); $GLOBALS['dbi'] = $dbi; $this->insertEdit = new InsertEdit( $GLOBALS['dbi'], new Relation($GLOBALS['dbi']), new Transformations(), new FileListing(), new Template() ); $this->assertEquals( [], $this->insertEdit->getCommentsMap('db', 'table') ); $GLOBALS['cfg']['ShowPropertyComments'] = true; $this->assertEquals( ['d' => 'b'], $this->insertEdit->getCommentsMap('db', 'table') ); } /** * Test for getHtmlForIgnoreOption */ public function testGetHtmlForIgnoreOption(): void { $expected = '
' . "\n"; $checked = 'checked="checked" '; $this->assertEquals( sprintf($expected, $checked), $this->insertEdit->getHtmlForIgnoreOption(1) ); $this->assertEquals( sprintf($expected, ''), $this->insertEdit->getHtmlForIgnoreOption(1, false) ); } /** * Test for getHtmlForInsertEditFormColumn */ public function testGetHtmlForInsertEditFormColumn(): void { $_SESSION[' HMAC_secret '] = hash('sha1', 'test'); $o_rows = 0; $tabindex = 0; $GLOBALS['plugin_scripts'] = []; $foreigners = ['foreign_keys_data' => []]; $table_column = [ 'Field' => 'col', 'Type' => 'varchar(20)', 'Null' => 'Yes', 'Privileges' => 'insert,update,select', ]; $repopulate = [md5('col') => 'val']; $column_mime = [ 'input_transformation' => 'Input/Image_JPEG_Upload.php', 'input_transformation_options' => '150', ]; $resultStub = $this->createMock(DummyResult::class); $resultStub->expects($this->any()) ->method('getFieldsMeta') ->will($this->returnValue([new FieldMetadata(0, 0, (object) ['length' => -1])])); // Test w/ input transformation $actual = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getHtmlForInsertEditFormColumn', [ $table_column, 0, [], false, $resultStub, '', '', '', false, [], &$o_rows, &$tabindex, 0, false, $foreigners, 0, 'table', 'db', 0, 0, '', '', $repopulate, $column_mime, '', ] ); $this->assertStringContainsString('col', $actual); $this->assertStringContainsString('', $actual); $this->assertStringContainsString('varchar(20)', $actual); $this->assertStringContainsString('', $actual); $this->assertStringContainsString('', $actual); $this->assertStringContainsString('Image preview here', $actual); $this->assertStringContainsString( '', $actual ); // Test w/o input_transformation $table_column = [ 'Field' => 'qwerty', 'Type' => 'datetime', 'Null' => 'Yes', 'Key' => '', 'Extra' => '', 'Default' => null, 'Privileges' => 'insert,update,select', ]; $repopulate = [md5('qwerty') => '12-10-14']; $actual = $this->callFunction( $this->insertEdit, InsertEdit::class, 'getHtmlForInsertEditFormColumn', [ $table_column, 0, [], false, $resultStub, '', '', '[a][0]', true, [], &$o_rows, &$tabindex, 0, false, $foreigners, 0, 'table', 'db', 0, 0, '', '', $repopulate, [], '', ] ); $this->assertStringContainsString('qwerty', $actual); $this->assertStringContainsString('', $actual); $this->assertStringContainsString('datetime', $actual); $this->assertStringContainsString( 'assertStringContainsString( '', $actual ); $this->assertStringContainsString( '', $actual ); $this->assertStringContainsString( 'assertStringContainsString( '', $actual ); $this->assertStringContainsString( ' []]; $table_columns = [ [ 'Field' => 'test', 'Extra' => '', 'Type' => 'longtext', 'Null' => 'Yes', 'pma_type' => 'longtext', 'True_Type' => 'longtext', 'Privileges' => 'select,insert,update,references', ], ]; $resultStub = $this->createMock(DummyResult::class); $resultStub->expects($this->any()) ->method('getFieldsMeta') ->will($this->returnValue([new FieldMetadata(0, 0, (object) ['length' => -1])])); $actual = $this->insertEdit->getHtmlForInsertEditRow( [], $table_columns, [], false, $resultStub, '', '', '', false, [], $o_rows, $tabindex, 1, false, $foreigners, 0, 'table', 'db', 0, 0, 'ltr', [], ['wc'] ); $this->assertStringContainsString('test', $actual); $this->assertStringContainsString('Column', $actual); $this->assertStringContainsString('assertStringContainsString('Value', $actual); $this->assertStringContainsString('longtext', $actual); $this->assertStringContainsString( '