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_err/wp-content/plugins/give/src/API/Endpoints/Migrations/RunMigration.php
<?php

namespace Give\API\Endpoints\Migrations;

use Exception;
use Give\Framework\Database\DB;
use Give\Framework\Migrations\Contracts\BatchMigration;
use Give\Framework\Migrations\Contracts\Migration;
use Give\Framework\Migrations\Contracts\ReversibleMigration;
use Give\Framework\Migrations\Controllers\BatchMigrationRunner;
use Give\Framework\Migrations\MigrationsRegister;
use Give\MigrationLog\MigrationLogFactory;
use Give\MigrationLog\MigrationLogStatus;
use WP_REST_Request;
use WP_REST_Response;

/**
 * Class RunMigration
 * @package    Give\API\Endpoints\Migrations
 *
 * @since      4.0.0 run batch migrations
 * @since      2.10.0
 */
class RunMigration extends Endpoint
{
    /**
     * @var MigrationsRegister
     */
    private $migrationRegister;

    /**
     * @var MigrationLogFactory
     */
    private $migrationLogFactory;

    /**
     * RunMigration constructor.
     *
     * @param MigrationsRegister  $migrationsRegister
     * @param MigrationLogFactory $migrationLogFactory
     */
    public function __construct(
        MigrationsRegister $migrationsRegister,
        MigrationLogFactory $migrationLogFactory
    ) {
        $this->migrationRegister = $migrationsRegister;
        $this->migrationLogFactory = $migrationLogFactory;
    }

    /**
     * @inheritDoc
     */
    public function registerRoute()
    {
        register_rest_route(
            'give-api/v2',
            'migrations/run-migration',
            [
                [
                    'methods' => 'POST',
                    'callback' => [$this, 'runMigration'],
                    'permission_callback' => [$this, 'permissionsCheck'],
                    'args' => [
                        'id' => [
                            'type' => 'string',
                            'required' => true,
                        ],
                    ],
                ],
                'schema' => [$this, 'getSchema'],
            ]
        );

        register_rest_route(
            'give-api/v2',
            'migrations/run-batch-migration',
            [
                [
                    'methods' => 'POST',
                    'callback' => [$this, 'runBatchMigration'],
                    'permission_callback' => [$this, 'permissionsCheck'],
                    'args' => [
                        'id' => [
                            'type' => 'string',
                            'required' => true,
                        ],
                    ],
                ],
            ]
        );

        register_rest_route(
            'give-api/v2',
            'migrations/reschedule-failed-actions',
            [
                [
                    'methods' => 'POST',
                    'callback' => [$this, 'rescheduleFailedActions'],
                    'permission_callback' => [$this, 'permissionsCheck'],
                    'args' => [
                        'id' => [
                            'type' => 'string',
                            'required' => true,
                        ],
                    ],
                ],
            ]
        );

        /**
         * @since 4.3.0
         */
        register_rest_route(
            'give-api/v2',
            'migrations/rollback-migration',
            [
                [
                    'methods' => 'POST',
                    'callback' => [$this, 'rollbackMigration'],
                    'permission_callback' => [$this, 'permissionsCheck'],
                    'args' => [
                        'id' => [
                            'type' => 'string',
                            'required' => true,
                            'validate_callback' => function ($param) {
                                $migrationClass = $this->migrationRegister->getMigration($param);

                                return is_subclass_of($migrationClass, ReversibleMigration::class);
                            },
                        ],
                    ],
                ],
                'schema' => [$this, 'getSchema'],
            ]
        );
    }

    /**
     * @return array
     */
    public function getSchema()
    {
        return [
            '$schema' => 'http://json-schema.org/draft-04/schema#',
            'title' => 'logs',
            'type' => 'object',
            'properties' => [
                'id' => [
                    'type' => 'string',
                    'description' => esc_html__('Migration ID', 'give'),
                ],
            ],
        ];
    }

    /**
     * @param WP_REST_Request $request
     *
     * @return WP_REST_Response
     */
    public function runMigration(WP_REST_Request $request): WP_REST_Response
    {
        $migrationId = $request->get_param('id');
        $migrationLog = $this->migrationLogFactory->make($migrationId);

        // Begin transaction
        DB::beginTransaction();

        try {
            $migrationClass = $this->migrationRegister->getMigration($migrationId);
            /**
             * @var Migration $migration
             */
            $migration = give($migrationClass);
            $migration->run();
            // Save migration status
            $migrationLog
                ->setStatus(MigrationLogStatus::SUCCESS)
                ->setError(null)
                ->save();

            DB::commit();

            return new WP_REST_Response(['status' => true]);
        } catch (Exception $exception) {
            DB::rollback();

            $migrationLog
                ->setStatus(MigrationLogStatus::FAILED)
                ->setError([
                    'status' => __('Migration failed', 'give'),
                    'error' => [
                        'message' => $exception->getMessage(),
                        'code' => $exception->getCode(),
                        'file' => $exception->getFile(),
                        'line' => $exception->getLine(),
                    ],
                ])
                ->save();
        }

        return new WP_REST_Response(
            [
                'status' => false,
                'message' => $exception->getMessage(),
            ]
        );
    }


    /**
     * Run batch migration
     *
     * @since 4.0.0
     */
    public function runBatchMigration(WP_REST_Request $request): WP_REST_Response
    {
        $migrationId = $request->get_param('id');
        $migrationClass = $this->migrationRegister->getMigration($migrationId);

        if ( ! is_subclass_of($migrationClass, BatchMigration::class)) {
            return new WP_REST_Response([
                'status' => false,
                'message' => 'Migration is not an instance of ' . BatchMigration::class,
            ]);
        }

        try {
            // We are not running migration directly,
            // we just have to set migration status to PENDING and Migration Runner will handle it
            $migrationLog = $this->migrationLogFactory->make($migrationId);
            $migrationLog->setStatus(MigrationLogStatus::PENDING);
            $migrationLog->save();
        } catch (Exception $e) {
            return new WP_REST_Response([
                'status' => false,
                'message' => $e->getMessage(),
            ]);
        }

        return new WP_REST_Response(['status' => true]);
    }

    /**
     * Reschedule failed actions
     *
     * @since 4.0.0
     */
    public function rescheduleFailedActions(WP_REST_Request $request): WP_REST_Response
    {
        $migrationId = $request->get_param('id');
        $migrationClass = $this->migrationRegister->getMigration($migrationId);
        $migration = give($migrationClass);

        if ( ! is_subclass_of($migration, BatchMigration::class)) {
            return new WP_REST_Response([
                'status' => false,
                'message' => 'Migration is not an instance of ' . BatchMigration::class,
            ]);
        }

        try {
            (new BatchMigrationRunner($migration))->rescheduleFailedActions();
        } catch (Exception $e) {
            return new WP_REST_Response([
                'status' => false,
                'message' => $e->getMessage(),
            ]);
        }

        return new WP_REST_Response(['status' => true]);
    }


    /**
     * @since 4.3.0
     */
    public function rollbackMigration(WP_REST_Request $request): WP_REST_Response
    {
        $migrationId = $request->get_param('id');
        $migrationClass = $this->migrationRegister->getMigration($migrationId);
        $migration = give($migrationClass);
        $migrationLog = $this->migrationLogFactory->make($migrationId);

        if ($migration instanceof ReversibleMigration) {
            try {
                $migration->reverse();
                $migrationLog->setStatus(MigrationLogStatus::REVERSED);
            } catch (Exception $e) {
                $migrationLog
                    ->setStatus(MigrationLogStatus::FAILED)
                    ->setError([
                        'status' => __('Rollback failed', 'give'),
                        'error' => [
                            'message' => $e->getMessage(),
                            'code' => $e->getCode(),
                            'file' => $e->getFile(),
                            'line' => $e->getLine(),
                        ],
                    ]);

                return new WP_REST_Response([
                    'status' => false,
                    'message' => $e->getMessage(),
                ]);
            }

            $migrationLog->save();

            return new WP_REST_Response(['status' => true]);
        }

        return new WP_REST_Response(['status' => false]);
    }

}