| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- <?php
- declare(strict_types=1);
- /**
- * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- * @link https://cakephp.org CakePHP(tm) Project
- * @since 3.0.0
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Test\TestCase\Database\Log;
- use Cake\Database\Driver\Sqlserver;
- use Cake\Database\Log\LoggedQuery;
- use Cake\Datasource\ConnectionManager;
- use Cake\TestSuite\TestCase;
- use Cake\Utility\Text;
- use Exception;
- /**
- * Tests LoggedQuery class
- */
- class LoggedQueryTest extends TestCase
- {
- protected $driver;
- protected $true = 'TRUE';
- protected $false = 'FALSE';
- public function setUp(): void
- {
- $this->driver = ConnectionManager::get('test')->getDriver();
- if ($this->driver instanceof Sqlserver) {
- $this->true = '1';
- $this->false = '0';
- }
- }
- /**
- * Tests that LoggedQuery can be converted to string
- */
- public function testStringConversion(): void
- {
- $logged = new LoggedQuery();
- $logged->setContext(['query' => 'SELECT foo FROM bar']);
- $this->assertSame('SELECT foo FROM bar', (string)$logged);
- }
- /**
- * Tests that query placeholders are replaced when logged
- */
- public function testStringInterpolation(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'driver' => $this->driver,
- 'query' => 'SELECT a FROM b where a = :p1 AND b = :p2 AND c = :p3 AND d = :p4 AND e = :p5 AND f = :p6',
- 'params' => ['p1' => 'string', 'p3' => null, 'p2' => 3, 'p4' => true, 'p5' => false, 'p6' => 0],
- ]);
- $expected = "SELECT a FROM b where a = 'string' AND b = 3 AND c = NULL AND d = $this->true AND e = $this->false AND f = 0";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that positional placeholders are replaced when logging a query
- */
- public function testStringInterpolationNotNamed(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'driver' => $this->driver,
- 'query' => 'SELECT a FROM b where a = ? AND b = ? AND c = ? AND d = ? AND e = ? AND f = ?',
- 'params' => ['string', '3', null, true, false, 0],
- ]);
- $expected = "SELECT a FROM b where a = 'string' AND b = '3' AND c = NULL AND d = $this->true AND e = $this->false AND f = 0";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that repeated placeholders are correctly replaced
- */
- public function testStringInterpolationDuplicate(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1 AND b = :p1 AND c = :p2 AND d = :p2',
- 'params' => ['p1' => 'string', 'p2' => 3],
- ]);
- $expected = "SELECT a FROM b where a = 'string' AND b = 'string' AND c = 3 AND d = 3";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that named placeholders
- */
- public function testStringInterpolationNamed(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1 AND b = :p11 AND c = :p20 AND d = :p2',
- 'params' => ['p11' => 'test', 'p1' => 'string', 'p2' => 3, 'p20' => 5],
- ]);
- $expected = "SELECT a FROM b where a = 'string' AND b = 'test' AND c = 5 AND d = 3";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that placeholders are replaced with correctly escaped strings
- */
- public function testStringInterpolationSpecialChars(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1 AND b = :p2 AND c = :p3 AND d = :p4',
- 'params' => ['p1' => '$2y$10$dUAIj', 'p2' => '$0.23', 'p3' => 'a\\0b\\1c\\d', 'p4' => "a'b"],
- ]);
- $expected = "SELECT a FROM b where a = '\$2y\$10\$dUAIj' AND b = '\$0.23' AND c = 'a\\\\0b\\\\1c\\\\d' AND d = 'a''b'";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that query placeholders are replaced when logged
- */
- public function testBinaryInterpolation(): void
- {
- $query = new LoggedQuery();
- $uuid = str_replace('-', '', Text::uuid());
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1',
- 'params' => ['p1' => hex2bin($uuid)],
- ]);
- $expected = "SELECT a FROM b where a = '{$uuid}'";
- $this->assertSame($expected, (string)$query);
- }
- /**
- * Tests that unknown possible binary data is not replaced to hex.
- */
- public function testBinaryInterpolationIgnored(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1',
- 'params' => ['p1' => "a\tz"],
- ]);
- $expected = "SELECT a FROM b where a = 'a\tz'";
- $this->assertSame($expected, (string)$query);
- }
- public function testGetContext(): void
- {
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1',
- 'numRows' => 10,
- 'took' => 15,
- ]);
- $expected = [
- 'numRows' => 10,
- 'took' => 15.0,
- ];
- $this->assertSame($expected, $query->getContext());
- }
- public function testJsonSerialize(): void
- {
- $error = new Exception('You fail!');
- $query = new LoggedQuery();
- $query->setContext([
- 'query' => 'SELECT a FROM b where a = :p1',
- 'params' => ['p1' => '$2y$10$dUAIj'],
- 'numRows' => 4,
- 'error' => $error,
- ]);
- $expected = json_encode([
- 'query' => 'SELECT a FROM b where a = :p1',
- 'numRows' => 4,
- 'params' => ['p1' => '$2y$10$dUAIj'],
- 'took' => 0,
- 'error' => [
- 'class' => get_class($error),
- 'message' => $error->getMessage(),
- 'code' => $error->getCode(),
- ],
- ]);
- $this->assertEquals($expected, json_encode($query));
- }
- }
|