123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- <?php
- /**
- * @link https://www.yiiframework.com/
- * @copyright Copyright (c) 2008 Yii Software LLC
- * @license https://www.yiiframework.com/license/
- */
- namespace yii\db\pgsql;
- use yii\db\ArrayExpression;
- use yii\db\ExpressionBuilderInterface;
- use yii\db\ExpressionBuilderTrait;
- use yii\db\ExpressionInterface;
- use yii\db\JsonExpression;
- use yii\db\Query;
- /**
- * Class ArrayExpressionBuilder builds [[ArrayExpression]] for PostgreSQL DBMS.
- *
- * @author Dmytro Naumenko <d.naumenko.a@gmail.com>
- * @since 2.0.14
- */
- class ArrayExpressionBuilder implements ExpressionBuilderInterface
- {
- use ExpressionBuilderTrait;
- /**
- * {@inheritdoc}
- * @param ArrayExpression|ExpressionInterface $expression the expression to be built
- */
- public function build(ExpressionInterface $expression, array &$params = [])
- {
- $value = $expression->getValue();
- if ($value === null) {
- return 'NULL';
- }
- if ($value instanceof Query) {
- list ($sql, $params) = $this->queryBuilder->build($value, $params);
- return $this->buildSubqueryArray($sql, $expression);
- }
- $placeholders = $this->buildPlaceholders($expression, $params);
- return 'ARRAY[' . implode(', ', $placeholders) . ']' . $this->getTypehint($expression);
- }
- /**
- * Builds placeholders array out of $expression values
- * @param ExpressionInterface|ArrayExpression $expression
- * @param array $params the binding parameters.
- * @return array
- */
- protected function buildPlaceholders(ExpressionInterface $expression, &$params)
- {
- $value = $expression->getValue();
- $placeholders = [];
- if ($value === null || !is_array($value) && !$value instanceof \Traversable) {
- return $placeholders;
- }
- if ($expression->getDimension() > 1) {
- foreach ($value as $item) {
- $placeholders[] = $this->build($this->unnestArrayExpression($expression, $item), $params);
- }
- return $placeholders;
- }
- foreach ($value as $item) {
- if ($item instanceof Query) {
- list ($sql, $params) = $this->queryBuilder->build($item, $params);
- $placeholders[] = $this->buildSubqueryArray($sql, $expression);
- continue;
- }
- $item = $this->typecastValue($expression, $item);
- if ($item instanceof ExpressionInterface) {
- $placeholders[] = $this->queryBuilder->buildExpression($item, $params);
- continue;
- }
- $placeholders[] = $this->queryBuilder->bindParam($item, $params);
- }
- return $placeholders;
- }
- /**
- * @param ArrayExpression $expression
- * @param mixed $value
- * @return ArrayExpression
- */
- private function unnestArrayExpression(ArrayExpression $expression, $value)
- {
- $expressionClass = get_class($expression);
- return new $expressionClass($value, $expression->getType(), $expression->getDimension()-1);
- }
- /**
- * @param ArrayExpression $expression
- * @return string the typecast expression based on [[type]].
- */
- protected function getTypehint(ArrayExpression $expression)
- {
- if ($expression->getType() === null) {
- return '';
- }
- $result = '::' . $expression->getType();
- $result .= str_repeat('[]', $expression->getDimension());
- return $result;
- }
- /**
- * Build an array expression from a subquery SQL.
- *
- * @param string $sql the subquery SQL.
- * @param ArrayExpression $expression
- * @return string the subquery array expression.
- */
- protected function buildSubqueryArray($sql, ArrayExpression $expression)
- {
- return 'ARRAY(' . $sql . ')' . $this->getTypehint($expression);
- }
- /**
- * Casts $value to use in $expression
- *
- * @param ArrayExpression $expression
- * @param mixed $value
- * @return JsonExpression
- */
- protected function typecastValue(ArrayExpression $expression, $value)
- {
- if ($value instanceof ExpressionInterface) {
- return $value;
- }
- if (in_array($expression->getType(), [Schema::TYPE_JSON, Schema::TYPE_JSONB], true)) {
- return new JsonExpression($value);
- }
- return $value;
- }
- }
|