vendor/contao/core-bundle/src/Resources/contao/library/Contao/Database/Statement.php line 247

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of Contao.
  4.  *
  5.  * (c) Leo Feyer
  6.  *
  7.  * @license LGPL-3.0-or-later
  8.  */
  9. namespace Contao\Database;
  10. use Contao\Database;
  11. use Doctrine\DBAL\Connection;
  12. use Doctrine\DBAL\Driver\Result as DoctrineResult;
  13. use Doctrine\DBAL\Exception\DriverException;
  14. /**
  15.  * Create and execute queries
  16.  *
  17.  * The class creates the database queries replacing the wildcards and escaping
  18.  * the values. It then executes the query and returns a result object.
  19.  *
  20.  * Usage:
  21.  *
  22.  *     $db = Database::getInstance();
  23.  *     $stmt = $db->prepare("SELECT * FROM tl_member WHERE city=?");
  24.  *     $stmt->limit(10);
  25.  *     $res = $stmt->execute('London');
  26.  *
  27.  * @property string  $query        The query string
  28.  * @property string  $error        The last error message
  29.  * @property integer $affectedRows The number of affected rows
  30.  * @property integer $insertId     The last insert ID
  31.  */
  32. class Statement
  33. {
  34.     /**
  35.      * Connection ID
  36.      * @var Connection
  37.      */
  38.     protected $resConnection;
  39.     /**
  40.      * Database statement
  41.      * @var DoctrineResult
  42.      */
  43.     protected $statement;
  44.     /**
  45.      * Query string
  46.      * @var string
  47.      */
  48.     protected $strQuery;
  49.     /**
  50.      * @var array
  51.      */
  52.     private $arrSetParams = array();
  53.     /**
  54.      * @var array
  55.      */
  56.     private $arrLastUsedParams = array();
  57.     /**
  58.      * Validate the connection resource and store the query string
  59.      *
  60.      * @param Connection $resConnection The connection resource
  61.      */
  62.     public function __construct(Connection $resConnection)
  63.     {
  64.         $this->resConnection $resConnection;
  65.     }
  66.     /**
  67.      * Return an object property
  68.      *
  69.      * @param string $strKey The property name
  70.      *
  71.      * @return mixed|null The property value or null
  72.      */
  73.     public function __get($strKey)
  74.     {
  75.         switch ($strKey)
  76.         {
  77.             case 'query':
  78.                 return $this->strQuery;
  79.             case 'affectedRows':
  80.                 return $this->statement->rowCount();
  81.             case 'insertId':
  82.                 return $this->resConnection->lastInsertId();
  83.         }
  84.         return null;
  85.     }
  86.     /**
  87.      * Prepare a query string so the following functions can handle it
  88.      *
  89.      * @param string $strQuery The query string
  90.      *
  91.      * @return Statement The statement object
  92.      *
  93.      * @throws \Exception If $strQuery is empty
  94.      */
  95.     public function prepare($strQuery)
  96.     {
  97.         if (!$strQuery)
  98.         {
  99.             throw new \Exception('Empty query string');
  100.         }
  101.         $this->strQuery trim($strQuery);
  102.         $this->arrLastUsedParams = array();
  103.         return $this;
  104.     }
  105.     /**
  106.      * Autogenerate the SET/VALUES subpart of a query from an associative array
  107.      *
  108.      * Usage:
  109.      *
  110.      *     $set = array(
  111.      *         'firstname' => 'Leo',
  112.      *         'lastname'  => 'Feyer'
  113.      *     );
  114.      *     $stmt->prepare("UPDATE tl_member %s")->set($set);
  115.      *
  116.      * @param array $arrParams The associative array
  117.      *
  118.      * @return Statement The statement object
  119.      */
  120.     public function set($arrParams)
  121.     {
  122.         if (substr_count($this->strQuery'%s') !== || !\in_array(strtoupper(substr($this->strQuery06)), array('INSERT''UPDATE'), true))
  123.         {
  124.             trigger_deprecation('contao/core-bundle''4.13''Using "%s()" is only supported for INSERT and UPDATE queries with the "%%s" placeholder. This will throw an exception in Contao 5.0.'__METHOD__);
  125.             return $this;
  126.         }
  127.         $this->arrSetParams array_values($arrParams);
  128.         $arrParamNames array_map(
  129.             static function ($strName)
  130.             {
  131.                 if (!preg_match('/^(?:[A-Za-z0-9_$]+|`[^`]+`)$/'$strName))
  132.                 {
  133.                     throw new \RuntimeException(sprintf('Invalid column name "%s" in %s()'$strName__METHOD__));
  134.                 }
  135.                 return Database::quoteIdentifier($strName);
  136.             },
  137.             array_keys($arrParams)
  138.         );
  139.         // INSERT
  140.         if (strncasecmp($this->strQuery'INSERT'6) === 0)
  141.         {
  142.             $strQuery sprintf(
  143.                 '(%s) VALUES (%s)',
  144.                 implode(', '$arrParamNames),
  145.                 implode(', 'array_fill(0\count($arrParams), '?'))
  146.             );
  147.         }
  148.         // UPDATE
  149.         else
  150.         {
  151.             if (!$arrParamNames)
  152.             {
  153.                 throw new \InvalidArgumentException('Set array must not be empty for UPDATE queries');
  154.             }
  155.             $strQuery 'SET ' implode('=?, '$arrParamNames) . '=?';
  156.         }
  157.         $this->strQuery str_replace('%s'$strQuery$this->strQuery);
  158.         return $this;
  159.     }
  160.     /**
  161.      * Handle limit and offset
  162.      *
  163.      * @param integer $intRows   The maximum number of rows
  164.      * @param integer $intOffset The number of rows to skip
  165.      *
  166.      * @return Statement The statement object
  167.      */
  168.     public function limit($intRows$intOffset=0)
  169.     {
  170.         if ($intRows <= 0)
  171.         {
  172.             $intRows 30;
  173.         }
  174.         if ($intOffset 0)
  175.         {
  176.             $intOffset 0;
  177.         }
  178.         if (strncasecmp($this->strQuery'SELECT'6) === 0)
  179.         {
  180.             $this->strQuery .= ' LIMIT ' $intOffset ',' $intRows;
  181.         }
  182.         else
  183.         {
  184.             $this->strQuery .= ' LIMIT ' $intRows;
  185.         }
  186.         return $this;
  187.     }
  188.     /**
  189.      * Execute the query and return the result object
  190.      *
  191.      * @return Result The result object
  192.      */
  193.     public function execute()
  194.     {
  195.         $arrParams \func_get_args();
  196.         if (\count($arrParams) === && \is_array($arrParams[0]))
  197.         {
  198.             trigger_deprecation('contao/core-bundle''4.13''Using "%s()" with an array parameter has been deprecated and will no longer work in Contao 5.0. Use argument unpacking via ... instead."'__METHOD__);
  199.             $arrParams array_values($arrParams[0]);
  200.         }
  201.         return $this->query(''array_merge($this->arrSetParams$arrParams));
  202.     }
  203.     /**
  204.      * Directly send a query string to the database
  205.      *
  206.      * @param string $strQuery The query string
  207.      *
  208.      * @return Result|Statement The result object or the statement object if there is no result set
  209.      *
  210.      * @throws \Exception If the query string is empty
  211.      */
  212.     public function query($strQuery='', array $arrParams = array(), array $arrTypes = array())
  213.     {
  214.         if (!empty($strQuery))
  215.         {
  216.             $this->strQuery trim($strQuery);
  217.         }
  218.         // Make sure there is a query string
  219.         if (!$this->strQuery)
  220.         {
  221.             throw new \Exception('Empty query string');
  222.         }
  223.         $arrParams array_map(
  224.             static function ($varParam)
  225.             {
  226.                 if (\is_string($varParam) || \is_bool($varParam) || \is_float($varParam) || \is_int($varParam) || $varParam === null)
  227.                 {
  228.                     return $varParam;
  229.                 }
  230.                 return serialize($varParam);
  231.             },
  232.             $arrParams
  233.         );
  234.         $this->arrLastUsedParams $arrParams;
  235.         // Execute the query
  236.         // TODO: remove the try/catch block in Contao 5.0
  237.         try
  238.         {
  239.             $this->statement $this->resConnection->executeQuery($this->strQuery$arrParams$arrTypes);
  240.         }
  241.         catch (DriverException|\ArgumentCountError $exception)
  242.         {
  243.             // SQLSTATE[HY000]: This command is not supported in the prepared statement protocol
  244.             if ($exception->getCode() === 1295)
  245.             {
  246.                 $this->resConnection->executeStatement($this->strQuery$arrParams$arrTypes);
  247.                 trigger_deprecation('contao/core-bundle''4.13''Using "%s()" for statements (instead of queries) has been deprecated and will no longer work in Contao 5.0. Use "%s::executeStatement()" instead.'__METHOD__Connection::class);
  248.                 return $this;
  249.             }
  250.             if (!$arrParams)
  251.             {
  252.                 throw $exception;
  253.             }
  254.             $intTokenCount substr_count(preg_replace("/('[^']*')/"''$this->strQuery), '?');
  255.             if (\count($arrParams) <= $intTokenCount)
  256.             {
  257.                 throw $exception;
  258.             }
  259.             // If we get here, there are more parameters than tokens, so we slice the array and try to execute the query again
  260.             $this->statement $this->resConnection->executeQuery($this->strQuery\array_slice($arrParams0$intTokenCount), $arrTypes);
  261.             // Only trigger the deprecation if the parameter count was the reason for the exception and the previous call did not throw
  262.             if ($this->arrLastUsedParams === array(null))
  263.             {
  264.                 trigger_deprecation('contao/core-bundle''4.13''Using "%s::execute(null)" has been deprecated and will no longer work in Contao 5.0. Omit the NULL parameters instead.'__CLASS__);
  265.             }
  266.             else
  267.             {
  268.                 trigger_deprecation('contao/core-bundle''4.13''Passing more parameters than "?" tokens has been deprecated and will no longer work in Contao 5.0. Use the correct number of parameters instead.');
  269.             }
  270.         }
  271.         // No result set available
  272.         if ($this->statement->columnCount() < 1)
  273.         {
  274.             return $this;
  275.         }
  276.         // Instantiate a result object
  277.         return new Result($this->statement$this->strQuery);
  278.     }
  279.     /**
  280.      * Replace the wildcards in the query string
  281.      *
  282.      * @param array $arrValues The values array
  283.      *
  284.      * @throws \Exception If $arrValues has too few values to replace the wildcards in the query string
  285.      *
  286.      * @deprecated Deprecated since Contao 4.13, to be removed in Contao 5.0.
  287.      */
  288.     protected function replaceWildcards($arrValues)
  289.     {
  290.         trigger_deprecation('contao/core-bundle''4.13''Using "%s()" has been deprecated and will no longer work in Contao 5.0.'__METHOD__);
  291.         $arrValues $this->escapeParams($arrValues);
  292.         $this->strQuery preg_replace('/(?<!%)%([^bcdufosxX%])/''%%$1'$this->strQuery);
  293.         // Replace wildcards
  294.         if (!$this->strQuery = @vsprintf($this->strQuery$arrValues))
  295.         {
  296.             throw new \Exception('Too few arguments to build the query string');
  297.         }
  298.     }
  299.     /**
  300.      * Escape the values and serialize objects and arrays
  301.      *
  302.      * @param array $arrValues The values array
  303.      *
  304.      * @return array The array with the escaped values
  305.      *
  306.      * @deprecated Deprecated since Contao 4.13, to be removed in Contao 5.0.
  307.      */
  308.     protected function escapeParams($arrValues)
  309.     {
  310.         trigger_deprecation('contao/core-bundle''4.13''Using "%s()" has been deprecated and will no longer work in Contao 5.0.'__METHOD__);
  311.         foreach ($arrValues as $k=>$v)
  312.         {
  313.             switch (\gettype($v))
  314.             {
  315.                 case 'string':
  316.                     $arrValues[$k] = $this->resConnection->quote($v);
  317.                     break;
  318.                 case 'boolean':
  319.                     $arrValues[$k] = ($v === true) ? 0;
  320.                     break;
  321.                 case 'object':
  322.                 case 'array':
  323.                     $arrValues[$k] = $this->resConnection->quote(serialize($v));
  324.                     break;
  325.                 default:
  326.                     $arrValues[$k] = $v ?? 'NULL';
  327.                     break;
  328.             }
  329.         }
  330.         return $arrValues;
  331.     }
  332.     /**
  333.      * Explain the current query
  334.      *
  335.      * @return string The explanation string
  336.      *
  337.      * @deprecated Deprecated since Contao 4.13, to be removed in Contao 5.0.
  338.      */
  339.     public function explain()
  340.     {
  341.         trigger_deprecation('contao/core-bundle''4.13''Using "%s()" has been deprecated and will no longer work in Contao 5.0.'__METHOD__);
  342.         return $this->resConnection->fetchAssociative('EXPLAIN ' $this->strQuery$this->arrLastUsedParams);
  343.     }
  344.     /**
  345.      * Bypass the cache and always execute the query
  346.      *
  347.      * @return Result The result object
  348.      *
  349.      * @deprecated Deprecated since Contao 4.0, to be removed in Contao 5.0.
  350.      *             Use Statement::execute() instead.
  351.      */
  352.     public function executeUncached()
  353.     {
  354.         trigger_deprecation('contao/core-bundle''4.0''Using "Contao\Statement::executeUncached()" has been deprecated and will no longer work in Contao 5.0. Use "Contao\Statement::execute()" instead.');
  355.         return \call_user_func_array(array($this'execute'), \func_get_args());
  356.     }
  357.     /**
  358.      * Always execute the query and add or replace an existing cache entry
  359.      *
  360.      * @return Result The result object
  361.      *
  362.      * @deprecated Deprecated since Contao 4.0, to be removed in Contao 5.0.
  363.      *             Use Statement::execute() instead.
  364.      */
  365.     public function executeCached()
  366.     {
  367.         trigger_deprecation('contao/core-bundle''4.0''Using "Contao\Statement::executeCached()" has been deprecated and will no longer work in Contao 5.0. Use "Contao\Statement::execute()" instead.');
  368.         return \call_user_func_array(array($this'execute'), \func_get_args());
  369.     }
  370. }
  371. class_alias(Statement::class, 'Database\Statement');