HEX
Server: Apache/2.4.52 (Ubuntu)
System: Linux WebLive 5.15.0-79-generic #86-Ubuntu SMP Mon Jul 10 16:07:21 UTC 2023 x86_64
User: ubuntu (1000)
PHP: 7.4.33
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: /var/www/html/wpmuhibbah/wp-content/plugins/give/src/Campaigns/Repositories/CampaignRepository.php
<?php

namespace Give\Campaigns\Repositories;

use Exception;
use Give\Campaigns\Models\Campaign;
use Give\Campaigns\ValueObjects\CampaignType;
use Give\Donations\ValueObjects\DonationMetaKeys;
use Give\Framework\Database\DB;
use Give\Framework\Exceptions\Primitives\InvalidArgumentException;
use Give\Framework\Models\ModelQueryBuilder;
use Give\Framework\Support\Facades\DateTime\Temporal;
use Give\Helpers\Hooks;
use Give\Log\Log;

/**
 * @since 4.0.0
 */
class CampaignRepository
{
    /**
     * @since 4.0.0
     *
     * @var string[]
     */
    private $requiredProperties = [
        'title',
        'status',
    ];

    /**
     * @since 4.0.0
     *
     * Get Campaign by ID
     *
     * @since 4.0.0
     */
    public function getById(int $id)
    {
        return $this->queryById($id)->get();
    }

    /**
     * @since 4.0.0
     *
     * Query Campaign by ID
     *
     * @since 4.0.0
     */
    public function queryById(int $id): ModelQueryBuilder
    {
        return $this->prepareQuery()
            ->where('id', $id);
    }

    /**
     * @since 4.0.0
     *
     * Get Campaign by Form ID using a lookup table
     */
    public function getByFormId(int $formId)
    {
        return $this->queryByFormId($formId)->get();
    }

    /**
     * @since 4.0.0
     *
     * @param int $formId
     *
     * @return ModelQueryBuilder<Campaign>
     */
    public function queryByFormId(int $formId): ModelQueryBuilder
    {
        return $this->prepareQuery()
            ->leftJoin('give_campaign_forms', 'campaigns.id', 'forms.campaign_id', 'forms')
            ->where('forms.form_id', $formId);
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception|InvalidArgumentException
     */
    public function insert(Campaign $campaign): void
    {
        $this->validateProperties($campaign);

        Hooks::doAction('givewp_campaign_creating', $campaign);

        $currentDate = Temporal::getCurrentDateTime();

        $dateCreated = Temporal::withoutMicroseconds($campaign->createdAt ?: $currentDate);
        $dateCreatedFormatted = Temporal::getFormattedDateTime($dateCreated);

        $startDate = Temporal::withoutMicroseconds($campaign->startDate ?: $currentDate);
        $startDateFormatted = Temporal::getFormattedDateTime($startDate);

        $endDateFormatted = $campaign->endDate ? Temporal::getFormattedDateTime($campaign->endDate) : $campaign->endDate;

        DB::query('START TRANSACTION');

        try {
            DB::table('give_campaigns')
                ->insert([
                    'campaign_type' => $campaign->type->getValue(),
                    'campaign_title' => wp_strip_all_tags($campaign->title, true),
                    'short_desc' => wp_strip_all_tags($campaign->shortDescription),
                    'long_desc' => wp_strip_all_tags($campaign->longDescription),
                    'campaign_logo' => $campaign->logo,
                    'campaign_image' => $campaign->image,
                    'primary_color' => $campaign->primaryColor,
                    'secondary_color' => $campaign->secondaryColor,
                    'campaign_goal' => $campaign->goal,
                    'goal_type' => $campaign->goalType->getValue(),
                    'status' => $campaign->status->getValue(),
                    'start_date' => $startDateFormatted,
                    'end_date' => $endDateFormatted,
                    'date_created' => $dateCreatedFormatted,
                ]);

            $campaignId = DB::last_insert_id();

        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed creating a campaign', compact('campaign'));

            throw new $exception('Failed creating a campaign');
        }

        DB::query('COMMIT');

        $campaign->id = $campaignId;
        $campaign->createdAt = $dateCreated;
        $campaign->startDate = $startDate;

        Hooks::doAction('givewp_campaign_created', $campaign);
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception|InvalidArgumentException
     */
    public function update(Campaign $campaign): void
    {
        $this->validateProperties($campaign);

        $startDateFormatted = Temporal::getFormattedDateTime($campaign->startDate);
        $endDateFormatted = $campaign->endDate ? Temporal::getFormattedDateTime($campaign->endDate) : $campaign->endDate;

        Hooks::doAction('givewp_campaign_updating', $campaign);

        DB::query('START TRANSACTION');

        try {
            DB::table('give_campaigns')
                ->where('id', $campaign->id)
                ->update([
                    'campaign_type' => $campaign->type->getValue(),
                    'campaign_page_id' => $campaign->pageId,
                    'campaign_title' => wp_strip_all_tags($campaign->title, true),
                    'short_desc' => wp_strip_all_tags($campaign->shortDescription),
                    'long_desc' => wp_strip_all_tags($campaign->longDescription),
                    'campaign_logo' => $campaign->logo,
                    'campaign_image' => $campaign->image,
                    'primary_color' => $campaign->primaryColor,
                    'secondary_color' => $campaign->secondaryColor,
                    'campaign_goal' => $campaign->goal,
                    'goal_type' => $campaign->goalType->getValue(),
                    'status' => $campaign->status->getValue(),
                    'start_date' => $startDateFormatted,
                    'end_date' => $endDateFormatted,
                ]);
        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed updating a campaign', compact('campaign'));

            throw new $exception('Failed updating a campaign');
        }

        DB::query('COMMIT');

        Hooks::doAction('givewp_campaign_updated', $campaign);
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception
     */
    public function addCampaignForm(Campaign $campaign, int $donationFormId, bool $isDefault = false)
    {
        Hooks::doAction('givewp_campaign_form_relationship_creating', $campaign, $donationFormId, $isDefault);

        DB::query('START TRANSACTION');

        try {
            if ($isDefault) {
                DB::table('give_campaigns')
                    ->where('id', $campaign->id)
                    ->update([
                        'form_id' => $donationFormId,
                    ]);

                $campaign->defaultFormId = $donationFormId;
            }

            DB::table('give_campaign_forms')
                ->insert([
                    'form_id' => $donationFormId,
                    'campaign_id' => $campaign->id,
                ]);
        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed creating a campaign form relationship', compact('campaign'));

            throw new $exception('Failed creating a campaign form relationship');
        }

        DB::query('COMMIT');

        Hooks::doAction('givewp_campaign_form_relationship_created', $campaign, $donationFormId, $isDefault);
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception
     */
    public function updateDefaultCampaignForm(Campaign $campaign, int $donationFormId)
    {
        Hooks::doAction('givewp_campaign_default_form_updating', $campaign, $donationFormId);

        DB::query('START TRANSACTION');

        try {
            DB::table('give_campaigns')
                ->where('id', $campaign->id)
                ->update([
                    'form_id' => $donationFormId
                ]);

            $campaign->defaultFormId = $donationFormId;
        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed updating the campaign default form', compact('campaign'));

            throw new $exception('Failed updating the campaign default form');
        }

        DB::query('COMMIT');

        Hooks::doAction('givewp_campaign_default_form_updated', $campaign, $donationFormId);
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception
     */
    public function delete(Campaign $campaign): bool
    {
        DB::query('START TRANSACTION');

        Hooks::doAction('givewp_campaign_deleting', $campaign);

        try {
            DB::table('give_campaigns')
                ->where('id', $campaign->id)
                ->delete();

        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed deleting a campaign', compact('campaign'));

            throw new $exception('Failed deleting a campaign');
        }

        DB::query('COMMIT');

        Hooks::doAction('givewp_campaign_deleted', $campaign);

        return true;
    }

    /**
     * @since 4.0.0
     *
     * @throws Exception
     */
    public function mergeCampaigns(Campaign $destinationCampaign, Campaign ...$campaignsToMerge): bool
    {
        // Make sure the destination campaign ID will not be included into $campaignsToMergeIds
        $campaignsToMergeIds = array_column($campaignsToMerge, 'id');
        if ($key = array_search($destinationCampaign->id, $campaignsToMergeIds, true)) {
            unset($campaignsToMergeIds[$key]);
        }

        Hooks::doAction('givewp_campaigns_merging', $destinationCampaign, $campaignsToMergeIds);

        DB::query('START TRANSACTION');

        try {
            // Convert $campaignsToMergeIds to string to use it in the queries
            $campaignsToMergeIdsString = implode(', ', $campaignsToMergeIds);

            // Migrate revenue entries from campaigns to merge to the destination campaign
            DB::query(
                DB::prepare("UPDATE " . DB::prefix('give_revenue') . " SET campaign_id = %d WHERE campaign_id IN ($campaignsToMergeIdsString)",
                    [
                        $destinationCampaign->id,
                    ])
            );

            // Migrate forms from campaigns to merge to the destination campaign
            DB::query(
                DB::prepare("UPDATE " . DB::prefix('give_campaign_forms') . " SET campaign_id = %d WHERE campaign_id IN ($campaignsToMergeIdsString)",
                    [
                        $destinationCampaign->id,
                    ])
            );

            // Update donations campaign id meta value
            DB::query(
                DB::prepare("UPDATE " . DB::prefix('give_donationmeta') . " SET meta_value = %d WHERE meta_key = %s AND meta_value IN ($campaignsToMergeIdsString)",
                    [
                        $destinationCampaign->id,
                        DonationMetaKeys::CAMPAIGN_ID
                    ])
            );

            // Delete campaigns to merge now that we already migrated the necessary data to the destination campaign
            DB::query("DELETE FROM " . DB::prefix('give_campaigns') . " WHERE id IN ($campaignsToMergeIdsString)");
        } catch (Exception $exception) {
            DB::query('ROLLBACK');

            Log::error('Failed merging campaigns into destination campaign', [
                'campaignsToMergeIds' => $campaignsToMergeIds,
                'destinationCampaign' => compact('destinationCampaign'),
            ]);

            throw new $exception('Failed merging campaigns into destination campaign');
        }

        DB::query('COMMIT');

        Hooks::doAction('givewp_campaigns_merged', $destinationCampaign, $campaignsToMergeIds);

        return true;
    }

    /**
     * @since 4.0.0
     */
    private function validateProperties(Campaign $campaign): void
    {
        foreach ($this->requiredProperties as $key) {
            if ( ! isset($campaign->$key)) {
                throw new InvalidArgumentException("'$key' is required.");
            }
        }
    }

    /**
     * @since 4.0.0
     *
     * @return ModelQueryBuilder<Campaign>
     */
    public function prepareQuery(): ModelQueryBuilder
    {
        $builder = new ModelQueryBuilder(Campaign::class);

        return $builder->from('give_campaigns', 'campaigns')
            ->select(
                ['campaigns.id', 'id'],
                ['campaigns.form_id', 'defaultFormId'], // Prefix the `form_id` column to avoid conflicts with the `give_campaign_forms` table.
                ['campaign_type', 'type'],
                ['campaign_page_id', 'pageId'],
                ['campaign_title', 'title'],
                ['short_desc', 'shortDescription'],
                ['long_desc', 'longDescription'],
                ['campaign_logo', 'logo'],
                ['campaign_image', 'image'],
                ['primary_color', 'primaryColor'],
                ['secondary_color', 'secondaryColor'],
                ['campaign_goal', 'goal'],
                ['goal_type', 'goalType'],
                'status',
                ['start_date', 'startDate'],
                ['end_date', 'endDate'],
                ['date_created', 'createdAt']
            )
            // Select only core campaign
            ->where('campaigns.campaign_type', CampaignType::CORE);
    }
}