From 7ca4c018b66d0dde0aa6be14cdee176074557699 Mon Sep 17 00:00:00 2001 From: Volodymyr Hadomskyi Date: Sun, 19 May 2024 14:45:06 +0200 Subject: [PATCH 1/2] Improve retry logic and logs ENG-15588 --- e2e/Makefile | 2 +- e2e/docker-compose.yml | 11 +- e2e/happy-lager-override/config/general.php | 1 + e2e/happy-lager-override/queue_listen.sh | 8 + .../job/PostJobRetryController.php | 12 + src/modules/AbstractRetryJob.php | 3 + ...tchInstantJobTranslationsFromConnector.php | 114 --------- src/modules/FetchJobStatusFromConnector.php | 2 +- src/modules/FetchTranslationFromConnector.php | 2 +- ...chVerifiedJobTranslationsFromConnector.php | 223 ------------------ src/modules/SendJobToConnector.php | 2 +- src/modules/SendTranslationToConnector.php | 2 +- .../SendJobToLiltConnectorHandler.php | 6 + src/services/listeners/AfterErrorListener.php | 95 ++++++-- .../listeners/QueueBeforePushListener.php | 4 - .../FetchJobStatusFromConnectorCest.php | 3 - .../FetchTranslationFromConnectorCest.php | 2 - 17 files changed, 123 insertions(+), 369 deletions(-) create mode 100755 e2e/happy-lager-override/queue_listen.sh delete mode 100644 src/modules/FetchInstantJobTranslationsFromConnector.php delete mode 100644 src/modules/FetchVerifiedJobTranslationsFromConnector.php diff --git a/e2e/Makefile b/e2e/Makefile index 6db49e91..9c662bad 100644 --- a/e2e/Makefile +++ b/e2e/Makefile @@ -54,7 +54,7 @@ up: clone down docker-compose exec -T app sh -c 'php craft project-config/rebuild' docker-compose exec -T app sh -c 'php craft up' docker-compose exec -T app sh -c 'php craft migrate/up' - docker-compose exec -T app sh -c 'php craft queue/run' + docker-compose exec -T app sh -c 'nohup ./queue_listen.sh > queue.log 2>&1 &' cli: docker-compose exec -T app sh diff --git a/e2e/docker-compose.yml b/e2e/docker-compose.yml index ca20f140..9f0655df 100644 --- a/e2e/docker-compose.yml +++ b/e2e/docker-compose.yml @@ -28,12 +28,19 @@ services: MOCKSERVER_CORS_ALLOW_METHODS: "CONNECT, DELETE, GET, HEAD, OPTIONS, POST, PUT, PATCH, TRACE" networks: - e2e-network + + mockserver-healthcheck: + image: alpine:latest healthcheck: - test: [ "CMD-SHELL", "curl --fail http://localhost:1080/status" ] + test: ["CMD-SHELL", "curl --fail -XPUT http://mockserver:1080/status || exit 1"] interval: 5s timeout: 5s retries: 3 - restart: on-failure + depends_on: + - mockserver + networks: + - e2e-network + entrypoint: ["sh", "-c", "apk add --no-cache curl && while sleep 3600; do :; done"] networks: e2e-network: diff --git a/e2e/happy-lager-override/config/general.php b/e2e/happy-lager-override/config/general.php index c4ba551a..44176301 100644 --- a/e2e/happy-lager-override/config/general.php +++ b/e2e/happy-lager-override/config/general.php @@ -38,6 +38,7 @@ // Whether crawlers should be allowed to index pages and following links 'disallowRobots' => !$isProd, + 'runQueueAutomatically' => false, 'aliases' => [ '@assetBasePath' => App::env('ASSET_BASE_PATH') ?: "./assets", diff --git a/e2e/happy-lager-override/queue_listen.sh b/e2e/happy-lager-override/queue_listen.sh new file mode 100755 index 00000000..153368a7 --- /dev/null +++ b/e2e/happy-lager-override/queue_listen.sh @@ -0,0 +1,8 @@ +#!/bin/sh + +while true; do + echo "Starting the queue listen process..." + php craft queue/listen -v + echo "Queue listen completed. Restarting in 30 seconds..." + sleep 1 +done diff --git a/src/controllers/job/PostJobRetryController.php b/src/controllers/job/PostJobRetryController.php index 6f54f6c7..4e0a145f 100644 --- a/src/controllers/job/PostJobRetryController.php +++ b/src/controllers/job/PostJobRetryController.php @@ -5,12 +5,24 @@ namespace lilthq\craftliltplugin\controllers\job; use Craft; +use craft\errors\ElementNotFoundException; use craft\web\Controller; +use LiltConnectorSDK\ApiException; use lilthq\craftliltplugin\Craftliltplugin; +use Throwable; +use yii\base\Exception; +use yii\db\StaleObjectException; use yii\web\Response; class PostJobRetryController extends Controller { + /** + * @throws ElementNotFoundException + * @throws Throwable + * @throws ApiException + * @throws StaleObjectException + * @throws Exception + */ public function actionInvoke(): Response { $request = Craft::$app->getRequest(); diff --git a/src/modules/AbstractRetryJob.php b/src/modules/AbstractRetryJob.php index a16be215..7ae4ee1c 100644 --- a/src/modules/AbstractRetryJob.php +++ b/src/modules/AbstractRetryJob.php @@ -37,6 +37,9 @@ abstract class AbstractRetryJob extends BaseJob */ abstract public function canRetry(): bool; + /** + * @return SendJobToConnector|SendTranslationToConnector|FetchJobStatusFromConnector|FetchTranslationFromConnector + */ abstract public function getRetryJob(): BaseJob; protected function getCommand(): ?Command diff --git a/src/modules/FetchInstantJobTranslationsFromConnector.php b/src/modules/FetchInstantJobTranslationsFromConnector.php deleted file mode 100644 index 82331967..00000000 --- a/src/modules/FetchInstantJobTranslationsFromConnector.php +++ /dev/null @@ -1,114 +0,0 @@ - $this->jobId]); - if (!$job || !$job->isInstantFlow()) { - $this->markAsDone($queue); - - return; - } - - $mutex = Craft::$app->getMutex(); - $mutexKey = __CLASS__ . '_' . __FUNCTION__ . '_' . $this->jobId; - if (!$mutex->acquire($mutexKey)) { - Craft::error(sprintf('Job %s is already processing job %d', __CLASS__, $this->jobId)); - - return; - } - - Craftliltplugin::$plugin->syncJobFromLiltConnectorHandler->__invoke($job); - - $liltJob = Craftliltplugin::getInstance()->connectorJobRepository->findOneById($this->liltJobId); - if ($liltJob->getStatus() === JobResponse::STATUS_COMPLETE) { - Craft::$app->elements->invalidateCachesForElementType( - Job::class - ); - } - - $this->markAsDone($queue); - $mutex->release($mutexKey); - } - - /** - * @inheritdoc - */ - protected function defaultDescription(): ?string - { - return Craft::t('app', 'Lilt translations'); - } - - /** - * @param $queue - * @return void - */ - private function markAsDone($queue): void - { - $this->setProgress( - $queue, - 1, - Craft::t( - 'app', - 'Fetching of translations for jobId: {jobId} liltJobId: {liltJobId} is done', - [ - 'jobId' => $this->jobId, - 'liltJobId' => $this->liltJobId, - ] - ) - ); - } - - public function getTtr(): int - { - return self::TTR; - } - - public function canRetry($attempt, $error): bool - { - return $attempt < self::RETRY_COUNT; - } -} diff --git a/src/modules/FetchJobStatusFromConnector.php b/src/modules/FetchJobStatusFromConnector.php index 435a4b8d..1e038b32 100644 --- a/src/modules/FetchJobStatusFromConnector.php +++ b/src/modules/FetchJobStatusFromConnector.php @@ -27,7 +27,7 @@ class FetchJobStatusFromConnector extends AbstractRetryJob public const PRIORITY = 1024; public const TTR = 60 * 30; - private const RETRY_COUNT = 3; + private const RETRY_COUNT = 10; /** * @var int $liltJobId diff --git a/src/modules/FetchTranslationFromConnector.php b/src/modules/FetchTranslationFromConnector.php index 7bd40c1e..5bffdf70 100644 --- a/src/modules/FetchTranslationFromConnector.php +++ b/src/modules/FetchTranslationFromConnector.php @@ -29,7 +29,7 @@ class FetchTranslationFromConnector extends AbstractRetryJob public const PRIORITY = 2048; public const TTR = 60 * 30; - private const RETRY_COUNT = 3; + private const RETRY_COUNT = 10; /** * @var int $liltJobId diff --git a/src/modules/FetchVerifiedJobTranslationsFromConnector.php b/src/modules/FetchVerifiedJobTranslationsFromConnector.php deleted file mode 100644 index a96430be..00000000 --- a/src/modules/FetchVerifiedJobTranslationsFromConnector.php +++ /dev/null @@ -1,223 +0,0 @@ - $this->jobId]); - $jobRecord = JobRecord::findOne(['id' => $this->jobId]); - - if (!$jobRecord || !$job) { - Craft::error( - sprintf( - 'Job record %d not found, looks like job was removed. Translation fetching aborted.', - $this->jobId - ) - ); - - $this->markAsDone($queue); - - return; - } - - if (!$job->isVerifiedFlow()) { - $this->markAsDone($queue); - - return; - } - - $mutex = Craft::$app->getMutex(); - $mutexKey = __CLASS__ . '_' . __FUNCTION__ . '_' . $this->jobId; - if (!$mutex->acquire($mutexKey)) { - Craft::error(sprintf('Job %s is already processing job %d', __CLASS__, $this->jobId)); - - return; - } - - $unprocessedTranslations = Craftliltplugin::getInstance() - ->translationRepository - ->findUnprocessedByJobIdMapped($job->id); - - if (empty($unprocessedTranslations)) { - $jobRecord->status = Job::STATUS_READY_FOR_REVIEW; - $jobRecord->save(); - - Craft::$app->elements->invalidateCachesForElementType( - Job::class - ); - - $this->markAsDone($queue); - } - - $translations = Craftliltplugin::getInstance()->connectorTranslationRepository->findByJobId( - $job->liltJobId - ); - - $statuses = array_unique( - array_map( - function (TranslationResponse $translationResponse) use ($job, $unprocessedTranslations) { - if ($translationResponse->getStatus() === TranslationResponse::STATUS_EXPORT_COMPLETE) { - try { - Craftliltplugin::getInstance()->syncJobFromLiltConnectorHandler->processTranslation( - $translationResponse, - $job - ); - } catch (Exception $ex) { - Craft::error([ - 'message' => "Can't fetch process translation!", - 'exception_message' => $ex->getMessage(), - 'exception_trace' => $ex->getTrace(), - 'exception' => $ex, - ]); - - Craftliltplugin::getInstance()->translationFailedHandler->__invoke( - $translationResponse, - $job, - $unprocessedTranslations - ); - - return TranslationRecord::STATUS_FAILED; - } - - return TranslationRecord::STATUS_READY_FOR_REVIEW; - } - - if ( - $translationResponse->getStatus() === TranslationResponse::STATUS_IMPORT_FAILED - || $translationResponse->getStatus() === TranslationResponse::STATUS_EXPORT_FAILED - ) { - Craftliltplugin::getInstance()->translationFailedHandler->__invoke( - $translationResponse, - $job, - $unprocessedTranslations - ); - - return TranslationRecord::STATUS_FAILED; - } - - return TranslationRecord::STATUS_IN_PROGRESS; - }, - $translations->getResults() - ) - ); - - if (in_array(TranslationRecord::STATUS_IN_PROGRESS, $statuses, true)) { - // One of translations still in progress, we are waiting till all of them are done - Queue::push( - (new FetchVerifiedJobTranslationsFromConnector( - [ - 'jobId' => $this->jobId, - 'liltJobId' => $this->liltJobId, - ] - )), - self::PRIORITY, - self::DELAY_IN_SECONDS - ); - } - - if ($statuses === ['failed', 'canceled']) { - $jobRecord->status = Job::STATUS_FAILED; - $jobRecord->save(); - } elseif (in_array('in-progress', $statuses, true)) { - $jobRecord->status = Job::STATUS_IN_PROGRESS; - $jobRecord->save(); - } else { - //TODO: can't be default, we need to reach all translations to status ready for review! - $jobRecord->status = Job::STATUS_READY_FOR_REVIEW; - $jobRecord->save(); - - Craft::$app->elements->invalidateCachesForElementType(Translation::class); - - Craftliltplugin::getInstance()->jobLogsRepository->create( - $jobRecord->id, - Craft::$app->getUser()->getId(), - 'Translations downloaded' - ); - } - - Craft::$app->elements->invalidateCachesForElement( - $job - ); - - $this->markAsDone($queue); - $mutex->release($mutexKey); - } - - /** - * @inheritdoc - */ - protected function defaultDescription(): ?string - { - return Craft::t('app', 'Lilt translations'); - } - - /** - * @param $queue - * @return void - */ - private function markAsDone($queue): void - { - $this->setProgress( - $queue, - 1, - Craft::t( - 'app', - 'Fetching of translations for jobId: {jobId} liltJobId: {liltJobId} is done', - [ - 'jobId' => $this->jobId, - 'liltJobId' => $this->liltJobId, - ] - ) - ); - } - - public function getTtr(): int - { - return self::TTR; - } - - public function canRetry($attempt, $error): bool - { - return $attempt < self::RETRY_COUNT; - } -} diff --git a/src/modules/SendJobToConnector.php b/src/modules/SendJobToConnector.php index a59fe3bb..fb8370e4 100644 --- a/src/modules/SendJobToConnector.php +++ b/src/modules/SendJobToConnector.php @@ -24,7 +24,7 @@ class SendJobToConnector extends AbstractRetryJob public const PRIORITY = 1024; public const TTR = 60 * 30; - private const RETRY_COUNT = 3; + private const RETRY_COUNT = 10; /** * @inheritdoc diff --git a/src/modules/SendTranslationToConnector.php b/src/modules/SendTranslationToConnector.php index 19c37380..be581765 100644 --- a/src/modules/SendTranslationToConnector.php +++ b/src/modules/SendTranslationToConnector.php @@ -29,7 +29,7 @@ class SendTranslationToConnector extends AbstractRetryJob public const PRIORITY = 1024; public const TTR = 60 * 30; - private const RETRY_COUNT = 3; + private const RETRY_COUNT = 10; /** * @var int diff --git a/src/services/handlers/SendJobToLiltConnectorHandler.php b/src/services/handlers/SendJobToLiltConnectorHandler.php index 720e7a82..2094d0d1 100644 --- a/src/services/handlers/SendJobToLiltConnectorHandler.php +++ b/src/services/handlers/SendJobToLiltConnectorHandler.php @@ -184,6 +184,12 @@ public function __invoke(Job $job): void $this->updateJob($job, $jobLilt->getId(), Job::STATUS_IN_PROGRESS); + // Set all translations to in progress + TranslationRecord::updateAll( + ['status' => TranslationRecord::STATUS_IN_PROGRESS], + ['jobId' => $job->id] + ); + if ($isQueueEachTranslationFileSeparately) { return; } diff --git a/src/services/listeners/AfterErrorListener.php b/src/services/listeners/AfterErrorListener.php index ed68a432..83397ae4 100644 --- a/src/services/listeners/AfterErrorListener.php +++ b/src/services/listeners/AfterErrorListener.php @@ -15,23 +15,21 @@ use lilthq\craftliltplugin\Craftliltplugin; use lilthq\craftliltplugin\elements\Job; use lilthq\craftliltplugin\elements\Translation; -use lilthq\craftliltplugin\modules\FetchInstantJobTranslationsFromConnector; +use lilthq\craftliltplugin\modules\AbstractRetryJob; use lilthq\craftliltplugin\modules\FetchJobStatusFromConnector; use lilthq\craftliltplugin\modules\FetchTranslationFromConnector; -use lilthq\craftliltplugin\modules\FetchVerifiedJobTranslationsFromConnector; use lilthq\craftliltplugin\modules\SendJobToConnector; use lilthq\craftliltplugin\modules\SendTranslationToConnector; use lilthq\craftliltplugin\records\JobRecord; use lilthq\craftliltplugin\records\TranslationRecord; use yii\base\Event; +use yii\db\Exception; use yii\queue\ExecEvent; class AfterErrorListener implements ListenerInterface { private const SUPPORTED_JOBS = [ FetchJobStatusFromConnector::class, - FetchInstantJobTranslationsFromConnector::class, - FetchVerifiedJobTranslationsFromConnector::class, FetchTranslationFromConnector::class, SendJobToConnector::class, SendTranslationToConnector::class, @@ -71,23 +69,39 @@ public function __invoke(Event $event): Event } /** - * @var FetchTranslationFromConnector|FetchJobStatusFromConnector|SendJobToConnector $queueJob + * @var AbstractRetryJob $queueJob */ $queueJob = $event->job; $jobRecord = JobRecord::findOne(['id' => $queueJob->jobId]); Craft::error([ - "message" => sprintf( + "message" => sprintf( 'Job %s failed due to: %s', get_class($queueJob), $event->error->getMessage() ), "queueJob" => $queueJob, + "attempt" => $queueJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, "jobRecord" => $jobRecord ]); if (!$queueJob->canRetry()) { + Craft::warning([ + "message" => sprintf( + 'Job %s can\'t be retried', + $event->id + ), + "queueJob" => $queueJob, + "attempt" => $queueJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, + "jobRecord" => $jobRecord + ]); + + $jobRecord->status = Job::STATUS_FAILED; $jobRecord->save(); @@ -104,7 +118,7 @@ public function __invoke(Event $event): Event ); Craft::error([ - "message" => sprintf( + "message" => sprintf( '[%s] Mark lilt job %d (%d) as failed due to: %s', get_class($queueJob), $jobRecord->liltJobId, @@ -112,6 +126,9 @@ public function __invoke(Event $event): Event $event->error->getMessage() ), "queueJob" => $queueJob, + "attempt" => $queueJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, "jobRecord" => $jobRecord ]); @@ -146,24 +163,70 @@ public function __invoke(Event $event): Event (string)$event->id ); - if ($event->error instanceof ApiException && $event->error->getCode() === 500) { + Craft::info([ + "message" => sprintf( + 'Released job %s', + $event->id + ), + "queueJob" => $queueJob, + "attempt" => $queueJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, + "jobRecord" => $jobRecord + ]); + + $retryJob = $queueJob->getRetryJob(); + + $isApiError = $event->error instanceof ApiException + && $event->error->getCode() === 500; + $isDeadlockError = $event->error instanceof Exception + && strpos( + $event->error->getMessage(), + 'Deadlock found when trying to get lock' + ) !== false; + + if ($isApiError || $isDeadlockError) { + // Infrastructure error, we need always retry + $retryJob->attempt = 0; + \craft\helpers\Queue::push( - $queueJob, - $queueJob::PRIORITY, - $queueJob::getDelay() + $retryJob->getRetryJob(), + $retryJob::PRIORITY, + $retryJob::getDelay() ); + Craft::info([ + "message" => 'Retried job due to infrastructure error', + "queueJob" => $retryJob, + "isApiError" => $isApiError, + "isDeadlockError" => $isDeadlockError, + "attempt" => $retryJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, + "jobRecord" => $jobRecord + ]); + return $event; } - ++$queueJob->attempt; - \craft\helpers\Queue::push( - $queueJob, - $queueJob::PRIORITY, - $queueJob::getDelay() + $retryJob, + $retryJob::PRIORITY, + $retryJob::getDelay() ); + Craft::info([ + "message" => sprintf( + 'Retried job, attempt %d', + $queueJob->attempt + ), + "queueJob" => $retryJob, + "attempt" => $retryJob->attempt, + "liltJobId" => $jobRecord->liltJobId, + "pluginJobId" => $jobRecord->id, + "jobRecord" => $jobRecord + ]); + return $event; } } diff --git a/src/services/listeners/QueueBeforePushListener.php b/src/services/listeners/QueueBeforePushListener.php index 51dfee6e..60c36b8a 100644 --- a/src/services/listeners/QueueBeforePushListener.php +++ b/src/services/listeners/QueueBeforePushListener.php @@ -11,10 +11,8 @@ use Craft; use craft\queue\Queue; -use lilthq\craftliltplugin\modules\FetchInstantJobTranslationsFromConnector; use lilthq\craftliltplugin\modules\FetchJobStatusFromConnector; use lilthq\craftliltplugin\modules\FetchTranslationFromConnector; -use lilthq\craftliltplugin\modules\FetchVerifiedJobTranslationsFromConnector; use lilthq\craftliltplugin\modules\SendJobToConnector; use lilthq\craftliltplugin\modules\SendTranslationToConnector; use yii\base\Event; @@ -25,8 +23,6 @@ class QueueBeforePushListener implements ListenerInterface { private const SUPPORTED_JOBS = [ FetchJobStatusFromConnector::class, - FetchInstantJobTranslationsFromConnector::class, - FetchVerifiedJobTranslationsFromConnector::class, FetchTranslationFromConnector::class, SendJobToConnector::class, SendTranslationToConnector::class, diff --git a/tests/integration/modules/FetchJobStatusFromConnectorCest.php b/tests/integration/modules/FetchJobStatusFromConnectorCest.php index 5f5591ae..950b663b 100644 --- a/tests/integration/modules/FetchJobStatusFromConnectorCest.php +++ b/tests/integration/modules/FetchJobStatusFromConnectorCest.php @@ -13,11 +13,8 @@ use LiltConnectorSDK\Model\JobResponse; use LiltConnectorSDK\Model\SettingsResponse; use lilthq\craftliltplugin\Craftliltplugin; -use lilthq\craftliltplugin\modules\FetchInstantJobTranslationsFromConnector; use lilthq\craftliltplugin\modules\FetchJobStatusFromConnector; use lilthq\craftliltplugin\modules\FetchTranslationFromConnector; -use lilthq\craftliltplugin\modules\FetchVerifiedJobTranslationsFromConnector; -use lilthq\craftliltplugin\parameters\CraftliltpluginParameters; use lilthq\craftliltplugin\records\TranslationRecord; use lilthq\craftliltplugin\services\repositories\SettingsRepository; use lilthq\craftliltplugintests\integration\AbstractIntegrationCest; diff --git a/tests/integration/modules/FetchTranslationFromConnectorCest.php b/tests/integration/modules/FetchTranslationFromConnectorCest.php index 45b5b467..a1412ee2 100644 --- a/tests/integration/modules/FetchTranslationFromConnectorCest.php +++ b/tests/integration/modules/FetchTranslationFromConnectorCest.php @@ -15,9 +15,7 @@ use LiltConnectorSDK\Model\TranslationResponse; use lilthq\craftliltplugin\Craftliltplugin; use lilthq\craftliltplugin\elements\Job; -use lilthq\craftliltplugin\modules\FetchInstantJobTranslationsFromConnector; use lilthq\craftliltplugin\modules\FetchTranslationFromConnector; -use lilthq\craftliltplugin\modules\FetchVerifiedJobTranslationsFromConnector; use lilthq\craftliltplugin\records\JobRecord; use lilthq\craftliltplugin\records\TranslationRecord; use lilthq\craftliltplugintests\integration\AbstractIntegrationCest; From f482d6a1cd5a492d557f72a4211c123e7f750190 Mon Sep 17 00:00:00 2001 From: Volodymyr Hadomskyi Date: Mon, 27 May 2024 21:13:28 +0200 Subject: [PATCH 2/2] Add job attempts Resolve translation if there duplicates ENG-15588 --- Makefile | 79 ++--- docker-compose.yml | 2 - src/Craftliltplugin.php | 6 +- .../job/PostJobRetryController.php | 2 +- src/elements/Job.php | 4 + src/migrations/Install.php | 1 + .../m240526_212007_add_job_attempts.php | 39 +++ src/modules/FetchJobStatusFromConnector.php | 94 ++---- src/modules/FetchTranslationFromConnector.php | 2 +- src/records/JobRecord.php | 1 + src/services/ServiceInitializer.php | 6 +- src/services/handlers/CreateJobHandler.php | 1 + src/services/handlers/EditJobHandler.php | 1 + src/services/handlers/ResendJobHandler.php | 67 ++++ .../ResolveTranslationsConnectorIds.php | 153 +++++++++ .../UpdateTranslationsConnectorIds.php | 75 ----- .../job/PostJobRetryControllerCest.php | 68 ++-- .../FetchJobStatusFromConnectorCest.php | 307 +++++++++++++++++- 18 files changed, 678 insertions(+), 230 deletions(-) create mode 100644 src/migrations/m240526_212007_add_job_attempts.php create mode 100644 src/services/handlers/ResendJobHandler.php create mode 100644 src/services/handlers/ResolveTranslationsConnectorIds.php delete mode 100644 src/services/handlers/UpdateTranslationsConnectorIds.php diff --git a/Makefile b/Makefile index bc31e989..a4de7c91 100644 --- a/Makefile +++ b/Makefile @@ -7,86 +7,89 @@ PHP_VERSION?=8.1 MYSQL_VERSION?=5.7 up: - docker-compose up -d - docker-compose exec -T mysql-test sh -c 'while ! mysqladmin ping -h"mysql-test" --silent; do sleep 1; done' + docker compose up -d + docker compose exec -T mysql-test sh -c 'while ! mysqladmin ping -h"mysql-test" --silent; do sleep 1; done' + +create-migration: + docker compose exec -T -u www-data cli-app sh -c "php craft migrate/create --plugin=craft-lilt-plugin add_job_attempts" down: - docker-compose down -v --remove-orphans + docker compose down -v --remove-orphans restart: down - docker-compose up -d + docker compose up -d cli: - docker-compose exec -u www-data cli-app sh + docker compose exec -u www-data cli-app sh root: - docker-compose exec -u root cli-app sh + docker compose exec -u root cli-app sh composer-install: - docker-compose exec -T -u root cli-app sh -c "apk add git" - docker-compose exec -T -u root cli-app sh -c "chown -R www-data:www-data /craft-lilt-plugin" - docker-compose exec -T -u root cli-app sh -c "rm -f composer.lock" - docker-compose exec -T -u root cli-app sh -c "rm -rf vendor" - docker-compose exec -T -u www-data cli-app sh -c "cp tests/.env.test tests/.env" - docker-compose exec -T -u www-data cli-app sh -c "curl -s https://getcomposer.org/installer | php" - docker-compose exec -T -u www-data cli-app sh -c "php composer.phar install" + docker compose exec -T -u root cli-app sh -c "apk add git" + docker compose exec -T -u root cli-app sh -c "chown -R www-data:www-data /craft-lilt-plugin" + docker compose exec -T -u root cli-app sh -c "rm -f composer.lock" + docker compose exec -T -u root cli-app sh -c "rm -rf vendor" + docker compose exec -T -u www-data cli-app sh -c "cp tests/.env.test tests/.env" + docker compose exec -T -u www-data cli-app sh -c "curl -s https://getcomposer.org/installer | php" + docker compose exec -T -u www-data cli-app sh -c "php composer.phar install" quality: - docker-compose exec -T -u www-data cli-app sh -c "curl -L -s https://phar.phpunit.de/phpcpd.phar --output phpcpd.phar" - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/phpcs" - docker-compose exec -T -u www-data cli-app sh -c "php phpcpd.phar src --exclude /craft-lilt-plugin/src/migrations" + docker compose exec -T -u www-data cli-app sh -c "curl -L -s https://phar.phpunit.de/phpcpd.phar --output phpcpd.phar" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/phpcs" + docker compose exec -T -u www-data cli-app sh -c "php phpcpd.phar src --exclude /craft-lilt-plugin/src/migrations" quality-fix: - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/phpcbf" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/phpcbf" codecept-build: - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept build" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept build" coverage-xdebug: - docker-compose exec -T -u www-data cli-app sh -c "php -dxdebug.mode=coverage vendor/bin/codecept run --coverage --coverage-xml --coverage-html" + docker compose exec -T -u www-data cli-app sh -c "php -dxdebug.mode=coverage vendor/bin/codecept run --coverage --coverage-xml --coverage-html" install-pcov: - docker-compose exec -T -u root cli-app sh -c "apk --no-cache add pcre-dev autoconf dpkg-dev dpkg file g++ gcc libc-dev make pkgconf re2c" - docker-compose exec -T -u root cli-app sh -c "pecl install pcov || true" - docker-compose exec -T -u root cli-app sh -c "docker-php-ext-enable pcov" + docker compose exec -T -u root cli-app sh -c "apk --no-cache add pcre-dev autoconf dpkg-dev dpkg file g++ gcc libc-dev make pkgconf re2c" + docker compose exec -T -u root cli-app sh -c "pecl install pcov || true" + docker compose exec -T -u root cli-app sh -c "docker-php-ext-enable pcov" coverage: install-pcov - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run --coverage --coverage-xml --coverage-html" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run --coverage --coverage-xml --coverage-html" tests-with-coverage: codecept-build install-pcov unit-coverage integration-coverage functional-coverage integration-coverage: - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run integration --coverage-xml=coverage-integration.xml" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run integration --coverage-xml=coverage-integration.xml" functional-coverage: - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run functional --coverage-xml=coverage-functional.xml" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run functional --coverage-xml=coverage-functional.xml" unit-coverage: - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run unit --coverage-xml=coverage-unit.xml" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run unit --coverage-xml=coverage-unit.xml" integration: codecept-build - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run integration" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run integration" functional: codecept-build - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run functional" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run functional" unit: codecept-build - docker-compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run unit" + docker compose exec -T -u www-data cli-app sh -c "php vendor/bin/codecept run unit" test: functional integration unit prepare-container: - PHP_VERSION=7.2 docker-compose up -d - docker-compose exec -T -u root cli-app sh -c "chown -R www-data:www-data /craft-lilt-plugin" - docker-compose exec -T -u root cli-app sh -c "apk --no-cache add bash make git" - docker-compose exec -T -u www-data cli-app sh -c "cp tests/.env.test tests/.env" - docker-compose exec -T -u root cli-app sh -c "curl -s https://getcomposer.org/installer | php" - docker-compose exec -T -u root cli-app sh -c "cp composer.phar /bin/composer" + PHP_VERSION=7.2 docker compose up -d + docker compose exec -T -u root cli-app sh -c "chown -R www-data:www-data /craft-lilt-plugin" + docker compose exec -T -u root cli-app sh -c "apk --no-cache add bash make git" + docker compose exec -T -u www-data cli-app sh -c "cp tests/.env.test tests/.env" + docker compose exec -T -u root cli-app sh -c "curl -s https://getcomposer.org/installer | php" + docker compose exec -T -u root cli-app sh -c "cp composer.phar /bin/composer" test-craft-versions: prepare-container - docker-compose exec -T -u www-data cli-app bash -c \ + docker compose exec -T -u www-data cli-app bash -c \ "./craft-versions.sh ${CRAFT_VERSION}" require-guzzle-v6: - docker-compose exec -T -u www-data cli-app sh -c "php composer.phar require guzzlehttp/guzzle:^6.0 -W --no-scripts || true" - docker-compose exec -T -u www-data cli-app sh -c 'if ! php composer.phar show -i | grep "guzzlehttp/guzzle" | grep "6."; then echo "Guzzle version 6 is not present."; exit 1; fi' + docker compose exec -T -u www-data cli-app sh -c "php composer.phar require guzzlehttp/guzzle:^6.0 -W --no-scripts || true" + docker compose exec -T -u www-data cli-app sh -c 'if ! php composer.phar show -i | grep "guzzlehttp/guzzle" | grep "6."; then echo "Guzzle version 6 is not present."; exit 1; fi' diff --git a/docker-compose.yml b/docker-compose.yml index 3820eba5..01c35244 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,5 +1,3 @@ -version: '3' - services: cli-app: image: craftcms/cli:${PHP_VERSION} diff --git a/src/Craftliltplugin.php b/src/Craftliltplugin.php index abe9a4ec..8dff1d04 100644 --- a/src/Craftliltplugin.php +++ b/src/Craftliltplugin.php @@ -36,13 +36,14 @@ use lilthq\craftliltplugin\services\handlers\LoadI18NHandler; use lilthq\craftliltplugin\services\handlers\PublishDraftHandler; use lilthq\craftliltplugin\services\handlers\RefreshJobStatusHandler; +use lilthq\craftliltplugin\services\handlers\ResendJobHandler; use lilthq\craftliltplugin\services\handlers\SendJobToLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\SendTranslationToLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\StartQueueManagerHandler; use lilthq\craftliltplugin\services\handlers\SyncJobFromLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\TranslationFailedHandler; use lilthq\craftliltplugin\services\handlers\UpdateJobStatusHandler; -use lilthq\craftliltplugin\services\handlers\UpdateTranslationsConnectorIds; +use lilthq\craftliltplugin\services\handlers\ResolveTranslationsConnectorIds; use lilthq\craftliltplugin\services\listeners\ListenerRegister; use lilthq\craftliltplugin\services\mappers\LanguageMapper; use lilthq\craftliltplugin\services\providers\ConnectorConfigurationProvider; @@ -109,8 +110,9 @@ * @property CreateDraftHandler $createDraftHandler * @property CopySourceTextHandler $copySourceTextHandler * @property UpdateJobStatusHandler $updateJobStatusHandler + * @property ResendJobHandler $resendJobHandler * @property SettingsRepository $settingsRepository - * @property UpdateTranslationsConnectorIds $updateTranslationsConnectorIds + * @property ResolveTranslationsConnectorIds $resolveTranslationsConnectorIds * @property PackagistRepository $packagistRepository * @property StartQueueManagerHandler $startQueueManagerHandler * @property ServiceInitializer $serviceInitializer diff --git a/src/controllers/job/PostJobRetryController.php b/src/controllers/job/PostJobRetryController.php index 4e0a145f..7d0c41b6 100644 --- a/src/controllers/job/PostJobRetryController.php +++ b/src/controllers/job/PostJobRetryController.php @@ -46,7 +46,7 @@ public function actionInvoke(): Response sprintf('Job retried (previous Lilt Job ID: %d)', $job->liltJobId) ); - Craftliltplugin::getInstance()->sendJobToLiltConnectorHandler->__invoke($job); + Craftliltplugin::getInstance()->resendJobHandler->__invoke($job->id); } diff --git a/src/elements/Job.php b/src/elements/Job.php index 540a694b..c6fb702c 100644 --- a/src/elements/Job.php +++ b/src/elements/Job.php @@ -43,6 +43,7 @@ class Job extends Element public const STATUS_FAILED = 'failed'; public const STATUS_NEEDS_ATTENTION = 'needs-attention'; + public const MAX_JOB_ATTEMPTS = 5; public $uid; public $authorId; @@ -59,6 +60,9 @@ class Job extends Element public $dateCreated; public $dateUpdated; + public $attempt = 0; + + // @codingStandardsIgnoreStart private $_author; private $_elements; diff --git a/src/migrations/Install.php b/src/migrations/Install.php index 4dcebac4..990e2ee0 100644 --- a/src/migrations/Install.php +++ b/src/migrations/Install.php @@ -32,6 +32,7 @@ public function safeUp(): void $this->createTable(CraftliltpluginParameters::JOB_TABLE_NAME, [ 'id' => $this->primaryKey()->unsigned(), + 'attempt' => $this->integer()->notNull()->defaultValue(0), 'title' => $this->string()->null(), 'authorId' => $this->integer()->null(), 'liltJobId' => $this->integer()->null(), diff --git a/src/migrations/m240526_212007_add_job_attempts.php b/src/migrations/m240526_212007_add_job_attempts.php new file mode 100644 index 00000000..2c731f12 --- /dev/null +++ b/src/migrations/m240526_212007_add_job_attempts.php @@ -0,0 +1,39 @@ +addColumn( + CraftliltpluginParameters::JOB_TABLE_NAME, + 'attempt', + $this->integer()->unsigned()->notNull()->defaultValue(0) + ); + } + + /** + * @inheritdoc + */ + public function safeDown() + { + // Remove the 'attempt' column from the 'lilt_jobs' table + $this->dropColumn( + CraftliltpluginParameters::JOB_TABLE_NAME, + 'attempt' + ); + } +} diff --git a/src/modules/FetchJobStatusFromConnector.php b/src/modules/FetchJobStatusFromConnector.php index 1e038b32..0380b719 100644 --- a/src/modules/FetchJobStatusFromConnector.php +++ b/src/modules/FetchJobStatusFromConnector.php @@ -79,14 +79,39 @@ public function execute($queue): void || $liltJob->getStatus() === JobResponse::STATUS_FAILED; if ($isJobFailed) { - $jobRecord->status = Job::STATUS_FAILED; - Craftliltplugin::getInstance()->jobLogsRepository->create( $jobRecord->id, Craft::$app->getUser()->getId(), sprintf('Job failed, received status: %s', $liltJob->getStatus()) ); + if ($jobRecord->attempt < Job::MAX_JOB_ATTEMPTS) { + // Retry job + $jobRecord->attempt++; + $jobRecord->save(); + + Craftliltplugin::getInstance()->resendJobHandler->__invoke( + $jobRecord->id + ); + + + Craftliltplugin::getInstance()->jobLogsRepository->create( + $jobRecord->id, + Craft::$app->getUser()->getId(), + sprintf( + 'Trying again to send job, attempt: %d/%d', + $jobRecord->attempt, + Job::MAX_JOB_ATTEMPTS + ) + ); + + $mutex->release($mutexKey); + $this->markAsDone($queue); + return; + } + + // Job failed, set status to failed + $jobRecord->status = Job::STATUS_FAILED; $jobRecord->save(); TranslationRecord::updateAll( @@ -108,7 +133,7 @@ public function execute($queue): void } if (!$isJobFinished) { - $queueDisableAutomaticSync = (bool) Craftliltplugin::getInstance()->settingsRepository->get( + $queueDisableAutomaticSync = (bool)Craftliltplugin::getInstance()->settingsRepository->get( SettingsRepository::QUEUE_DISABLE_AUTOMATIC_SYNC ); @@ -132,15 +157,12 @@ public function execute($queue): void return; } - $connectorTranslations = Craftliltplugin::getInstance()->connectorTranslationRepository->findByJobId( - $job->liltJobId - ); - + $connectorTranslations = Craftliltplugin::getInstance()->resolveTranslationsConnectorIds->update($job); $connectorTranslationsStatuses = array_map( function (TranslationResponse $connectorTranslation) { return $connectorTranslation->getStatus(); }, - $connectorTranslations->getResults() + $connectorTranslations ); $translationFinished = @@ -184,7 +206,7 @@ function (TranslationResponse $connectorTranslation) { return; } - $queueDisableAutomaticSync = (bool) Craftliltplugin::getInstance()->settingsRepository->get( + $queueDisableAutomaticSync = (bool)Craftliltplugin::getInstance()->settingsRepository->get( SettingsRepository::QUEUE_DISABLE_AUTOMATIC_SYNC ); @@ -208,7 +230,7 @@ function (TranslationResponse $connectorTranslation) { return; } - if ($jobRecord->isVerifiedFlow()) { + if ($jobRecord->isVerifiedFlow() || $jobRecord->isInstantFlow()) { #LILT_TRANSLATION_WORKFLOW_VERIFIED $jobRecord->status = Job::STATUS_IN_PROGRESS; @@ -217,58 +239,6 @@ function (TranslationResponse $connectorTranslation) { $this->jobId ); - Craftliltplugin::getInstance()->updateTranslationsConnectorIds->update($job); - - foreach ($translations as $translation) { - Queue::push( - new FetchTranslationFromConnector( - [ - 'jobId' => $this->jobId, - 'liltJobId' => $this->liltJobId, - 'translationId' => $translation->id, - ] - ), - FetchTranslationFromConnector::PRIORITY, - 10, //10 seconds for first job - FetchTranslationFromConnector::TTR - ); - } - } - - if ($jobRecord->isInstantFlow()) { - #LILT_TRANSLATION_WORKFLOW_INSTANT - - if ( - $liltJob->getStatus() === JobResponse::STATUS_FAILED - || $liltJob->getStatus() === JobResponse::STATUS_CANCELED - ) { - $jobRecord->status = Job::STATUS_FAILED; - - TranslationRecord::updateAll( - ['status' => TranslationRecord::STATUS_FAILED], - ['jobId' => $jobRecord->id] - ); - - Craft::error([ - "message" => sprintf( - 'Set job %d and translations to status failed due to failed/cancel status from lilt', - $jobRecord->id - ), - "jobRecord" => $jobRecord, - ]); - - $mutex->release($mutexKey); - $this->markAsDone($queue); - - return; - } - - $translations = Craftliltplugin::getInstance()->translationRepository->findByJobId( - $this->jobId - ); - - Craftliltplugin::getInstance()->updateTranslationsConnectorIds->update($job); - foreach ($translations as $translation) { Queue::push( new FetchTranslationFromConnector( diff --git a/src/modules/FetchTranslationFromConnector.php b/src/modules/FetchTranslationFromConnector.php index 5bffdf70..b003acf4 100644 --- a/src/modules/FetchTranslationFromConnector.php +++ b/src/modules/FetchTranslationFromConnector.php @@ -76,7 +76,7 @@ public function execute($queue): void } if (empty($translationRecord->connectorTranslationId)) { - Craftliltplugin::getInstance()->updateTranslationsConnectorIds->update($job); + Craftliltplugin::getInstance()->resolveTranslationsConnectorIds->update($job); } $translationRecord->refresh(); diff --git a/src/records/JobRecord.php b/src/records/JobRecord.php index f2ac53ed..7dcfbc20 100644 --- a/src/records/JobRecord.php +++ b/src/records/JobRecord.php @@ -26,6 +26,7 @@ * @property int $sourceSiteLanguage [int(11) unsigned] * @property string $targetSiteIds [json] * @property string $dueDate [datetime] + * @property int $attempt [int(11) unsigned] * * @property-read ActiveQueryInterface $element * @property string $translationWorkflow [varchar(50)] diff --git a/src/services/ServiceInitializer.php b/src/services/ServiceInitializer.php index 8206d499..9f9dc7a6 100644 --- a/src/services/ServiceInitializer.php +++ b/src/services/ServiceInitializer.php @@ -36,13 +36,14 @@ use lilthq\craftliltplugin\services\handlers\LoadI18NHandler; use lilthq\craftliltplugin\services\handlers\PublishDraftHandler; use lilthq\craftliltplugin\services\handlers\RefreshJobStatusHandler; +use lilthq\craftliltplugin\services\handlers\ResendJobHandler; use lilthq\craftliltplugin\services\handlers\SendJobToLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\SendTranslationToLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\StartQueueManagerHandler; use lilthq\craftliltplugin\services\handlers\SyncJobFromLiltConnectorHandler; use lilthq\craftliltplugin\services\handlers\TranslationFailedHandler; use lilthq\craftliltplugin\services\handlers\UpdateJobStatusHandler; -use lilthq\craftliltplugin\services\handlers\UpdateTranslationsConnectorIds; +use lilthq\craftliltplugin\services\handlers\ResolveTranslationsConnectorIds; use lilthq\craftliltplugin\services\listeners\ListenerRegister; use lilthq\craftliltplugin\services\mappers\LanguageMapper; use lilthq\craftliltplugin\services\providers\ConnectorConfigurationProvider; @@ -79,6 +80,7 @@ public function run(): void $pluginInstance->setComponents([ 'createJobHandler' => CreateJobHandler::class, + 'resendJobHandler' => ResendJobHandler::class, 'sendJobToLiltConnectorHandler' => SendJobToLiltConnectorHandler::class, 'copySourceTextHandler' => CopySourceTextHandler::class, 'syncJobFromLiltConnectorHandler' => SyncJobFromLiltConnectorHandler::class, @@ -93,7 +95,7 @@ public function run(): void 'createTranslationsHandler' => CreateTranslationsHandler::class, 'refreshJobStatusHandler' => RefreshJobStatusHandler::class, 'updateJobStatusHandler' => UpdateJobStatusHandler::class, - 'updateTranslationsConnectorIds' => UpdateTranslationsConnectorIds::class, + 'resolveTranslationsConnectorIds' => ResolveTranslationsConnectorIds::class, 'packagistRepository' => PackagistRepository::class, 'startQueueManagerHandler' => StartQueueManagerHandler::class, 'listenerRegister' => [ diff --git a/src/services/handlers/CreateJobHandler.php b/src/services/handlers/CreateJobHandler.php index 8cbac2e3..20d2f877 100644 --- a/src/services/handlers/CreateJobHandler.php +++ b/src/services/handlers/CreateJobHandler.php @@ -24,6 +24,7 @@ public function __invoke(CreateJobCommand $command, bool $asDraft = false): Job $job->authorId = $command->getAuthorId(); $job->title = $command->getTitle(); $job->liltJobId = null; + $job->attempt = 0; $job->status = $asDraft ? Job::STATUS_DRAFT : Job::STATUS_NEW; $job->sourceSiteId = $command->getSourceSiteId(); diff --git a/src/services/handlers/EditJobHandler.php b/src/services/handlers/EditJobHandler.php index 8a43cf1f..5a5a82b2 100644 --- a/src/services/handlers/EditJobHandler.php +++ b/src/services/handlers/EditJobHandler.php @@ -58,6 +58,7 @@ public function __invoke(EditJobCommand $command): Job $job->elementIds = $command->getEntries(); $job->translationWorkflow = $command->getTranslationWorkflow(); $job->versions = $command->getVersions(); + $job->attempt = 0; $jobRecord->setAttributes($job->getAttributes(), false); diff --git a/src/services/handlers/ResendJobHandler.php b/src/services/handlers/ResendJobHandler.php new file mode 100644 index 00000000..b9310766 --- /dev/null +++ b/src/services/handlers/ResendJobHandler.php @@ -0,0 +1,67 @@ + $jobId]); + $jobRecord->status = Job::STATUS_IN_PROGRESS; + $jobRecord->liltJobId = null; + $jobRecord->save(); + + // Remove all drafts related to job + $translationRecords = TranslationRecord::findAll(['jobId' => $jobId]); + array_map(static function (TranslationRecord $t) { + if ($t->translatedDraftId !== null) { + Craft::$app->elements->deleteElementById( + $t->translatedDraftId + ); + } + }, $translationRecords); + + + // Set translation status to in progress and connector translation id to null + TranslationRecord::updateAll([ + 'status' => TranslationRecord::STATUS_IN_PROGRESS, + 'connectorTranslationId' => null, + 'translatedDraftId' => null, + 'sourceContent' => null + ], ['jobId' => $jobId]); + + // Invalidate caches for job and translation types + Craft::$app->getElements()->invalidateCachesForElementType(Job::class); + Craft::$app->getElements()->invalidateCachesForElementType(Translation::class); + + // Trigger send job to connector + Queue::push( + new SendJobToConnector(['jobId' => $jobId]), + SendJobToConnector::PRIORITY, + 10 + ); + } +} diff --git a/src/services/handlers/ResolveTranslationsConnectorIds.php b/src/services/handlers/ResolveTranslationsConnectorIds.php new file mode 100644 index 00000000..cb19f0ac --- /dev/null +++ b/src/services/handlers/ResolveTranslationsConnectorIds.php @@ -0,0 +1,153 @@ +connectorTranslationRepository + ->findByJobId($job->liltJobId); + + $connectorTranslationsMapped = $this->mapConnectorTranslations($connectorTranslations->getResults()); + + $translationResponses = $this->filterTranslationResponses($connectorTranslationsMapped); + + return $this->updateTranslationRecords($translationResponses, $job); + } + + private function mapConnectorTranslations(array $connectorTranslations): array + { + $connectorTranslationsMapped = []; + foreach ($connectorTranslations as $translationResponse) { + try { + $elementId = Craftliltplugin::getInstance() + ->connectorTranslationRepository + ->getElementIdFromTranslationResponse($translationResponse); + } catch (WrongTranslationFilenameException $ex) { + Craft::error(sprintf("Can't get element id from file: %s", $translationResponse->getName())); + continue; + } + + $targetLanguage = $this->getTranslationTargetLanguage($translationResponse); + $connectorTranslationsMapped[$elementId][$targetLanguage][] = $translationResponse; + } + + return $connectorTranslationsMapped; + } + + private function filterTranslationResponses(array $connectorTranslationsMapped): array + { + $translationResponses = []; + foreach ($connectorTranslationsMapped as $elementId => $targetLanguages) { + foreach ($targetLanguages as $targetLanguage => $translations) { + if (count($translations) === 1) { + $translationResponses[] = $translations[0]; + continue; + } + + $translationResponses[] = $this->findCompleteTranslationResponses($translations); + } + } + + return $translationResponses; + } + + /** + * @param TranslationResponse[] $translations + * @return TranslationResponse + */ + private function findCompleteTranslationResponses(array $translations): TranslationResponse + { + $result = null; + foreach ($translations as $translation) { + // Search for complete translation + if ( + $translation->getStatus() == TranslationResponse::STATUS_EXPORT_COMPLETE + || $translation->getStatus() == TranslationResponse::STATUS_MT_COMPLETE + ) { + $result = $translation; + break; + } + } + + // Fallback to first translation + if ($result === null) { + $result = $translations[0]; + } + + return $result; + } + + private function updateTranslationRecords(array $translationResponses, Job $job): array + { + $result = []; + foreach ($translationResponses as $translationResponse) { + try { + $elementId = Craftliltplugin::getInstance() + ->connectorTranslationRepository + ->getElementIdFromTranslationResponse($translationResponse); + } catch (WrongTranslationFilenameException $ex) { + Craft::error(sprintf("Can't get element id from file: %s", $translationResponse->getName())); + continue; + } + + $targetLanguage = $this->getTranslationTargetLanguage($translationResponse); + $siteId = Craftliltplugin::getInstance() + ->languageMapper + ->getSiteIdByLanguage(trim($targetLanguage, '-')); + $versionId = $job->getElementVersionId($elementId); + + $translationRecord = TranslationRecord::findOne([ + 'targetSiteId' => $siteId, + 'versionId' => $versionId, + 'jobId' => $job->id + ]); + + if ($translationRecord === null) { + throw new RuntimeException(sprintf( + "Can't find translation for target %s, jobId %d, versionId %d", + trim($targetLanguage, '-'), + $job->id, + $versionId + )); + } + + $translationRecord->connectorTranslationId = $translationResponse->getId(); + $translationRecord->save(); + + $result[$translationResponse->getId()] = $translationResponse; + } + + return array_values($result); + } + + private function getTranslationTargetLanguage(TranslationResponse $translationResponse): string + { + if (empty($translationResponse->getTrgLocale())) { + return $translationResponse->getTrgLang(); + } + + return sprintf('%s-%s', $translationResponse->getTrgLang(), $translationResponse->getTrgLocale()); + } +} diff --git a/src/services/handlers/UpdateTranslationsConnectorIds.php b/src/services/handlers/UpdateTranslationsConnectorIds.php deleted file mode 100644 index 2cb159c8..00000000 --- a/src/services/handlers/UpdateTranslationsConnectorIds.php +++ /dev/null @@ -1,75 +0,0 @@ -connectorTranslationRepository->findByJobId( - $job->liltJobId - ); - - foreach ($connectorTranslations->getResults() as $translationResponse) { - try { - $elementId = Craftliltplugin::getInstance() - ->connectorTranslationRepository - ->getElementIdFromTranslationResponse($translationResponse); - } catch (WrongTranslationFilenameException $ex) { - Craft::error( - sprintf("Can't get element id from file: %s", $translationResponse->getName()) - ); - - continue; - } - - if (empty($translationResponse->getTrgLocale())) { - $targetLanguage = $translationResponse->getTrgLang(); - } else { - $targetLanguage = sprintf( - '%s-%s', - $translationResponse->getTrgLang(), - $translationResponse->getTrgLocale() - ); - } - - $translationRecord = TranslationRecord::findOne([ - 'targetSiteId' => Craftliltplugin::getInstance() - ->languageMapper - ->getSiteIdByLanguage( - trim($targetLanguage, '-') - ), - 'versionId' => $job->getElementVersionId($elementId), - 'jobId' => $job->id - ]); - - if ($translationRecord === null) { - throw new RuntimeException( - sprintf( - "Can't find translation for target %s, jobId %d, versionId %d", - trim($targetLanguage, '-'), - $job->id, - $job->getElementVersionId($elementId) - ) - ); - } - - $translationRecord->connectorTranslationId = $translationResponse->getId(); - $translationRecord->save(); - } - } -} diff --git a/tests/integration/controllers/job/PostJobRetryControllerCest.php b/tests/integration/controllers/job/PostJobRetryControllerCest.php index a017c062..edb3d2e0 100644 --- a/tests/integration/controllers/job/PostJobRetryControllerCest.php +++ b/tests/integration/controllers/job/PostJobRetryControllerCest.php @@ -19,6 +19,7 @@ use lilthq\craftliltplugin\Craftliltplugin; use lilthq\craftliltplugin\elements\Job; use lilthq\craftliltplugin\modules\FetchJobStatusFromConnector; +use lilthq\craftliltplugin\modules\SendJobToConnector; use lilthq\craftliltplugin\parameters\CraftliltpluginParameters; use lilthq\craftliltplugin\records\JobRecord; use lilthq\craftliltplugin\records\TranslationRecord; @@ -63,45 +64,30 @@ public function testRetrySuccess(IntegrationTester $I): void [$job, $translations] = $I->createJobWithTranslations([ 'title' => 'Awesome test job', 'elementIds' => [$element->id], - 'targetSiteIds' => [Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('es-ES')], + 'targetSiteIds' => [ + Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('es-ES'), + Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('ru-RU'), + Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('de-DE'), + ], 'sourceSiteId' => Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('en-US'), 'translationWorkflow' => SettingsResponse::LILT_TRANSLATION_WORKFLOW_VERIFIED, 'versions' => [], 'authorId' => 1, ]); + foreach ($translations as $translationRecord) { + $translationRecord->status = Job::STATUS_FAILED; + $translationRecord->save(); + } + $jobRecord = JobRecord::findOne(['id' => $job->id]); $jobRecord->status = Job::STATUS_FAILED; $jobRecord->save(); - $expectQueueJob = new FetchJobStatusFromConnector([ - 'liltJobId' => 777, + $expectQueueJob = new SendJobToConnector([ 'jobId' => $job->id ]); - $I->expectJobCreateRequest( - [ - 'project_prefix' => 'Awesome test job', - 'lilt_translation_workflow' => 'VERIFIED', - ], - 200, - ['id' => 777,] - ); - - $expectedUrl = sprintf( - '/api/v1.0/jobs/777/files?name=%s' - . '&srclang=en-US' - . '&trglang=es-ES' . - '&due=', - urlencode( - sprintf('element_%d_first-entry-user-1.json+html', $element->getId()) - ) - ); - $expectedBody = ExpectedElementContent::getExpectedBody($element); - - $I->expectJobTranslationsRequest($expectedUrl, $expectedBody, HttpCode::OK); - $I->expectJobStartRequest(777, HttpCode::OK); - $I->sendAjaxPostRequest( sprintf( '?p=admin/%s', @@ -111,22 +97,16 @@ public function testRetrySuccess(IntegrationTester $I): void ); $translations = array_map(static function (TranslationRecord $translationRecord) use ($element) { - $expectedDraftBody = ExpectedElementContent::getExpectedBody( - Craft::$app->elements->getElementById( - $translationRecord->translatedDraftId, - null, - $translationRecord->targetSiteId - ) - ); - Assert::assertSame(Job::STATUS_IN_PROGRESS, $translationRecord->status); Assert::assertSame($element->id, $translationRecord->versionId); - Assert::assertEquals($expectedDraftBody, $translationRecord->sourceContent); + Assert::assertNull($translationRecord->sourceContent); + Assert::assertNull($translationRecord->translatedDraftId); + Assert::assertNull($translationRecord->connectorTranslationId); + Assert::assertSame( Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('en-US'), $translationRecord->sourceSiteId ); - Assert::assertNotNull($translationRecord->translatedDraftId); return [ 'versionId' => $translationRecord->versionId, @@ -139,12 +119,18 @@ public function testRetrySuccess(IntegrationTester $I): void ]; }, TranslationRecord::findAll(['jobId' => $job->id, 'elementId' => $element->id])); - Assert::assertCount(1, $translations); + $expectedLanguages = ['es-ES', 'ru-RU', 'de-DE']; + $actualLanguages = Craftliltplugin::getInstance()->languageMapper->getLanguagesBySiteIds( + array_column($translations, 'targetSiteId') + ); + + sort($expectedLanguages); + sort($actualLanguages); + + Assert::assertCount(3, $translations); Assert::assertEquals( - ['es-ES'], - Craftliltplugin::getInstance()->languageMapper->getLanguagesBySiteIds( - array_column($translations, 'targetSiteId') - ) + $expectedLanguages, + $actualLanguages ); $I->assertJobInQueue($expectQueueJob); diff --git a/tests/integration/modules/FetchJobStatusFromConnectorCest.php b/tests/integration/modules/FetchJobStatusFromConnectorCest.php index 950b663b..a013d850 100644 --- a/tests/integration/modules/FetchJobStatusFromConnectorCest.php +++ b/tests/integration/modules/FetchJobStatusFromConnectorCest.php @@ -12,9 +12,14 @@ use IntegrationTester; use LiltConnectorSDK\Model\JobResponse; use LiltConnectorSDK\Model\SettingsResponse; +use LiltConnectorSDK\Model\TranslationResponse; use lilthq\craftliltplugin\Craftliltplugin; +use lilthq\craftliltplugin\elements\Job; +use lilthq\craftliltplugin\elements\Translation; use lilthq\craftliltplugin\modules\FetchJobStatusFromConnector; use lilthq\craftliltplugin\modules\FetchTranslationFromConnector; +use lilthq\craftliltplugin\modules\SendJobToConnector; +use lilthq\craftliltplugin\records\JobRecord; use lilthq\craftliltplugin\records\TranslationRecord; use lilthq\craftliltplugin\services\repositories\SettingsRepository; use lilthq\craftliltplugintests\integration\AbstractIntegrationCest; @@ -79,7 +84,7 @@ public function testExecuteSuccessVerified(IntegrationTester $I): void 'errorMsg' => null, 'id' => 11111, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'export_complete', + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, 'trgLang' => 'es', 'trgLocale' => 'ES', 'updatedAt' => '2022-06-02T23:01:42', @@ -89,7 +94,7 @@ public function testExecuteSuccessVerified(IntegrationTester $I): void 'errorMsg' => null, 'id' => 22222, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'export_complete', + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, 'trgLang' => 'de', 'trgLocale' => 'DE', 'updatedAt' => '2022-06-02T23:01:42', @@ -99,7 +104,7 @@ public function testExecuteSuccessVerified(IntegrationTester $I): void 'errorMsg' => null, 'id' => 33333, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'export_complete', + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, 'trgLang' => 'ru', 'trgLocale' => 'RU', 'updatedAt' => '2022-06-02T23:01:42', @@ -167,6 +172,170 @@ public function testExecuteSuccessVerified(IntegrationTester $I): void } } + /** + * @throws Exception + * @throws ModuleException + */ + public function testExecuteSuccessVerifiedDuplicate(IntegrationTester $I): void + { + $I->clearQueue(); + $I->disableOption(SettingsRepository::QUEUE_DISABLE_AUTOMATIC_SYNC); + + Db::truncateTable(Craft::$app->queue->tableName); + + $user = Craft::$app->getUsers()->getUserById(1); + $I->amLoggedInAs($user); + + $element = Entry::find() + ->where(['authorId' => 1]) + ->orderBy(['id' => SORT_DESC]) + ->one(); + + [$job, $translations] = $I->createJobWithTranslations([ + 'title' => 'Awesome test job', + 'elementIds' => [$element->id], + 'targetSiteIds' => '*', + 'sourceSiteId' => Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('en-US'), + 'translationWorkflow' => SettingsResponse::LILT_TRANSLATION_WORKFLOW_VERIFIED, + 'versions' => [], + 'authorId' => 1, + 'liltJobId' => 777, + ]); + + $I->expectJobGetRequest( + 777, + 200, + [ + 'status' => JobResponse::STATUS_COMPLETE + ] + ); + + $responseBody = [ + 'limit' => 25, + 'results' => [ + 0 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 11111, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, + 'trgLang' => 'es', + 'trgLocale' => 'ES', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 1 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 22222, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, + 'trgLang' => 'de', + 'trgLocale' => 'DE', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 2 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 33333, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_IMPORT_COMPLETE, + 'trgLang' => 'ru', + 'trgLocale' => 'RU', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 3 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 44444, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_IMPORT_COMPLETE, + 'trgLang' => 'ru', + 'trgLocale' => 'RU', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 4 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 55555, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_EXPORT_COMPLETE, + 'trgLang' => 'ru', + 'trgLocale' => 'RU', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 5 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 66666, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_IMPORT_COMPLETE, + 'trgLang' => 'ru', + 'trgLocale' => 'RU', + 'updatedAt' => '2022-06-02T23:01:42', + ], + ], + 'start' => 0, + ]; + + $I->expectTranslationsGetRequest( + 777, + 0, + 100, + HttpCode::OK, + $responseBody + ); + + $I->runQueue( + FetchJobStatusFromConnector::class, + [ + 'liltJobId' => $job->liltJobId, + 'jobId' => $job->id, + ] + ); + + $totalJobs = Craft::$app->queue->getJobInfo(); + + Assert::assertCount(3, $totalJobs); + $I->assertJobInQueue( + new FetchTranslationFromConnector([ + 'jobId' => $job->id, + 'translationId' => $translations[0]->id, + 'liltJobId' => 777 + ]) + ); + + $I->assertJobInQueue( + new FetchTranslationFromConnector([ + 'jobId' => $job->id, + 'translationId' => $translations[1]->id, + 'liltJobId' => 777 + ]) + ); + + $I->assertJobInQueue( + new FetchTranslationFromConnector([ + 'jobId' => $job->id, + 'translationId' => $translations[2]->id, + 'liltJobId' => 777 + ]) + ); + + $translationAssertions = [ + 'es-ES' => 11111, + 'de-DE' => 22222, + 'ru-RU' => 55555, + ]; + foreach ($translationAssertions as $language => $expectedConnectorTranslationId) { + $translationEs = TranslationRecord::findOne([ + 'jobId' => $job->id, + 'elementId' => $element->id, + 'targetSiteId' => Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage($language) + ]); + + Assert::assertSame($expectedConnectorTranslationId, $translationEs->connectorTranslationId); + } + } + /** * @throws Exception * @throws ModuleException @@ -213,7 +382,7 @@ public function testExecuteSuccessInstant(IntegrationTester $I): void 'errorMsg' => null, 'id' => 11111, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'mt_complete', + 'status' => TranslationResponse::STATUS_MT_COMPLETE, 'trgLang' => 'es', 'trgLocale' => 'ES', 'updatedAt' => '2022-06-02T23:01:42', @@ -223,7 +392,7 @@ public function testExecuteSuccessInstant(IntegrationTester $I): void 'errorMsg' => null, 'id' => 22222, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'mt_complete', + 'status' => TranslationResponse::STATUS_MT_COMPLETE, 'trgLang' => 'de', 'trgLocale' => 'DE', 'updatedAt' => '2022-06-02T23:01:42', @@ -233,7 +402,7 @@ public function testExecuteSuccessInstant(IntegrationTester $I): void 'errorMsg' => null, 'id' => 33333, 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), - 'status' => 'mt_complete', + 'status' => TranslationResponse::STATUS_MT_COMPLETE, 'trgLang' => 'ru', 'trgLocale' => 'RU', 'updatedAt' => '2022-06-02T23:01:42', @@ -301,6 +470,132 @@ public function testExecuteSuccessInstant(IntegrationTester $I): void } } + /** + * @throws Exception + * @throws ModuleException + */ + public function testExecuteSuccessInstantRetry(IntegrationTester $I): void + { + $I->clearQueue(); + $I->disableOption(SettingsRepository::QUEUE_DISABLE_AUTOMATIC_SYNC); + + Db::truncateTable(Craft::$app->queue->tableName); + + $user = Craft::$app->getUsers()->getUserById(1); + $I->amLoggedInAs($user); + + $element = Entry::find() + ->where(['authorId' => 1]) + ->orderBy(['id' => SORT_DESC]) + ->one(); + + [$job, $translations] = $I->createJobWithTranslations([ + 'title' => 'Awesome test job', + 'elementIds' => [$element->id], + 'targetSiteIds' => '*', + 'sourceSiteId' => Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage('en-US'), + 'translationWorkflow' => SettingsResponse::LILT_TRANSLATION_WORKFLOW_INSTANT, + 'versions' => [], + 'authorId' => 1, + 'liltJobId' => 777, + ]); + + $I->expectJobGetRequest( + 777, + 200, + [ + 'status' => JobResponse::STATUS_FAILED + ] + ); + + $responseBody = [ + 'limit' => 25, + 'results' => [ + 0 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 11111, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_MT_COMPLETE, + 'trgLang' => 'es', + 'trgLocale' => 'ES', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 1 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 22222, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_MT_COMPLETE, + 'trgLang' => 'de', + 'trgLocale' => 'DE', + 'updatedAt' => '2022-06-02T23:01:42', + ], + 2 => [ + 'createdAt' => '2022-05-29T11:31:58', + 'errorMsg' => null, + 'id' => 33333, + 'name' => sprintf('497058_element_%d_first-entry-user-1.json+html', $element->id), + 'status' => TranslationResponse::STATUS_MT_COMPLETE, + 'trgLang' => 'ru', + 'trgLocale' => 'RU', + 'updatedAt' => '2022-06-02T23:01:42', + ], + ], + 'start' => 0, + ]; + + $I->expectTranslationsGetRequest( + 777, + 0, + 100, + HttpCode::OK, + $responseBody + ); + + $I->runQueue( + FetchJobStatusFromConnector::class, + [ + 'liltJobId' => $job->liltJobId, + 'jobId' => $job->id, + ] + ); + + $totalJobs = Craft::$app->queue->getJobInfo(); + + Assert::assertCount(1, $totalJobs); + $I->assertJobInQueue( + new SendJobToConnector([ + 'jobId' => $job->id, + ]) + ); + + $jobRecord = JobRecord::findOne(['id' => $job->id]); + Assert::assertSame(Job::STATUS_IN_PROGRESS, $jobRecord->status); + Assert::assertNull($jobRecord->liltJobId); + Assert::assertSame(1, $jobRecord->attempt); + + $translationAssertions = [ + 'es-ES' => null, + 'de-DE' => null, + 'ru-RU' => null, + ]; + foreach ($translationAssertions as $language => $expectedConnectorTranslationId) { + $actualTranslation = TranslationRecord::findOne([ + 'jobId' => $job->id, + 'elementId' => $element->id, + 'targetSiteId' => Craftliltplugin::getInstance()->languageMapper->getSiteIdByLanguage($language) + ]); + + Assert::assertSame($expectedConnectorTranslationId, $actualTranslation->connectorTranslationId); + Assert::assertSame(TranslationRecord::STATUS_IN_PROGRESS, $actualTranslation->status); + + Assert::assertNull($actualTranslation->connectorTranslationId); + Assert::assertNull($actualTranslation->translatedDraftId); + Assert::assertNull($actualTranslation->sourceContent); + } + } + /** * @throws Exception * @throws ModuleException