Install the Push Notification feature
Edit on GitHubThis document describes how to install the Push Notification feature.
Install feature core
Follow the steps below to install the Push Notification feature core.
Prerequisites
Install the required features:
NAME | VERSION | INSTALLATION GUIDE |
---|---|---|
Spryker Core | 202404.0 | Install the Spryker Core feature |
1) Install the required modules
Install the required modules using Composer:
composer require spryker-feature/push-notification: "202404.0" --update-with-dependencies
Make sure that the following modules have been installed:
MODULE | EXPECTED DIRECTORY |
---|---|
PushNotification | vendor/spryker/push-notification |
PushNotificationExtension | vendor/spryker/push-notification-extension |
PushNotificationWebPushPhp | vendor/spryker/push-notification-web-push-php |
PushNotificationsBackendApi | vendor/spryker/push-notifications-backend-api |
2) Set up database schema and transfer objects
Apply the database changes and generate entity and transfer changes:
console propel:install
console transfer:generate
Make sure that the following changes have been applied by checking your database:
DATABASE ENTITY | TYPE | EVENT |
---|---|---|
spy_push_notification | table | created |
spy_push_notification_subscription | table | created |
spy_push_notification_group | table | created |
spy_push_notification_provider | table | created |
spy_push_notification_subscription_delivery_log | table | created |
Make sure that the following changes have been triggered in transfer objects:
TRANSFER | TYPE | EVENT | PATH |
---|---|---|---|
PushNotificationSubscriptionCollectionRequest | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionCollectionRequestTransfer |
PushNotificationSubscriptionCollectionResponse | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionCollectionResponseTransfer |
PushNotificationSubscriptionCollectionDeleteCriteria | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionCollectionDeleteCriteriaTransfer |
PushNotificationSubscriptionCollection | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionCollectionTransfer |
PushNotificationSubscription | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionTransfer |
PushNotificationSubscriptionDeliveryLog | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionDeliveryLogTransfer |
PushNotificationUser | class | created | src/Generated/Shared/Transfer/PushNotificationUserTransfer |
PushNotificationCollectionRequest | class | created | src/Generated/Shared/Transfer/PushNotificationCollectionRequestTransfer |
PushNotificationCollectionResponse | class | created | src/Generated/Shared/Transfer/PushNotificationCollectionResponseTransfer |
PushNotification | class | created | src/Generated/Shared/Transfer/PushNotificationTransfer |
PushNotificationGroup | class | created | src/Generated/Shared/Transfer/PushNotificationGroupTransfer |
PushNotificationGroupCriteria | class | created | src/Generated/Shared/Transfer/PushNotificationGroupCriteriaTransfer |
PushNotificationGroupConditions | class | created | src/Generated/Shared/Transfer/PushNotificationGroupConditionsTransfer |
PushNotificationGroupCollection | class | created | src/Generated/Shared/Transfer/PushNotificationGroupCollectionTransfer |
PushNotificationProviderCriteria | class | created | src/Generated/Shared/Transfer/PushNotificationProviderCriteriaTransfer |
PushNotificationSubscriptionCriteria | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionCriteriaTransfer |
PushNotificationProviderCollection | class | created | src/Generated/Shared/Transfer/PushNotificationProviderCollectionTransfer |
PushNotificationProviderCollectionRequest | class | created | src/Generated/Shared/Transfer/PushNotificationProviderCollectionRequestTransfer |
PushNotificationProviderCollectionResponse | class | created | src/Generated/Shared/Transfer/PushNotificationProviderCollectionResponseTransfer |
PushNotificationProviderCollectionDeleteCriteria | class | created | src/Generated/Shared/Transfer/ApiPushNotificationGroupsAttributesTransferTransfer |
PushNotificationProvider | class | created | src/Generated/Shared/Transfer/PushNotificationProviderTransfer |
PushNotificationProviderConditions | class | created | src/Generated/Shared/Transfer/PushNotificationProviderConditionsTransfer |
PushNotificationSubscriptionConditions | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionConditionsTransfer |
ErrorCollection | class | created | src/Generated/Shared/Transfer/ErrorCollectionTransfer |
Error | class | created | src/Generated/Shared/Transfer/ErrorTransfer |
PushNotificationCollection | class | created | src/Generated/Shared/Transfer/PushNotificationCollectionTransfer |
PushNotificationCriteria | class | created | src/Generated/Shared/Transfer/PushNotificationCriteriaTransfer |
PushNotificationConditions | class | created | src/Generated/Shared/Transfer/PushNotificationConditionsTransfer |
PushNotificationSubscriptionsBackendApiAttributes | class | created | src/Generated/Shared/Transfer/PushNotificationSubscriptionsBackendApiAttributesTransfer |
PushNotificationGroupsBackendApiAttributes | class | created | src/Generated/Shared/Transfer/PushNotificationGroupsBackendApiAttributesTransfer |
PushNotificationProvidersBackendApiAttributes | class | created | src/Generated/Shared/Transfer/PushNotificationProvidersBackendApiAttributesTransfer |
GlueRequest | class | created | src/Generated/Shared/Transfer/GlueRequestTransfer |
GlueRequestUser | class | created | src/Generated/Shared/Transfer/GlueRequestUserTransfer |
GlueResponse | class | created | src/Generated/Shared/Transfer/GlueResponseTransfer |
GlueResource | class | created | src/Generated/Shared/Transfer/GlueResourceTransfer |
GlueResourceMethodConfiguration | class | created | src/Generated/Shared/Transfer/GlueResourceMethodConfigurationTransfer |
GlueError | class | created | src/Generated/Shared/Transfer/GlueErrorTransfer |
Sort | class | created | src/Generated/Shared/Transfer/SortTransfer |
Pagination | class | created | src/Generated/Shared/Transfer/PaginationTransfer |
LocaleCriteria | class | created | src/Generated/Shared/Transfer/LocaleCriteriaTransfer |
LocaleConditions | class | created | src/Generated/Shared/Transfer/LocaleConditionsTransfer |
Locale | class | created | src/Generated/Shared/Transfer/LocaleTransfer |
3) Set up configuration
- To make the
push-notification-subscriptions
andpush-notification-providers
resources protected, adjust the protected paths configuration:
src/Pyz/Shared/GlueBackendApiApplicationAuthorizationConnector/GlueBackendApiApplicationAuthorizationConnectorConfig.php
<?php
namespace Pyz\Shared\GlueBackendApiApplicationAuthorizationConnector;
use Spryker\Shared\GlueBackendApiApplicationAuthorizationConnector\GlueBackendApiApplicationAuthorizationConnectorConfig as SprykerGlueBackendApiApplicationAuthorizationConnectorConfig;
class GlueBackendApiApplicationAuthorizationConnectorConfig extends SprykerGlueBackendApiApplicationAuthorizationConnectorConfig
{
/**
* @return array<string, mixed>
*/
public function getProtectedPaths(): array
{
return [
'/push-notification-subscriptions' => [
'isRegularExpression' => false,
],
'/\/push-notification-providers.*/' => [
'isRegularExpression' => true,
],
];
}
}
- Add the configuration defining *Voluntary Application Server Identity (VAPID) keys, which are used by the push notification:
CONFIGURATION | SPECIFICATION | NAMESPACE |
---|---|---|
PushNotificationWebPushPhpConstants::VAPID_PUBLIC_KEY | Provides the VAPID public key. Used for authentication to send push notifications. |
Spryker\Shared\PushNotificationWebPushPhp |
PushNotificationWebPushPhpConstants::VAPID_PRIVATE_KEY | Provides the VAPID private key. Used for authentication to send push notifications. |
Spryker\Shared\PushNotificationWebPushPhp |
PushNotificationWebPushPhpConstants::VAPID_SUBJECT | Provides the VAPID subject. Used for authentication to send push notifications. |
Spryker\Shared\PushNotificationWebPushPhp |
config/Shared/config_default.php
<?php
use Spryker\Shared\PushNotificationWebPushPhp\PushNotificationWebPushPhpConstants;
// >>> Push Notification Web Push Php
$config[PushNotificationWebPushPhpConstants::VAPID_PUBLIC_KEY] = getenv('SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_PUBLIC_KEY');
$config[PushNotificationWebPushPhpConstants::VAPID_PRIVATE_KEY] = getenv('SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_PRIVATE_KEY');
$config[PushNotificationWebPushPhpConstants::VAPID_SUJECT] = getenv('SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_SUBJECT');
- Add the VAPID keys to the needed deploy file:
image:
tag: spryker/php:8.1
environment:
SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_PUBLIC_KEY: 'Your public key.'
SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_PRIVATE_KEY: 'Your private key.'
SPRYKER_PUSH_NOTIFICATION_WEB_PUSH_PHP_VAPID_SUBJECT: 'https://your.subject'
VAPID is a new way to send and receive website push notifications. Your VAPID keys let you send web push campaigns without sending them through a service like Firebase Cloud Messaging (FCM).
To generate VAPID
keys, you can use the following tools:
https://vapidkeys.com/
: an online tool to generate keys.https://www.npmjs.com/package//web-push
: a Node.js package that generates VAPID keys.
The following example command generates VAPID
keys by using the web-push
Node.js library:
web-push generate-vapid-keys --json
4) Add translations
- Append glossary according to your configuration:
>push_notification.validation.error.push_notification_provider_not_found,The push notification provider was not found.,en_US
push_notification.validation.error.push_notification_provider_not_found,Der Push-Benachrichtigungsanbieters wurde nicht gefunden.,de_DE
push_notification.validation.error.push_notification_not_found,Push notification not found.,en_US
push_notification.validation.error.push_notification_not_found,Push-Nachricht nicht gefunden.,de_DE
push_notification.validation.error.wrong_group_name,Wrong group name.,en_US
push_notification.validation.error.wrong_group_name,Falscher Gruppenname.,de_DE
push_notification.validation.error.push_notification_already_exists,Push notification subscription already exists.,en_US
push_notification.validation.error.push_notification_already_exists,Das Abonnement für Push-Benachrichtigungen existiert bereits.,de_DE
push_notification.validation.error.locale_not_found,Provided locale not found.,en_US
push_notification.validation.error.locale_not_found,Bereitgestellte Sprache nicht gefunden.,de_DE
push_notification_web_push_php.validation.error.invalid_payload_structure,Invalid payload structure.,en_US
push_notification_web_push_php.validation.error.invalid_payload_structure,Ungültige Nutzlaststruktur.,de_DE
push_notification_web_push_php.validation.error.payload_length_exceeded,The maximum payload length exceeded.,en_US
push_notification_web_push_php.validation.error.payload_length_exceeded,Die maximale Länge der Nutzlast wurde überschritten.,de_DE
push_notification.validation.push_notification_provider_name_exists,A push notification provider with the same name already exists.,en_US
push_notification.validation.push_notification_provider_name_exists,Ein Push-Benachrichtigungsanbieter mit demselben Namen ist bereits vorhanden.,de_DE
push_notification.validation.push_notification_provider_name_wrong_length,A push notification provider name must have length from %min% to %max% characters.,en_US
push_notification.validation.push_notification_provider_name_wrong_length,Ein Push-Benachrichtigungsanbietername muss eine Länge von %min% bis %max% Zeichen haben.,de_DE
push_notification.validation.push_notification_provider_name_is_not_unique,A push notification provider with the same name already exists in request.,en_US
push_notification.validation.push_notification_provider_name_is_not_unique,Ein Push-Benachrichtigungsanbieter mit demselben Namen ist bereits in der Anfrage vorhanden.,de_DE
push_notification.validation.push_notification_exists,Unable to delete push notification provider while push notification exists.,en_US
push_notification.validation.push_notification_exists,Der Push-Benachrichtigungsanbieter kann nicht gelöscht werden solange Push-Benachrichtigungen vorhanden sind.,de_DE
push_notification.validation.push_notification_subscription_exists,Unable to delete push notification provider while push notification subscription exists.,en_US
push_notification.validation.push_notification_subscription_exists,Der Push-Benachrichtigungsanbieter kann nicht gelöscht werden solange Push-Benachrichtigungsabonnements vorhanden sind.,de_DE
push_notification.validation.wrong_request_body,Wrong request body.,en_US
push_notification.validation.wrong_request_body,Falscher Anforderungstext.,de_DE
- Import data:
console data:import glossary
5) Set up behavior
- Enable the following behaviors by registering the plugins:
PLUGIN | SPECIFICATION | PREREQUISITES | NAMESPACE |
---|---|---|---|
PushNotificationWebPushPhpPushNotificationSubscriptionValidatorPlugin | Validates the push notification subscription payload. | Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification | |
PushNotificationWebPushPhpPayloadLengthPushNotificationValidatorPlugin | Validates the push notification payload. | Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification | |
PushNotificationWebPushPhpPushNotificationSenderPlugin | Sends the push notification collection. | Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification |
src/Pyz/Zed/PushNotification/PushNotificationDependencyProvider.php
<?php
namespace Pyz\Zed\PushNotification;
use Spryker\Zed\PushNotification\PushNotificationDependencyProvider as SprykerPushNotificationDependencyProvider;
use Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification\PushNotificationWebPushPhpPayloadLengthPushNotificationValidatorPlugin;
use Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification\PushNotificationWebPushPhpPushNotificationSenderPlugin;
use Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\PushNotification\PushNotificationWebPushPhpPushNotificationSubscriptionValidatorPlugin;
class PushNotificationDependencyProvider extends SprykerPushNotificationDependencyProvider
{
/**
* @return list<\Spryker\Zed\PushNotificationExtension\Dependency\Plugin\PushNotificationSubscriptionValidatorPluginInterface>
*/
protected function getPushNotificationSubscriptionValidatorPlugins(): array
{
return [
new PushNotificationWebPushPhpPushNotificationSubscriptionValidatorPlugin(),
];
}
/**
* @return list<\Spryker\Zed\PushNotificationExtension\Dependency\Plugin\PushNotificationValidatorPluginInterface>
*/
protected function getPushNotificationValidatorPlugins(): array
{
return [
new PushNotificationWebPushPhpPayloadLengthPushNotificationValidatorPlugin(),
];
}
/**
* @return list<\Spryker\Zed\PushNotificationExtension\Dependency\Plugin\PushNotificationSenderPluginInterface>
*/
protected function getPushNotificationSenderPlugins(): array
{
return [
new PushNotificationWebPushPhpPushNotificationSenderPlugin(),
];
}
}
- Enable the following installer plugins:
PLUGIN | SPECIFICATION | PREREQUISITES | NAMESPACE |
---|---|---|---|
PushNotificationWebPushPhpProviderInstallerPlugin | Adds the web-push provider to the list of available push notification providers. |
Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\Installer |
src/Pyz/Zed/Installer/InstallerDependencyProvider.php
<?php
namespace Pyz\Zed\Installer;
use Spryker\Zed\Installer\InstallerDependencyProvider as SprykerInstallerDependencyProvider;
use Spryker\Zed\PushNotificationWebPushPhp\Communication\Plugin\Installer\PushNotificationWebPushPhpProviderInstallerPlugin;
class InstallerDependencyProvider extends SprykerInstallerDependencyProvider
{
/**
* @return list<\Spryker\Zed\Installer\Dependency\Plugin\InstallerPluginInterface|\Spryker\Zed\InstallerExtension\Dependency\Plugin\InstallerPluginInterface>
*/
public function getInstallerPlugins(): array
{
return [
new PushNotificationWebPushPhpProviderInstallerPlugin(),
];
}
}
Ensure that the installer plugin works correctly:
- Execute install plugins:
docker/sdk console setup:init-db
- Check that the
web-push-php
push notification provider exists in thespy_push_notification_provider
database table.
- Enable the following console command plugins:
PLUGIN | SPECIFICATION | PREREQUISITES | NAMESPACE |
---|---|---|---|
SendPushNotificationConsole | Sends notifications in an async way. | Spryker\Zed\PushNotification\Communication\Console | |
DeleteExpiredPushNotificationSubscriptionConsole | Deletes expired push notification subscriptions from Persistence . |
Spryker\Zed\PushNotification\Communication\Console |
src/Pyz/Zed/Console/ConsoleDependencyProvider.php
<?php
namespace Pyz\Zed\Console;
use Spryker\Zed\Console\ConsoleDependencyProvider as SprykerConsoleDependencyProvider;
use Spryker\Zed\PushNotification\Communication\Console\DeleteExpiredPushNotificationSubscriptionConsole;
use Spryker\Zed\PushNotification\Communication\Console\SendPushNotificationConsole;
class ConsoleDependencyProvider extends SprykerConsoleDependencyProvider
{
/**
* @param \Spryker\Zed\Kernel\Container $container
*
* @return list<\Symfony\Component\Console\Command\Command>
*/
protected function getConsoleCommands(Container $container): array
{
return [
new DeleteExpiredPushNotificationSubscriptionConsole(),
new SendPushNotificationConsole(),
];
}
}
- To enable the Backend API, register the plugin:
PLUGIN | SPECIFICATION | PREREQUISITES | NAMESPACE |
---|---|---|---|
PushNotificationSubscriptionsBackendResourcePlugin | Registers the push-notification-subscriptions resource. |
Spryker\Glue\PushNotificationsBackendApi\Plugin\GlueBackendApiApplication | |
PushNotificationProvidersBackendResourcePlugin | Registers the push-notification-providers resource. |
Spryker\Glue\PushNotificationsBackendApi\Plugin\GlueBackendApiApplication |
src/Pyz/Glue/GlueBackendApiApplication/GlueBackendApiApplicationDependencyProvider.php
<?php
namespace Pyz\Glue\GlueBackendApiApplication;
use Spryker\Glue\GlueBackendApiApplication\GlueBackendApiApplicationDependencyProvider as SprykerGlueBackendApiApplicationDependencyProvider;
use Spryker\Glue\PushNotificationsBackendApi\Plugin\GlueBackendApiApplication\PushNotificationSubscriptionsBackendResourcePlugin;
use Spryker\Glue\PushNotificationsBackendApi\Plugin\GlueBackendApiApplication\PushNotificationProvidersBackendResourcePlugin;
class GlueBackendApiApplicationDependencyProvider extends SprykerGlueBackendApiApplicationDependencyProvider
{
/**
* @return array<\Spryker\Glue\GlueApplicationExtension\Dependency\Plugin\ResourceInterface>
*/
protected function getResourcePlugins(): array
{
return [
new PushNotificationSubscriptionsBackendResourcePlugin(),
new PushNotificationProvidersBackendResourcePlugin(),
];
}
}
Verify the push notifications
- Create a simple single-page demo application.
- Generate
VAPID
keys athttps://vapidkeys.com/
. - Create a directory for the demo application:
mkdir push_notification_spa
- In the
push_notification_spa
directory, create the following files:
…/push_notification_spa/index.html
<html>
<head>
<title>Web Push sandbox</title>
</head>
<body>
<h1>Web Push sandbox</h1>
<button id="push-button">Enable browser notifications</button>
<button id="subscription-button">Create Push notification subscription</button>
<script type="text/javascript" src="app.js"></script>
</body>
</html>
.../push_notification_spa/app.js
document.addEventListener('DOMContentLoaded', () => {
const applicationServerKey = '';
let isPushEnabled = false;
const pushButton = document.querySelector('#push-button');
if (!pushButton) {
return;
}
pushButton.addEventListener('click', function () {
if (isPushEnabled) {
disablePush();
} else {
enablePush();
}
});
const subscriptionButton = document.querySelector('#subscription-button');
if (!subscriptionButton) {
return;
}
navigator.serviceWorker.register('serviceWorker.js').then(
() => {
console.log('[SW] Service worker has been registered');
},
e => {
console.error('[SW] Service worker registration failed', e);
changePushButtonState('incompatible');
}
);
subscriptionButton.addEventListener('click', async function () {
let serviceWorkerRegistration = await navigator.serviceWorker.ready;
let subscription = await serviceWorkerRegistration.pushManager.subscribe({
userVisibleOnly: true,
applicationServerKey: urlBase64ToUint8Array(applicationServerKey),
})
push_sendSubscriptionToServer(subscription); // pass subscription here
});
if (!('serviceWorker' in navigator)) {
console.warn('Service workers are not supported by this browser');
changePushButtonState('incompatible');
return;
}
if (!('PushManager' in window)) {
console.warn('Push notifications are not supported by this browser');
changePushButtonState('incompatible');
return;
}
if (!('showNotification' in ServiceWorkerRegistration.prototype)) {
console.warn('Notifications are not supported by this browser');
changePushButtonState('incompatible');
return;
}
// Check the current Notification permission.
// If its denied, the button should appear as such until the user changes the permission manually
if (Notification.permission === 'denied') {
console.warn('Notifications are denied by the user');
changePushButtonState('incompatible');
}
function changePushButtonState(state) {
switch (state) {
case 'enabled':
pushButton.disabled = false;
pushButton.textContent = 'Disable Push notifications';
isPushEnabled = true;
break;
case 'disabled':
pushButton.disabled = false;
pushButton.textContent = 'Enable Push notifications';
isPushEnabled = false;
break;
case 'computing':
pushButton.disabled = true;
pushButton.textContent = 'Loading...';
break;
case 'incompatible':
pushButton.disabled = true;
pushButton.textContent = 'Push notifications are not compatible with this browser';
break;
default:
console.error('Unhandled push button state', state);
break;
}
}
function urlBase64ToUint8Array(base64String) {
const padding = '='.repeat((4 - (base64String.length % 4)) % 4);
const base64 = (base64String + padding).replace(/\-/g, '+').replace(/_/g, '/');
const rawData = window.atob(base64);
const outputArray = new Uint8Array(rawData.length);
for (let i = 0; i < rawData.length; ++i) {
outputArray[i] = rawData.charCodeAt(i);
}
return outputArray;
}
function checkNotificationPermission() {
return new Promise((resolve, reject) => {
if (Notification.permission === 'denied') {
return reject(new Error('Push messages are blocked.'));
}
if (Notification.permission === 'granted') {
return resolve();
}
if (Notification.permission === 'default') {
return Notification.requestPermission().then(result => {
if (result !== 'granted') {
reject(new Error('Bad permission result'));
} else {
resolve();
}
});
}
return reject(new Error('Unknown permission'));
});
}
function enablePush() {
changePushButtonState('computing');
checkNotificationPermission()
.then(() => navigator.serviceWorker.ready)
.then(serviceWorkerRegistration =>
serviceWorkerRegistration.pushManager.subscribe({
userVisibleOnly: true,
applicationServerKey: urlBase64ToUint8Array(applicationServerKey),
})
)
.then(subscription => {
console.info('Browser notifications are activated!');
changePushButtonState('enabled')
})
.then(subscription => subscription && changePushButtonState('enabled')) // update your UI
.catch(e => {
if (Notification.permission === 'denied') {
console.warn('Notifications are denied by the user.');
changePushButtonState('incompatible');
} else {
console.error('Impossible to subscribe to push notifications', e);
changePushButtonState('disabled');
}
});
}
function disablePush() {
changePushButtonState('computing');
// To unsubscribe from push messaging, you need to get the subscription object
navigator.serviceWorker.ready
.then(serviceWorkerRegistration => serviceWorkerRegistration.pushManager.getSubscription())
.then(subscription => {
changePushButtonState('disabled');
})
.then(subscription => subscription.unsubscribe())
.then(() => changePushButtonState('disabled'))
.catch(e => {
// We failed to unsubscribe, this can lead to
// an unusual state, so it may be best to remove
// the users data from your data store and
// inform the user that you have done so
console.error('Error when unsubscribing the user', e);
changePushButtonState('disabled');
});
}
async function getToken() {
const response = await fetch('http://glue-backend.de.spryker.local/token', {
method: "POST",
body: JSON.stringify({
grantType: "password",
username: "",
password: ""
}),
});
return response.json()
}
async function push_sendSubscriptionToServer(subscription) {
const tokenScos = await getToken();
const accessToken = tokenScos[0].access_token;
const key = subscription.getKey('p256dh');
const token = subscription.getKey('auth');
return fetch('http://glue-backend.de.spryker.local/push-notification-subscriptions', {
method: 'POST',
headers: {
'Accept': 'application/json',
'Authorization': `Bearer ${accessToken}`,
'Content-Type': 'application/vnd.api+json'
},
body: JSON.stringify({
data: {
type: "push-notification-subscriptions",
attributes: {
providerName: "web-push-php",
group: {
name: "warehouse",
identifier: 1234
},
'payload': {
endpoint: subscription.endpoint,
publicKey: key ? btoa(String.fromCharCode.apply(null, new Uint8Array(key))) : null,
authToken: token ? btoa(String.fromCharCode.apply(null, new Uint8Array(token))) : null,
}
}
}
}),
}).then(() => subscription);
}
});
…/push_notification_spa/serviceWorker.js
self.addEventListener('push', function (event) {
if (!(self.Notification && self.Notification.permission === 'granted')) {
return;
}
const sendNotification = body => {
const title = "Web Push example";
return self.registration.showNotification(title, {
body,
});
};
if (event.data) {
const message = event.data.text();
event.waitUntil(sendNotification(message));
}
});
- To set up credential, in
.../push_notification_spa/app.js
, set your VAPID public key for theapplicationServerKey
variable. - In
.../push_notification_spa/app.js
, find thegetToken()
method and set the credentials of the user that works in your system. - Run the local HTTP server with the demo app:
php -S localhost:8000
- To enable the push notification, on the Web Push sandbox page, click Enable browser notifications.
- To create a push notification subscription, on the Web Push sandbox page, click Create Push notification subscription.
- Create the push notification by adding it manually to the
spy_push_notification
database table. Use the same group and notification provider that is used by the subscription. - Send the push notification:
docker/sdk console send-push-notifications
The notification is displayed with content from the `spy_push_notification.payload` database field.
- Change the
spy_push_notification_subscription.expired_at
subscription expiration date to the previous year’s date. - Remove the outdated subscriptions:
docker/sdk console delete-expired-push-notification-subscriptions
6) Set up a cron job to automate push notifications
In the cron-job list, enable the send-push-notifications
and delete-expired-push-notification-subscriptions
console commands:
config/Zed/cronjobs/jobs.php
<?php
/**
* Notes:
*
* - jobs[]['name'] must not contain spaces or any other characters that have to be urlencode()'d
* - jobs[]['role'] default value is 'admin'
*/
$stores = require(APPLICATION_ROOT_DIR . '/config/Shared/stores.php');
$allStores = array_keys($stores);
/* Push notification */
$jobs[] = [
'name' => 'delete-expired-push-notification-subscriptions',
'command' => '$PHP_BIN vendor/bin/console push-notification:delete-expired-push-notification-subscriptions',
'schedule' => '0 0 * * 0',
'enable' => true,
'stores' => $allStores,
];
$jobs[] = [
'name' => 'send-push-notifications',
'command' => '$PHP_BIN vendor/bin/console push-notification:send-push-notifications',
'schedule' => '* * * * *',
'enable' => true,
'stores' => $allStores,
];
- Create a push notification and subscribe to it.
Make sure that push notifications are sent based on the cron-job schedule by checking the
spy_push_notification_subscription_delivery_log
database table. This table contains a record for each unique combination of push notification and push notification subscription. - Create the push notification subscription record with
spy_push_notification_subscription.expired_at
with last year’s date. Make sure that outdated push notification subscription is removed based on the cron-job schedule by checking thespy_push_notification_subscription
database table.
To optimize job execution and manage job concurrency, we recommend configuring and using the following pre-installed Jenkins plugins:
Priority Sorter Plugin
: lets you set priorities for your Jenkins jobs. You can assign higher priorities to critical jobs, ensuring they are executed before lower-priority jobs.Throttle Concurrent Builds Plugin
: provides fine-grained control over how many concurrent builds of a job can run. You can use it to limit the number of parallel executions for resource-intensive jobs, preventing overloading your Jenkins environment.
Thank you!
For submitting the form