DoctrineDbalAdapter.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\Cache\Adapter;
  11. use Doctrine\DBAL\Connection;
  12. use Doctrine\DBAL\Driver\ServerInfoAwareConnection;
  13. use Doctrine\DBAL\DriverManager;
  14. use Doctrine\DBAL\Exception as DBALException;
  15. use Doctrine\DBAL\Exception\TableNotFoundException;
  16. use Doctrine\DBAL\ParameterType;
  17. use Doctrine\DBAL\Schema\Schema;
  18. use Symfony\Component\Cache\Exception\InvalidArgumentException;
  19. use Symfony\Component\Cache\Marshaller\DefaultMarshaller;
  20. use Symfony\Component\Cache\Marshaller\MarshallerInterface;
  21. use Symfony\Component\Cache\PruneableInterface;
  22. class DoctrineDbalAdapter extends AbstractAdapter implements PruneableInterface
  23. {
  24. protected $maxIdLength = 255;
  25. private $marshaller;
  26. private $conn;
  27. private $platformName;
  28. private $serverVersion;
  29. private $table = 'cache_items';
  30. private $idCol = 'item_id';
  31. private $dataCol = 'item_data';
  32. private $lifetimeCol = 'item_lifetime';
  33. private $timeCol = 'item_time';
  34. private $namespace;
  35. /**
  36. * You can either pass an existing database Doctrine DBAL Connection or
  37. * a DSN string that will be used to connect to the database.
  38. *
  39. * The cache table is created automatically when possible.
  40. * Otherwise, use the createTable() method.
  41. *
  42. * List of available options:
  43. * * db_table: The name of the table [default: cache_items]
  44. * * db_id_col: The column where to store the cache id [default: item_id]
  45. * * db_data_col: The column where to store the cache data [default: item_data]
  46. * * db_lifetime_col: The column where to store the lifetime [default: item_lifetime]
  47. * * db_time_col: The column where to store the timestamp [default: item_time]
  48. *
  49. * @param Connection|string $connOrDsn
  50. *
  51. * @throws InvalidArgumentException When namespace contains invalid characters
  52. */
  53. public function __construct($connOrDsn, string $namespace = '', int $defaultLifetime = 0, array $options = [], MarshallerInterface $marshaller = null)
  54. {
  55. if (isset($namespace[0]) && preg_match('#[^-+.A-Za-z0-9]#', $namespace, $match)) {
  56. throw new InvalidArgumentException(sprintf('Namespace contains "%s" but only characters in [-+.A-Za-z0-9] are allowed.', $match[0]));
  57. }
  58. if ($connOrDsn instanceof Connection) {
  59. $this->conn = $connOrDsn;
  60. } elseif (\is_string($connOrDsn)) {
  61. if (!class_exists(DriverManager::class)) {
  62. throw new InvalidArgumentException(sprintf('Failed to parse the DSN "%s". Try running "composer require doctrine/dbal".', $connOrDsn));
  63. }
  64. $this->conn = DriverManager::getConnection(['url' => $connOrDsn]);
  65. } else {
  66. throw new \TypeError(sprintf('Argument 1 passed to "%s()" must be "%s" or string, "%s" given.', __METHOD__, Connection::class, get_debug_type($connOrDsn)));
  67. }
  68. $this->table = $options['db_table'] ?? $this->table;
  69. $this->idCol = $options['db_id_col'] ?? $this->idCol;
  70. $this->dataCol = $options['db_data_col'] ?? $this->dataCol;
  71. $this->lifetimeCol = $options['db_lifetime_col'] ?? $this->lifetimeCol;
  72. $this->timeCol = $options['db_time_col'] ?? $this->timeCol;
  73. $this->namespace = $namespace;
  74. $this->marshaller = $marshaller ?? new DefaultMarshaller();
  75. parent::__construct($namespace, $defaultLifetime);
  76. }
  77. /**
  78. * Creates the table to store cache items which can be called once for setup.
  79. *
  80. * Cache ID are saved in a column of maximum length 255. Cache data is
  81. * saved in a BLOB.
  82. *
  83. * @throws DBALException When the table already exists
  84. */
  85. public function createTable()
  86. {
  87. $schema = new Schema();
  88. $this->addTableToSchema($schema);
  89. foreach ($schema->toSql($this->conn->getDatabasePlatform()) as $sql) {
  90. $this->conn->executeStatement($sql);
  91. }
  92. }
  93. /**
  94. * {@inheritdoc}
  95. */
  96. public function configureSchema(Schema $schema, Connection $forConnection): void
  97. {
  98. // only update the schema for this connection
  99. if ($forConnection !== $this->conn) {
  100. return;
  101. }
  102. if ($schema->hasTable($this->table)) {
  103. return;
  104. }
  105. $this->addTableToSchema($schema);
  106. }
  107. /**
  108. * {@inheritdoc}
  109. */
  110. public function prune(): bool
  111. {
  112. $deleteSql = "DELETE FROM $this->table WHERE $this->lifetimeCol + $this->timeCol <= ?";
  113. $params = [time()];
  114. $paramTypes = [ParameterType::INTEGER];
  115. if ('' !== $this->namespace) {
  116. $deleteSql .= " AND $this->idCol LIKE ?";
  117. $params[] = sprintf('%s%%', $this->namespace);
  118. $paramTypes[] = ParameterType::STRING;
  119. }
  120. try {
  121. $this->conn->executeStatement($deleteSql, $params, $paramTypes);
  122. } catch (TableNotFoundException $e) {
  123. }
  124. return true;
  125. }
  126. /**
  127. * {@inheritdoc}
  128. */
  129. protected function doFetch(array $ids): iterable
  130. {
  131. $now = time();
  132. $expired = [];
  133. $sql = "SELECT $this->idCol, CASE WHEN $this->lifetimeCol IS NULL OR $this->lifetimeCol + $this->timeCol > ? THEN $this->dataCol ELSE NULL END FROM $this->table WHERE $this->idCol IN (?)";
  134. $result = $this->conn->executeQuery($sql, [
  135. $now,
  136. $ids,
  137. ], [
  138. ParameterType::INTEGER,
  139. Connection::PARAM_STR_ARRAY,
  140. ])->iterateNumeric();
  141. foreach ($result as $row) {
  142. if (null === $row[1]) {
  143. $expired[] = $row[0];
  144. } else {
  145. yield $row[0] => $this->marshaller->unmarshall(\is_resource($row[1]) ? stream_get_contents($row[1]) : $row[1]);
  146. }
  147. }
  148. if ($expired) {
  149. $sql = "DELETE FROM $this->table WHERE $this->lifetimeCol + $this->timeCol <= ? AND $this->idCol IN (?)";
  150. $this->conn->executeStatement($sql, [
  151. $now,
  152. $expired,
  153. ], [
  154. ParameterType::INTEGER,
  155. Connection::PARAM_STR_ARRAY,
  156. ]);
  157. }
  158. }
  159. /**
  160. * {@inheritdoc}
  161. */
  162. protected function doHave(string $id): bool
  163. {
  164. $sql = "SELECT 1 FROM $this->table WHERE $this->idCol = ? AND ($this->lifetimeCol IS NULL OR $this->lifetimeCol + $this->timeCol > ?)";
  165. $result = $this->conn->executeQuery($sql, [
  166. $id,
  167. time(),
  168. ], [
  169. ParameterType::STRING,
  170. ParameterType::INTEGER,
  171. ]);
  172. return (bool) $result->fetchOne();
  173. }
  174. /**
  175. * {@inheritdoc}
  176. */
  177. protected function doClear(string $namespace): bool
  178. {
  179. if ('' === $namespace) {
  180. if ('sqlite' === $this->getPlatformName()) {
  181. $sql = "DELETE FROM $this->table";
  182. } else {
  183. $sql = "TRUNCATE TABLE $this->table";
  184. }
  185. } else {
  186. $sql = "DELETE FROM $this->table WHERE $this->idCol LIKE '$namespace%'";
  187. }
  188. try {
  189. $this->conn->executeStatement($sql);
  190. } catch (TableNotFoundException $e) {
  191. }
  192. return true;
  193. }
  194. /**
  195. * {@inheritdoc}
  196. */
  197. protected function doDelete(array $ids): bool
  198. {
  199. $sql = "DELETE FROM $this->table WHERE $this->idCol IN (?)";
  200. try {
  201. $this->conn->executeStatement($sql, [array_values($ids)], [Connection::PARAM_STR_ARRAY]);
  202. } catch (TableNotFoundException $e) {
  203. }
  204. return true;
  205. }
  206. /**
  207. * {@inheritdoc}
  208. */
  209. protected function doSave(array $values, int $lifetime)
  210. {
  211. if (!$values = $this->marshaller->marshall($values, $failed)) {
  212. return $failed;
  213. }
  214. $platformName = $this->getPlatformName();
  215. $insertSql = "INSERT INTO $this->table ($this->idCol, $this->dataCol, $this->lifetimeCol, $this->timeCol) VALUES (?, ?, ?, ?)";
  216. switch (true) {
  217. case 'mysql' === $platformName:
  218. $sql = $insertSql." ON DUPLICATE KEY UPDATE $this->dataCol = VALUES($this->dataCol), $this->lifetimeCol = VALUES($this->lifetimeCol), $this->timeCol = VALUES($this->timeCol)";
  219. break;
  220. case 'oci' === $platformName:
  221. // DUAL is Oracle specific dummy table
  222. $sql = "MERGE INTO $this->table USING DUAL ON ($this->idCol = ?) ".
  223. "WHEN NOT MATCHED THEN INSERT ($this->idCol, $this->dataCol, $this->lifetimeCol, $this->timeCol) VALUES (?, ?, ?, ?) ".
  224. "WHEN MATCHED THEN UPDATE SET $this->dataCol = ?, $this->lifetimeCol = ?, $this->timeCol = ?";
  225. break;
  226. case 'sqlsrv' === $platformName && version_compare($this->getServerVersion(), '10', '>='):
  227. // MERGE is only available since SQL Server 2008 and must be terminated by semicolon
  228. // It also requires HOLDLOCK according to http://weblogs.sqlteam.com/dang/archive/2009/01/31/UPSERT-Race-Condition-With-MERGE.aspx
  229. $sql = "MERGE INTO $this->table WITH (HOLDLOCK) USING (SELECT 1 AS dummy) AS src ON ($this->idCol = ?) ".
  230. "WHEN NOT MATCHED THEN INSERT ($this->idCol, $this->dataCol, $this->lifetimeCol, $this->timeCol) VALUES (?, ?, ?, ?) ".
  231. "WHEN MATCHED THEN UPDATE SET $this->dataCol = ?, $this->lifetimeCol = ?, $this->timeCol = ?;";
  232. break;
  233. case 'sqlite' === $platformName:
  234. $sql = 'INSERT OR REPLACE'.substr($insertSql, 6);
  235. break;
  236. case 'pgsql' === $platformName && version_compare($this->getServerVersion(), '9.5', '>='):
  237. $sql = $insertSql." ON CONFLICT ($this->idCol) DO UPDATE SET ($this->dataCol, $this->lifetimeCol, $this->timeCol) = (EXCLUDED.$this->dataCol, EXCLUDED.$this->lifetimeCol, EXCLUDED.$this->timeCol)";
  238. break;
  239. default:
  240. $platformName = null;
  241. $sql = "UPDATE $this->table SET $this->dataCol = ?, $this->lifetimeCol = ?, $this->timeCol = ? WHERE $this->idCol = ?";
  242. break;
  243. }
  244. $now = time();
  245. $lifetime = $lifetime ?: null;
  246. try {
  247. $stmt = $this->conn->prepare($sql);
  248. } catch (TableNotFoundException $e) {
  249. if (!$this->conn->isTransactionActive() || \in_array($platformName, ['pgsql', 'sqlite', 'sqlsrv'], true)) {
  250. $this->createTable();
  251. }
  252. $stmt = $this->conn->prepare($sql);
  253. }
  254. // $id and $data are defined later in the loop. Binding is done by reference, values are read on execution.
  255. if ('sqlsrv' === $platformName || 'oci' === $platformName) {
  256. $stmt->bindParam(1, $id);
  257. $stmt->bindParam(2, $id);
  258. $stmt->bindParam(3, $data, ParameterType::LARGE_OBJECT);
  259. $stmt->bindValue(4, $lifetime, ParameterType::INTEGER);
  260. $stmt->bindValue(5, $now, ParameterType::INTEGER);
  261. $stmt->bindParam(6, $data, ParameterType::LARGE_OBJECT);
  262. $stmt->bindValue(7, $lifetime, ParameterType::INTEGER);
  263. $stmt->bindValue(8, $now, ParameterType::INTEGER);
  264. } elseif (null !== $platformName) {
  265. $stmt->bindParam(1, $id);
  266. $stmt->bindParam(2, $data, ParameterType::LARGE_OBJECT);
  267. $stmt->bindValue(3, $lifetime, ParameterType::INTEGER);
  268. $stmt->bindValue(4, $now, ParameterType::INTEGER);
  269. } else {
  270. $stmt->bindParam(1, $data, ParameterType::LARGE_OBJECT);
  271. $stmt->bindValue(2, $lifetime, ParameterType::INTEGER);
  272. $stmt->bindValue(3, $now, ParameterType::INTEGER);
  273. $stmt->bindParam(4, $id);
  274. $insertStmt = $this->conn->prepare($insertSql);
  275. $insertStmt->bindParam(1, $id);
  276. $insertStmt->bindParam(2, $data, ParameterType::LARGE_OBJECT);
  277. $insertStmt->bindValue(3, $lifetime, ParameterType::INTEGER);
  278. $insertStmt->bindValue(4, $now, ParameterType::INTEGER);
  279. }
  280. foreach ($values as $id => $data) {
  281. try {
  282. $rowCount = $stmt->executeStatement();
  283. } catch (TableNotFoundException $e) {
  284. if (!$this->conn->isTransactionActive() || \in_array($platformName, ['pgsql', 'sqlite', 'sqlsrv'], true)) {
  285. $this->createTable();
  286. }
  287. $rowCount = $stmt->executeStatement();
  288. }
  289. if (null === $platformName && 0 === $rowCount) {
  290. try {
  291. $insertStmt->executeStatement();
  292. } catch (DBALException $e) {
  293. // A concurrent write won, let it be
  294. }
  295. }
  296. }
  297. return $failed;
  298. }
  299. /**
  300. * @internal
  301. */
  302. protected function getId($key)
  303. {
  304. if ('pgsql' !== $this->getPlatformName()) {
  305. return parent::getId($key);
  306. }
  307. if (str_contains($key, "\0") || str_contains($key, '%') || !preg_match('//u', $key)) {
  308. $key = rawurlencode($key);
  309. }
  310. return parent::getId($key);
  311. }
  312. private function getPlatformName(): string
  313. {
  314. if (isset($this->platformName)) {
  315. return $this->platformName;
  316. }
  317. $platform = $this->conn->getDatabasePlatform();
  318. switch (true) {
  319. case $platform instanceof \Doctrine\DBAL\Platforms\MySQLPlatform:
  320. case $platform instanceof \Doctrine\DBAL\Platforms\MySQL57Platform:
  321. return $this->platformName = 'mysql';
  322. case $platform instanceof \Doctrine\DBAL\Platforms\SqlitePlatform:
  323. return $this->platformName = 'sqlite';
  324. case $platform instanceof \Doctrine\DBAL\Platforms\PostgreSQLPlatform:
  325. case $platform instanceof \Doctrine\DBAL\Platforms\PostgreSQL94Platform:
  326. return $this->platformName = 'pgsql';
  327. case $platform instanceof \Doctrine\DBAL\Platforms\OraclePlatform:
  328. return $this->platformName = 'oci';
  329. case $platform instanceof \Doctrine\DBAL\Platforms\SQLServerPlatform:
  330. case $platform instanceof \Doctrine\DBAL\Platforms\SQLServer2012Platform:
  331. return $this->platformName = 'sqlsrv';
  332. default:
  333. return $this->platformName = \get_class($platform);
  334. }
  335. }
  336. private function getServerVersion(): string
  337. {
  338. if (isset($this->serverVersion)) {
  339. return $this->serverVersion;
  340. }
  341. $conn = $this->conn->getWrappedConnection();
  342. if ($conn instanceof ServerInfoAwareConnection) {
  343. return $this->serverVersion = $conn->getServerVersion();
  344. }
  345. return $this->serverVersion = '0';
  346. }
  347. private function addTableToSchema(Schema $schema): void
  348. {
  349. $types = [
  350. 'mysql' => 'binary',
  351. 'sqlite' => 'text',
  352. ];
  353. $table = $schema->createTable($this->table);
  354. $table->addColumn($this->idCol, $types[$this->getPlatformName()] ?? 'string', ['length' => 255]);
  355. $table->addColumn($this->dataCol, 'blob', ['length' => 16777215]);
  356. $table->addColumn($this->lifetimeCol, 'integer', ['unsigned' => true, 'notnull' => false]);
  357. $table->addColumn($this->timeCol, 'integer', ['unsigned' => true]);
  358. $table->setPrimaryKey([$this->idCol]);
  359. }
  360. }