GetKeywords.php 8.72 KB
<?php

namespace App\Service\Requests\Direct;

use App\Jobs\ProcessCallLimitedAPI;
use App\Models\AdGroup;
use App\Models\Campaigns;
use App\Models\Keyword;
use App\Models\Pivots\GoalAdGroup;
use App\Models\Pivots\GoalKeyword;
use App\Service\Contract\APIRequest;
use App\Service\Requests\DirectRequest;
use Carbon\Carbon;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Log;

class GetKeywords extends DirectRequest
{
    protected $max_count = 10000;
    protected $max_count_CampaignIds = 10;
    protected $max_count_AdGroupIds = 1000;
    protected $max_count_Ids = 10000;

    function call($params = null)
    {
        $this->requestPrepare($params);
        $process = new ProcessCallLimitedAPI($this);
        dispatch($process)->onQueue('limits');
    }

    public function getObjectsCount()
    {
        $params = $this->getParams();
//        if (isset($params['SelectionCriteria']['CampaignIds'])) {
//            return -1;
//        }
//        if (isset($params['SelectionCriteria']['AdGroupIds'])) {
//            return -1;
//        }
        if (isset($params['SelectionCriteria']['Ids'])) {
            return count($params['SelectionCriteria']['Ids']);
        }
        return $this->getMaxCount();
    }

    public function slice($maxObjects): ?APIRequest
    {
        $params = $this->getParams();

        if (isset($params['SelectionCriteria']['Ids'])) {
            return $this->sliceByKey($maxObjects, ['SelectionCriteria', 'Ids']);
        }

        return null;
    }

    function handle($response)
    {

        $connection = config('database.default');
        $now = $connection=='sqlite'? DB::raw("strftime('%Y-%m-%d %H:%M:%S', datetime('now'))") : DB::raw('now()');

        try {
            $external_ids = [];

            if (!$this->getToken()->isMain()) {
                return;//добавлять фразы будем только по основному. По цеелвым будем наоборот выгружать.
            }

            if (!isset($response['result']['Keywords'])) {
                return;
            }

            foreach ($response['result']['Keywords'] as $keyword) {
                $adGroupId = (string)$keyword['AdGroupId'];

                if (isset($external_ids[$adGroupId])) {
                    continue;
                }
                $external_ids[$adGroupId] = true;
            }

            if (!count($external_ids)) {
                return;
            }

            if ($this->getToken()->isMain()) {
                $ad_groups = AdGroup::whereIn('external_id', array_keys($external_ids))
                    ->get()
                    ->keyBy('external_id');
            }

//            $ids = [];
//            $campaign_ids_synced_need = [];
            $insertData = [];

            $cnt = 0;
            $items = [];
            foreach ($response['result']['Keywords'] as $keyword) {

                $ad_group = $ad_groups->get((string)$keyword['AdGroupId']);

                if (!$ad_group) {
                    continue;
                }

                $external_id = (string)$keyword['Id'];

                $data = [
                    'external_id' => $external_id,
                    'campaign_external_id' => $ad_group->campaign_external_id,
                    'ad_group_external_id' => $ad_group->external_id,
                    'ad_group_id' => $ad_group->getKey(),
                    'campaign_id' => $ad_group->campaign_id,
                    'keyword' => $keyword['Keyword'],
                    'user_param_1' => $keyword['UserParam1'] ?? null,
                    'user_param_2' => $keyword['UserParam2'] ?? null,
                    'bid' => $keyword['Bid'],
                    'context_bid' => $keyword['ContextBid'],
                    'strategy_priority' => $keyword['StrategyPriority'] ?? null,
                    'state' => $keyword['State'],
                    'status' => $keyword['Status'],
                    'serving_status' => $keyword['ServingStatus'],
                    'updated_at' => $now,
                    'deleted_at' => null //не забыть убрать признак удаления, если вдруг опять пришла удаленная ранее фраза
                ];

                $keyword_data = Keyword::where('external_id', $external_id)->get()->first();
                if ($keyword_data){
                    if (!$keyword_data->equals($data)){
                        //если данные изменились, то надо обновить и в кеше и пометить на то что надо их выгрузить
                        $k_upd = Keyword::where('external_id', $external_id);
                        unset($data['external_id']);
                        $k_upd->update($data);
                        $k_upd->get()->first()->goalKeywords()->update(['updated_need' => $now]);
                    }else{
                        $items[] = $external_id;
                    }
                } else{
                    //новый фразы
                    $insertData[$external_id] = $data;
                }

            }

            Keyword::whereIn('external_id', $items)->update([
                'updated_at'    =>  $now
            ]);

            foreach (array_chunk($insertData, 1000) as $data) {
                $items = [];
                foreach ($data as $item){
                    $items[] = $item['external_id'];
                }
                Keyword::insertOrIgnore($data);
                Keyword::whereIn('external_id', $items)->update([
                    'updated_at' => $now
                ]);
            }

            //удаление будет толко когда получаем изменения по группам, иначе это либо загрузка все фраз либо как то избранных
            if  ( !empty($this->getParams()['SelectionCriteria']['AdGroupIds']) &&
                !isset($response['result']['LimitedBy']) ){
                $ag_groups = $this->getParams()['SelectionCriteria']['AdGroupIds'];
                //последний блок с данными, можно удалять те, что так и не получили
                //Выбираем все те фразы, которые обновлены позже чем группа.
                //это означает что этой фразы не было в результатах и в БД она по этой причине не обновилась
                //надо такие пометить на удаление
                //при удалении для всех таких надо будет удалить фразы из целевых и потом удалить их сами
                if ($connection=='sqlite'){
                    $sql = "UPDATE keywords
                        SET deleted_at = {$now->getValue()}
                        WHERE updated_at<=(SELECT ag.keywords_loaded_at FROM ad_groups ag WHERE keywords.ad_group_id=ag.id)
                            AND deleted_at is null
                            AND ad_group_external_id in (" . implode(", ", $ag_groups) . ")";
                } else{
                    $sql = "UPDATE keywords k
                                INNER JOIN ad_groups ag ON k.ad_group_id=ag.id
                        SET k.deleted_at = {$now->getValue()}
                        WHERE k.updated_at<=ag.keywords_loaded_at
                            AND k.deleted_at is null
                            AND ag.external_id in (" . implode(", ", $ag_groups) . ")";
                }

                DB::update($sql);
                AdGroup::whereIn('external_id', $ag_groups)->update(['keywords_loaded_at'   =>  $now]);
            }
        } catch (\Exception $e) {
            Log::debug($e);
            throw $e;
        }
    }

    private function requestPrepare($filter)
    {
        $this->setService('Keywords');
        $this->setMethod('get');
        $params = [
            "FieldNames" => [
                "Id", "Keyword", "AdGroupId", "CampaignId",
                "UserParam1", "UserParam2", "Bid",
                "ContextBid", "State", "Status", "ServingStatus",
            ],
        ];
        if (isset($filter['CampaignIds'])) {
            $params['SelectionCriteria'] = [
                'CampaignIds' => $filter['CampaignIds'],
            ];
        }
        if (isset($filter['AdGroupIds'])) {
            $params['SelectionCriteria'] = [
                'AdGroupIds' => $filter['AdGroupIds'],
            ];
        }
        if (isset($filter['Ids'])) {

            $this->max_count = $this->max_count_Ids;

            $params['SelectionCriteria'] = [
                'Ids' => $filter['Ids'],
            ];
        }
        $this->setParams($params);
    }
}