Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,12 @@
# Release Notes for KlaviyoConnect

## 7.2.4 - 2026-05-05

### Fixed

- Klaviyo API failures no longer break host requests. Errors are caught, logged to the `klaviyoconnect` category, and the request continues.
- Klaviyo API calls now hard-cap at 5s (5s connect) with retries disabled, so an unresponsive Klaviyo can't stall the host request.

### 7.2.3 - 2025-06-21

### Fixed
Expand Down
2 changes: 1 addition & 1 deletion composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
"description": "Craft Commerce",
"license": "proprietary",
"type": "craft-plugin",
"version": "7.2.3",
"version": "7.2.4",
"keywords": [
"klaviyo"
],
Expand Down
25 changes: 21 additions & 4 deletions src/Plugin.php
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
use fostercommerce\klaviyoconnect\queue\jobs\TrackOrderComplete;
use fostercommerce\klaviyoconnect\utilities\KCUtilities;
use fostercommerce\klaviyoconnect\variables\Variable;
use Throwable;
use yii\base\Event;

/**
Expand Down Expand Up @@ -71,14 +72,22 @@ static function (RegisterUrlRulesEvent $event): void {

if ($settings->trackSaveUser) {
Event::on(User::class, User::EVENT_AFTER_SAVE, static function (Event $event): void {
self::getInstance()->track->onSaveUser($event);
try {
self::getInstance()->track->onSaveUser($event);
} catch (Throwable $throwable) {
Craft::error('Klaviyo onSaveUser failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}
});
}

if (Craft::$app->plugins->isPluginEnabled('commerce')) {
if ($settings->trackCommerceCartUpdated) {
Event::on(Order::class, Order::EVENT_AFTER_SAVE, static function (Event $e): void {
self::getInstance()->track->onCartUpdated($e);
try {
self::getInstance()->track->onCartUpdated($e);
} catch (Throwable $throwable) {
Craft::error('Klaviyo onCartUpdated failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}
});
}

Expand All @@ -98,7 +107,11 @@ static function (RegisterUrlRulesEvent $event): void {
OrderHistories::class,
OrderHistories::EVENT_ORDER_STATUS_CHANGE,
static function (OrderStatusEvent $e): void {
self::getInstance()->track->onStatusChanged($e);
try {
self::getInstance()->track->onStatusChanged($e);
} catch (Throwable $throwable) {
Craft::error('Klaviyo onStatusChanged failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}
}
);
}
Expand All @@ -108,7 +121,11 @@ static function (OrderStatusEvent $e): void {
Payments::class,
Payments::EVENT_AFTER_REFUND_TRANSACTION,
static function (RefundTransactionEvent $e): void {
self::getInstance()->track->onOrderRefunded($e);
try {
self::getInstance()->track->onOrderRefunded($e);
} catch (Throwable $throwable) {
Craft::error('Klaviyo onOrderRefunded failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}
}
);
}
Expand Down
34 changes: 27 additions & 7 deletions src/controllers/ApiController.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
use fostercommerce\klaviyoconnect\models\EventProperties;
use fostercommerce\klaviyoconnect\Plugin;
use fostercommerce\klaviyoconnect\queue\jobs\SyncOrders;
use GuzzleHttp\Exception\RequestException;
use Throwable;
use yii\web\NotFoundHttpException;
use yii\web\Response as YiiResponse;

Expand All @@ -29,9 +29,23 @@ public function actionTrack(): mixed
{
$this->requirePostRequest();

$this->identify();
$this->trackEvent();
$this->addProfileToLists();
try {
$this->identify();
} catch (Throwable $throwable) {
Craft::error('Klaviyo identify failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}

try {
$this->trackEvent();
} catch (Throwable $throwable) {
Craft::error('Klaviyo trackEvent failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}

try {
$this->addProfileToLists();
} catch (Throwable $throwable) {
Craft::error('Klaviyo addProfileToLists failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}

$request = Craft::$app->getRequest();
if ($request->isAjax && ! $request->getParam('forward')) {
Expand Down Expand Up @@ -76,7 +90,12 @@ public function actionSyncOrders(): void
*/
public function actionIdentify(): void
{
$this->identify();
try {
$this->identify();
} catch (Throwable $throwable) {
Craft::error('Klaviyo identify failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}

$this->forwardOrRedirect();
}

Expand Down Expand Up @@ -169,8 +188,9 @@ private function identify(): void
$profile = $this->mapProfile();
try {
Plugin::getInstance()->track->identifyUser($profile);
} catch (RequestException) {
// Swallow. Klaviyo responds with a 200.
} catch (Throwable $throwable) {
// Swallow. Klaviyo flake/outage must not break host request flow.
Craft::error('Klaviyo identifyUser failed: ' . $throwable->getMessage(), 'klaviyoconnect');
}
}

Expand Down
9 changes: 8 additions & 1 deletion src/services/Api.php
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,14 @@ public function init(): void
{
parent::init();

$this->api = new KlaviyoAPI($this->getSetting('klaviyoApiKey'));
$this->api = new KlaviyoAPI(
$this->getSetting('klaviyoApiKey'),
num_retries: 0,
guzzle_options: [
'connect_timeout' => 5,
'timeout' => 5,
],
);
}

public function track(string $event, array $profile, ?EventProperties $eventProperties = null, ?string $timestamp = null): void
Expand Down
Loading