<?php

namespace Alexmg86\LaravelSubQuery;

use Closure;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Concerns\QueriesRelationships;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Relations\HasManyThrough;
use Illuminate\Database\Query\Expression;
use Illuminate\Support\Str;

class LaravelSubQuery extends Builder
{
    use QueriesRelationships;

    /**
     * The relationship sums that should be eager loaded on every query.
     *
     * @var array
     */
    protected $withSum = [];

    /**
     * The relationship min value that should be eager loaded on every query.
     *
     * @var array
     */
    protected $withMin = [];

    /**
     * The relationship max value that should be eager loaded on every query.
     *
     * @var array
     */
    protected $withMax = [];

    /**
     * The relationship avg value that should be eager loaded on every query.
     *
     * @var array
     */
    protected $withAvg = [];

    public function withSum($relations)
    {
        return $this->withSubQuery($relations, 'sum');
    }

    public function withMin($relations)
    {
        return $this->withSubQuery($relations, 'min');
    }

    public function withMax($relations)
    {
        return $this->withSubQuery($relations, 'max');
    }

    public function withAvg($relations)
    {
        return $this->withSubQuery($relations, 'avg');
    }

    public function orderByRelation($relations, $orderType = 'desc', $type = 'max')
    {
        if (is_array($relations)) {
            $orderType = isset($relations[0]) ? $relations[0] : $orderType;
            $type = isset($relations[1]) ? $relations[1] : $type;
            unset($relations[0], $relations[1]);
        }

        $column = is_array($relations) ? array_key_first($relations) : $relations;
        if (! strpos($column, ':')) {
            return $this->orderBy($column, $orderType);
        }

        return $this->withSubQuery($relations, $type, $orderType);
    }

    protected function withSubQuery($relations, $type, $orderType = null)
    {
        if (empty($relations)) {
            return $this;
        }

        if (is_null($this->query->columns)) {
            $this->query->select([$this->query->from . '.*']);
        }

        $relations = is_array($relations) ? $relations : array_slice(func_get_args(), 0, 1);

        foreach ($this->parseForSubQueryRelations($relations) as $name => $constraints) {
            $segments = explode(' ', $name);

            unset($alias);

            if (count($segments) === 3 && Str::lower($segments[1]) === 'as') {
                [$name, $alias] = [$segments[0], $segments[2]];
            }

            $nameExplode = explode(':', $name);
            $name = $nameExplode[0];
            $columns = isset($nameExplode[1]) ? explode(',', $nameExplode[1]) : [];

            $relation = $this->getRelationWithoutConstraints($name);

            // Here we will get the relationship sum query and prepare to add it to the main query
            // as a sub-select. First, we'll get the "has" query and use that to get the relation
            // sum query. We will normalize the relation name then append _{column}_sum as the name.
            foreach ($columns as $column) {
                $query = $relation->getRelationExistenceQuery(
                    $relation->getRelated()->newQuery(),
                    $this,
                    new Expression('' . $type . '(' . $column . ')')
                )->setBindings([], 'select');

                $query->callScope($constraints);

                $query = $query->mergeConstraintsFrom($relation->getQuery())->toBase();

                $query->orders = null;

                $query->setBindings([], 'order');

                if (count($query->columns) > 1) {
                    $query->columns = [$query->columns[0]];

                    $query->bindings['select'] = [];
                }

                // Finally we will add the proper result column alias to the query and run the subselect
                // statement against the query builder. Then we will return the builder instance back
                // to the developer for further constraint chaining that needs to take place on it.
                $column = $alias ?? Str::snake($name . '_' . $column . '_' . $type);

                if (strpos($this->getSql($this), $this->getSql($query)) === false) {
                    $this->selectSub($query, $column);
                }

                // Add sorting
                if ($orderType) {
                    $this->orderBy($column, $orderType);
                }
            }
        }

        return $this;
    }

    /**
     * Convert the request to a full one.
     * @param  [object] $builder
     * @return [string]
     */
    protected function getSql($builder)
    {
        $sql = $builder->toSql();
        $bindings = $builder->getBindings();
        foreach ($bindings as $binding) {
            $value = is_numeric($binding) ? $binding : "'" . $binding . "'";
            $sql = preg_replace('/\?/', $value, $sql, 1);
        }

        return $sql;
    }

    /**
     * Parse a list of relations into individuals.
     *
     * @param  array  $relations
     * @return array
     */
    protected function parseForSubQueryRelations(array $relations)
    {
        $results = [];

        foreach ($relations as $name => $constraints) {
            // If the "name" value is a numeric key, we can assume that no constraints
            // have been specified. We will just put an empty Closure there so that
            // we can treat these all the same while we are looping through them.
            if (is_numeric($name)) {
                $name = $constraints;

                [$name, $constraints] = [$name, static function () {
                }];
            }

            $results[$name] = $constraints;
        }

        return $results;
    }

    public function setWithSum($withSum)
    {
        return $this->withSum($withSum);
    }

    public function setWithMin($withMin)
    {
        return $this->withMin($withMin);
    }

    public function setWithMax($withMax)
    {
        return $this->withMax($withMax);
    }

    public function setWithAvg($withAvg)
    {
        return $this->withAvg($withAvg);
    }

    /**
     * Eager load the relationships for the models.
     * Overwriting Illuminate\Database\Eloquent\Builder@eagerLoadRelations
     *
     * @param  array  $models
     * @return array
     */
    public function eagerLoadRelationsOne(array $models, string $type)
    {
        foreach ($this->eagerLoad as $name => $constraints) {
            if (strpos($name, '.') === false) {
                $models = $this->eagerLoadRelationOne($models, $name, $constraints, $type);
            }
        }

        return $models;
    }

    /**
     * Eagerly load the relationship on a set of models.
     * Overwriting Illuminate\Database\Eloquent\Builder@eagerLoadRelation
     *
     * @param  array  $models
     * @param  string  $name
     * @param  \Closure  $constraints
     * @return array
     */
    protected function eagerLoadRelationOne(array $models, $name, Closure $constraints, string $type)
    {
        $relation = $this->getRelation($name);

        $relation->addEagerConstraints($models);

        $constraints($relation);

        $parseData = $this->parseRelationToKeys($relation, $type);

        return $relation->match(
            $relation->initRelation($models, $name),
            $relation
            ->whereIn($parseData[0], $parseData[1])
            ->getEager(),
            $name
        );
    }

    /**
     * Getting only need ids
     * @param  object $relation
     * @param  string $type
     * @return array
     */
    private function parseRelationToKeys(object $relation, string $type)
    {
        $relationCopy = clone $relation;

        $typeRelation = get_class($relationCopy);
        switch (get_class($relationCopy)) {
            case HasMany::class:
                $maxKey = $asKey = $relation->getLocalKeyName();
                $groupby = $relation->getForeignKeyName();
                break;
            case BelongsToMany::class:
                $maxKey = $asKey = $relation->getRelatedKeyName();
                $groupby = $relation->getRelated()->getTable() . '.' . $relation->getForeignPivotKeyName();
                break;
            case HasManyThrough::class:
                $asKey = $relation->getSecondLocalKeyName();
                $maxKey = $relation->getRelated()->getTable() . '.' . $asKey;
                $groupby = $relation->getParent()->getTable() . '.' . $relation->getFirstKeyName();
                break;
        }

        return [$maxKey, $relationCopy->selectRaw("$type($maxKey) as $asKey")->groupby($groupby)->pluck($asKey)];
    }
}
