ConnectionManagerTest.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. <?php
  2. declare(strict_types=1);
  3. /**
  4. * Licensed under The MIT License
  5. * For full copyright and license information, please see the LICENSE.txt
  6. * Redistributions of files must retain the above copyright notice
  7. *
  8. * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
  9. * @link https://cakephp.org CakePHP(tm) Project
  10. * @since 1.2.0
  11. * @license https://opensource.org/licenses/mit-license.php MIT License
  12. */
  13. namespace Cake\Test\TestCase\Datasource;
  14. use BadMethodCallException;
  15. use Cake\Core\Exception\CakeException;
  16. use Cake\Database\Connection;
  17. use Cake\Database\Driver\Mysql;
  18. use Cake\Database\Driver\Sqlite;
  19. use Cake\Database\Driver\Sqlserver;
  20. use Cake\Datasource\ConnectionManager;
  21. use Cake\Datasource\Exception\MissingDatasourceException;
  22. use Cake\TestSuite\TestCase;
  23. use InvalidArgumentException;
  24. use PHPUnit\Framework\Attributes\DataProvider;
  25. use TestApp\Datasource\FakeConnection;
  26. use TestPlugin\Datasource\TestSource;
  27. /**
  28. * ConnectionManager Test
  29. */
  30. class ConnectionManagerTest extends TestCase
  31. {
  32. /**
  33. * tearDown method
  34. */
  35. public function tearDown(): void
  36. {
  37. parent::tearDown();
  38. $this->clearPlugins();
  39. ConnectionManager::drop('test_variant');
  40. ConnectionManager::dropAlias('other_name');
  41. ConnectionManager::dropAlias('test2');
  42. }
  43. /**
  44. * Data provider for valid config data sets.
  45. *
  46. * @return array
  47. */
  48. public static function configProvider(): array
  49. {
  50. return [
  51. 'Array of data using classname key.' => [[
  52. 'className' => FakeConnection::class,
  53. 'instance' => 'Sqlite',
  54. 'database' => ':memory:',
  55. ]],
  56. 'Direct instance' => [new FakeConnection()],
  57. ];
  58. }
  59. /**
  60. * Test the various valid config() calls.
  61. *
  62. * @param \Cake\Datasource\ConnectionInterface|array $settings
  63. */
  64. #[DataProvider('configProvider')]
  65. public function testConfigVariants($settings): void
  66. {
  67. $this->assertNotContains('test_variant', ConnectionManager::configured(), 'test_variant config should not exist.');
  68. ConnectionManager::setConfig('test_variant', $settings);
  69. $ds = ConnectionManager::get('test_variant');
  70. $this->assertInstanceOf(FakeConnection::class, $ds);
  71. $this->assertContains('test_variant', ConnectionManager::configured());
  72. }
  73. /**
  74. * Test invalid classes cause exceptions
  75. */
  76. public function testConfigInvalidOptions(): void
  77. {
  78. $this->expectException(MissingDatasourceException::class);
  79. ConnectionManager::setConfig('test_variant', [
  80. 'className' => 'Herp\Derp',
  81. ]);
  82. ConnectionManager::get('test_variant');
  83. }
  84. /**
  85. * Test for errors on duplicate config.
  86. */
  87. public function testConfigDuplicateConfig(): void
  88. {
  89. $this->expectException(BadMethodCallException::class);
  90. $this->expectExceptionMessage('Cannot reconfigure existing key `test_variant`');
  91. $settings = [
  92. 'className' => FakeConnection::class,
  93. 'database' => ':memory:',
  94. ];
  95. ConnectionManager::setConfig('test_variant', $settings);
  96. ConnectionManager::setConfig('test_variant', $settings);
  97. }
  98. /**
  99. * Test get() failing on missing config.
  100. */
  101. public function testGetFailOnMissingConfig(): void
  102. {
  103. $this->expectException(CakeException::class);
  104. $this->expectExceptionMessage('The datasource configuration `test_variant` was not found.');
  105. ConnectionManager::get('test_variant');
  106. }
  107. /**
  108. * Test loading configured connections.
  109. */
  110. public function testGet(): void
  111. {
  112. $config = ConnectionManager::getConfig('test');
  113. $this->skipIf(empty($config), 'No test config, skipping');
  114. $ds = ConnectionManager::get('test');
  115. $this->assertSame($ds, ConnectionManager::get('test'));
  116. $this->assertInstanceOf(Connection::class, $ds);
  117. $this->assertSame('test', $ds->configName());
  118. }
  119. /**
  120. * Test loading connections without aliases
  121. */
  122. public function testGetNoAlias(): void
  123. {
  124. $this->expectException(CakeException::class);
  125. $this->expectExceptionMessage('The datasource configuration `other_name` was not found.');
  126. $config = ConnectionManager::getConfig('test');
  127. $this->skipIf(empty($config), 'No test config, skipping');
  128. ConnectionManager::alias('test', 'other_name');
  129. ConnectionManager::get('other_name', false);
  130. }
  131. /**
  132. * Test that configured() finds configured sources.
  133. */
  134. public function testConfigured(): void
  135. {
  136. ConnectionManager::setConfig('test_variant', [
  137. 'className' => FakeConnection::class,
  138. 'database' => ':memory:',
  139. ]);
  140. $results = ConnectionManager::configured();
  141. $this->assertContains('test_variant', $results);
  142. }
  143. /**
  144. * testGetPluginDataSource method
  145. */
  146. public function testGetPluginDataSource(): void
  147. {
  148. $this->loadPlugins(['TestPlugin']);
  149. $name = 'test_variant';
  150. $config = ['className' => 'TestPlugin.TestSource', 'foo' => 'bar'];
  151. ConnectionManager::setConfig($name, $config);
  152. $connection = ConnectionManager::get($name);
  153. $this->assertInstanceOf(TestSource::class, $connection);
  154. unset($config['className']);
  155. $this->assertSame($config + ['name' => 'test_variant'], $connection->config());
  156. }
  157. /**
  158. * Tests that a connection configuration can be deleted in runtime
  159. */
  160. public function testDrop(): void
  161. {
  162. ConnectionManager::setConfig('test_variant', [
  163. 'className' => FakeConnection::class,
  164. 'database' => ':memory:',
  165. ]);
  166. $result = ConnectionManager::configured();
  167. $this->assertContains('test_variant', $result);
  168. $this->assertTrue(ConnectionManager::drop('test_variant'));
  169. $result = ConnectionManager::configured();
  170. $this->assertNotContains('test_variant', $result);
  171. $this->assertFalse(ConnectionManager::drop('probably_does_not_exist'), 'Should return false on failure.');
  172. }
  173. public function testAliases(): void
  174. {
  175. $this->assertSame(['default' => 'test'], ConnectionManager::aliases());
  176. ConnectionManager::alias('test', 'test2');
  177. $this->assertSame(['default' => 'test', 'test2' => 'test'], ConnectionManager::aliases());
  178. ConnectionManager::dropAlias('test2');
  179. $this->assertSame(['default' => 'test'], ConnectionManager::aliases());
  180. }
  181. /**
  182. * Test aliasing connections.
  183. */
  184. public function testAlias(): void
  185. {
  186. ConnectionManager::setConfig('test_variant', [
  187. 'className' => FakeConnection::class,
  188. 'database' => ':memory:',
  189. ]);
  190. ConnectionManager::alias('test_variant', 'other_name');
  191. $result = ConnectionManager::get('test_variant');
  192. $this->assertSame($result, ConnectionManager::get('other_name'));
  193. }
  194. /**
  195. * provider for DSN strings.
  196. *
  197. * @return array
  198. */
  199. public static function dsnProvider(): array
  200. {
  201. return [
  202. 'no user' => [
  203. 'mysql://localhost:3306/database',
  204. [
  205. 'className' => Connection::class,
  206. 'driver' => Mysql::class,
  207. 'host' => 'localhost',
  208. 'database' => 'database',
  209. 'port' => 3306,
  210. 'scheme' => 'mysql',
  211. ],
  212. ],
  213. 'subdomain host' => [
  214. 'mysql://my.host-name.com:3306/database',
  215. [
  216. 'className' => Connection::class,
  217. 'driver' => Mysql::class,
  218. 'host' => 'my.host-name.com',
  219. 'database' => 'database',
  220. 'port' => 3306,
  221. 'scheme' => 'mysql',
  222. ],
  223. ],
  224. 'user & pass' => [
  225. 'mysql://root:secret@localhost:3306/database?log=1',
  226. [
  227. 'scheme' => 'mysql',
  228. 'className' => Connection::class,
  229. 'driver' => Mysql::class,
  230. 'host' => 'localhost',
  231. 'username' => 'root',
  232. 'password' => 'secret',
  233. 'port' => 3306,
  234. 'database' => 'database',
  235. 'log' => '1',
  236. ],
  237. ],
  238. 'no password' => [
  239. 'mysql://user@localhost:3306/database',
  240. [
  241. 'className' => Connection::class,
  242. 'driver' => Mysql::class,
  243. 'host' => 'localhost',
  244. 'database' => 'database',
  245. 'port' => 3306,
  246. 'scheme' => 'mysql',
  247. 'username' => 'user',
  248. ],
  249. ],
  250. 'empty password' => [
  251. 'mysql://user:@localhost:3306/database',
  252. [
  253. 'className' => Connection::class,
  254. 'driver' => Mysql::class,
  255. 'host' => 'localhost',
  256. 'database' => 'database',
  257. 'port' => 3306,
  258. 'scheme' => 'mysql',
  259. 'username' => 'user',
  260. 'password' => '',
  261. ],
  262. ],
  263. 'sqlite memory' => [
  264. 'sqlite:///:memory:',
  265. [
  266. 'className' => Connection::class,
  267. 'driver' => Sqlite::class,
  268. 'database' => ':memory:',
  269. 'scheme' => 'sqlite',
  270. ],
  271. ],
  272. 'sqlite path' => [
  273. 'sqlite:////absolute/path',
  274. [
  275. 'className' => Connection::class,
  276. 'driver' => Sqlite::class,
  277. 'database' => '/absolute/path',
  278. 'scheme' => 'sqlite',
  279. ],
  280. ],
  281. 'sqlite database query' => [
  282. 'sqlite:///?database=:memory:',
  283. [
  284. 'className' => Connection::class,
  285. 'driver' => Sqlite::class,
  286. 'database' => ':memory:',
  287. 'scheme' => 'sqlite',
  288. ],
  289. ],
  290. 'sqlserver' => [
  291. 'sqlserver://sa:Password12!@.\SQL2012SP1/cakephp?MultipleActiveResultSets=false',
  292. [
  293. 'className' => Connection::class,
  294. 'driver' => Sqlserver::class,
  295. 'host' => '.\SQL2012SP1',
  296. 'MultipleActiveResultSets' => false,
  297. 'password' => 'Password12!',
  298. 'database' => 'cakephp',
  299. 'scheme' => 'sqlserver',
  300. 'username' => 'sa',
  301. ],
  302. ],
  303. 'sqllocaldb' => [
  304. 'sqlserver://username:password@(localdb)\.\DeptSharedLocalDB/database',
  305. [
  306. 'className' => Connection::class,
  307. 'driver' => Sqlserver::class,
  308. 'host' => '(localdb)\.\DeptSharedLocalDB',
  309. 'password' => 'password',
  310. 'database' => 'database',
  311. 'scheme' => 'sqlserver',
  312. 'username' => 'username',
  313. ],
  314. ],
  315. 'classname query arg' => [
  316. 'mysql://localhost/database?className=Custom\Driver',
  317. [
  318. 'className' => Connection::class,
  319. 'database' => 'database',
  320. 'driver' => 'Custom\Driver',
  321. 'host' => 'localhost',
  322. 'scheme' => 'mysql',
  323. ],
  324. ],
  325. 'classname and port' => [
  326. 'mysql://localhost:3306/database?className=Custom\Driver',
  327. [
  328. 'className' => Connection::class,
  329. 'database' => 'database',
  330. 'driver' => 'Custom\Driver',
  331. 'host' => 'localhost',
  332. 'scheme' => 'mysql',
  333. 'port' => 3306,
  334. ],
  335. ],
  336. 'custom connection class' => [
  337. 'Cake\Database\Connection://localhost:3306/database?driver=Cake\Database\Driver\Mysql',
  338. [
  339. 'className' => Connection::class,
  340. 'database' => 'database',
  341. 'driver' => Mysql::class,
  342. 'host' => 'localhost',
  343. 'scheme' => Connection::class,
  344. 'port' => 3306,
  345. ],
  346. ],
  347. 'complex password' => [
  348. 'mysql://user:/?#][{}$%20@!@localhost:3306/database?log=1&quoteIdentifiers=1',
  349. [
  350. 'className' => Connection::class,
  351. 'database' => 'database',
  352. 'driver' => Mysql::class,
  353. 'host' => 'localhost',
  354. 'password' => '/?#][{}$%20@!',
  355. 'port' => 3306,
  356. 'scheme' => 'mysql',
  357. 'username' => 'user',
  358. 'log' => 1,
  359. 'quoteIdentifiers' => 1,
  360. ],
  361. ],
  362. ];
  363. }
  364. /**
  365. * Test parseDsn method.
  366. */
  367. #[DataProvider('dsnProvider')]
  368. public function testParseDsn(string $dsn, array $expected): void
  369. {
  370. $result = ConnectionManager::parseDsn($dsn);
  371. $this->assertEquals($expected, $result);
  372. }
  373. /**
  374. * Test parseDsn invalid.
  375. */
  376. public function testParseDsnInvalid(): void
  377. {
  378. $this->expectException(InvalidArgumentException::class);
  379. $this->expectExceptionMessage('The DSN string `bagof:nope` could not be parsed.');
  380. ConnectionManager::parseDsn('bagof:nope');
  381. }
  382. /**
  383. * Tests that directly setting an instance in a config, will not return a different
  384. * instance later on
  385. */
  386. public function testConfigWithObject(): void
  387. {
  388. $connection = new FakeConnection();
  389. ConnectionManager::setConfig('test_variant', $connection);
  390. $this->assertSame($connection, ConnectionManager::get('test_variant'));
  391. }
  392. /**
  393. * Tests configuring an instance with a callable
  394. */
  395. public function testConfigWithCallable(): void
  396. {
  397. $connection = new FakeConnection();
  398. $callable = function ($alias) use ($connection) {
  399. $this->assertSame('test_variant', $alias);
  400. return $connection;
  401. };
  402. ConnectionManager::setConfig('test_variant', $callable);
  403. $this->assertSame($connection, ConnectionManager::get('test_variant'));
  404. }
  405. /**
  406. * Tests that setting a config will also correctly set the name for the connection
  407. */
  408. public function testSetConfigName(): void
  409. {
  410. //Set with explicit name
  411. ConnectionManager::setConfig('test_variant', [
  412. 'className' => FakeConnection::class,
  413. 'database' => ':memory:',
  414. ]);
  415. $result = ConnectionManager::get('test_variant');
  416. $this->assertSame('test_variant', $result->configName());
  417. ConnectionManager::drop('test_variant');
  418. ConnectionManager::setConfig([
  419. 'test_variant' => [
  420. 'className' => FakeConnection::class,
  421. 'database' => ':memory:',
  422. ],
  423. ]);
  424. $result = ConnectionManager::get('test_variant');
  425. $this->assertSame('test_variant', $result->configName());
  426. }
  427. }