<?php

namespace AnourValar\EloquentSerialize\Grammars;

trait EloquentBuilderGrammar
{
    /**
     * Serialize state for \Illuminate\Database\Eloquent\Builder
     *
     * @param \Illuminate\Database\Eloquent\Builder $builder
     * @return array
     */
    protected function packEloquentBuilder(\Illuminate\Database\Eloquent\Builder $builder): array
    {
        return [
            'with' => $this->getEagers($builder), // preloaded ("eager") relations
            'removed_scopes' => $builder->removedScopes(), // global scopes
            'casts' => $builder->getModel()->getCasts(),
        ];
    }

    /**
     * Unserialize state for \Illuminate\Database\Eloquent\Builder
     *
     * @param array $data
     * @param \Illuminate\Database\Eloquent\Builder $builder
     * @return void
     */
    protected function unpackEloquentBuilder(array $data, \Illuminate\Database\Eloquent\Builder &$builder): void
    {
        // Preloaded ("eager") relations
        $this->setEagers($builder, $data['with']);

        // Global scopes
        if ($data['removed_scopes']) {
            $builder->withoutGlobalScopes($data['removed_scopes']);
        }

        // Casts
        if (method_exists($builder->getModel(), 'mergeCasts')) { // old versions support
            $builder->getModel()->mergeCasts($data['casts']);
        }
    }

    /**
     * @param \Illuminate\Database\Eloquent\Builder $builder
     * @return array
     */
    private function getEagers(\Illuminate\Database\Eloquent\Builder $builder): array
    {
        $result = [];

        foreach ($builder->getEagerLoads() as $name => $value) {
            $relation = $builder;
            foreach (explode('.', $name) as $part) {
                $relation = $relation->getRelation($part); // get a relation without "constraints"
            }
            $referenceRelation = clone $relation;

            $value($relation); // apply closure
            $result[$name] = [
                'query' => $this->packQueryBuilder($relation->getQuery()->getQuery()),
                'eloquent' => $this->packEloquentBuilder($relation->getQuery()),
                'extra' => $this->relationExtraData($relation) ? $relation->exportExtraParametersForSerialize() : null,
            ];

            $relation->getQuery()->getModel()->newInstance()->with($name)->getEagerLoads()[$name]($referenceRelation);
            $this->cleanStaticConstraints($result[$name]['query'], $this->packQueryBuilder($referenceRelation->getQuery()->getQuery()));
        }

        return $result;
    }

    /**
     * @param \Illuminate\Database\Eloquent\Builder $builder
     * @param array $eagers
     * @return void
     */
    private function setEagers(\Illuminate\Database\Eloquent\Builder $builder, array $eagers): void
    {
        foreach ($eagers as &$value) {
            $value = function ($query) use ($value) {
                if (isset($value['extra'])) {
                    $query->importExtraParametersForSerialize($value['extra']);
                }

                // Input argument may be different depends on context
                while (! ($query instanceof \Illuminate\Database\Eloquent\Builder)) {
                    $query = $query->getQuery();
                }
                if (isset($value['eloquent'])) {
                    $this->unpackEloquentBuilder($value['eloquent'], $query);
                }

                // Input argument may be different depends on context
                while (! ($query instanceof \Illuminate\Database\Query\Builder)) {
                    $query = $query->getQuery();
                }

                $this->unpackQueryBuilder(isset($value['query']) ? $value['query'] : $value, $query);
            };
        }
        unset($value);

        $builder->setEagerLoads($eagers);
    }

    /**
     * @param array $packedQueryBuilder
     * @param array $packedReferenceQueryBuilder
     * @return void
     */
    private function cleanStaticConstraints(array &$packedQueryBuilder, array $packedReferenceQueryBuilder): void
    {
        $properties = [
            'aggregate', 'columns', 'distinct', 'wheres', 'groups', 'havings', 'orders', 'limit', 'offset', 'unions',
            'unionLimit', 'unionOffset', 'unionOrders', 'joins', 'groupLimit',
        ];

        foreach ($properties as $property) {
            if (! is_array($packedQueryBuilder[$property] ?? null)) {
                continue;
            }

            foreach ($packedQueryBuilder[$property] as $key => $item) {
                if (in_array($item, (array) ($packedReferenceQueryBuilder[$property] ?? null), true)) {
                    unset($packedQueryBuilder[$property][$key]);
                }
            }
        }

        foreach ($packedQueryBuilder['bindings'] as $binding => $data) {
            if (! is_array($data)) {
                continue; // just in case ;)
            }

            foreach ($data as $key => $value) {
                if (
                    isset($packedReferenceQueryBuilder['bindings'][$binding][$key])
                    && $packedReferenceQueryBuilder['bindings'][$binding][$key] === $value
                ) {
                    unset($packedQueryBuilder['bindings'][$binding][$key]);
                }
            }
        }
    }

    /**
     * @param \Illuminate\Database\Eloquent\Relations\Relation $relation
     * @return bool
     */
    private function relationExtraData(\Illuminate\Database\Eloquent\Relations\Relation $relation): bool
    {
        \Illuminate\Database\Eloquent\Relations\MorphTo::macro('importExtraParametersForSerialize', function (array $params) {
            foreach ($params as $key => $value) {
                $this->$key = $value;
            }
        });

        if ($relation instanceof \Illuminate\Database\Eloquent\Relations\MorphTo) {
            \Illuminate\Database\Eloquent\Relations\MorphTo::macro('exportExtraParametersForSerialize', function () {
                return [
                    'morphableEagerLoads' => $this->morphableEagerLoads,
                    'morphableEagerLoadCounts' => $this->morphableEagerLoadCounts,
                    'morphableConstraints' => $this->morphableConstraints,
                ];
            });

            return true;
        }

        return false;
    }
}
